为什么setState()不更新React上的渲染?

时间:2020-10-04 17:08:30

标签: javascript node.js reactjs

我正在尝试制作一个包含“单元”的2D游戏板,该单元具有名为isAlive的二进制状态值。用户可以通过单击它们来更改此值。该值使用不同的颜色指示给用户。 Board.js构造函数使用false值创建每个单元格。 面板上还有一个“重置”按钮,用于将所有单元的重置为false

当我单击它们时,单元格的值会正确更改。单击“重置”按钮时,我希望每个单元格的isAlive值为false。但是,在视觉上按下“重置”按钮时,板上没有任何变化(单元的颜色没有变化)。 console.log(this.state.cells);handleClick()方法上的Board.js行为所有单元格打印props: Object { isAlive: false }。那么,为什么细胞没有在视觉上更新呢?我在这里做什么错了?

Board.js

export default class Board extends React.Component {
    constructor(props) {
        super(props);
        this.state = {
            cells: []
        };
        this.rowNum = 10;
        this.colNum = 10;

        for (let i = 0; i < (this.rowNum * this.colNum); i++) {
            this.state.cells.push(<Cell isAlive={false}/>);
        }
        this.handleClick = this.handleClick.bind(this);
        this.getCols = this.getCols.bind(this);

    }

    getCols() {
        let cols = [];
        for (let i = this.rowNum; i <= (this.rowNum * this.colNum); i += this.rowNum) {
            cols.push(this.state.cells.slice(i - this.rowNum, i).map(function (item, i) {
                return <div key={i}>{item}</div>
            }));
        }
        return cols;
    }

    handleClick() {
        const newBoard = this.state.cells;
        newBoard.forEach((item, index, arr) => {
            arr[index] = <Cell isAlive={false}/>
        });
        this.setState({
                cells: newBoard
            }, function () {
                console.log(this.state.cells);
            }
        );
    }

    render() {
        return (<div className="background">
                <table className="gameBoard">
                    <tbody>
                    <tr>
                        {this.getCols().map(function (item, index) {
                            return <td className="col" key={index}>{item}</td>
                        })}
                    </tr>
                    </tbody>
                </table>
                <button onClick={this.handleClick}> Reset</button>
            </div>
        );
    }

}

Cell.js的重要部分:

export default class Cell extends React.Component {
    constructor(props) {
        super(props);
        this.colors = {
            dead: '#041b40',
            alive: '#b2e8f7',
            hover_dead: '#495d76',
            hover_alive: '#e6fffd'
        };

        this.state = {
            isAlive: this.props.isAlive,
            isMouseDown: false,
            isHovering: false
        };


        this.handleClick = this.handleClick.bind(this)
    }

    handleClick() {
        this.setState({
            isAlive: !this.state.isAlive,
            isMouseDown: this.state.isMouseDown,
            isHovering: this.state.isHovering

        })
    }

    ...

    determineColor() {
        if (this.state.isAlive && this.state.isHovering) {
            return this.colors.hover_alive;
        } else if (!this.state.isAlive && this.state.isHovering) {
            return this.colors.hover_dead;
        } else if (this.state.isAlive && !this.state.isHovering) {
            return this.colors.alive;
        }else {
            return this.colors.dead;
        }
    }

    render() {
        return (
            <div>
                <button className='square'
                        onClick={this.handleClick}
                        type='button'
                        style={{
                            backgroundColor: this.determineColor()
                        }}>
                </button>
            </div>
        )
    };


}

2 个答案:

答案 0 :(得分:0)

您需要在道具更改时更新内部状态:

export default class Cell extends React.Component {
    constructor(props) {
        super(props);
        this.colors = {
            dead: '#041b40',
            alive: '#b2e8f7',
            hover_dead: '#495d76',
            hover_alive: '#e6fffd'
        };

        this.state = {
            isAlive: this.props.isAlive,
            isMouseDown: false,
            isHovering: false
        };


        this.handleClick = this.handleClick.bind(this)
    }
    componentDidUpdate(prevProps, prevState) { //<--HERE
       this.setState({isAlive: this.props.isAlive})
    }
    handleClick() {
        this.setState({
            isAlive: !this.state.isAlive,
            isMouseDown: this.state.isMouseDown,
            isHovering: this.state.isHovering

        })
    }

    ...

    determineColor() {
        if (this.state.isAlive && this.state.isHovering) {
            return this.colors.hover_alive;
        } else if (!this.state.isAlive && this.state.isHovering) {
            return this.colors.hover_dead;
        } else if (this.state.isAlive && !this.state.isHovering) {
            return this.colors.alive;
        }else {
            return this.colors.dead;
        }
    }

    render() {
        return (
            <div>
                <button className='square'
                        onClick={this.handleClick}
                        type='button'
                        style={{
                            backgroundColor: this.determineColor()
                        }}>
                </button>
            </div>
        )
    };

答案 1 :(得分:0)

您必须使用componentDidUpdate

 componentDidUpdate(prevProps) {
    
      if (this.props.isAlive!== prevProps.isAlive) {
      this.setState({isAlive: this.props.isAlive})
      }
    }