【react】记录一次react组件props依赖异步数据(setFieldsValue)后传递form给子组件,再逐层传递给孙子组件引起的未渲染异常

背景

react祖父组件设置异步数据(setFieldsValue)后传递form给子组件,再逐层传递给孙子组件引起的未渲染异常,孙子组件如果不设置useEffect和useState去监听value的值进行重渲染,会出现获取得到value最新值,但是不进行渲染的异常

解决前后的代码对比

完整代码(异常):
javascript 复制代码
const GetText = (props) => {
  const { value, config, containerId, getFieldValue } = props;
  const [itemValue, setItemValue] = useState(value);
  const { formId } = config;
  const getFormId = useCallback(
    (suffix = '_param_countryPhoneCode') => {
      if (Array.isArray(formId)) {
        return [containerId, formId[0], `${formId[1]}${suffix}`];
      }
      return `${formId}${suffix}`;
    },
    [formId, containerId],
  );
  
  useEffect(() => {
    const pObj = getFieldValue(getFormId()) || {};
    if (!value.includes('+')) {
      const _itemValue = `${pObj.countryPhoneCode ? '+' : ''}${pObj.countryPhoneCode} ${value}`;
      setItemValue(_itemValue);
    }
  }, [getFieldValue, getFormId, value, getFieldValue(getFormId())]);

  return itemValue ? (
    <>
      <div style={{ fontSize: 12 }}>{itemValue}</div>
    </>
  ) : null;
};
完整代码(正常):
javascript 复制代码
const GetText = (props) => {
  const { value, config, containerId, getFieldValue } = props;
  const [itemValue, setItemValue] = useState();
  const { formId } = config;
  const getFormId = useCallback(
    (suffix = '_param_countryPhoneCode') => {
      if (Array.isArray(formId)) {
        return [containerId, formId[0], `${formId[1]}${suffix}`];
      }
      return `${formId}${suffix}`;
    },
    [formId, containerId],
  );

  // 注意,由于在changeRecordModal中有对变更记录表单项做了顺序标记,
  // 因此formId会多一个序号,在变更记录弹窗里,首次加载可能因为更新是异步的导致value没有正常渲染,
  // 所以这边必须借用useEffect和useState控制value的更新渲染
  useEffect(() => {
    const pObj = getFieldValue(getFormId()) || {};
    if (!value.includes('+')) {
      const _itemValue = `${pObj.countryPhoneCode ? '+' : ''}${pObj.countryPhoneCode} ${value}`;
      setItemValue(_itemValue);
    }
  }, [getFieldValue, getFormId, itemValue, value]);

  return itemValue ? (
    <>
      <div style={{ fontSize: 12 }}>{itemValue}</div>
    </>
  ) : null;
};
相关推荐
永乐春秋29 分钟前
WEB攻防-通用漏洞&文件上传&js验证&mime&user.ini&语言特性
前端
鸽鸽程序猿30 分钟前
【前端】CSS
前端·css
ggdpzhk32 分钟前
VUE:基于MVVN的前端js框架
前端·javascript·vue.js
小曲曲2 小时前
接口上传视频和oss直传视频到阿里云组件
javascript·阿里云·音视频
学不会•3 小时前
css数据不固定情况下,循环加不同背景颜色
前端·javascript·html
EasyNTS4 小时前
H.264/H.265播放器EasyPlayer.js视频流媒体播放器关于websocket1006的异常断连
javascript·h.265·h.264
活宝小娜5 小时前
vue不刷新浏览器更新页面的方法
前端·javascript·vue.js
程序视点5 小时前
【Vue3新工具】Pinia.js:提升开发效率,更轻量、更高效的状态管理方案!
前端·javascript·vue.js·typescript·vue·ecmascript
coldriversnow5 小时前
在Vue中,vue document.onkeydown 无效
前端·javascript·vue.js
我开心就好o5 小时前
uniapp点左上角返回键, 重复来回跳转的问题 解决方案
前端·javascript·uni-app