gpt4 book ai didi

javascript - 使用 Context API 在非嵌套组件之间传递变量

转载 作者:行者123 更新时间:2023-12-01 00:48:41 24 4
gpt4 key购买 nike

假设我有两个未嵌套的组件:一个按钮和一个面板。单击按钮时,面板将根据之前的状态显示或隐藏(如开/关开关)。它们不是嵌套组件,因此结构如下所示:

<div>
<Toolbar>
<Button />
</Toolbar>
<Content>
...
<ButtonPanel />
</Content>
</div>

我无法更改 DOM 的结构。除了按钮和面板组件之外,我也无法修改任何其他组件。

但是,ButtonButtonPanel 组件是相关的,并且将在整个解决方案中一起使用。我需要将属性传递给面板以让它知道何时显示或何时隐藏。我正在考虑使用 Context API 来完成此操作,但我认为我做错了,并且该属性永远不会更新。

这是我的代码:

上下文

import React from 'react';

export const ButtonContext = React.createContext({
showPanel: false,
});

按钮

import React, { Component } from 'react';
import { ButtonContext } from './ButtonContext';

class Button extends Component {
constructor() {
super();
this.state = {
showPanel: false,
};
}

render() {
return (
<ButtonContext.Provider value={{ showPanel: this.state.showPanel }}>
<li>
<a
onClick={() => this.setState({ showPanel: !this.state.showPanel }, () => console.log('Changed'))}
>
<span>Button</span>
</a>
</li>
</ButtonContext.Provider>
);
}
}

export { Button };

面板

import React, { Component } from 'react';
import { Panel, ListGroup, ListGroupItem } from 'react-bootstrap';
import { ButtonContext } from './ButtonContext';

class ButtonPanel extends Component {
static contextType = ButtonContext;
render() {
return (
<ButtonContext.Consumer>
{
({ showPanel }) => {
if (showPanel) {
return (
<Panel id="tasksPanel">
<Panel.Heading >Panel Heading</Panel.Heading>
<ListGroup>
<ListGroupItem>No Items.</ListGroupItem>
</ListGroup>
</Panel>
);
}
return null;
}
}
</ButtonContext.Consumer>
);
}
}

export { ButtonPanel };

我还尝试简单地访问 ButtonPanel 组件中的上下文,如下所示:

render() {
const context = this.context;
return context.showPanel ?
(
<Panel id="tasksPanel">
<Panel.Heading >Tasks</Panel.Heading>
<ListGroup>
<ListGroupItem className="tasks-empty-state">No tasks available.</ListGroupItem>
</ListGroup>
</Panel>
)
:
null;
}

我做错了什么?

谢谢

最佳答案

来自 React docs :

Accepts a value prop to be passed to consuming components that are descendants of this Provider.

所以这意味着 <ButtonContext.Provider>必须包裹<ButtonContext.Consumer>或者它必须位于组件层次结构的较高位置。

因此,根据您的用例,您可以这样做:

// This app component is the div that wraps both Toolbar and Content. You can name it as you want
class App extends Component {
state = {
showPanel: false,
}

handleTogglePanel = () => this.setState(prevState => ({ togglePanel: !prevState.togglePanel }));

render() {
return (
<ButtonContext.Provider value={{ showPanel: this.state.showPanel, handleTogglePanel: this.handleTogglePanel }}>
<Toolbar>
<Button />
</Toolbar>
<Content>
<ButtonPanel />
</Content>
</ButtonContext.Provider>
);
}
}

class Button extends Component {
...
<ButtonContext.Consumer>
{({ handleTogglePanel }) => <a onClick={handleTogglePanel} />}
</ButtonContext.Consumer>
}

class ButtonPanel extends Component {
...
<ButtonContext.Consumer>
{({ showPanel }) => showPanel && <Panel>...</Panel>}
</ButtonContext.Consumer>
}

关于javascript - 使用 Context API 在非嵌套组件之间传递变量,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57167905/

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