javascript - react 保持渲染 Prop 包裹组件的状态

标签 javascript reactjs

我通过 children Prop 作为函数创建了以下渲染 Prop 组件:

export class GenericAbstractLoader extends React.Component<IGenericAbstractLoaderRenderProps, any> {
    constructor(props: IGenericAbstractLoaderRenderProps) {
        super(props);
    }

    public render() {
        return this.props.isLoading ? (
            <div className="generic-abstract-loader-wrapper">
                <div className="generic-abstract-loader">
                    <img src={loader} />
                </div>
                {this.props.children(this.props)}
            </div>
        ) : (
            this.props.children(this.props)
        );
    }
}

如果 isLoading 为 false,它实际上只是渲染包装的组件,或者如果 prop 为 true,则添加一层加载。 它“非常完美”地工作。

我确实在另一个组件中以这种方式调用它:

public render() {
    return (
        <div>
            <button onClick={this.justDisable}>JUST SET FALSE TO STATE</button>
            <br />
            <button onClick={this.enableDisable}>Disable/Enable Elements</button>
            <br />

            <GenericAbstractLoader customProp="custom prop" isLoading={this.state.shouldDisable}>
                {props => <HomeTestForm {...props} />}
            </GenericAbstractLoader>
        </div>
    );
}

如您所见,包装组件是一个简单的 HomeTestForm,它只包含一个输入文本:

export class HomeTestForm extends React.Component<IGenericAbstractLoaderHOCProps, any> {
    constructor(props: IGenericAbstractLoaderHOCProps) {
        super(props);
        this.handleChange = this.handleChange.bind(this);
        this.state = { value: 'Initial Value' };
    }

    public render() {
        return (
            <div>
                    <input type="text" value={this.state.value} onChange={this.handleChange} />
            </div>
        );
    }

    public handleChange(event: any) {
        this.setState({ value: event.target.value });
    }
}

我的问题是,当我切换 isLoading Prop 时,包装组件不会保留状态,因此如果我更改内部文本字段的值,在添加加载层时,并且在删除它时,不采用该值,因此输入字段仍使用初始值呈现。

使用 render props 创建持久包装组件的正确方法是什么?

最佳答案

您的组件状态正在重置,因为当您的加载状态发生变化时,React 正在创建一个全新的组件。如果将 console.log 添加到 HomeTestForm 的构造函数中,就可以看到这些。 React 认为这是必要的,因为 GenericAbstractLoader 中的三元运算符。如果您可以将 GenericAbstractLoader 中的 render 函数重构为类似于下面的示例,您可以为 React 提供跨渲染持久保存组件实例所需的上下文。

render() 
  return (
    <div className={this.props.isLoading ? "generic-abstract-loader-wrapper" : ""}>
      {this.props.isLoading && <img src={loader} />}
      {this.props.children(this.props)}
    </div>
  );
}

React 的文档在 Reconciliation 下对此有一个简短的部分。 .这种情况属于不同类型的元素标题。

此外,与您的问题无关,根据您当前的示例,您不需要使用渲染 Prop 。 GenericAbstractLoader 中的 render 函数可以只渲染 {this.props.children} 而无需函数调用,您可以将 Prop 直接放在 child 身上组件如下例所示。您可能只是简化了 SO 的示例,并且遇到需要渲染 Prop 的情况,但我想指出这一点以防万一。

<GenericAbstractLoader isLoading={this.state.shouldDisable}>
  <HomeTestForm customProp="custom prop" />
</GenericAbstractLoader

关于javascript - react 保持渲染 Prop 包裹组件的状态,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53366686/

相关文章:

javascript - 在成功函数内的 AJAX 请求中访问 "data"对象

javascript - 如何获取dojo/store/JsonRest中的响应头?

javascript - 检查 lynx 中的元素

reactjs - Typescript + Recompose - 生命周期中的类型错误

reactjs - 如何在 React 中重定向到外部链接?

javascript - 为什么多次调用 setState() 时状态不能正确更新

javascript - Shadowbox 3.0.3 在 Shadowboxed 窗口关闭时重新加载父级

javascript - 如何使用 Javascript 获取选定的多个下拉列表的值?

javascript - 从默认路由重定向到另一条路由时出现滞后

javascript - 为什么这个handleBlur函数带有事件参数,而函数内部没有使用事件参数呢?