дочерний компонент не перерисовывается при смене реквизита - PullRequest
0 голосов
/ 11 ноября 2018

Проблема:

На первой странице моего приложения отображается список счетов на продажу.

Я выбираю тот, который запускает функцию, принимающую в качестве аргумента идентификатор инвойса, по которому щелкнули, и получает его детали с помощью вызова API.

Компонент

Customer, отображаемый из SaleInvoice, имеет поле ввода (компонент Typeahead), которое должно показывать customerName, переданное из SaleInvoice, но не делает это правильно. Иногда оно пустое, и когда я возвращаюсь на первую страницу (список счетов-фактур) и выбираю другой счет-фактуру, customerName предыдущего счета-фактуры. Я проверил журнал консоли (см. Строку после <React.Fragment> в Customer компоненте) и вижу правильные значения customerName в состоянии, показанном редукторами.

Первоначально SaleInvoice был компонентом с состоянием, но не имел объекта состояния. Я сделал состояние с возможностью извлечения моих данных через API в componentWillMount. Из-за вышеуказанной проблемы, я попробовал это:

добавлено customerName в состоянии в SaleInvoice

изменено this.props.customerName на this.state.customerName в реквизитах клиента

используется

getDerivedStateFromProps ()

, который говорит, что я не могу использовать componentWillMount

Также попробовал shouldComponentUpdate и некоторые другие вещи. Ничего не работает Пожалуйста, помогите. Если нужно разместить больше кода, пожалуйста, дайте мне знать.

соответствующий срез редуктора

case actionTypes.INVOICE_BY_ID_SUCCESS:
  let customerData = action.payload[0];
  let saleInvoiceData = action.payload[1];
  let newState = Object.assign({}, state);
  newState.loading = false;
  newState.error = null;
  newState.customerInfo = {
    ...state.customerInfo,
    id: customerData.id,
    place: customerData.place,
    addressLineOne: customerData.address_line_one,
  };
  newState.saleInvoiceId = saleInvoiceData.id;
  newState.customerName = saleInvoiceData.customer_name;
  newState.serialNumber = saleInvoiceData.serial_number;
  newState.amountBeforeFreight = saleInvoiceData.amount_before_freight;
  newState.freight = saleInvoiceData.freight;
  newState.amountAfterFreight = saleInvoiceData.amount_after_freight;
  return newState;

SaleInvoiceContainer.js (исключая импорт)

const mapStateToProps = (state, ownProps) => {
  console.log(`ownProps ${ownProps}`);
  console.log(ownProps);
  return {
    customerLoading: state.saleInvoiceReducer.customerLoading,
    customerError: state.saleInvoiceReducer.customerError,

    productError: state.lineItemsReducer.error,
    productLoading: state.lineItemsReducer.loading,

    saleInvoiceError: state.saleInvoiceReducer.error,
    saleInvoiceLoading: state.lineItemsReducer.error,

saleInvoiceId: state.saleInvoiceReducer.saleInvoiceId,
customerData: state.saleInvoiceReducer.customerData, // data of all customers
productData: state.lineItemsReducer.productData, // data of all products
customerInfo: state.saleInvoiceReducer.customerInfo, // data of current customer
addingCustomer: state.saleInvoiceReducer.addingCustomer, // modal show/hide
customerName: state.saleInvoiceReducer.customerName, // input field name
grandTotal: subTotalSelector(state),
  };
};
const mapDispatchToProps = (dispatch, ownProps) => {
  return {
    fetchCustomer: () => dispatch(fetchCustomer()),
    fetchProduct: () => dispatch(fetchProduct()),
    getInvoiceById: () =>
      dispatch(getInvoiceById(ownProps.location.state.id)),
    onBlurCustomerName: event => dispatch(onBlurCustomerName(event)),
    stopAddingCustomer: () => dispatch(stopAddingCustomer()),
  };
};




const SaleInvoiceContainer = connect(
  mapStateToProps,
  mapDispatchToProps,
)(SaleInvoice);

SaleInvoice.js (без учета импорта)

class SaleInvoice extends React.Component {
  state = {
    customerName: '',
  };

