ant-design/components/form/demo/validate-customized.md
2016-01-25 15:00:03 +08:00

6.2 KiB

自定义校验规则

  • order: 13

密码校验实例。

这里使用了 validation 的 forceValidate(fields, callback) 方法,在对第一次输入的密码进行校验时会触发二次密码的校验。


import { Button, Form, Input, Row, Col } from 'antd';
import classNames from 'classnames';
const createForm = Form.create;
const FormItem = Form.Item;

function noop() {
  return false;
}

let Demo = React.createClass({
  getInitialState() {
    return {
      passBarShow: false, // 是否显示密码强度提示条
      rePassBarShow: false,
      passStrength: 'L', // 密码强度
      rePassStrength: 'L'
    };
  },

  handleSubmit(e) {
    e.preventDefault();
    this.props.form.validateFields((errors, values) => {
      if (!!errors) {
        console.log('Errors in form!!!');
        return;
      }
      console.log('Submit!!!');
      console.log(values);
    });
  },

  handleReset(e) {
    e.preventDefault();
    this.props.form.resetFields();
  },

  getPassStrenth(value, type) {
    if (value) {
      let strength;
      // 密码强度的校验规则自定义,这里只是做个简单的示例
      if (value.length < 6) {
        strength = 'L';
      } else if (value.length <= 9) {
        strength = 'M';
      } else {
        strength = 'H';
      }
      if (type === 'pass') {
        this.setState({ passBarShow: true, passStrength: strength });
      } else {
        this.setState({ rePassBarShow: true, rePassStrength: strength });
      }
    } else {
      if (type === 'pass') {
        this.setState({ passBarShow: false });
      } else {
        this.setState({ rePassBarShow: false });
      }
    }
  },

  checkPass(rule, value, callback) {
    const form = this.props.form;
    this.getPassStrenth(value, 'pass');

    if (form.getFieldValue('pass')) {
      form.validateFields(['rePass'], { force: true });
    }

    callback();
  },

  checkPass2(rule, value, callback) {
    const form = this.props.form;
    this.getPassStrenth(value, 'rePass');

    if (value && value !== form.getFieldValue('pass')) {
      callback('两次输入密码不一致!');
    } else {
      callback();
    }
  },

  renderPassStrengthBar(type) {
    const strength = type === 'pass' ? this.state.passStrength : this.state.rePassStrength;
    const classSet = classNames({
      'ant-pwd-strength': true,
      'ant-pwd-strength-low': strength === 'L',
      'ant-pwd-strength-medium': strength === 'M',
      'ant-pwd-strength-high': strength === 'H'
    });
    const level = {
      L: '低',
      M: '中',
      H: '高'
    };

    return (
      <div>
        <ul className={classSet}>
          <li className="ant-pwd-strength-item ant-pwd-strength-item-1"></li>
          <li className="ant-pwd-strength-item ant-pwd-strength-item-2"></li>
          <li className="ant-pwd-strength-item ant-pwd-strength-item-3"></li>
          <span className="ant-form-text">
            {level[strength]}
          </span>
        </ul>
      </div>
    );
  },

  render() {
    return (
      <Form horizontal form={this.props.form}>
        <Row>
          <Col span="18">
            <FormItem
              label="密码:"
              labelCol={{span: 6}}
              wrapperCol={{span: 18}}
              id="pass"
              options={{
                rules: [
                  {required: true, whitespace: true, message: '请填写密码'},
                  {validator: this.checkPass}
                ]
              }}>
              <Input type="password"
                onContextMenu={noop} onPaste={noop} onCopy={noop} onCut={noop}
                autoComplete="off"/>
            </FormItem>
          </Col>
          <Col span="6">
            {this.state.passBarShow ? this.renderPassStrengthBar('pass') : null}
          </Col>
        </Row>

        <Row>
          <Col span="18">
            <FormItem
              label="确认密码:"
              labelCol={{span: 6}}
              wrapperCol={{span: 18}}
              id="rePass"
              options={{
                rules: [{
                  required: true,
                  whitespace: true,
                  message: '请再次输入密码',
                }, {
                  validator: this.checkPass2,
                }],
              }}>
              <Input type="password"
                onContextMenu={noop} onPaste={noop} onCopy={noop} onCut={noop}
                autoComplete="off" />
            </FormItem>
          </Col>
          <Col span="6">
            {this.state.rePassBarShow ? this.renderPassStrengthBar('rePass') : null}
          </Col>
        </Row>

        <Row>
          <Col span="18">
            <FormItem
              wrapperCol={{span: 12, offset: 6}}>
              <Button type="primary" onClick={this.handleSubmit}>确定</Button>
              &nbsp;&nbsp;&nbsp;
              <Button type="ghost" onClick={this.handleReset}>重置</Button>
            </FormItem>
          </Col>
          <Col span="6" />
        </Row>
      </Form>
    );
  }
});

Demo = createForm()(Demo);

ReactDOM.render(<Demo />, mountNode);
.ant-pwd-strength {
  display: inline-block;
  margin-left: 8px;
  line-height: 32px;
  height: 32px;
  vertical-align: middle;
}

.ant-pwd-strength-item {
  float: left;
  margin-right: 1px;
  margin-top: 12px;
  width: 19px;
  height: 8px;
  line-height: 8px;
  list-style: none;
  background-color: #f3f3f3;
  transition: all 0.3s cubic-bezier(0.645, 0.045, 0.355, 1);
}

.ant-pwd-strength-item-1 {
  border-top-left-radius: 6px;
  border-bottom-left-radius: 6px;
}

.ant-pwd-strength-item-2 {
  width: 20px;
}

.ant-pwd-strength-item-3 {
  border-top-right-radius: 6px;
  border-bottom-right-radius: 6px;
  margin-right: 8px;
}

.ant-pwd-strength-low .ant-pwd-strength-item-1, .ant-pwd-strength-medium .ant-pwd-strength-item-1, .ant-pwd-strength-high .ant-pwd-strength-item-1 {
  background-color: #FAC450;
}

.ant-pwd-strength-medium .ant-pwd-strength-item-2, .ant-pwd-strength-high .ant-pwd-strength-item-2 {
  background-color: rgba(135, 208, 104, .6);
  filter:progid:DXImageTransform.Microsoft.gradient(startColorstr=#9987D068,endColorstr=#9987D068);
}

.ant-pwd-strength-high .ant-pwd-strength-item-3 {
  background-color: #87D068;
}