ReactJS:点击时动态添加组件

我有一个菜单按钮,按下时必须添加一个新组件.它似乎有效(如果我手动调用该函数来添加它们显示的组件).问题是,如果我单击按钮,它们不会显示出来,我想是因为我应该使用 setState 来重绘它们.我不确定如何在另一个函数/组件中调用另一个组件的 setState.

这是我的 index.js

从'react'导入反应;从 'react-dom' 导入 ReactDOM;导入'./index.css';从'./Menu'导入菜单;从 './serviceWorker' 导入 * 作为 serviceWorker;从'./Block.js'导入块;ReactDOM.render(<div className="主容器"><菜单/><方块/></div>, document.getElementById('root'));//如果你想让你的应用离线工作和加载更快,你可以改变//unregister() 到 register() 下面.请注意,这带有一些陷阱.//了解更多关于服务工作者的信息:serviceWorker.unregister();

然后我有 Menu.js

从'react'导入反应;导入'./Menu.css';从'./Block.js'导入{blocksHandler};类菜单扩展 React.Component {构造函数(道具){超级(道具);this.state = {值:''};this.handleAdd = this.handleAdd.bind(this);}处理添加(事件){blocksHandler.add('lol');console.log(blocksHandler.render());}使成为() {返回 (<div className="菜单"><header className="菜单标题"><button className="Menu-button" onClick={this.handleAdd}>添加块</button></标题></div>);}}导出默认菜单;

最后是 Block.js

从'react'导入反应;导入'./Block.css';//此函数将组件添加到数组并返回它们让 blocksHandler = (function() {让块 = [];返回 {添加:功能(块){blocks.push(block);},渲染:函数(){返回块;}}})();类块扩展 React.Component {构造函数(道具){超级(道具);这个.state = {标题: '',内容: ''};this.handleChange = this.handleChange.bind(this);this.handleSubmit = this.handleSubmit.bind(this);}处理变化(事件){this.setState({[event.target.name]: event.target.value});}处理提交(事件){alert('提交了一个名字:' + this.state.title);event.preventDefault();}使成为() {返回 (<div className="块容器"><form onSubmit={this.handleSubmit}><div className="块标题"><标签>块标题:{blocksHandler.render().map(i => (<区块键={i}/>))}</div>)}}导出默认块;导出 {blocksHandler};

我是 React 的初学者,所以我什至不确定我的方法是否正确.感谢您提供的任何帮助.

解决方案

下面我敲了一个非常简单的父/子类型设置,..

Parent 负责渲染 Button,这里我只是使用了一个简单的编号数组.当您单击任何按钮时,它会调用 Parent 中的 setState,这反过来会导致 Parent 重新渲染它的 Children.

<块引用>

注意:我也使用过 React Hooks 来做这件事,我只是找到了更多自然且更易于使用.可以使用Classes,原理相同适用.

const {useState} = React;功能孩子(道具){常量 {标题} = 道具;常量 {lines, setLines} = props.pstate;返回<按钮点击={()=>{setLines([...lines,lines.length]);}}>{标题}</按钮>;}功能父(道具){const [行,setLines] = useState([0]);return lines.map(m => <子键={m} caption={`点击${m}`} pstate={{lines, setLines}}/>);}ReactDOM.render(<React.Fragment><父母/></React.Fragment>, document.querySelector('#mount'));

<script crossorigin src="https://unpkg.com/react@16/umd/react.development.js"></脚本><script crossorigin src="https://unpkg.com/react-dom@16/umd/react-dom.development.js"></script><div id="mount"></div>

I have a menu button that when pressed has to add a new component. It seems to work (if I manually call the function to add the components they are shown). The problem is that if I click the button they are not shown, and I suppose because I should use setState to redraw them. I am not sure how to call the setState of another component within another function/component.

This is my index.js

import React from 'react';
import ReactDOM from 'react-dom';
import './index.css';
import Menu from './Menu';
import * as serviceWorker from './serviceWorker';
import Blocks from './Block.js';


ReactDOM.render(
    <div className="Main-container">
        <Menu />
        <Blocks />
    </div>
    , document.getElementById('root'));

// If you want your app to work offline and load faster, you can change
// unregister() to register() below. Note this comes with some pitfalls.
// Learn more about service workers:
serviceWorker.unregister();

Then I have the Menu.js

import React from 'react';
import './Menu.css';
import {blocksHandler} from './Block.js';

class Menu extends React.Component {

  constructor(props) {

    super(props);
    this.state = {value: ''};

    this.handleAdd = this.handleAdd.bind(this);

  }

  handleAdd(event) {
    blocksHandler.add('lol');
    console.log(blocksHandler.render());
  }

  render() {
    return (
      <div className="Menu">
        <header className="Menu-header">
          <button className="Menu-button" onClick={this.handleAdd}>Add block</button>
        </header>
      </div>
    );
  }
}

export default Menu;

And finally the Block.js

import React from 'react';
import './Block.css';

// this function adds components to an array and returns them

let blocksHandler = (function() {
    let blocks = [];
    return {
        add: function(block) {
            blocks.push(block);
        },
        render: function() {
            return blocks;
        }
    }
})();

class Block extends React.Component {
    constructor(props) {
        super(props);
        this.state = {
            title: '',
            content: ''
        };

        this.handleChange = this.handleChange.bind(this);
        this.handleSubmit = this.handleSubmit.bind(this);
    }

    handleChange(event) {
        this.setState({[event.target.name]: event.target.value});
    }

    handleSubmit(event) {
        alert('A name was submitted: ' + this.state.title);
        event.preventDefault();
    }

    render() {
      return (
        <div className="Block-container">
            <form onSubmit={this.handleSubmit}>
            <div className="Block-title">
                <label>
                    Block title:
                    <input type="text" name="title" value={this.state.value} onChange={this.handleChange} />
                </label>
            </div>
            <div className="Block-content">
                <label>
                    Block content:
                    <input type="text" name="content" value={this.state.value} onChange={this.handleChange} />
                </label>
            </div>
            <input type="submit" value="Save" />
            </form>
        </div>
      );
    }
}

class Blocks extends React.Component {

    render() {
        return (
            <div>
                {blocksHandler.render().map(i => (
                    <Block key={i} />
                ))}
            </div>
        )
    }
}


export default Blocks;
export {blocksHandler};

I am a React complete beginner so I'm not even sure my approach is correct. Thank you for any help you can provide.

解决方案

Below I've knocked up a really simple Parent / Child type setup,..

The Parent is responsible for rendering the Buttons, I just used a simple numbered array here. When you click any of the buttons, it calls the setState in the Parent, and this in turns causes the Parent to re-render it's Children.

Note: I've also used React Hooks to do this, I just find them more natural and easier to use. You can use Classes, the same principle applies.

const {useState} = React;

function Child(props) {
  const {caption} = props;
  const {lines, setLines} = props.pstate;
  return <button onClick={() => {
    setLines([...lines, lines.length]);
  }}>
    {caption}
  </button>;
}

function Parent(props) {
  const [lines, setLines] = useState([0]);  
  return lines.map(m => <Child key={m} caption={`Click ${m}`} pstate={{lines, setLines}}/>);
}


ReactDOM.render(<React.Fragment>
  <Parent/>
</React.Fragment>, document.querySelector('#mount'));

<script crossorigin src="https://unpkg.com/react@16/umd/react.development.js"></script>
<script crossorigin src="https://unpkg.com/react-dom@16/umd/react-dom.development.js"></script>
<div id="mount"></div>

相关文章