render.js 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. import { deepClone } from '@/utils/index'
  2. const componentChild = {}
  3. /**
  4. * 将./slots中的文件挂载到对象componentChild上
  5. * 文件名为key,对应JSON配置中的__config__.tag
  6. * 文件内容为value,解析JSON配置中的__slot__
  7. */
  8. const slotsFiles = require.context('./slots', false, /\.js$/)
  9. const keys = slotsFiles.keys() || []
  10. keys.forEach(key => {
  11. const tag = key.replace(/^\.\/(.*)\.\w+$/, '$1')
  12. const value = slotsFiles(key).default
  13. componentChild[tag] = value
  14. })
  15. function vModel(dataObject, defaultValue, config) {
  16. // 获取上传表单元素组件上传的文件
  17. if (config.tag === "el-upload") {
  18. // 上传表单元素组件的成功和移除事件
  19. dataObject.attrs["on-success"] = (response, file, fileList) => {
  20. this.$emit("upload", response, file, fileList);
  21. };
  22. dataObject.attrs["on-remove"] = (file, fileList) => {
  23. this.$emit("deleteUpload", file, fileList);
  24. };
  25. dataObject.attrs["on-preview"] = (file, fileList) => {
  26. this.$emit("previewUpload", file, fileList);
  27. };
  28. // 获取上传表单元素的默认值
  29. try {
  30. dataObject.props["file-list"] = JSON.parse(defaultValue);
  31. } catch (err) {
  32. console.warn(err);
  33. }
  34. return;
  35. }
  36. // 获取普通表单元素的值
  37. dataObject.props.value = defaultValue;
  38. dataObject.on.input = (val) => {
  39. this.$emit("input", val);
  40. };
  41. }
  42. function mountSlotFiles(h, confClone, children) {
  43. const childObjs = componentChild[confClone.__config__.tag]
  44. if (childObjs) {
  45. Object.keys(childObjs).forEach(key => {
  46. const childFunc = childObjs[key]
  47. if (confClone.__slot__ && confClone.__slot__[key]) {
  48. children.push(childFunc(h, confClone, key))
  49. }
  50. })
  51. }
  52. }
  53. function emitEvents(confClone) {
  54. ['on', 'nativeOn'].forEach(attr => {
  55. const eventKeyList = Object.keys(confClone[attr] || {})
  56. eventKeyList.forEach(key => {
  57. const val = confClone[attr][key]
  58. if (typeof val === 'string') {
  59. confClone[attr][key] = event => this.$emit(val, event)
  60. }
  61. })
  62. })
  63. }
  64. function buildDataObject(confClone, dataObject) {
  65. Object.keys(confClone).forEach(key => {
  66. const val = confClone[key]
  67. if (key === '__vModel__') {
  68. vModel.call(this, dataObject, confClone.__config__.defaultValue, confClone.__config__)
  69. } else if (dataObject[key] !== undefined) {
  70. if (dataObject[key] === null
  71. || dataObject[key] instanceof RegExp
  72. || ['boolean', 'string', 'number', 'function'].includes(typeof dataObject[key])) {
  73. dataObject[key] = val
  74. } else if (Array.isArray(dataObject[key])) {
  75. dataObject[key] = [...dataObject[key], ...val]
  76. } else {
  77. dataObject[key] = { ...dataObject[key], ...val }
  78. }
  79. } else {
  80. dataObject.attrs[key] = val
  81. }
  82. })
  83. // 清理属性
  84. clearAttrs(dataObject)
  85. }
  86. function clearAttrs(dataObject) {
  87. delete dataObject.attrs.__config__
  88. delete dataObject.attrs.__slot__
  89. delete dataObject.attrs.__methods__
  90. }
  91. function makeDataObject() {
  92. // 深入数据对象:
  93. // https://cn.vuejs.org/v2/guide/render-function.html#%E6%B7%B1%E5%85%A5%E6%95%B0%E6%8D%AE%E5%AF%B9%E8%B1%A1
  94. return {
  95. class: {},
  96. attrs: {},
  97. props: {},
  98. domProps: {},
  99. nativeOn: {},
  100. on: {},
  101. style: {},
  102. directives: [],
  103. scopedSlots: {},
  104. slot: null,
  105. key: null,
  106. ref: null,
  107. refInFor: true
  108. }
  109. }
  110. export default {
  111. props: {
  112. conf: {
  113. type: Object,
  114. required: true
  115. }
  116. },
  117. render(h) {
  118. const dataObject = makeDataObject()
  119. const confClone = deepClone(this.conf)
  120. const children = this.$slots.default || []
  121. // 如果slots文件夹存在与当前tag同名的文件,则执行文件中的代码
  122. mountSlotFiles.call(this, h, confClone, children)
  123. // 将字符串类型的事件,发送为消息
  124. emitEvents.call(this, confClone)
  125. // 将json表单配置转化为vue render可以识别的 “数据对象(dataObject)”
  126. buildDataObject.call(this, confClone, dataObject)
  127. return h(this.conf.__config__.tag, dataObject, children)
  128. }
  129. }