render.js 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  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) {
  16. dataObject.props.value = defaultValue
  17. dataObject.on.input = val => {
  18. this.$emit('input', val)
  19. }
  20. }
  21. function mountSlotFiles(h, confClone, children) {
  22. const childObjs = componentChild[confClone.__config__.tag]
  23. if (childObjs) {
  24. Object.keys(childObjs).forEach(key => {
  25. const childFunc = childObjs[key]
  26. if (confClone.__slot__ && confClone.__slot__[key]) {
  27. children.push(childFunc(h, confClone, key))
  28. }
  29. })
  30. }
  31. }
  32. function emitEvents(confClone) {
  33. ['on', 'nativeOn'].forEach(attr => {
  34. const eventKeyList = Object.keys(confClone[attr] || {})
  35. eventKeyList.forEach(key => {
  36. const val = confClone[attr][key]
  37. if (typeof val === 'string') {
  38. confClone[attr][key] = event => this.$emit(val, event)
  39. }
  40. })
  41. })
  42. }
  43. function buildDataObject(confClone, dataObject) {
  44. Object.keys(confClone).forEach(key => {
  45. const val = confClone[key]
  46. if (key === '__vModel__') {
  47. vModel.call(this, dataObject, confClone.__config__.defaultValue)
  48. } else if (dataObject[key] !== undefined) {
  49. if (dataObject[key] === null
  50. || dataObject[key] instanceof RegExp
  51. || ['boolean', 'string', 'number', 'function'].includes(typeof dataObject[key])) {
  52. dataObject[key] = val
  53. } else if (Array.isArray(dataObject[key])) {
  54. dataObject[key] = [...dataObject[key], ...val]
  55. } else {
  56. dataObject[key] = { ...dataObject[key], ...val }
  57. }
  58. } else {
  59. dataObject.attrs[key] = val
  60. }
  61. })
  62. // 清理属性
  63. clearAttrs(dataObject)
  64. }
  65. function clearAttrs(dataObject) {
  66. delete dataObject.attrs.__config__
  67. delete dataObject.attrs.__slot__
  68. delete dataObject.attrs.__methods__
  69. }
  70. function makeDataObject() {
  71. // 深入数据对象:
  72. // 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
  73. return {
  74. class: {},
  75. attrs: {},
  76. props: {},
  77. domProps: {},
  78. nativeOn: {},
  79. on: {},
  80. style: {},
  81. directives: [],
  82. scopedSlots: {},
  83. slot: null,
  84. key: null,
  85. ref: null,
  86. refInFor: true
  87. }
  88. }
  89. export default {
  90. props: {
  91. conf: {
  92. type: Object,
  93. required: true
  94. }
  95. },
  96. render(h) {
  97. const dataObject = makeDataObject()
  98. const confClone = deepClone(this.conf)
  99. const children = this.$slots.default || []
  100. // 如果slots文件夹存在与当前tag同名的文件,则执行文件中的代码
  101. mountSlotFiles.call(this, h, confClone, children)
  102. // 将字符串类型的事件,发送为消息
  103. emitEvents.call(this, confClone)
  104. // 将json表单配置转化为vue render可以识别的 “数据对象(dataObject)”
  105. buildDataObject.call(this, confClone, dataObject)
  106. return h(this.conf.__config__.tag, dataObject, children)
  107. }
  108. }