| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293 |
- "use strict";
- var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- exports.default = void 0;
- var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends"));
- var _vue = require("vue");
- var _cloneDeep = _interopRequireDefault(require("lodash/cloneDeep"));
- var _intersection = _interopRequireDefault(require("lodash/intersection"));
- var _isEqual = _interopRequireDefault(require("lodash/isEqual"));
- var _debounce = _interopRequireDefault(require("lodash/debounce"));
- var _omit = _interopRequireDefault(require("lodash/omit"));
- var _validateUtil = require("./utils/validateUtil");
- var _messages = require("./utils/messages");
- var _asyncUtil = require("./utils/asyncUtil");
- function isRequired(rules) {
- let isRequired = false;
- if (rules && rules.length) {
- rules.every(rule => {
- if (rule.required) {
- isRequired = true;
- return false;
- }
- return true;
- });
- }
- return isRequired;
- }
- function toArray(value) {
- if (value === undefined || value === null) {
- return [];
- }
- return Array.isArray(value) ? value : [value];
- }
- function getPropByPath(obj, path, strict) {
- let tempObj = obj;
- path = path.replace(/\[(\w+)\]/g, '.$1');
- path = path.replace(/^\./, '');
- const keyArr = path.split('.');
- let i = 0;
- for (let len = keyArr.length; i < len - 1; ++i) {
- if (!tempObj && !strict) break;
- const key = keyArr[i];
- if (key in tempObj) {
- tempObj = tempObj[key];
- } else {
- if (strict) {
- throw new Error('please transfer a valid name path to validate!');
- }
- break;
- }
- }
- return {
- o: tempObj,
- k: keyArr[i],
- v: tempObj ? tempObj[keyArr[i]] : null,
- isValid: tempObj && keyArr[i] in tempObj
- };
- }
- function useForm(modelRef) {
- let rulesRef = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : (0, _vue.ref)({});
- let options = arguments.length > 2 ? arguments[2] : undefined;
- const initialModel = (0, _cloneDeep.default)((0, _vue.unref)(modelRef));
- const validateInfos = (0, _vue.reactive)({});
- const rulesKeys = (0, _vue.shallowRef)([]);
- const resetFields = newValues => {
- (0, _extends2.default)((0, _vue.unref)(modelRef), (0, _extends2.default)((0, _extends2.default)({}, (0, _cloneDeep.default)(initialModel)), newValues));
- (0, _vue.nextTick)(() => {
- Object.keys(validateInfos).forEach(key => {
- validateInfos[key] = {
- autoLink: false,
- required: isRequired((0, _vue.unref)(rulesRef)[key])
- };
- });
- });
- };
- const filterRules = function () {
- let rules = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
- let trigger = arguments.length > 1 ? arguments[1] : undefined;
- if (!trigger.length) {
- return rules;
- } else {
- return rules.filter(rule => {
- const triggerList = toArray(rule.trigger || 'change');
- return (0, _intersection.default)(triggerList, trigger).length;
- });
- }
- };
- let lastValidatePromise = null;
- const validateFields = function (names) {
- let option = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
- let strict = arguments.length > 2 ? arguments[2] : undefined;
- // Collect result in promise list
- const promiseList = [];
- const values = {};
- for (let i = 0; i < names.length; i++) {
- const name = names[i];
- const prop = getPropByPath((0, _vue.unref)(modelRef), name, strict);
- if (!prop.isValid) continue;
- values[name] = prop.v;
- const rules = filterRules((0, _vue.unref)(rulesRef)[name], toArray(option && option.trigger));
- if (rules.length) {
- promiseList.push(validateField(name, prop.v, rules, option || {}).then(() => ({
- name,
- errors: [],
- warnings: []
- })).catch(ruleErrors => {
- const mergedErrors = [];
- const mergedWarnings = [];
- ruleErrors.forEach(_ref => {
- let {
- rule: {
- warningOnly
- },
- errors
- } = _ref;
- if (warningOnly) {
- mergedWarnings.push(...errors);
- } else {
- mergedErrors.push(...errors);
- }
- });
- if (mergedErrors.length) {
- return Promise.reject({
- name,
- errors: mergedErrors,
- warnings: mergedWarnings
- });
- }
- return {
- name,
- errors: mergedErrors,
- warnings: mergedWarnings
- };
- }));
- }
- }
- const summaryPromise = (0, _asyncUtil.allPromiseFinish)(promiseList);
- lastValidatePromise = summaryPromise;
- const returnPromise = summaryPromise.then(() => {
- if (lastValidatePromise === summaryPromise) {
- return Promise.resolve(values);
- }
- return Promise.reject([]);
- }).catch(results => {
- const errorList = results.filter(result => result && result.errors.length);
- return errorList.length ? Promise.reject({
- values,
- errorFields: errorList,
- outOfDate: lastValidatePromise !== summaryPromise
- }) : Promise.resolve(values);
- });
- // Do not throw in console
- returnPromise.catch(e => e);
- return returnPromise;
- };
- const validateField = function (name, value, rules) {
- let option = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : {};
- const promise = (0, _validateUtil.validateRules)([name], value, rules, (0, _extends2.default)({
- validateMessages: _messages.defaultValidateMessages
- }, option), !!option.validateFirst);
- if (!validateInfos[name]) {
- return promise.catch(e => e);
- }
- validateInfos[name].validateStatus = 'validating';
- promise.catch(e => e).then(function () {
- let results = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
- var _a;
- if (validateInfos[name].validateStatus === 'validating') {
- const res = results.filter(result => result && result.errors.length);
- validateInfos[name].validateStatus = res.length ? 'error' : 'success';
- validateInfos[name].help = res.length ? res.map(r => r.errors) : null;
- (_a = options === null || options === void 0 ? void 0 : options.onValidate) === null || _a === void 0 ? void 0 : _a.call(options, name, !res.length, res.length ? (0, _vue.toRaw)(validateInfos[name].help[0]) : null);
- }
- });
- return promise;
- };
- const validate = (names, option) => {
- let keys = [];
- let strict = true;
- if (!names) {
- strict = false;
- keys = rulesKeys.value;
- } else if (Array.isArray(names)) {
- keys = names;
- } else {
- keys = [names];
- }
- const promises = validateFields(keys, option || {}, strict);
- // Do not throw in console
- promises.catch(e => e);
- return promises;
- };
- const clearValidate = names => {
- let keys = [];
- if (!names) {
- keys = rulesKeys.value;
- } else if (Array.isArray(names)) {
- keys = names;
- } else {
- keys = [names];
- }
- keys.forEach(key => {
- validateInfos[key] && (0, _extends2.default)(validateInfos[key], {
- validateStatus: '',
- help: null
- });
- });
- };
- const mergeValidateInfo = items => {
- const info = {
- autoLink: false
- };
- const help = [];
- const infos = Array.isArray(items) ? items : [items];
- for (let i = 0; i < infos.length; i++) {
- const arg = infos[i];
- if ((arg === null || arg === void 0 ? void 0 : arg.validateStatus) === 'error') {
- info.validateStatus = 'error';
- arg.help && help.push(arg.help);
- }
- info.required = info.required || (arg === null || arg === void 0 ? void 0 : arg.required);
- }
- info.help = help;
- return info;
- };
- let oldModel = initialModel;
- let isFirstTime = true;
- const modelFn = model => {
- const names = [];
- rulesKeys.value.forEach(key => {
- const prop = getPropByPath(model, key, false);
- const oldProp = getPropByPath(oldModel, key, false);
- const isFirstValidation = isFirstTime && (options === null || options === void 0 ? void 0 : options.immediate) && prop.isValid;
- if (isFirstValidation || !(0, _isEqual.default)(prop.v, oldProp.v)) {
- names.push(key);
- }
- });
- validate(names, {
- trigger: 'change'
- });
- isFirstTime = false;
- oldModel = (0, _cloneDeep.default)((0, _vue.toRaw)(model));
- };
- const debounceOptions = options === null || options === void 0 ? void 0 : options.debounce;
- let first = true;
- (0, _vue.watch)(rulesRef, () => {
- rulesKeys.value = rulesRef ? Object.keys((0, _vue.unref)(rulesRef)) : [];
- if (!first && options && options.validateOnRuleChange) {
- validate();
- }
- first = false;
- }, {
- deep: true,
- immediate: true
- });
- (0, _vue.watch)(rulesKeys, () => {
- const newValidateInfos = {};
- rulesKeys.value.forEach(key => {
- newValidateInfos[key] = (0, _extends2.default)({}, validateInfos[key], {
- autoLink: false,
- required: isRequired((0, _vue.unref)(rulesRef)[key])
- });
- delete validateInfos[key];
- });
- for (const key in validateInfos) {
- if (Object.prototype.hasOwnProperty.call(validateInfos, key)) {
- delete validateInfos[key];
- }
- }
- (0, _extends2.default)(validateInfos, newValidateInfos);
- }, {
- immediate: true
- });
- (0, _vue.watch)(modelRef, debounceOptions && debounceOptions.wait ? (0, _debounce.default)(modelFn, debounceOptions.wait, (0, _omit.default)(debounceOptions, ['wait'])) : modelFn, {
- immediate: options && !!options.immediate,
- deep: true
- });
- return {
- modelRef,
- rulesRef,
- initialModel,
- validateInfos,
- resetFields,
- validate,
- validateField,
- mergeValidateInfo,
- clearValidate
- };
- }
- var _default = exports.default = useForm;
|