【问题标题】:Re-render component with useEffect when something changes in a child component当子组件发生变化时,使用 useEffect 重新渲染组件
【发布时间】:2020-02-27 19:09:51
【问题描述】:

我有一个允许用户在列表中添加任务的应用程序。任务从 API 中获取,并与“列表”组件一起显示。当用户从“AddButton”组件添加新任务时,该任务将存储在数据库中。 我希望在“AddButton”组件上发生handleSubmit 函数并将任务添加到数据库时重新渲染“List”组件。 “addTask”和“getTasks”正在从 API 获取数据。 提前感谢您的帮助。

列表组件

import React, { useState, useEffect } from 'react';
import { makeStyles } from '@material-ui/styles';
import List from '@material-ui/core/List';
import ListItem from '@material-ui/core/ListItem';
import ListItemText from '@material-ui/core/ListItemText';
import Moment from 'react-moment';
import { getTasks } from './services/getTasks';
import AddButton from './AddButton';
import './App.css';

const useStyles = makeStyles(theme => ({
  root: {
    display: 'flex',
    flexDirection: 'column',
    width: '100%',
    justifyContent: 'space-between',
    height: '100%',
    fontSize: '16px',
  },
  listItemLinkRoot: {
    paddingLeft: theme.spacing(3),
    width: '100%',
    '&:hover': {
      backgroundColor: '#212121',
      color: 'white',
    },
    display: 'flex',
    flexDirection: 'column',
    alignItems: 'flex-start',
  },

  buttonContainer: {
    display: 'flex',
    width: '100%',
    flexDirection: 'column',
    justifyContent: 'flex-end',
  },

  list: {
    flexGrow: 1,
    overflow: 'auto',
  },

  listItemText: {
    marginBottom: 8,
    // fontSize: 20,
  },
}));

function ListItemLink(props) {
  return <ListItem button component="a" {...props} />;
}

export default function TaskList() {
  const classes = useStyles();
  const [tasks, setTasks] = useState([]);

  useEffect(() => {
    const fetchData = async () => {
      const result = await getTasks();
      setTasks(result);
    };

    fetchData();
  }, []);

  return (
    <div className={classes.root}>
      <List classes={{ root: classes.list }}>
        {tasks.map(task => (
          <ListItemLink
            divider
            key={task.id}
            classes={{ root: classes.listItemLinkRoot }}
            href="simple-list"
          >
            <ListItemText
              classes={{ root: classes.listItemText }}
              primary={task.description}
            />
            <Moment
              classes={{ root: classes.listItemDate }}
              format="DD/MM/YYYY"
            >
              {task.createdAt}
            </Moment>
          </ListItemLink>
        ))}
      </List>
      <div className={classes.buttonContainer}>
        <AddButton classes={{ root: classes.add }} />
      </div>
    </div>
  );
}


添加按钮组件


import React, { useState } from 'react';
import { makeStyles } from '@material-ui/core/styles';
import Fab from '@material-ui/core/Fab';
import AddIcon from '@material-ui/icons/Add';
import TextField from '@material-ui/core/TextField';
import { addTask } from './services/postTask';

const useStyles = makeStyles(theme => ({
  cont: {
    display: 'flex',
    flexDirection: 'row',
    paddingBottom: '24px',
    justifyContent: 'space-between',
    backgroundColor: '#e0e0e0',
    width: '100%',
    alignItems: 'center',
    felxGrow: 1,
  },
  fab: {
    marginTop: theme.spacing(2),
    marginRight: theme.spacing(2),
    width: '100%',
  },

  textField: {
    marginLeft: theme.spacing(3),
    marginTop: 0,
    marginBottom: 0,
    flexGrow: 1,
  },
}));

export default function AddButton() {
  const classes = useStyles();
  const [task, setTask] = useState({
    description: '',
    completed: false,
  });

  const handleChange = ev => {
    setTask({ ...task, [ev.target.id]: ev.target.value });
  };

  const handleSubmit = () => {
    addTask(task);
  };

  return (
    <div className={classes.cont}>
      <TextField
        onChange={handleChange}
        id="description"
        label="Add a task"
        rowsMax="4"
        className={classes.textField}
        margin="normal"
      />
      <Fab
        onClick={handleSubmit}
        variant="extended"
        size="small"
        color="primary"
        aria-label="add"
        className={classes.fab}
      >
        <AddIcon />
        Add
      </Fab>
    </div>
  );
}

【问题讨论】:

  • 我会使用 name 而不是 id 作为输入,因为 id 对于整个文档必须是唯一的,并且没有组件可以或应该保证某个 id 已经被另一个组件占用。
  • 如果 List 管理任务,为什么不直接将 addTask 传递给列表中的按钮并让它在添加任务时设置状态?
  • 您好,感谢您的宝贵时间。您能否澄清您的第二条评论?
  • 关于您的第一条评论,您的意思是在 handleChange 函数中使用 name 而不是 id 吗?再次感谢

标签: javascript reactjs rerender use-effect


【解决方案1】:

在您的列表组件中,您可以使用 handleSubmit 函数并将其传递给您的子 AddButton 组件:

<AddButton classes={{ root: classes.add }} handleSubmit={handleSubmit} />

【讨论】:

  • 您好,感谢您的回答。如果我在 List 组件中有 handleSubmit 函数,由于 handleChange 函数位于不同的组件中,现在怎么可能将哪个任务添加到列表中?
【解决方案2】:

我能想到的一个解决方案是将 fetchData 函数移到 useEffect 钩子之外,并将其作为道具传递给 Button:


const fetchData = async () => {
     const result = await getTasks();
     setTasks(result);
 };

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

...

<AddButton classes={{ root: classes.add }} refetch={fetchData}/>

然后在 AddButton 中(假设 addTask() 是异步的)。

const handleSubmit = () => {
    addTask(task)
      .then(res => props.refetch())
  };

虽然在父组件中处理所有状态功能可能更有意义。

【讨论】:

  • 您好,感谢您的回复。我也想到了这一点,但我想像你提到的那样处理 List 组件中的状态。我认为我在按钮组件中处理它变得更加复杂。不过还是谢谢你的时间。
猜你喜欢
  • 2021-07-09
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2022-08-10
  • 2021-12-01
  • 2022-06-30
相关资源
最近更新 更多