DraggableItem.vue 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. <script>
  2. import draggable from 'vuedraggable'
  3. import render from '@/utils/generator/render'
  4. const components = {
  5. itemBtns(h, element, index, parent) {
  6. const { copyItem, deleteItem } = this.$listeners
  7. return [
  8. <span class="drawing-item-copy" title="复制" onClick={event => {
  9. copyItem(element, parent); event.stopPropagation()
  10. }}>
  11. <i class="el-icon-copy-document" />
  12. </span>,
  13. <span class="drawing-item-delete" title="删除" onClick={event => {
  14. deleteItem(index, parent); event.stopPropagation()
  15. }}>
  16. <i class="el-icon-delete" />
  17. </span>
  18. ]
  19. }
  20. }
  21. const layouts = {
  22. colFormItem(h, element, index, parent) {
  23. const { activeItem } = this.$listeners
  24. let className = this.activeId === element.formId ? 'drawing-item active-from-item' : 'drawing-item'
  25. if (this.formConf.unFocusedComponentBorder) className += ' unfocus-bordered'
  26. return (
  27. <el-col span={element.span} class={className}
  28. nativeOnClick={event => { activeItem(element); event.stopPropagation() }}>
  29. <el-form-item label-width={element.labelWidth ? `${element.labelWidth}px` : null}
  30. label={element.label} required={element.required}>
  31. <render key={element.renderKey} conf={element} onInput={ event => {
  32. this.$set(element, 'defaultValue', event)
  33. }} />
  34. </el-form-item>
  35. {components.itemBtns.apply(this, arguments)}
  36. </el-col>
  37. )
  38. },
  39. rowFormItem(h, element, index, parent) {
  40. const { activeItem } = this.$listeners
  41. const className = this.activeId === element.formId ? 'drawing-row-item active-from-item' : 'drawing-row-item'
  42. let child = renderChildren.apply(this, arguments)
  43. if (element.type === 'flex') {
  44. child = <el-row type={element.type} justify={element.justify} align={element.align}>
  45. {child}
  46. </el-row>
  47. }
  48. return (
  49. <el-col span={element.span}>
  50. <el-row gutter={element.gutter} class={className}
  51. nativeOnClick={event => { activeItem(element); event.stopPropagation() }}>
  52. <span class="component-name">{element.componentName}</span>
  53. <draggable list={element.children} animation={340} group="componentsGroup" class="drag-wrapper">
  54. {child}
  55. </draggable>
  56. {components.itemBtns.apply(this, arguments)}
  57. </el-row>
  58. </el-col>
  59. )
  60. }
  61. }
  62. function renderChildren(h, element, index, parent) {
  63. if (!Array.isArray(element.children)) return null
  64. return element.children.map((el, i) => {
  65. const layout = layouts[el.layout]
  66. if (layout) {
  67. return layout.call(this, h, el, i, element.children)
  68. }
  69. return layoutIsNotFound()
  70. })
  71. }
  72. function layoutIsNotFound() {
  73. throw new Error(`没有与${this.element.layout}匹配的layout`)
  74. }
  75. export default {
  76. components: {
  77. render,
  78. draggable
  79. },
  80. props: [
  81. 'element',
  82. 'index',
  83. 'drawingList',
  84. 'activeId',
  85. 'formConf'
  86. ],
  87. render(h) {
  88. const layout = layouts[this.element.layout]
  89. if (layout) {
  90. return layout.call(this, h, this.element, this.index, this.drawingList)
  91. }
  92. return layoutIsNotFound()
  93. }
  94. }
  95. </script>