You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
145 lines
4.4 KiB
145 lines
4.4 KiB
import {deepClone} from '@/utils/index'
|
|
import SignPad from '@/components/form/SignPad/index'
|
|
|
|
const componentChild = {}
|
|
/**
|
|
* 将./slots中的文件挂载到对象componentChild上
|
|
* 文件名为key,对应JSON配置中的__config__.tag
|
|
* 文件内容为value,解析JSON配置中的__slot__
|
|
*/
|
|
const slotsFiles = require.context('./slots', false, /\.js$/)
|
|
const keys = slotsFiles.keys() || []
|
|
keys.forEach(key => {
|
|
const tag = key.replace(/^\.\/(.*)\.\w+$/, '$1')
|
|
const value = slotsFiles(key).default
|
|
componentChild[tag] = value
|
|
})
|
|
|
|
function vModel(dataObject, defaultValue) {
|
|
let config = dataObject.attrs.__config__
|
|
// 表单组件特殊处理
|
|
if (config.tag === 'el-upload') {
|
|
// 增加上传回调事件
|
|
dataObject.attrs['on-success'] = (response, file, fileList) => {
|
|
this.$emit('upload', response, file, fileList)
|
|
}
|
|
dataObject.attrs['on-remove'] = (file, fileList) => {
|
|
// eslint-disable-next-line vue/custom-event-name-casing
|
|
this.$emit('deleteUpload', file, fileList)
|
|
}
|
|
// eslint-disable-next-line no-unused-vars
|
|
dataObject.attrs['on-exceed'] = (files, fileList) => {
|
|
this.$message.error(`最多上传${config.limit}个文件`)
|
|
},
|
|
dataObject.attrs['before-upload'] = file => {
|
|
let sizeUnitNum = 1
|
|
// 文件大小判断
|
|
switch (config.sizeUnit) {
|
|
case 'KB':
|
|
sizeUnitNum = 1024
|
|
break
|
|
case 'MB':
|
|
sizeUnitNum = 1024 * 1024
|
|
break
|
|
case 'GB':
|
|
sizeUnitNum = 1024 * 1024 * 1024
|
|
break
|
|
}
|
|
let totalSize = config.fileSize * sizeUnitNum
|
|
if (file.size > totalSize) {
|
|
this.$message.error(`上传图片大小不能超过${config.fileSize}${config.sizeUnit}`)
|
|
return false
|
|
}
|
|
}
|
|
} else {
|
|
dataObject.props.value = defaultValue
|
|
|
|
dataObject.on.input = val => {
|
|
this.$emit('input', val)
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
function mountSlotFlies(h, confClone, children) {
|
|
const childObjs = componentChild[confClone.__config__.tag]
|
|
if (childObjs) {
|
|
Object.keys(childObjs).forEach(key => {
|
|
const childFunc = childObjs[key]
|
|
if (confClone.__slot__ && confClone.__slot__[key]) {
|
|
children.push(childFunc(h, confClone, key, this))
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
function emitEvents(confClone) {
|
|
['on', 'nativeOn'].forEach(attr => {
|
|
const eventKeyList = Object.keys(confClone[attr] || {})
|
|
eventKeyList.forEach(key => {
|
|
const val = confClone[attr][key]
|
|
if (typeof val === 'string') {
|
|
confClone[attr][key] = event => this.$emit(val, event)
|
|
}
|
|
})
|
|
})
|
|
}
|
|
|
|
function buildDataObject(confClone, dataObject) {
|
|
Object.keys(confClone).forEach(key => {
|
|
const val = confClone[key]
|
|
if (key === '__vModel__') {
|
|
vModel.call(this, dataObject, confClone.__config__.defaultValue)
|
|
} else if (dataObject[key]) {
|
|
dataObject[key] = {...dataObject[key], ...val}
|
|
} else {
|
|
dataObject.attrs[key] = val
|
|
}
|
|
})
|
|
|
|
// 清理属性
|
|
clearAttrs(dataObject)
|
|
}
|
|
|
|
function clearAttrs(dataObject) {
|
|
delete dataObject.attrs.__config__
|
|
delete dataObject.attrs.__slot__
|
|
delete dataObject.attrs.__methods__
|
|
}
|
|
|
|
function makeDataObject() {
|
|
return {
|
|
attrs: {},
|
|
props: {},
|
|
nativeOn: {},
|
|
on: {},
|
|
style: {}
|
|
}
|
|
}
|
|
|
|
export default {
|
|
components:{
|
|
SignPad
|
|
},
|
|
props: {
|
|
conf: {
|
|
type: Object,
|
|
required: true
|
|
}
|
|
},
|
|
render(h) {
|
|
const dataObject = makeDataObject()
|
|
const confClone = deepClone(this.conf)
|
|
const children = this.$slots.default || []
|
|
// 如果slots文件夹存在与当前tag同名的文件,则执行文件中的代码
|
|
mountSlotFlies.call(this, h, confClone, children)
|
|
|
|
// 将字符串类型的事件,发送为消息
|
|
emitEvents.call(this, confClone)
|
|
|
|
// 将json表单配置转化为vue render可以识别的 “数据对象(dataObject)”
|
|
buildDataObject.call(this, confClone, dataObject)
|
|
|
|
return h(this.conf.__config__.tag, dataObject, children)
|
|
}
|
|
}
|
|
|