UNPKG

fastlion-amis

Version:

一种MIS页面生成工具

503 lines (472 loc) 15.3 kB
import { createObject, isMobile, isObject } from './helper'; import { filter } from './tpl'; import { isPureVariable, resolveVariableAndFilter } from './tpl-builtin'; const isExisty = (value: any) => value !== null && value !== undefined; const isEmpty = (value: any) => value === ''; const makeRegexp = (reg: string | RegExp) => { if (reg instanceof RegExp) { return reg; } else if (/^(?:matchRegexp\:)?\/(.+)\/([gimuy]*)$/.test(reg)) { return new RegExp(RegExp.$1, RegExp.$2 || ''); } else if (typeof reg === 'string') { return new RegExp(reg); } return /^$/; }; import memoize from 'lodash/memoize'; import isPlainObject from 'lodash/isPlainObject'; const makeUrlRegexp = memoize(function (options: any) { options = { schemes: ['http', 'https', 'ftp', 'sftp'], allowLocal: true, allowDataUrl: false, ...(isPlainObject(options) ? options : {}) }; // https://github.com/ansman/validate.js/blob/master/validate.js#L1098-L1164 let { schemes, allowLocal, allowDataUrl } = options; if (!Array.isArray(schemes)) { schemes = ['http', 'https', 'ftp', 'sftp']; } let regex = '^' + // protocol identifier '(?:(?:' + schemes.join('|') + ')://)' + // user:pass authentication '(?:\\S+(?::\\S*)?@)?' + '(?:'; var tld = '(?:\\.(?:[a-z\\u00a1-\\uffff]{2,}))'; if (allowLocal) { tld += '?'; } else { regex += // IP address exclusion // private & local networks '(?!(?:10|127)(?:\\.\\d{1,3}){3})' + '(?!(?:169\\.254|192\\.168)(?:\\.\\d{1,3}){2})' + '(?!172\\.(?:1[6-9]|2\\d|3[0-1])(?:\\.\\d{1,3}){2})'; } regex += // IP address dotted notation octets // excludes loopback network 0.0.0.0 // excludes reserved space >= 224.0.0.0 // excludes network & broacast addresses // (first & last IP address of each class) '(?:[1-9]\\d?|1\\d\\d|2[01]\\d|22[0-3])' + '(?:\\.(?:1?\\d{1,2}|2[0-4]\\d|25[0-5])){2}' + '(?:\\.(?:[1-9]\\d?|1\\d\\d|2[0-4]\\d|25[0-4]))' + '|' + // host name '(?:(?:[a-z\\u00a1-\\uffff0-9]-*)*[a-z\\u00a1-\\uffff0-9]+)' + // domain name '(?:\\.(?:[a-z\\u00a1-\\uffff0-9]-*)*[a-z\\u00a1-\\uffff0-9]+)*' + tld + ')' + // port number '(?::\\d{2,5})?' + // resource path '(?:[/?#]\\S*)?' + '$'; if (allowDataUrl) { // RFC 2397 var mediaType = '\\w+\\/[-+.\\w]+(?:;[\\w=]+)*'; var urlchar = "[A-Za-z0-9-_.!~\\*'();\\/?:@&=+$,%]*"; var dataurl = 'data:(?:' + mediaType + ')?(?:;base64)?,' + urlchar; regex = '(?:' + regex + ')|(?:^' + dataurl + '$)'; } return new RegExp(regex, 'i'); }); export interface ValidateFn { ( values: { [propsName: string]: any }, value: any, arg1?: any, arg2?: any, arg3?: any ): boolean; } const isExpr = (value: string) => { const isExpr = /\[(.*?)\]/g.test(value) if (isExpr) { return true } return false } export const validations: { [propsName: string]: ValidateFn; } = { isRequired: function (values, value: any) { if (isObject(value)) { return Array.isArray(value.info) && value.info.length > 0 } return ( value !== undefined && value !== '' && value !== null && (!Array.isArray(value) || !!value.length) && (value?.info !== null) && (!isObject(value) || !!value.value) ); }, isExisty: function (values, value) { return isExisty(value); }, matchRegexp: function (values, value, regexp) { return !isExisty(value) || isEmpty(value) || makeRegexp(regexp).test(value); }, isUndefined: function (values, value) { return value === undefined; }, isEmptyString: function (values, value) { return isEmpty(value); }, isEmail: function (values, value) { return validations.matchRegexp( values, value, /^((([a-z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+(\.([a-z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+)*)|((\x22)((((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(([\x01-\x08\x0b\x0c\x0e-\x1f\x7f]|\x21|[\x23-\x5b]|[\x5d-\x7e]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(\\([\x01-\x09\x0b\x0c\x0d-\x7f]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]))))*(((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(\x22)))@((([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.)+(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))$/i ); }, isUrl: function (values, value, options) { return validations.matchRegexp(values, value, makeUrlRegexp(options)); }, isTrue: function (values, value) { return value === true; }, isFalse: function (values, value) { return value === false; }, isNumeric: function (values, value) { if (typeof value === 'number') { return true; } return validations.matchRegexp(values, value, /^[-+]?(?:\d*[.])?\d+$/); }, isExprOrNumeric: function (values, value) { if (isExpr(value)) { return true } if (typeof value === 'number') { return true; } // 匹配,号分隔的部分 return validations.matchRegexp(values, value, /^(?:[-+]?\d+(\.\d+)?,)*[-+]?\d+(\.\d+)?$/); }, isExprOrDate: function (values, value) { if (isExpr(value)) { return true } // 没有或者是正常日期都通过 return !value || !isNaN(new Date(value).getTime()) }, isAlpha: function (values, value) { return validations.matchRegexp(values, value, /^[A-Z]+$/i); }, isAlphanumeric: function (values, value) { return validations.matchRegexp(values, value, /^[0-9A-Z]+$/i); }, isInt: function (values, value) { return validations.matchRegexp(values, value, /^(?:[-+]?(?:0|[1-9]\d*))$/); }, isFloat: function (values, value) { return validations.matchRegexp( values, value, /^(?:[-+]?(?:\d+))?(?:\.\d*)?(?:[eE][\+\-]?(?:\d+))?$/ ); }, isWords: function (values, value) { return validations.matchRegexp(values, value, /^[A-Z\s]+$/i); }, isSpecialWords: function (values, value) { return validations.matchRegexp(values, value, /^[A-Z\s\u00C0-\u017F]+$/i); }, isLength: function (values, value, length) { return !isExisty(value) || isEmpty(value) || value.length === length; }, equals: function (values, value, eql) { return !isExisty(value) || isEmpty(value) || value == eql; }, equalsField: function (values, value, field) { return value == values[field]; }, maxLength: function (values, value, length) { // 此方法应该判断文本长度,如果传入数据为number,导致 maxLength 和 maximum 表现一致了,默认转成string if (typeof value === 'number') { value = String(value); } //中文占两个字符长度 if (typeof value === 'string') { value = value.replace(/[^\u0000-\u00ff]/g, 'aa') } return !isExisty(value) || value.length <= length; }, minLength: function (values, value, length) { return !isExisty(value) || isEmpty(value) || value.length >= length; }, isUrlPath: function (values, value, regexp) { return !isExisty(value) || isEmpty(value) || /^[a-z0-9_\\-]+$/i.test(value); }, maximum: function (values, value, maximum) { return ( !isExisty(value) || isEmpty(value) || (parseFloat(value) || 0) <= (parseFloat(maximum) || 0) ); }, lt: function (values, value, maximum) { return ( !isExisty(value) || isEmpty(value) || (parseFloat(value) || 0) < (parseFloat(maximum) || 0) ); }, minimum: function (values, value, minimum) { return ( !isExisty(value) || isEmpty(value) || (parseFloat(value) || 0) >= (parseFloat(minimum) || 0) ); }, gt: function (values, value, minimum) { return ( !isExisty(value) || isEmpty(value) || (parseFloat(value) || 0) > (parseFloat(minimum) || 0) ); }, isJson: function (values, value, minimum) { if (isExisty(value) && !isEmpty(value) && typeof value === 'string') { try { const result = JSON.parse(value); if (typeof result === 'object' && result) { return true; } return false; } catch (e) { return false; } } return true; }, isPhoneNumber: function (values, value) { return ( !isExisty(value) || isEmpty(value) || /^[1]([3-9])[0-9]{9}$/.test(value) ); }, isTelNumber: function (values, value) { return ( !isExisty(value) || isEmpty(value) || /^(\(\d{3,4}\)|\d{3,4}-|\s)?\d{7,14}$/.test(value) ); }, isZipcode: function (values, value) { return ( !isExisty(value) || isEmpty(value) || /^[1-9]{1}(\d+){5}$/.test(value) ); }, isId: function (values, value) { return ( !isExisty(value) || isEmpty(value) || /(^[1-9]\d{5}(18|19|([23]\d))\d{2}((0[1-9])|(10|11|12))(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$)|(^[1-9]\d{5}\d{2}((0[1-9])|(10|11|12))(([0-2][1-9])|10|20|30|31)\d{3}$)/.test( value ) ); }, notEmptyString: function (values, value) { return !isExisty(value) || !(String(value) && String(value).trim() === ''); }, matchRegexp1: function (values, value, regexp) { return validations.matchRegexp(values, value, regexp); }, matchRegexp2: function (values, value, regexp) { return validations.matchRegexp(values, value, regexp); }, matchRegexp3: function (values, value, regexp) { return validations.matchRegexp(values, value, regexp); }, matchRegexp4: function (values, value, regexp) { return validations.matchRegexp(values, value, regexp); }, matchRegexp5: function (values, value, regexp) { return validations.matchRegexp(values, value, regexp); }, matchRegexp6: function (values, value, regexp) { return validations.matchRegexp(values, value, regexp); }, matchRegexp7: function (values, value, regexp) { return validations.matchRegexp(values, value, regexp); }, matchRegexp8: function (values, value, regexp) { return validations.matchRegexp(values, value, regexp); }, matchRegexp9: function (values, value, regexp) { return validations.matchRegexp(values, value, regexp); } }; export function addRule( ruleName: string, fn: ValidateFn, message: string = '' ) { validations[ruleName] = fn; validateMessages[ruleName] = message; } export const validateMessages: { [propName: string]: string; } = { isEmail: 'validate.isEmail', isRequired: 'validate.isRequired', isUrl: 'validate.isUrl', isInt: 'validate.isInt', isAlpha: 'validate.isAlpha', isNumeric: 'validate.isNumeric', isExprOrNumeric: 'validate.isExprOrNumeric', isExprOrDate: 'validate.isExprOrDate', isAlphanumeric: 'validate.isAlphanumeric', isFloat: 'validate.isFloat', isWords: 'validate.isWords', isUrlPath: 'validate.isUrlPath', matchRegexp: 'validate.matchRegexp', minLength: 'validate.minLength', maxLength: 'validate.maxLength', maximum: 'validate.maximum', lt: 'validate.lt', minimum: 'validate.minimum', gt: 'validate.gt', isJson: 'validate.isJson', isLength: 'validate.isLength', notEmptyString: 'validate.notEmptyString', equalsField: 'validate.equalsField', equals: 'validate.equals', isPhoneNumber: 'validate.isPhoneNumber', isTelNumber: 'validate.isTelNumber', isZipcode: 'validate.isZipcode', isId: 'validate.isId' }; export function validate( value: any, values: { [propName: string]: any }, rules: { [propName: string]: any }, messages?: { [propName: string]: string }, __ = (str: string, data: any) => str, label?: string ): Array<{ rule: string; msg: string; }> { const errors: Array<{ rule: string; msg: string; }> = []; rules && Object.keys(rules).forEach(ruleName => { // 如果ruleName 为空 则不处理这条规则 if ((!rules[ruleName] && rules[ruleName] !== 0) || !ruleName) { return; } else if (typeof validations[ruleName] !== 'function') { throw new Error('Validation `' + ruleName + '` not exists!'); } const fn = validations[ruleName]; const args = ( Array.isArray(rules[ruleName]) ? rules[ruleName] : [rules[ruleName]] ).map((item: any) => { if (typeof item === 'string' && isPureVariable(item)) { return resolveVariableAndFilter(item, values, '|raw'); } return item; }); // 移动端端还没有做必填 if (!fn(values, value, ...args)) { errors.push({ rule: ruleName, msg: filter( __(messages?.[ruleName] || '', rules) || (label ? `【${label}】` : '') + __(validateMessages[ruleName], rules), // isMobile() ? __(validateMessages[ruleName]) : __(messages?.[ruleName] || '') || (label ? `【${label}】` : '') + __(validateMessages[ruleName]), { ...[''].concat(args) } ) }); } }); return errors; } export function validateObject( values: { [propName: string]: any }, rules: { [propName: string]: any }, messages?: { [propName: string]: string }, __ = (str: string) => str ) { const ret: { [propName: string]: { rule: string; msg: string; }[]; } = {}; Object.keys(rules).forEach(key => { const msgs = validate( values[key], values, rules[key] === true ? { isRequired: true } : rules[key], messages, __ ); if (msgs.length) { ret[key] = msgs; } }); return ret; } const splitValidations = function (str: string): Array<string> { let i = 0; const placeholder: { [propName: string]: string } = {}; return str .replace(/matchRegexp\d*\s*\:\s*\/.*?\/[igm]*/g, raw => { placeholder[`__${i}`] = raw; return `__${i++}`; }) .split(/,(?![^{\[]*[}\]])/g) .map(str => (/^__\d+$/.test(str) ? placeholder[str] : str.trim())); }; export function str2rules(validations: string | { [propName: string]: any }): { [propName: string]: any; } { if (typeof validations === 'string') { return validations ? splitValidations(validations).reduce(function ( validations: { [propName: string]: any }, validation ) { const idx = validation.indexOf(':'); let validateMethod = validation; let args = []; if (~idx) { validateMethod = validation.substring(0, idx); args = /^matchRegexp/.test(validateMethod) ? [validation.substring(idx + 1).trim()] : validation .substring(idx + 1) .split(',') .map(function (arg) { try { return JSON.parse(arg); } catch (e) { return arg; } }); } validations[validateMethod] = args.length ? args : true; return validations; }, {}) : {}; } return validations || {}; }