Parser.vue 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. <script>
  2. import { deepClone } from '@/utils/index'
  3. import render from '@/components/render/render.js'
  4. import {getAccessToken} from "@/utils/auth";
  5. const ruleTrigger = {
  6. 'el-input': 'blur',
  7. 'el-input-number': 'blur',
  8. 'el-select': 'change',
  9. 'el-radio-group': 'change',
  10. 'el-checkbox-group': 'change',
  11. 'el-cascader': 'change',
  12. 'el-time-picker': 'change',
  13. 'el-date-picker': 'change',
  14. 'el-rate': 'change'
  15. }
  16. const layouts = {
  17. colFormItem(h, scheme) {
  18. const config = scheme.__config__
  19. const listeners = buildListeners.call(this, scheme)
  20. let labelWidth = config.labelWidth ? `${config.labelWidth}px` : null
  21. if (config.showLabel === false) labelWidth = '0'
  22. return (
  23. <el-col span={config.span}>
  24. <el-form-item label-width={labelWidth} prop={scheme.__vModel__}
  25. label={config.showLabel ? config.label : ''}>
  26. <render conf={scheme} on={listeners} />
  27. </el-form-item>
  28. </el-col>
  29. )
  30. },
  31. rowFormItem(h, scheme) {
  32. let child = renderChildren.apply(this, arguments)
  33. if (scheme.type === 'flex') {
  34. child = <el-row type={scheme.type} justify={scheme.justify} align={scheme.align}>
  35. {child}
  36. </el-row>
  37. }
  38. return (
  39. <el-col span={scheme.span}>
  40. <el-row gutter={scheme.gutter}>
  41. {child}
  42. </el-row>
  43. </el-col>
  44. )
  45. }
  46. }
  47. function renderFrom(h) {
  48. const { formConfCopy } = this
  49. return (
  50. <el-row gutter={formConfCopy.gutter}>
  51. <el-form
  52. size={formConfCopy.size}
  53. label-position={formConfCopy.labelPosition}
  54. disabled={formConfCopy.disabled}
  55. label-width={`${formConfCopy.labelWidth}px`}
  56. ref={formConfCopy.formRef}
  57. // model不能直接赋值 https://github.com/vuejs/jsx/issues/49#issuecomment-472013664
  58. props={{ model: this[formConfCopy.formModel] }}
  59. rules={this[formConfCopy.formRules]}
  60. >
  61. {renderFormItem.call(this, h, formConfCopy.fields)}
  62. {formConfCopy.formBtns && formBtns.call(this, h)}
  63. </el-form>
  64. </el-row>
  65. )
  66. }
  67. function formBtns(h) {
  68. return <el-col>
  69. <el-form-item size="large">
  70. <el-button type="primary" onClick={this.submitForm}>提交</el-button>
  71. <el-button onClick={this.resetForm}>重置</el-button>
  72. </el-form-item>
  73. </el-col>
  74. }
  75. function renderFormItem(h, elementList) {
  76. const that = this
  77. const data = this[this.formConf.formModel]
  78. // const formRef = that.$refs[that.formConf.formRef] // 这里直接添加有问题,此时还找不到表单 $refs
  79. return elementList.map(scheme => {
  80. const config = scheme.__config__
  81. const layout = layouts[config.layout]
  82. // edit by 芋道源码,解决 el-upload 上传的问题
  83. // 参考 https://github.com/JakHuang/form-generator/blob/master/src/components/parser/example/Index.vue 实现
  84. const vModel = scheme.__vModel__
  85. const val = data[vModel]
  86. if (scheme.__config__.tag === 'el-upload') {
  87. // 回显图片
  88. scheme['file-list'] = (val || []).map(url => ({ name: url, url }))
  89. // 上传地址 + 请求头
  90. scheme.action = process.env.VUE_APP_BASE_API + "/admin-api/infra/file/upload"
  91. scheme.headers = { Authorization: "Bearer " + getAccessToken() }
  92. // 注意 on-success 不能绑定箭头函数!!!
  93. scheme['on-success'] = function (response, file, fileList) {
  94. if (response.code !== 0) {
  95. return;
  96. }
  97. // 添加到 data 中
  98. const prev = data[vModel] || []
  99. this.$set(data, vModel, [
  100. ...prev,
  101. response.data
  102. ])
  103. // 发起表单校验
  104. that.$refs[that.formConf.formRef].validateField(vModel)
  105. }
  106. // 注意 on-remove 不能绑定箭头函数!!!
  107. scheme['on-remove'] = function (file, fileList) {
  108. // 移除从 data 中
  109. const prev = data[vModel] || []
  110. const index = prev.indexOf(file.response.data)
  111. if (index === -1) {
  112. return
  113. }
  114. prev.splice(index, 1) // 直接移除即可,无需重复 set,因为 array 是引用
  115. // 发起表单校验
  116. that.$refs[that.formConf.formRef].validateField(vModel)
  117. }
  118. }
  119. if (layout) {
  120. return layout.call(this, h, scheme)
  121. }
  122. throw new Error(`没有与${config.layout}匹配的layout`)
  123. })
  124. }
  125. function renderChildren(h, scheme) {
  126. const config = scheme.__config__
  127. if (!Array.isArray(config.children)) return null
  128. return renderFormItem.call(this, h, config.children)
  129. }
  130. function setValue(event, config, scheme) {
  131. this.$set(config, 'defaultValue', event)
  132. this.$set(this[this.formConf.formModel], scheme.__vModel__, event)
  133. }
  134. function buildListeners(scheme) {
  135. const config = scheme.__config__
  136. const methods = this.formConf.__methods__ || {}
  137. const listeners = {}
  138. // 给__methods__中的方法绑定this和event
  139. Object.keys(methods).forEach(key => {
  140. listeners[key] = event => methods[key].call(this, event)
  141. })
  142. // 响应 render.js 中的 vModel $emit('input', val)
  143. listeners.input = event => setValue.call(this, event, config, scheme)
  144. return listeners
  145. }
  146. export default {
  147. components: {
  148. render
  149. },
  150. props: {
  151. formConf: {
  152. type: Object,
  153. required: true
  154. }
  155. },
  156. data() {
  157. const data = {
  158. formConfCopy: deepClone(this.formConf),
  159. [this.formConf.formModel]: {},
  160. [this.formConf.formRules]: {}
  161. }
  162. this.initFormData(data.formConfCopy.fields, data[this.formConf.formModel])
  163. this.buildRules(data.formConfCopy.fields, data[this.formConf.formRules])
  164. return data
  165. },
  166. methods: {
  167. initFormData(componentList, formData) {
  168. componentList.forEach(cur => {
  169. const config = cur.__config__
  170. if (cur.__vModel__) formData[cur.__vModel__] = config.defaultValue
  171. if (config.children) this.initFormData(config.children, formData)
  172. })
  173. },
  174. buildRules(componentList, rules) {
  175. componentList.forEach(cur => {
  176. const config = cur.__config__
  177. if (Array.isArray(config.regList)) {
  178. if (config.required) {
  179. const required = { required: config.required, message: cur.placeholder }
  180. if (Array.isArray(config.defaultValue)) {
  181. required.type = 'array'
  182. required.message = `请至少选择一个${config.label}`
  183. }
  184. required.message === undefined && (required.message = `${config.label}不能为空`)
  185. config.regList.push(required)
  186. }
  187. rules[cur.__vModel__] = config.regList.map(item => {
  188. item.pattern && (item.pattern = eval(item.pattern))
  189. item.trigger = ruleTrigger && ruleTrigger[config.tag]
  190. return item
  191. })
  192. }
  193. if (config.children) this.buildRules(config.children, rules)
  194. })
  195. },
  196. resetForm() {
  197. this.formConfCopy = deepClone(this.formConf)
  198. this.$refs[this.formConf.formRef].resetFields()
  199. },
  200. submitForm() {
  201. this.$refs[this.formConf.formRef].validate(valid => {
  202. if (!valid) return false
  203. // 触发 submit 事件
  204. // update by 芋道源码
  205. // this.$emit('submit', this[this.formConf.formModel])
  206. this.$emit('submit', {
  207. conf: this.formConfCopy,
  208. values: this[this.formConf.formModel]
  209. })
  210. return true
  211. })
  212. }
  213. },
  214. render(h) {
  215. return renderFrom.call(this, h)
  216. }
  217. }
  218. </script>