在调用高阶组件时,在React中访问类外的prop

时间:2016-10-06 01:39:28

标签: javascript reactjs react-redux redux-form

我正在尝试使用高阶组件(HOC)模式来重用一些连接到state的代码并使用Redux Form formValueSelector方法。

formValueSelector需要一个引用表单名称的sting。我想动态设置它,并且只要我需要项目的值就可以使用这个HOC。我使用项目值进行计算。

在下面的代码中,HOC传递了组件和字符串。我想将它设置为从父(表单)传入的prop formName。

我是HOC模式的新手,所以任何提示都会受到最高的赞赏。

HOC

import React, { Component } from 'react';
import { connect } from 'react-redux';
import { formValueSelector } from 'redux-form';

function FormItemsValueSelectorHOC(FormElement, formName) {
  const selector = formValueSelector(formName);
  @connect(state => {
    console.log(state);
    const items = selector(state, 'items');
    return {
      items
    };
  }, null)
  class Base extends Component {
    render() {
      return (
        <FormElement {...this.props} />
      );
    }
  }
  return Base;
}
export default FormItemsValueSelectorHOC;

包装组件

import React, { Component, PropTypes } from 'react';
import { Field } from 'redux-form';
import formItemsValueSelectorHOC from '../Utilities/FormItemsValueSelectorHOC';

const renderField = ({ placeholder, input, type}) => {
  return (
    <input
      {...input}
      placeholder={placeholder}
      type={type}
    />
  );
};

class StatementLineItemDesktop extends Component {
  static propTypes = {
    items: PropTypes.array.isRequired,
    index: PropTypes.number.isRequired,
    item: PropTypes.string.isRequired,
    fields: PropTypes.object.isRequired,
    formName: PropTypes.string.isRequired
  };

  calculateLineTotal(items, index) {
    let unitPrice = '0';
    let quantity = '0';
    let lineTotal = '0.00';
    if (items) {
      if (items[index].price) {
        unitPrice = items[index].price.amountInCents;
      }
      quantity = items[index].quantity;
    }
    if (unitPrice && quantity) {
      lineTotal = unitPrice * quantity;
      lineTotal = Number(Math.round(lineTotal+'e2')+'e-2').toFixed(2); 
    }
    return <input value={lineTotal} readOnly placeholder="0.00" />;
  }

  render() {
    const { items, index, item, fields, formName} = this.props;
    return (
      <tr id={`item-row-${index}`} key={index} className="desktop-only">
        <td>
          <Field
            name={`${item}.text`}
            type="text"
            component={renderField}
            placeholder="Description"
          />
        </td>
        <td>
          <Field
            name={`${item}.quantity`}
            type="text"
            component={renderField}
            placeholder="0.00"
          />
        </td>
        <td>
          <Field
            name={`${item}.price.amountInCents`}
            type="text"
            component={renderField}
            placeholder="0.00"
          />
        </td>
        <td className="last-col">
          <Field
            name={`${item}.price.taxInclusive`}
            type="hidden"
            component="input"
          />
          {::this.calculateLineTotal(items, index)}
          <a
            className="remove-icon"
            onClick={() => fields.remove(index)}
          >
            <span className="icon icon-bridge_close" />
          </a>
        </td>
      </tr>
    );
  }
}

export default formItemsValueSelectorHOC(StatementLineItemDesktop, 'editQuote');

1 个答案:

答案 0 :(得分:6)

TLDR:使用ownProps参数

你应该做的草案

import React, { Component } from 'react';
import { connect } from 'react-redux';
import { formValueSelector } from 'redux-form';

function FormItemsValueSelectorHOC(FormElement) {
  @connect((state, ownProps) => {
    const formName = ownProps.formName;
    const selector = formValueSelector(formName);
    const items = selector(state, 'items');
    return {
      items
    };
  }, null)
  class Base extends Component {
    render() {
      // Now in here you should omit `formName` from the props you are
      // passing to your Form Element since it's not used overthere
      return (
        <FormElement {...this.props} />
      );
    }
  }
  return Base;
}
export default FormItemsValueSelectorHOC;

这将是您创建连接组件的方式

formItemsValueSelectorHOC(StatementLineItemDesktop);

这就是你使用它的方式

<ConnectedStatementLineItemDesktop formName={"editQuote"} />

让我再解释一下这项工作的方法

你缺少的东西是React-Redux API,你应该更多地探索它,因为它已经考虑了很多这种用例

因此,React-Redux的connect函数的第一个参数称为mapStateToProps

这是它的签名:

mapStateToProps(state, [ownProps]): stateProps

我要注意的是ownProps参数。

ownProps包含传递给连接组件的所有道具。

只是为了澄清,你有这些组件

  • 常规组件:即StatementLineItemDesktopBase
  • 连接组件:即ConnectedBase = connect(mapStateToProps)(Base)

因此,根据此信息,您的名为FormItemsValueSelectorHOC的HOC函数会返回ConnectedBase的变体。

因此,无论您传递给ConnectedBase的道具或从FormItemsValueSelectorHOC返回的任何组件,您都可以从ownProps

访问它们

BTW,在您的特定情况下,这是您的mapStateToProps

function mapStateToProps(state, ownProps) {
  const formName = ownProps.formName;
  const selector = formValueSelector(formName);
  const items = selector(state, 'items');
  return {
    items
  };
}

需要注意的是connect也是一个HOC,所以从逻辑上来说,你可以认为你使用普通组件做的大部分工作都可以用连接的组件完成,I建议你阅读connect来源,它不长也不困难,所以你可以探索和理解更多这个答案。

我希望这会有所帮助。