我对React还是很陌生,我正在尝试通过构建一个简单的notes应用程序进行练习。 Afaik /可以告诉您一切顺利,但是!我读到状态不应该手动更新,所以我正在复制状态数组并过滤出删除操作的结果。
但是失败! 相反,如果我控制台日志,它会从状态数组中正确删除要删除的元素,但是,当我在副本上调用setState()更新视图时,该列表是错误的!
由于某种原因,我的React列表总是从页面上直观地移除最后一个元素,然后出现,然后与我的状态不同步。
应用程序本身是带有嵌套列表和列表项组件的各种Form容器,它们使用来自form类的道具进行管理。
有人可以帮我弄清楚我在做什么错吗? 谢谢!
表单类
class NotesForm extends Component {
constructor(props) {
super(props);
const list = [
{ text: "Build out UI" },
{ text: "Add new note" },
{ text: "delete notes" },
{ text: "edit notes" }
];
this.state = {
'notes': list
};
// this.notes = list;
this.handleSubmit = this.handleSubmit.bind(this);
this.deleteNote = this.deleteNote.bind(this);
}
handleSubmit(e) {
e.preventDefault();
if (this.input.value.length === 0) { return; }
this.state.notes.push({text: this.input.value});
this.setState({ notes: this.state.notes });
this.input.value = "";
}
// BUG - deletes WRONG note!!
deleteNote(note) {
console.log({'DELETE_NOTE': note.text})
// var list = _.clone(this.state.notes);
var list = [...this.state.notes];
var filteredNotes = _.filter(list, function(n) {
return (n.text !== note.text);
})
console.log({
'list': list,
'filteredNotes': filteredNotes
})
this.setState({ notes: filteredNotes });
}
render() {
return (
<div className="row notes-form">
<div className="col-xs-12">
<form onSubmit={this.handleSubmit}>
<input type="text" className="new-note-input" ref={(input) => this.input = input} />
<br />
<button className="add-btn btn btn-info btn-block" type="button" onClick={this.handleSubmit}>Add</button>
<br />
<NotesList notes={this.state.notes} deleteNote={this.deleteNote} />
</form>
</div>
</div>
);
}
}
列表类别
class NotesList extends Component {
constructor(props) {
super(props);
}
render() {
return (
<ul className="notes-list">
{this.props.notes.map((n, index) => <NotesListItem key={index} note={n} deleteNote={this.props.deleteNote} />)}
</ul>
);
}
}
列出商品类别
class NotesListItem extends Component {
constructor(props) {
super(props);
this.state = {
'text': props.note.text
};
this.delete = this.delete.bind(this);
}
delete() {
this.props.deleteNote(this.props.note);
}
render() {
return (
<li className="notes-list-item">
<span className="item-text">{this.state.text}</span>
<div className="notes-btn-group btn-group" role="group">
<button className="delete-btn btn btn-danger" type="button" onClick={this.delete}>×</button>
</div>
</li>
);
}
}
答案 0 :(得分:1)
对于index
中的每个key
,请尝试使用诸如唯一ID之类的东西代替NotesListItem
作为NotesList
。请参阅与此相关的question(实际上可能是重复的副本):
import React, { Component } from 'react';
import NotesListItem from './NotesListItem';
class NotesList extends Component {
constructor(props) {
super(props);
}
render() {
return (
<ul className="notes-list">
{this.props.notes.map((n, index) => <NotesListItem key={n.id} note={n} deleteNote={this.props.deleteNote} />)}
</ul>
);
}
}
export default NotesList;
您可以使用类似uuid之类的东西来生成“唯一” ID。生成唯一密钥的方法有很多,但这取决于您的数据结构。另外,使用唯一ID和基于ID进行过滤,可以帮助避免数组中的两个音符与基于text
值进行过滤的文本相同的情况。
import uuidv1 from 'uuid/v1';
// ...
handleSubmit(e) {
e.preventDefault();
if (this.input.value.length === 0) { return; }
this.state.notes.push({id: uuidv1(), text: this.input.value});
this.setState({ notes: this.state.notes });
this.input.value = "";
}
我只建议使用类似这样的内容,因为您的文本可能会重复。您甚至可以使用类似的方法来摆脱困境:
{this.props.notes.map((n, index) => <NotesListItem key={index + n.text} note={n} deleteNote={this.props.deleteNote} />)}
此外,您不应该像this.state.notes.push({text: this.input.value});
这样直接改变状态。尝试这样的事情:
handleSubmit(e) {
e.preventDefault();
if (this.input.value.length === 0) { return; }
const note = { id: uuidv1(), text: this.input.value };
const notes = [...this.state.notes, note];
this.setState({ notes });
this.input.value = "";
}
此外,我避免使用ref
处理受控输入,尤其是设置值。为什么不在状态上创建一个属性来与简单的onChange
事件处理程序结合使用来处理输入的值。这将与React Forms文档以及处理输入值更新的“标准” React方法保持一致:
handleChange(e) {
this.setState({ text: e.target.value });
}
handleSubmit(e) {
e.preventDefault();
if (this.state.text.length === 0) { return; }
const note = { id: uuidv1(), text: this.state.text };
const notes = [...this.state.notes, note];
this.setState({ text: '', notes });
}
render() {
// ...
<input type="text" className="new-note-input" value={this.state.text} onChange={this.handleChange} />
// ...
}
这里是example的动作。
另一个答案可能足以解决您的问题。我建议回顾一下React article文档中提到/链接的以下Keys,其中讨论了将索引用作键的潜在负面影响。
希望有帮助!
答案 1 :(得分:0)
一个组件的构造函数只运行一次。 React将重用组件实例,并为其传递新的道具。这里的问题是NodeListItem以其自身的本地状态缓存笔记的文本,并在render方法中使用该文本。当其父项通过道具向其传递新的音符时,它将不使用它。它使用的状态现在已经过时。
子级组件通常应使用父级传递的道具中的数据。
class NotesListItem extends Component {
constructor(props) {
super(props);
// The problem is this statement here
this.state = {
'text': props.note.text
};
this.delete = this.delete.bind(this);
}
}
这是NotesListItem
类的固定版本。
class NotesListItem extends Component {
constructor(props) {
super(props);
this.delete = this.delete.bind(this);
}
delete() {
this.props.deleteNote(this.props.note);
}
render() {
return (
<li className="notes-list-item">
<span className="item-text">{this.props.note.text}</span> {/* <-- using props */}
<div className="notes-btn-group btn-group" role="group">
<button
className="delete-btn btn btn-danger"
type="button"
onClick={this.delete}
>
×
</button>
</div>
</li>
);
}
}