gpt4 book ai didi

javascript - 在 React 应用程序中处理复选框

转载 作者:行者123 更新时间:2023-11-29 18:51:14 24 4
gpt4 key购买 nike

在我的 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/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com