- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在我的 React 应用程序中,我试图在单击复选框时捕获一些事件,以便进行一些状态过滤,并仅显示需要的项目。event
来自带有一些 name
的 child 的复选框。有 3 个复选框,所以我需要知道点击了哪个 name
。
其中一些
<input
type="checkbox"
name="name1"
onClick={filterHandler}
/>Something</div>
状态类似于
state = {
items = [
{
name: "name1",
useful: true
},{
name: "name2",
useful: false
}],
filteredItems: []
}
这是处理程序
filterHandler = (evt) => {
let checked = evt.target.checked;
let name = evt.target.name;
let filteredItems = this.state.items.filter(item => {
return item[name] === checked; // filtered [{...},{...}...]
});
// when filtered - push to state and show all from here in <ItemsList />
this.setState({ filteredItems })
}
“显示”的 ItemsList 组件是这样的:
<ItemsList
items={this.state.filteredItems.length === 0 ? this.state.items : this.state.filteredItems}
/>
当复选框是唯一的时 - 它工作正常。但我有三个这样的盒子——出现了并发症:
1)在选中下一个框时,我将使用原始的未过滤项目数组操作 - 因此为此,我需要已经过滤后的数组。2) 我不能使用我的 filteredItems
数组 reviously filtered,因为取消选中该框时该数组变空。拥有第三个“临时”数组似乎有点奇怪。
我试过这种方式,也很相似
this.setState({
filteredItems: this.state.items.filter(item => {
if (item[name] === checked) {
console.log('catch');
return Object.assign({}, item)
} else {
console.log('no hits')
}
})
这几乎是好的,但是当取消选中 filteredItems
时,它们会填充相反的值((
感觉有更好的方法,请指教。
最佳答案
您可以通过存储过滤器的选中状态来实现。
例如,您的状态可能类似于:
state = {
items: [
{
name: "name1",
useful: true
},
{
name: "name2",
useful: false
}
],
filters: { 'name1': false, 'name2': false}, // key value pair name:checked
filteredItems: []
};
然后您的点击/更改处理程序将更新过滤列表和实际过滤器状态(已检查的内容)。
举个例子:
(更新:根据评论中的要求大量评论)
// Using syntax: someFunc = (params) => { ... }
// To avoid having to bind(this) in constructor
onChange = evt => {
// const is like var and let but doesn't change
// We need to capture anything dependent on
// evt.target in synchronous code, and
// and setState below is asynchronous
const name = evt.target.name;
const checked = evt.target.checked;
// Passing function instead of object to setState
// This is the recommended way if
// new state depends on existing state
this.setState(prevState => {
// We create a new object for filters
const filters = {
// We add all existing filters
// This adds them with their existing values
...prevState.filters,
// This is like:
// filters[name] = checked
// which just overrides the value of
// the prop that has the name of checkbox
[name]: checked
};
// Object.keys() will return ["name1", "name2"]
// But make sure that "filters" in
// our initial state has all values
const activeFilterNames = Object.keys(filters).filter(
// We then filter this list to only the ones that
// have their value set to true
// (meaning: checked)
// We set this in the `const filter =` part above
filterName => filters[filterName]
);
// We get the full list of items
// (Make sure it's set in initial state)
// Then we filter it to match only checked
const filteredItems = prevState.items.filter(item =>
// For each item, we loop over
// all checked filters
// some() means: return true if any of the
// array elements in `activeFilterNames`
// matches the condition
activeFilterNames.some(
// The condition is simply the filter name is
// the same as the item name
activeFilterName => activeFilterName === item.name
)
);
// The object returned from setState function
// is what we want to change in the state
return {
// this is the same as
// { filter: filters,
// filteredItems: filteredItems }
// Just taking advantage of how const names
// are the same as prop names
filters,
filteredItems
};
});
};
我在这里使用 JS/Babel 的最新功能,但希望代码是清晰的。在输入 setState()
evt.target
这是一个完整的组件示例:
import * as React from "react";
import ReactDOM from "react-dom";
import "./styles.css";
class App extends React.Component {
state = {
items: [
{
name: "name1",
useful: true
},
{
name: "name2",
useful: false
}
],
filters: { name1: false, name2: false },
filteredItems: []
};
// Using syntax: someFunc = (params) => { ... }
// To avoid having to bind(this) in constructor
onChange = evt => {
// const is like var and let but doesn't change
// We need to capture anything dependent on
// evt.target in synchronous code, and
// and setState below is asynchronous
const name = evt.target.name;
const checked = evt.target.checked;
// Passing function instead of object to setState
// This is the recommended way if
// new state depends on existing state
this.setState(prevState => {
// We create a new object for filters
const filters = {
// We add all existing filters
// This adds them with their existing values
...prevState.filters,
// This is like:
// filters[name] = checked
// which just overrides the value of
// the prop that has the name of checkbox
[name]: checked
};
// Object.keys() will return ["name1", "name2"]
// But make sure that "filters" in
// our initial state has all values
const activeFilterNames = Object.keys(filters).filter(
// We then filter this list to only the ones that
// have their value set to true
// (meaning: checked)
// We set this in the `const filter =` part above
filterName => filters[filterName]
);
// We get the full list of items
// (Make sure it's set in initial state)
// Then we filter it to match only checked
const filteredItems = prevState.items.filter(item =>
// For each item, we loop over
// all checked filters
// some() means: return true if any of the
// array elements in `activeFilterNames`
// matches the condition
activeFilterNames.some(
// The condition is simply the filter name is
// the same as the item name
activeFilterName => activeFilterName === item.name
)
);
// The object returned from setState function
// is what we want to change in the state
return {
// this is the same as
// { filter: filters,
// filteredItems: filteredItems }
// Just taking advantage of how const names
// are the same as prop names
filters,
filteredItems
};
});
};
renderCheckboxes() {
return Object.keys(this.state.filters).map((name, index) => {
return (
<label key={index}>
<input
onChange={this.onChange}
type="checkbox"
checked={this.state.filters[name]}
name={name}
/>
{name}
</label>
);
});
}
render() {
const items = this.state.filteredItems.length
? this.state.filteredItems
: this.state.items;
return (
<div>
<div>{this.renderCheckboxes()}</div>
<ul>
{items.map(item => (
<li key={item.name}>
{item.name}
{item.useful && " (useful)"}
</li>
))}
</ul>
</div>
);
}
}
const rootElement = document.getElementById("root");
ReactDOM.render(<App />, rootElement);
您可以从这里现场试用:
https://codesandbox.io/embed/6z8754nq1n
您当然可以根据需要创建不同的变体。例如,您可以选择将过滤移动到渲染函数而不是更改事件,或者存储您如何存储所选过滤器等,或者按原样使用它。最适合你的:)
关于javascript - 在 React 应用程序中处理复选框,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51264335/
我是一名优秀的程序员,十分优秀!