  componentWillMount() {
    // if api is called from here, state will not update when api updates
    // props change cause re-render
    this.props.getInvoiceById();
    this.props.fetchCustomer();
    this.props.fetchProduct();
  }
  static getDerivedStateFromProps(nextProps, prevState) {
    if (nextProps.customeName !== prevState.customerName) {
      return {customerName: nextProps.customerName};
    } else return null;
  }
  componentDidUpdate(prevProps, prevState) {
    if (prevProps.customerName !== this.state.customerName) {
      let customerName = this.state.customerName;
      //Perform some operation here
      this.setState({customerName});
    }
  }



render() {
    console.log(this.props);
let ui = this.props.customerError ? (
  <p>Customers failed to load!</p>
) : (
  <Spinner />
);

let printLink = '/sale-invoice/' + this.props.saleInvoiceId + '/print';
let sui = this.props.productError ? (
  <p>Products failed to load!</p>
) : (
  <Spinner />
);
if (
  !this.props.customerLoading &&
  !this.props.customerError &&
  !this.props.error
) {
  console.log('Customers have been loaded');
  ui = (
    <React.Fragment>
      <Modal
        show={this.props.addingCustomer}
        modalClosed={this.props.stopAddingCustomer}
        customerData={this.props.customerData}
        name={this.state.customerName}
      />
      <div className={classes.mainContainerTitle}>
        {console.log(this.props.grandTotal)}
        <h5 className={classes.pageTitle}>Sale Invoice</h5>
        <NavLink className={classes.NavLink} to={printLink}>
          Print
        </NavLink>
        {/*<button>Print</button>*/}
      </div>
      <rs.Container
        fluid
        className={[classes.mainContainer, classes.containerFluid].join(
          '',
        )}>
        <rs.Row className={classes.firstRow}>
          <Customer
            customerData={this.props.customerData}
            onBlurCustomerName={this.props.onBlurCustomerName}
            customerInfo={this.props.customerInfo}
            customerName={this.state.customerName}
          />
          <SaleInvoiceSummary grandTotal={this.props.grandTotal} />
        </rs.Row>
      </rs.Container>
    </React.Fragment>
  );
}
if (
  !this.props.productLoading &&
  !this.props.productError &&
  !this.props.error
) {
  console.log('Products have been loaded');
  sui = (
    <React.Fragment>
      <rs.Container fluid className={classes.gridContainer}>
        <LineItemsContainer />
      </rs.Container>
    </React.Fragment>
  );
}
return (
  <React.Fragment>
    {ui}
    {sui}
  </React.Fragment>


   );
  }

    }

Customer.js (исключая импорт)

const Customer = props => {
  function _renderMenuItemChildren(option, props, index) {
    return [
      <Highlighter key="name" search={props.text}>
        {option.name}
      </Highlighter>,
      <div key="place">
        <small>Place: {option.place}</small>
      </div>,
    ];
  }
  return (
    <React.Fragment>
      {console.log(props.customerName)}
      <rs.Card col="sm-4" className={classes.firstCard}>
        <rs.CardHeader className={classes.cardHeader}>
          Customer Details
        </rs.CardHeader>
        <rs.CardBody className={classes.cardBodySaleInvoice}>
          <rs.Label>Name</rs.Label>
          <React.Fragment>
            <Typeahead
              className={classes.customerTypeahead}
              defaultInputValue={props.customerName}
              allowNew={true}
              newSelectionPrefix="Add New: "
              disabled={false}
              labelKey="name" // this determines what array key value to show
              multiple={false}
              options={props.customerData}
              placeholder="Choose a customer..."
              onBlur={event => props.onBlurCustomerName(event)}
              renderMenuItemChildren={_renderMenuItemChildren}
            />
            <rs.FormGroup />
          </React.Fragment>
          <div className={classes.customerCardBody}>
            <rs.Label>Address</rs.Label>
            <div className={classes.address}>
              {props.customerInfo.addressLineOne}
              <br />
              {props.customerInfo.addressLineTwo}
              <br />
              {props.customerInfo.address_line_three}
              <br />
              {props.customerInfo.contact_no_one}
              <br />
              {props.customerInfo.gst_number}
              <br />
              <button>Edit</button>
            </div>
          </div>
        </rs.CardBody>
      </rs.Card>
    </React.Fragment>
  );
};

(PS: я новичок в React, дополнительные комментарии / критика в отношении кода будут полезны)

1 Ответ

0 голосов
/ 12 ноября 2018

Я добавил компонент ввода html и обнаружил, что он работает правильно. Проблема была с компонентом Typeahead.

https://github.com/fmoo/react-typeahead/issues/74#issuecomment-112552406

Кроме того, теперь используется react-select вместо Typeahead, и это тоже работает правильно.

...