Доступ к useEffect в Parent из дочернего компонента - PullRequest
1 голос
/ 28 марта 2020

Я использую реагирование и пытаюсь выяснить, как вызвать useEffect (для родителя) из дочернего компонента.

У меня есть дочерний компонент, который содержит модальное поле с полями ввода, которые передаются в мою БД .

import React, { useState } from 'react';
import { Modal, Button, Form } from 'react-bootstrap';

const AddContact = () => {

  // Modal Setup
  const [ showModal, setShowModal ] = useState(false);
  const handleClose = () => setShowModal(false);
  const handleShow = () => setShowModal(true);

  // Using state to hold the values for each field in the form
  const [first, setFirst] = useState('');
  const [last, setLast] = useState('');

  const handleSubmit = (e) => {
    e.preventDefault(); // prevents form default action
    const addContact = async () => {
      const result = await fetch('/api/crm/add', {
      method: 'POST',
      body: JSON.stringify({ first_name: first, last_name: last }),
      headers: {
        'Content-Type': 'application/json',
      }
      });
      const body = await result.json();
    }
    addContact();
    handleClose();
  };

  return (
    <>
      <Button onClick={handleShow} variant="primary" size="sm">Add Contact</Button>
      <Modal show={showModal} onHide={handleClose}>
        <Modal.Header closeButton>
          <Modal.Title>Add Contact</Modal.Title>
        </Modal.Header>
        <Modal.Body>
            <label for="formfirst">First Name: </label><br />
            <input 
              id="formfirst" 
              name="formfirst" 
              type="text" 
              value={first} 
              onChange={e => setFirst(e.target.value)}
            />
            <br/>
            <label for="formlast">Last Name: </label><br />
            <input 
              id="last" 
              name="last" 
              type="text" 
              value={last} 
              onChange={e => setLast(e.target.value)} 
            /> <br/>
            <Form.Group>

            </Form.Group>
        </Modal.Body>
        <Modal.Footer>
          <Button variant="secondary" onClick={handleClose}>Cancel</Button>
          <Button variant="primary" onClick={handleSubmit}>Submit</Button>
        </Modal.Footer>
      </Modal>
    </>
  );
}
export default AddContact;

У родительского компонента есть таблица данных, которую я хотел бы обновить sh, чтобы она отображала текущие данные:

import React, { useState, useEffect } from 'react';
import styled from 'styled-components';
import { Table } from 'react-bootstrap';

import AddContactForm from '../components/AddContactForm';

// React-Table
import {
  useTable,
} from 'react-table'

const ContactsPage = () => { 
  const [ tableData, setTableData ] = useState([]);

  useEffect( () => {
    const fetchData = async () => {
      const result = await fetch(`api/crm/get`);
      const body = await result.json();
      setTableData(body);
    }
    fetchData();
  },[]);

  const columns = React.useMemo(
    () => [
          {
            Header: 'First Name',
            accessor: 'first_name',
          },
          {
            Header: 'Last Name',
            accessor: 'last_name',
          } 
          ... SIMPLIFIED FOR CONCISENESS
    ],
    []
  )

  function ReactTable({ columns, data }) {
    const {
      getTableProps,
      ... REMOVED SOME REACT-TABLE FOR CONCISENESS
    } = useTable({
      columns,
      data,
      })

    return (
      <>
        <h2>Contacts</h2>
        <hr></hr>
        <div>
          <AddContactForm />
        </div>

        <Table striped bordered hover size="sm" {...getTableProps()}>
         ... REMOVED TABLE TO CONCISENESS
        </Table>
      </>
    );
  }


  const data = React.useMemo(() => tableData)

  return (
    <Styles>
      <ReactTable columns={columns} data={data} />
    </Styles>
  )
}

export default ContactsPage;

Как этого добиться? Я попытался превратить свой хук useEffect в функцию, которую я мог бы передать своему дочернему компоненту, но я получил сообщение о том, что эта новая функция не была функцией ???

1 Ответ

1 голос
/ 28 марта 2020

Не уверен, что имеет смысл передавать useEffect в качестве реквизита, вместо этого передайте функцию, используемую внутри useEffect, в качестве реквизита. Здесь

useEffect( () => {
    const fetchData = async () => {
      const result = await fetch(`api/crm/get`);
      const body = await result.json();
      setTableData(body);
    }
    fetchData();
  },[]);

Рефакторинг так:

// Wrapped in useCallback otherwise it would be recreated each time 
// this component rerenders, hence triggering useEffect below
let fetchData = React.useCallback(async () => {
      const result = await fetch(`api/crm/get`);
      const body = await result.json();
      setTableData(body);
    },[])

useEffect(() => {
  fetchData();
},[fetchData]);

и pass fetchData как реквизиты для модального , который вы можете вызвать после submit.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...