Schema
Schema 可以定义一个数据模型,用于对数据进行校验,可以对 <Form>
组件数据进行校验
- Schema.Model 定义数据模型对象
- Schema.Types 提供了一组数据类型:
- StringType
- NumberType
- ArrayType
- DateType
- ObjectType
- BooleanType
如何使用?
import { Schema } from 'rsuite';
const { StringType, NumberType } = Schema.Types;
const userModel = Schema.Model({
username: StringType().isRequired('用户名不能为空'),
email: StringType().isEmail('请输入正确的邮箱'),
age: NumberType('年龄应该是一个数字').range(18, 30, '年应该在 18 到 30 岁')
});
const checkResult = userModel.check({
username: 'foobar',
email: 'foo@bar.com',
age: 40
});
console.log(checkResult);
Output: checkResult
{
username: { hasError: false },
email: { hasError: false },
age: { hasError: true, errorMessage: '年应该在 18 到 30 岁' }
}
多重验证
StringType()
.minLength(6, '不能少于 6 个字符')
.maxLength(30, '不能大于 30 个字符')
.isRequired('该字段不能为空');
自定义验证
通过 addRule
函数自定义一个规则。
如果是对一个字符串类型的数据进行验证,可以通过 pattern
方法设置一个正则表达式进行自定义验证。
const model = Schema.Model({
field1: StringType().addRule(value => {
return /^[1-9][0-9]{3}\s?[a-zA-Z]{2}$/.test(value);
}, '请输入合法字符'),
field2: StringType().pattern(
/^[1-9][0-9]{3}\s?[a-zA-Z]{2}$/,
'请输入合法字符'
)
});
model.check({ field1: '', field2: '' });
/**
{
field1: {
hasError: true,
errorMessage: '请输入合法字符'
},
field2: {
hasError: true,
errorMessage: '请输入合法字符'
}
};
**/
自定义验证 - 多字段交叉验证
例如,验证两次输入密码是否一致
const model = Schema.Model({
password1: StringType().isRequired('该字段不能为空'),
password2: StringType().addRule((value, data) => {
if (value !== data.password1) {
return false;
}
return true;
}, '两次密码不一致')
});
model.check({ password1: '123456', password2: 'root' });
/**
{
password1: { hasError: false },
password2: {
hasError: true,
errorMessage: '两次密码不一致'
}
};
**/
自定义验证 - 异步校验
例如,校验邮箱是否重复
function asyncCheckEmail(email) {
return new Promise(resolve => {
setTimeout(() => {
if (email === 'foo@domain.com') {
resolve(false);
} else {
resolve(true);
}
}, 500);
});
}
const model = SchemaModel({
email: StringType()
.isEmail('请输入正确的邮箱')
.addRule((value, data) => {
return asyncCheckEmail(value);
}, '邮箱地址已经存在')
.isRequired('该字段不能为空')
});
model.checkAsync({ email: 'foo@domain.com' }).then(result => {
console.log(result);
});
/**
{
email: {
hasError: true,
errorMessage: '邮箱地址已经存在'
}
};
**/
嵌套对象
对于复杂的嵌套的 Object , 可以使用 ObjectType().shape 方法进行定义,比如:
const model = Schema.Model({
id: NumberType().isRequired('该字段不能为空'),
name: StringType().isRequired('用户名不能为空'),
info: ObjectType().shape({
email: StringType().isEmail('应该是一个 email'),
age: NumberType().min(18, '年龄应该大于18岁')
});
});
另外,更推荐把对象扁平化设计
import { flaser } from 'object-flaser';
const model = Schema.Model({
id: NumberType().isRequired('该字段不能为空'),
name: StringType().isRequired('用户名不能为空'),
'info.email': StringType().isEmail('应该是一个 email'),
'info.age': NumberType().min(18, '年龄应该大于18岁')
});
const user = flaser({
id: 1,
name: 'schema-type',
info: {
email: 'schema-type@gmail.com',
age: 17
}
});
model.check(data);
组合
Schema.Model
提供了一个静态方法 combine
, 可以对多个 Schema.Model
合并返回一个新的 Schema.Model
。
const model1 = Schema.Model({
username: StringType().isRequired('用户名不能为空'),
email: StringType().isEmail('请输入正确的邮箱')
});
const model2 = Schema.Model({
username: StringType().minLength(7, '最少7个字符'),
age: NumberType().range(18, 30, '年应该在 18 到 30 岁')
});
const model3 = Schema.Model({
groupId: NumberType().isRequired('该字段不能为空')
});
const model4 = Schema.Model.combine(model1, model2, model3);
model4.check({
username: 'foobar',
email: 'foo@bar.com',
age: 40,
groupId: 1
});
Schema.Types
StringType
NumberType
ArrayType
DateType
ObjectType
BooleanType
StringType
- isRequired(errorMessage: string, trim: boolean = true)
StringType().isRequired('该字段不能为空');
- isEmail(errorMessage: string)
StringType().isEmail('请输入正确的邮箱地址');
- isURL(errorMessage: string)
StringType().isURL('请输入正确的URL地址');
- isOneOf(items: Array, errorMessage: string)
StringType().isOneOf(['Javascript', 'CSS'], '只能输入 `Javascript`和 `CSS`');
- containsLetter(errorMessage: string)
StringType().containsLetter('必须包含英文字符');
- containsUppercaseLetter(errorMessage: string)
StringType().containsUppercaseLetter('必须包含大写的英文字符');
- containsLowercaseLetter(errorMessage: string)
StringType().containsLowercaseLetter('必须包含小写的英文字符');
- containsLetterOnly(errorMessage: string)
StringType().containsLetterOnly('只能包含的英文字符');
- containsNumber(errorMessage: string)
StringType().containsNumber('必须包含数字');
- pattern(regExp: RegExp, errorMessage: string)
StringType().pattern(/^[1-9][0-9]{3}\s?[a-zA-Z]{2}$/, '请输入合法字符');
- rangeLength(minLength: number, maxLength: number, errorMessage: string)
StringType().rangeLength(6, 30, '字符个数只能在 6 - 30 之间');
- minLength(minLength: number, errorMessage: string)
StringType().minLength(6, '最小需要6个字符');
- maxLength(maxLength: number, errorMessage: string)
StringType().maxLength(30, '最大只能30个字符');
- addRule(onValid: Function, errorMessage: string)
StringType().addRule((value, data) => {
return /^[1-9][0-9]{3}\s?[a-zA-Z]{2}$/.test(value);
}, '请输入合法字符');
NumberType
- isRequired(errorMessage: string)
NumberType().isRequired('该字段必填');
- isInteger(errorMessage: string)
NumberType().isInteger('只能是整型');
- isOneOf(items: Array, errorMessage: string)
NumberType().isOneOf([5, 10, 15], '只能是`5`,`10`,`15`');
- pattern(regExp: RegExp, errorMessage: string)
NumberType().pattern(/^[1-9][0-9]{3}$/, '请输入合法字符');
- range(minLength: number, maxLength: number, errorMessage: string)
NumberType().range(18, 40, '请输入 18 - 40 之间的数字');
- min(min: number, errorMessage: string)
NumberType().min(18, '最小值 18');
- max(max: number, errorMessage: string)
NumberType().max(40, '最大值 40');
- addRule(onValid: Function, errorMessage: string)
NumberType().addRule((value, data) => {
return value % 5 === 0;
}, '请输入有效的数字');
ArrayType
- isRequired(errorMessage: string)
ArrayType().isRequired('该字段必填');
- rangeLength(minLength: number, maxLength: number, errorMessage: string)
ArrayType().rangeLength(1, 3, '至少选择1个,但不能超过3个');
- minLength(minLength: number, errorMessage: string)
ArrayType().minLength(1, '至少选择1个');
- maxLength(maxLength: number, errorMessage: string)
ArrayType().maxLength(3, '不能超过3个');
- unrepeatable(errorMessage: string)
ArrayType().unrepeatable('不能出现重复选项');
- of(type: Object, errorMessage: string)
ArrayType().of(StringType().isEmail(), '格式错误');
- addRule(onValid: Function, errorMessage: string)
ArrayType().addRule((value, data) => {
return value.length % 2 === 0;
}, '好事成双');
DateType
- isRequired(errorMessage: string)
DateType().isRequired('日期不能为空');
- range(min: Date, max: Date, errorMessage: string)
DateType().range(
new Date('08/01/2017'),
new Date('08/30/2017'),
'时间应该在 08/01/2017 - 08/30/2017 之间'
);
- min(min: Date, errorMessage: string)
DateType().min(new Date('08/01/2017'), '时间的最小值 08/01/2017');
- max(max: Date, errorMessage: string)
DateType().max(new Date('08/30/2017'), '时间的最大值 08/30/2017');
- addRule(onValid: Function, errorMessage: string)
DateType().addRule((value, data) => {
return value.getDay() === 2;
}, '只能选择周二');
ObjectType
- isRequired(errorMessage: string)
ObjectType().isRequired('该对象不能为空');
- shape(type: Object)
ObjectType().shape({
email: StringType().isEmail('应该是一个 email'),
age: NumberType().min(18, '年龄应该大于18岁')
});
- addRule(onValid: Function, errorMessage: string)
ObjectType().addRule((value, data) => {
if (value.id || value.email) {
return true;
}
return false;
}, 'id 与 email 必须有一个不能为空');
BooleanType
- isRequired(errorMessage: string)
BooleanType().isRequired('该字段不能为空');
- addRule(onValid: Function, errorMessage: string)
ObjectType().addRule((value, data) => {
if (typeof value === 'undefined' && A === 10) {
return false;
}
return true;
}, '当 A 等于 10 的时候,该值必须为空');
⚠️ 注意事项
默认检查优先级:
- 1.isRequired
- 2.所有其他校验规则按顺序执行
如果 addRule
的第三个参数是 true
,则检查的优先级如下:
- 1.addRule
- 2.isRequired
- 3.预定义规则(如果未设置
isRequired
,并且值为空,则不执行规则)