antd3的表单实现(HOC解决方案)

使用的是装饰器 @createForm

js 复制代码
// MyRCFormPage.js
import React, { Component } from "react";
import createForm from "../components/my-rc-form/index";
import Input from "../components/input";

const nameRules = { required: true, message: "请输入姓名!" };
const passwordRules = { required: true, message: "请输入密码!" };

@createForm
class MyRCFormPage extends Component {
  componentDidMount() {
    this.props.form.setFieldsValue({ username: "default" });
  }
  submit = () => {
    this.props.form.validateFields((err, values) => {
      if (err) {
        console.log(err);
      } else {
        console.log(values);
      }
    });
  };
  render() {
    const { getFieldDecorator } = this.props.form;
	console.log('render');
		
    return (
      <div>
        <h3>MyRCFormPage</h3>
        {getFieldDecorator("username", { rules: [nameRules] })(
          <Input placeholder="Username" />
        )}
        {getFieldDecorator("password", { rules: [passwordRules] })(
          <Input placeholder="Password" />
        )}
        <button onClick={this.submit}>submit</button>
      </div>
    );
  }
}

export default MyRCFormPage;
js 复制代码
// input.js
import { Component } from "react";

const Input = (props) => {
  return <input {...props} />;
};

class CustomizeInput extends Component {
  render() {
    const { value = "", ...otherProps } = this.props;
    return (
      <div style={{ padding: 10 }}>
        <Input style={{ outline: "none" }} value={value} {...otherProps} />
      </div>
    );
  }
}

export default CustomizeInput;
js 复制代码
// my-rc-form/index.js
import React, { Component } from "react";

export default function createForm(Cmp) {
  return class extends Component {
    constructor(props) {
      super(props);
      this.state = {};
      this.options = {}; // 保存表单项的配置
    }
    handleChange = (e) => {
      const { name, value } = e.target;
      this.setState({
        [name]: value,
      });
    };
    getFieldDecorator = (field, option) => (InputCmp) => {
      this.options[field] = option;
      return React.cloneElement(InputCmp, {
        name: field,
        value: this.state[field],
        onChange: this.handleChange,
      });
    };
    setFieldsValue = (newStore) => {
      this.setState(newStore);
    };
    getFieldsValue = () => {
      return {
        ...this.state,
      };
    };
    validateFields = (callback) => {
      let err = [];
      // 校验
      for (let field in this.options) {
        if (this.state[field] === undefined) {
          err.push({ [field]: "error" });
        }
      }
      callback(err.length ? err : null, this.state);
    };
    getForm = () => {
      return {
        form: {
          getFieldDecorator: this.getFieldDecorator,
          setFieldsValue: this.setFieldsValue,
          getFieldsValue: this.getFieldsValue,
          validateFields: this.validateFields,
        },
      };
    };
    render() {
      return <Cmp {...this.props} {...this.getForm()} />;
    }
  };
}

现在的问题是当一个表单item发生改变时,整个表单都要重新执行render,如果表单非常大的时候。。。

相关推荐
梦帮科技6 小时前
Node.js配置生成器CLI工具开发实战
前端·人工智能·windows·前端框架·node.js·json
VT.馒头7 小时前
【力扣】2695. 包装数组
前端·javascript·算法·leetcode·职场和发展·typescript
css趣多多7 小时前
一个UI内置组件el-scrollbar
前端·javascript·vue.js
-凌凌漆-7 小时前
【vue】pinia中的值使用 v-model绑定出现[object Object]
javascript·vue.js·ecmascript
C澒8 小时前
前端整洁架构(Clean Architecture)实战解析:从理论到 Todo 项目落地
前端·架构·系统架构·前端框架
C澒8 小时前
Remesh 框架详解:基于 CQRS 的前端领域驱动设计方案
前端·架构·前端框架·状态模式
Charlie_lll8 小时前
学习Three.js–雪花
前端·three.js
onebyte8bits8 小时前
前端国际化(i18n)体系设计与工程化落地
前端·国际化·i18n·工程化
C澒8 小时前
前端分层架构实战:DDD 与 Clean Architecture 在大型业务系统中的落地路径与项目实践
前端·架构·系统架构·前端框架
BestSongC8 小时前
行人摔倒检测系统 - 前端文档(1)
前端·人工智能·目标检测