formData.js 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294
  1. import FieldCheck from "./fieldCheck";
  2. let hasOwnProperty = Object.prototype.hasOwnProperty;
  3. function hasOwn (obj, key) {
  4. return hasOwnProperty.call(obj, key)
  5. }
  6. /**
  7. * @typedef {object} formItemData 表单项数据
  8. * @property {string} [val] 表单项值
  9. * @property {string} [msg] 表单项错误信息
  10. * @property {number} [state] 表单项状态 0 通过 1 通过 2 不通过
  11. * @property {string} [showText] 表单项显示文本,用于在某些
  12. * @property {string} [label] 表单项显示文本
  13. * @property {string} [init] 表单项初始值
  14. * @property {Array} [options] 表单项枚举值
  15. * @property {string} [depend] 依赖字段, 该项存在将使用依赖字段的option中的checkField字段进行匹配验证规则
  16. * @property {string} [reCheckField] 该表单项用于匹配规则的字段
  17. * @property {Array} [disables] 禁用项
  18. */
  19. /**
  20. * @class FormItem
  21. * @description 表单项
  22. * @param {object} object 表单项数据
  23. * @param {FieldCheck} [fieldCheck] 字段验证对象
  24. * @param {object} [option] 配置项
  25. */
  26. class FormItem {
  27. formData;
  28. /**
  29. * @type {FieldCheck} 字段验证对象
  30. */
  31. fieldCheck;
  32. // 表单状态 默认 0 通过 1 不通过 2
  33. formState_default = 0;
  34. formState_pass = 1;
  35. formState_notPass = 2;
  36. defaultOption = {
  37. isMustMatchRule: false,// 表单字段是否必须匹配到验证规则
  38. }
  39. option = {};
  40. /**
  41. *
  42. * @param object
  43. * @param {FieldCheck} [fieldCheck] 字段验证对象
  44. * @param {object} [option] 配置项
  45. */
  46. constructor(object, fieldCheck, option) {
  47. this.fieldCheck = fieldCheck || new FieldCheck();
  48. // 合并配置项
  49. this.option = Object.assign(this.defaultOption, option);
  50. let errMsg;
  51. // 拿出其中的每一项来构建对应的表单项
  52. for (let key in object) {
  53. this[key] = object[key];
  54. // 验证表单项是否符合要求,不符合要求则抛出错误
  55. errMsg = FormItem.buildFormItem(object, key, object[key], this.fieldCheck, this.option.isMustMatchRule);
  56. if (errMsg) {
  57. throw new Error(`表单项${key}不符合要求,err:${errMsg}`);
  58. }
  59. }
  60. this.formData = object;
  61. }
  62. static isObject (obj) {
  63. return obj !== null && typeof obj === 'object'
  64. }
  65. /**
  66. * 检查表单项是否符合要求
  67. * @param { object } object 表单项数据
  68. * @param { string } field 字段名
  69. * @param { formItemData } formItemData 表单项数据
  70. * @param { FieldCheck } fieldCheck 字段验证对象
  71. * @param { boolean } isMustMatchRule 表单字段是否必须匹配到验证规则
  72. * @returns { string } errMsg 错误信息
  73. */
  74. static buildFormItem(object, field, formItemData, fieldCheck, isMustMatchRule) {
  75. if ( !FormItem.isObject(formItemData) ){
  76. return `form item ${field} must be object`;
  77. }
  78. // 是否需要从验证规则表中查找对应的验证规则
  79. let isNeedMatchRule = true;
  80. // 用于匹配的字段
  81. let checkFieldStr = field;
  82. let disables = formItemData.disables;
  83. // 设置默认值
  84. formItemData.val = formItemData.val || formItemData.init || '';
  85. // 设置默认提示词
  86. formItemData.msg = formItemData.msg || '';
  87. // 设置默认状态
  88. formItemData.state = formItemData.state || FormData.formState_default;
  89. // 设置默认显示文本
  90. formItemData.label = formItemData.label || '';
  91. // 判断是否有 options 选项有则判断是否有 init 选项,没有则设置第一个为 init
  92. if ( formItemData.options ){
  93. if( !formItemData.options.length || !formItemData.options[0] ){
  94. return `form item ${field} options must be array and has item`;
  95. }
  96. if ( !formItemData.init ){
  97. formItemData.init = formItemData.options[0].key;
  98. }
  99. // 判断 val 与 init 是否存在于 options 中
  100. let hasInit = false;
  101. for (let i = 0; i < formItemData.options.length; i++) {
  102. let option = formItemData.options[i];
  103. if ( option.key === formItemData.init ){
  104. hasInit = true;
  105. }
  106. // 判断该options是否为禁用项
  107. if ( disables && disables.indexOf(option.key) !== -1 ){
  108. option.disabled = true;
  109. }
  110. }
  111. if ( !hasInit ){
  112. return `form item ${field} init value must be in options`;
  113. }
  114. }
  115. // 判断是否有 depend 依赖字段 有依赖字段则依据依赖字段中的 option 中的 checkField 字段进行判断
  116. if( formItemData.depend && formItemData.reCheckField ){
  117. return `form item ${field} has depend and reCheckField, but depend and reCheckField can not exist at the same time`;
  118. }
  119. // 判断是否有 depend 依赖字段 有依赖字段则依据依赖字段中的 option 中的 checkField 字段进行判断
  120. if ( formItemData.depend ){
  121. // 判断依赖字段是否存在
  122. if ( !object[formItemData.depend] ){
  123. return `form item ${field} depend field ${formItemData.depend} but the field not exist`;
  124. }
  125. // 判断依赖字段的 option 是否存在
  126. if ( !object[formItemData.depend].options ){
  127. return `form item ${field} depend field ${formItemData.depend} has no options`;
  128. }
  129. // 判断依赖字段的 options 中是否有 checkField 字段
  130. let hasCheckField = false;
  131. for (let i = 0; i < object[formItemData.depend].options.length; i++) {
  132. let option = object[formItemData.depend].options[i];
  133. if ( option.checkField ){
  134. hasCheckField = true;
  135. checkFieldStr = option.checkField;
  136. break;
  137. }
  138. }
  139. if ( !hasCheckField ){
  140. return `form item ${field} depend field ${formItemData.depend} has no checkField`;
  141. }
  142. }
  143. // 判断是否有 reCheckField 有则使用该字段的值进行规则验证
  144. if ( formItemData.reCheckField ){
  145. checkFieldStr = formItemData.reCheckField;
  146. }
  147. // 判断是否有 rules 规则
  148. if(isMustMatchRule){
  149. if(fieldCheck.getRuleItem(checkFieldStr)){
  150. return `form item ${field} has no rules`;
  151. }
  152. }
  153. return '';
  154. }
  155. /**
  156. * 初始化表单项数据
  157. * @param { formObject } formObject 表单对象
  158. */
  159. static initFormItemData ( formObject ) {
  160. let keys = Object.keys(formObject);
  161. for(let i = 0; i < keys.length; i++){
  162. let key = keys[i];
  163. formObject[key].val = formObject[key].init;
  164. formObject[key].msg = '';
  165. formObject[key].state = FormData.formState_default;
  166. formObject[key].showText = '';
  167. }
  168. }
  169. /**
  170. * 检查表单项是否符合要求
  171. * @param {object} form 表单对象
  172. * @param isMustMatch 是否必须全部匹配到验证规则
  173. * @returns {boolean}
  174. */
  175. checkForm (form, isMustMatch) {
  176. let r = true;
  177. let n_checkPass = 0,
  178. n_checkTotal = 0;
  179. let msg = '';
  180. for (const fieldKey in form) {
  181. let formItem = form[fieldKey];
  182. let depend = form[formItem.depend];
  183. let checkField = fieldKey;
  184. let tmpInd = -1;
  185. n_checkTotal++;
  186. if(formItem.reCheckField){
  187. checkField = formItem.reCheckField;
  188. }
  189. // 禁用值判断 array
  190. if(formItem.disables){
  191. if(formItem.disables.indexOf(formItem.val) !== -1){
  192. formItem.msg = '该项内容不合法';
  193. r = false;
  194. }
  195. }
  196. // 枚举值判断
  197. if(formItem.options){
  198. // 有枚举字段,只判断是否在枚举中
  199. // console.log(`检测枚举字段:${checkField},值:${formItem.val}`);
  200. tmpInd = formItem.options.findIndex(item=>item.value == formItem.val);
  201. if(tmpInd === -1){
  202. console.log(`检测枚举字段:${checkField},值:${formItem.val}不在范围内`);
  203. formItem.msg = '选项不在范围内';
  204. formItem.state = 1;
  205. r = false;
  206. }else{
  207. // 判断值是否为禁用项
  208. if(formItem.options[tmpInd].disabled){
  209. formItem.msg = '该选项已经被禁用';
  210. r = false;
  211. }
  212. }
  213. // 枚举值判断完毕,继续下一个字段
  214. n_checkPass++;
  215. continue;
  216. }
  217. // 依赖字段判断
  218. if(depend){
  219. if(depend.options){
  220. // 依赖的对象有枚举类型,检查该枚举类型是否有有检测值
  221. let optionItem = depend.options.find(item=>item.value == depend.val);
  222. if(!optionItem){
  223. depend.msg = '选项不在范围内';
  224. formItem.msg = '该值依赖项输入异常';
  225. r = false;
  226. // continue;
  227. }
  228. if(optionItem.checkField){
  229. // console.log(`采用依赖项的检测字段${optionItem.checkField}`)
  230. checkField = optionItem.checkField;
  231. }
  232. }else{
  233. r = false;
  234. }
  235. if(!r)
  236. {
  237. depend.msg = '该项依赖项输入异常';
  238. formItem.msg = '该值依赖项输入异常';
  239. }
  240. }
  241. // 使用验证规则进行
  242. formItem.msg = this.fieldCheck.verify({
  243. [checkField]:formItem.val,
  244. })
  245. if (formItem.msg) r = false;
  246. if(r){
  247. n_checkPass++;
  248. formItem.state = this.formState_pass
  249. }else{
  250. formItem.state = this.formState_notPass
  251. }
  252. }
  253. msg = `检查表单项通过率:${n_checkPass}/${n_checkTotal}`;
  254. console.log(msg);
  255. return r;
  256. }
  257. }
  258. export default FormItem;