带有从状态初始化的 react-final-form 中的 react-select 的条件下拉列表

IT技术 reactjs react-select react-final-form react-final-form-arrays
2021-05-03 14:40:16

我正在使用react-selectreact-final-form条件下拉菜单,其中第二个选择的选项由<PickOptions/>基于第一个选择的值的组件提供(感谢这个 SO answer)。

这是组件:

/** Changes options and clears field B when field A changes */
const PickOptions = ({ a, b, optionsMap, children }) => {
  const aField = useField(a, { subscription: { value: 1 } });
  const bField = useField(b, { subscription: {} });
  const aValue = aField.input.value.value;
  const changeB = bField.input.onChange;
  const [options, setOptions] = React.useState(optionsMap[aValue]);

  React.useEffect(() => {
    changeB(undefined); // clear B
    setOptions(optionsMap[aValue]);
  }, [aValue, changeB, optionsMap]);
  return children(options || []);
};

当第一个选择的值改变 时,它清除第二个选择changeB(undefined)我还通过将第二个选择设置为数组中的第一个选项initialValue由于我需要从状态初始化值,我最终得到了以下代码:

initialValue={
  this.state.data.options[index] &&
  this.state.data.options[index].secondOption
    ? this.state.data.options[index]
        .secondOption
    : options.filter(
        option => option.type === "option"
      )[0]
}

但它不起作用。状态的初始值不会传递到由 呈现的字段<PickOptions/>如果我changeB(undefined)从组件中删除,则传递值,但是当第一个选择的值发生更改时(即使选项已更新),第二个选择的输入值不会更新。这是我的代码和框的链接

我该如何解决?

2个回答

我能够通过获取该fields.map()部分映射的所有内容并将其包装在它自己的组件中以确保它们中的每个都有单独的状态来使其工作。然后我只是将changeB(undefined)函数放在useEffect钩子的返回调用中,以在用户为第一个选择选择不同的选项后清除辅助选择,如下所示:

React.useEffect(() => {
  setOptions(optionsMap[aValue]);

  return function cleanup() {
    changeB(undefined) // clear B
  };
}, [aValue, changeB, optionsMap]);

你可以在这个沙箱中看到它是如何工作的:React Final Form - Clear Secondary Selects

要更改辅助选择字段,您需要为PickOptions数组对应的选项类型传递一个额外的props我还订阅并跟踪前一个bValue以检查它是否存在于当前bValueSet数组中。如果它存在,我们不理会它,否则我们用其对应optionType数组中的第一个值更新它

  // subscibe to keep track of previous bValue
  const bFieldSubscription = useField(b, { subscription: { value: 1 } })
  const bValue = bFieldSubscription.input.value.value

  React.useEffect(() => {
    setOptions(optionsMap[aValue]);

    if (optionsMap[aValue]) {
      // set of bValues defined in array
      const bValueSet = optionsMap[aValue].filter(x => x.type === optionType);

      // if the previous bValue does not exist in the current bValueSet then changeB
      if (!bValueSet.some(x => x.value === bValue)) {
        changeB(bValueSet[0]); // change B
      }
    }

  }, [aValue, changeB, optionsMap]);

这是该方法的沙箱:React Final Form - Update Secondary Selects

我还将您的类组件更改为函数式组件,因为我可以更轻松地查看和测试正在发生的事情,但此方法也应该适用于您的类组件。

根据之前的答案,我在我的组件中得到了以下代码:

// subscibe to keep track of aField has been changed
const aFieldSubscription = useField(a, { subscription: { dirty: 1 } });

React.useEffect(() => {
  setOptions(optionsMap[aValue]);

  if (optionsMap[aValue]) {
    // set of bValues defined in array
    const bValueSet = optionsMap[aValue].filter(x => x.type === optionType);

    if (aFieldSubscription.meta.dirty) {
      changeB(bValueSet[0]); // change B
    }
  }
}, [aValue, changeB, optionsMap]);

通过这种方式,它检查 aField 是否已被用户更改,如果为真,则将 bField 的值设置为数组中的第一个选项。