amis-editor 4.1.0-beta.4 → 4.2.0-beta.3

Sign up to get free protection for your applications and to get access to all the features.
Files changed (218) hide show
  1. package/dist/component/Breadcrumb.d.ts +23 -2
  2. package/dist/component/ClassNameControl.d.ts +1 -1
  3. package/dist/component/Control/APIControl.d.ts +2 -2
  4. package/dist/component/Control/OptionControl.d.ts +3 -3
  5. package/dist/component/Control/PopoverEdit.d.ts +4 -4
  6. package/dist/component/Control/ValidationControl.d.ts +1 -1
  7. package/dist/component/Editor.d.ts +20 -1
  8. package/dist/component/IFramePreview.d.ts +2 -1
  9. package/dist/component/NodeWrapper.d.ts +1 -1
  10. package/dist/component/Panel/RenderersPanel.d.ts +1 -2
  11. package/dist/component/Preview.d.ts +3 -1
  12. package/dist/component/RegionWrapper.d.ts +4 -0
  13. package/dist/component/ScaffoldModal.d.ts +1 -1
  14. package/dist/component/VRenderer.d.ts +4 -0
  15. package/dist/component/base/InputComponentName.d.ts +1 -1
  16. package/dist/component/base/SearchCustomRendererPanel.d.ts +15 -0
  17. package/dist/component/base/SearchPanel.d.ts +83 -0
  18. package/dist/component/base/SearchRendererPanel.d.ts +2 -42
  19. package/dist/component/base/ShortcutKey.d.ts +23 -0
  20. package/dist/env.d.ts +1 -1
  21. package/dist/exports.min.js +1 -1
  22. package/dist/index.d.ts +5 -1
  23. package/dist/index.min.js +1 -1
  24. package/dist/manager.d.ts +16 -4
  25. package/dist/plugin/Alert.d.ts +1 -1
  26. package/dist/plugin/Button.d.ts +1 -1
  27. package/dist/plugin/Card.d.ts +1 -0
  28. package/dist/plugin/Cards.d.ts +2 -1
  29. package/dist/plugin/Carousel.d.ts +1 -0
  30. package/dist/plugin/Chart.d.ts +1 -0
  31. package/dist/plugin/Collapse.d.ts +1 -0
  32. package/dist/plugin/Custom.d.ts +3 -0
  33. package/dist/plugin/CustomRegion.d.ts +40 -0
  34. package/dist/plugin/Dialog.d.ts +1 -1
  35. package/dist/plugin/Drawer.d.ts +1 -1
  36. package/dist/plugin/DropDownButton.d.ts +1 -1
  37. package/dist/plugin/Each.d.ts +1 -1
  38. package/dist/plugin/Flex.d.ts +1 -0
  39. package/dist/plugin/Form/Control.d.ts +1 -0
  40. package/dist/plugin/Form/Form.d.ts +19 -3
  41. package/dist/plugin/Form/Formula.d.ts +3 -3
  42. package/dist/plugin/Form/InputURL.d.ts +1 -0
  43. package/dist/plugin/Form/Switch.d.ts +2 -0
  44. package/dist/plugin/Grid.d.ts +1 -0
  45. package/dist/plugin/HBox.d.ts +1 -0
  46. package/dist/plugin/IFrame.d.ts +3 -3
  47. package/dist/plugin/Json.d.ts +1 -0
  48. package/dist/plugin/List.d.ts +2 -1
  49. package/dist/plugin/Mapping.d.ts +1 -0
  50. package/dist/plugin/Markdown.d.ts +1 -0
  51. package/dist/plugin/Nav.d.ts +1 -0
  52. package/dist/plugin/Page.d.ts +1 -1
  53. package/dist/plugin/Panel/Outline.d.ts +8 -0
  54. package/dist/plugin/Progress.d.ts +1 -0
  55. package/dist/plugin/QRCode.d.ts +1 -0
  56. package/dist/plugin/Reset.d.ts +0 -1
  57. package/dist/plugin/Service.d.ts +1 -0
  58. package/dist/plugin/Sparkline.d.ts +1 -0
  59. package/dist/plugin/Status.d.ts +1 -0
  60. package/dist/plugin/Steps.d.ts +1 -0
  61. package/dist/plugin/Submit.d.ts +0 -1
  62. package/dist/plugin/Table.d.ts +1 -1
  63. package/dist/plugin/TableView.d.ts +1 -0
  64. package/dist/plugin/Tasks.d.ts +1 -0
  65. package/dist/plugin/TooltipWrapper.d.ts +1 -0
  66. package/dist/plugin/Video.d.ts +1 -0
  67. package/dist/plugin/WebComponent.d.ts +1 -0
  68. package/dist/plugin/Wizard.d.ts +11 -2
  69. package/dist/plugin/Wrapper.d.ts +1 -0
  70. package/dist/plugin.d.ts +8 -7
  71. package/dist/store/editor.d.ts +58 -22
  72. package/dist/store/node.d.ts +6 -0
  73. package/dist/style.css +1 -1
  74. package/dist/util.d.ts +1 -1
  75. package/package.json +9 -3
  76. package/src/component/schemaTpl.tsx +2157 -0
  77. package/src/plugin/Alert.tsx +87 -0
  78. package/src/plugin/AnchorNav.tsx +233 -0
  79. package/src/plugin/Audio.tsx +154 -0
  80. package/src/plugin/Avatar.tsx +77 -0
  81. package/src/plugin/Breadcrumb.tsx +107 -0
  82. package/src/plugin/Button.tsx +281 -0
  83. package/src/plugin/ButtonGroup.tsx +85 -0
  84. package/src/plugin/ButtonToolbar.tsx +87 -0
  85. package/src/plugin/CRUD.tsx +1835 -0
  86. package/src/plugin/Card.tsx +290 -0
  87. package/src/plugin/Cards.tsx +318 -0
  88. package/src/plugin/Carousel.tsx +377 -0
  89. package/src/plugin/Chart.tsx +218 -0
  90. package/src/plugin/CodeView.tsx +60 -0
  91. package/src/plugin/Collapse.tsx +136 -0
  92. package/src/plugin/CollapseGroup.tsx +167 -0
  93. package/src/plugin/Container.tsx +40 -0
  94. package/src/plugin/Custom.tsx +128 -0
  95. package/src/plugin/CustomRegion.tsx +156 -0
  96. package/src/plugin/Date.tsx +74 -0
  97. package/src/plugin/Datetime.tsx +68 -0
  98. package/src/plugin/Dialog.tsx +176 -0
  99. package/src/plugin/Divider.tsx +36 -0
  100. package/src/plugin/Drawer.tsx +214 -0
  101. package/src/plugin/DropDownButton.tsx +235 -0
  102. package/src/plugin/Each.tsx +150 -0
  103. package/src/plugin/ErrorRenderer.tsx +15 -0
  104. package/src/plugin/Flex.tsx +151 -0
  105. package/src/plugin/Form/ButtonGroupSelect.tsx +75 -0
  106. package/src/plugin/Form/ButtonToolbar.tsx +110 -0
  107. package/src/plugin/Form/ChainedSelect.tsx +70 -0
  108. package/src/plugin/Form/Checkbox.tsx +87 -0
  109. package/src/plugin/Form/Checkboxes.tsx +167 -0
  110. package/src/plugin/Form/CodeEditor.tsx +81 -0
  111. package/src/plugin/Form/Combo.tsx +582 -0
  112. package/src/plugin/Form/ConditionBuilder.tsx +315 -0
  113. package/src/plugin/Form/Control.tsx +139 -0
  114. package/src/plugin/Form/DiffEditor.tsx +111 -0
  115. package/src/plugin/Form/FieldSet.tsx +163 -0
  116. package/src/plugin/Form/Form.tsx +687 -0
  117. package/src/plugin/Form/Formula.tsx +79 -0
  118. package/src/plugin/Form/Group.tsx +295 -0
  119. package/src/plugin/Form/Hidden.tsx +44 -0
  120. package/src/plugin/Form/InputArray.tsx +228 -0
  121. package/src/plugin/Form/InputCity.tsx +93 -0
  122. package/src/plugin/Form/InputColor.tsx +123 -0
  123. package/src/plugin/Form/InputDate.tsx +175 -0
  124. package/src/plugin/Form/InputDateRange.tsx +225 -0
  125. package/src/plugin/Form/InputDateTime.tsx +183 -0
  126. package/src/plugin/Form/InputDateTimeRange.tsx +221 -0
  127. package/src/plugin/Form/InputEmail.tsx +33 -0
  128. package/src/plugin/Form/InputExcel.tsx +85 -0
  129. package/src/plugin/Form/InputFile.tsx +221 -0
  130. package/src/plugin/Form/InputGroup.tsx +96 -0
  131. package/src/plugin/Form/InputImage.tsx +266 -0
  132. package/src/plugin/Form/InputKV.tsx +72 -0
  133. package/src/plugin/Form/InputMonth.tsx +35 -0
  134. package/src/plugin/Form/InputMonthRange.tsx +195 -0
  135. package/src/plugin/Form/InputNumber.tsx +89 -0
  136. package/src/plugin/Form/InputPassword.tsx +33 -0
  137. package/src/plugin/Form/InputQuarter.tsx +35 -0
  138. package/src/plugin/Form/InputQuarterRange.tsx +195 -0
  139. package/src/plugin/Form/InputRange.tsx +121 -0
  140. package/src/plugin/Form/InputRating.tsx +78 -0
  141. package/src/plugin/Form/InputRepeat.tsx +57 -0
  142. package/src/plugin/Form/InputRichText.tsx +186 -0
  143. package/src/plugin/Form/InputSubForm.tsx +189 -0
  144. package/src/plugin/Form/InputTable.tsx +434 -0
  145. package/src/plugin/Form/InputTag.tsx +70 -0
  146. package/src/plugin/Form/InputText.tsx +186 -0
  147. package/src/plugin/Form/InputTime.tsx +85 -0
  148. package/src/plugin/Form/InputTree.tsx +229 -0
  149. package/src/plugin/Form/InputURL.tsx +34 -0
  150. package/src/plugin/Form/InputYear.tsx +35 -0
  151. package/src/plugin/Form/Item.tsx +327 -0
  152. package/src/plugin/Form/ListSelect.tsx +73 -0
  153. package/src/plugin/Form/LocationPicker.tsx +62 -0
  154. package/src/plugin/Form/MatrixCheckboxes.tsx +136 -0
  155. package/src/plugin/Form/NestedSelect.tsx +211 -0
  156. package/src/plugin/Form/Picker.tsx +209 -0
  157. package/src/plugin/Form/Radios.tsx +119 -0
  158. package/src/plugin/Form/Select.tsx +233 -0
  159. package/src/plugin/Form/Static.tsx +322 -0
  160. package/src/plugin/Form/Switch.tsx +107 -0
  161. package/src/plugin/Form/TabsTransfer.tsx +259 -0
  162. package/src/plugin/Form/Textarea.tsx +83 -0
  163. package/src/plugin/Form/Transfer.tsx +368 -0
  164. package/src/plugin/Form/TreeSelect.tsx +263 -0
  165. package/src/plugin/Form/UUID.tsx +48 -0
  166. package/src/plugin/Grid.tsx +798 -0
  167. package/src/plugin/HBox.tsx +727 -0
  168. package/src/plugin/IFrame.tsx +70 -0
  169. package/src/plugin/Image.tsx +314 -0
  170. package/src/plugin/Images.tsx +231 -0
  171. package/src/plugin/Json.tsx +69 -0
  172. package/src/plugin/Link.tsx +93 -0
  173. package/src/plugin/List.tsx +278 -0
  174. package/src/plugin/ListItem.tsx +229 -0
  175. package/src/plugin/Log.tsx +52 -0
  176. package/src/plugin/Mapping.tsx +149 -0
  177. package/src/plugin/Markdown.tsx +47 -0
  178. package/src/plugin/Nav.tsx +184 -0
  179. package/src/plugin/Operation.tsx +95 -0
  180. package/src/plugin/Others/Action.tsx +426 -0
  181. package/src/plugin/Others/BasicToolbar.tsx +585 -0
  182. package/src/plugin/Others/DataDebug.tsx +134 -0
  183. package/src/plugin/Others/TableCell.tsx +480 -0
  184. package/src/plugin/Others/Unknown.tsx +37 -0
  185. package/src/plugin/Page.tsx +306 -0
  186. package/src/plugin/Panel/AvailableRenderers.tsx +41 -0
  187. package/src/plugin/Panel/Code.tsx +44 -0
  188. package/src/plugin/Panel/Name.tsx +26 -0
  189. package/src/plugin/Panel/Outline.tsx +40 -0
  190. package/src/plugin/Panel.tsx +243 -0
  191. package/src/plugin/Plain.tsx +84 -0
  192. package/src/plugin/Progress.tsx +125 -0
  193. package/src/plugin/Property.tsx +139 -0
  194. package/src/plugin/QRCode.tsx +96 -0
  195. package/src/plugin/Reset.tsx +23 -0
  196. package/src/plugin/Service.tsx +162 -0
  197. package/src/plugin/Sparkline.tsx +40 -0
  198. package/src/plugin/Status.tsx +76 -0
  199. package/src/plugin/Steps.tsx +128 -0
  200. package/src/plugin/Submit.tsx +23 -0
  201. package/src/plugin/Table.tsx +428 -0
  202. package/src/plugin/TableView.tsx +709 -0
  203. package/src/plugin/Tabs.tsx +362 -0
  204. package/src/plugin/Tasks.tsx +274 -0
  205. package/src/plugin/Time.tsx +68 -0
  206. package/src/plugin/TooltipWrapper.tsx +193 -0
  207. package/src/plugin/Tpl.tsx +158 -0
  208. package/src/plugin/Video.tsx +158 -0
  209. package/src/plugin/WebComponent.tsx +53 -0
  210. package/src/plugin/Wizard.tsx +740 -0
  211. package/src/plugin/Wrapper.tsx +107 -0
  212. package/src/plugin.ts +1050 -0
  213. package/dist/150a58f3318ca7541ed9.png +0 -0
  214. package/dist/471adb97c322b226e589.png +0 -0
  215. package/dist/4de5f42360bc5946c3c2.png +0 -0
  216. package/dist/4e9968bba3855f088fed.png +0 -0
  217. package/dist/7f09c38ebc687fea847a.png +0 -0
  218. package/dist/c94073576487510314ea.png +0 -0
@@ -0,0 +1,585 @@
1
+ import {
2
+ BasePlugin,
3
+ BaseEventContext,
4
+ BasicToolbarItem,
5
+ ContextMenuItem,
6
+ ContextMenuEventContext,
7
+ BasicPanelItem,
8
+ BuildPanelEventContext,
9
+ PluginEvent,
10
+ InsertEventContext
11
+ } from '../../plugin';
12
+ import {registerEditorPlugin} from '../../manager';
13
+ import {EditorNodeType} from '../../store/node';
14
+ import type {MenuItem} from 'amis-ui/lib/components/ContextMenu';
15
+
16
+ /**
17
+ * 用来给当前选中的元素添加一些基本的工具栏按钮。
18
+ */
19
+ export class BasicToolbarPlugin extends BasePlugin {
20
+ order = -9999;
21
+
22
+ buildEditorToolbar(
23
+ {id, schema}: BaseEventContext,
24
+ toolbars: Array<BasicToolbarItem>
25
+ ) {
26
+ const store = this.manager.store;
27
+ const node = store.getNodeById(id)!;
28
+ const parent = store.getSchemaParentById(id);
29
+ // let vertical = true;
30
+ const regionNode = node.parent as EditorNodeType; // 父级节点
31
+ if (Array.isArray(parent) && regionNode?.isRegion) {
32
+ const host = node.host as EditorNodeType;
33
+
34
+ if (node.draggable) {
35
+ toolbars.push({
36
+ iconSvg: 'drag-btn',
37
+ icon: 'fa fa-arrows',
38
+ tooltip: '按住拖动调整位置',
39
+ placement: 'bottom',
40
+ draggable: true,
41
+ order: -1000,
42
+ onDragStart: this.manager.startDrag.bind(this.manager, id)
43
+ });
44
+ }
45
+
46
+ const idx = parent.indexOf(schema);
47
+
48
+ // if (idx > 0 && node.moveable) {
49
+ // let icon = 'fa fa-arrow-up';
50
+ // let tooltip = '向上移动';
51
+
52
+ // const dom = this.manager.store
53
+ // .getDoc()
54
+ // .querySelector(`[data-editor-id="${id}"]`);
55
+ // const prevDom = this.manager.store
56
+ // .getDoc()
57
+ // .querySelector(`[data-editor-id="${parent[idx - 1]?.$$id}"]`);
58
+
59
+ // if (dom && prevDom) {
60
+ // const prevRect = prevDom.getBoundingClientRect();
61
+ // const rect = dom.getBoundingClientRect();
62
+
63
+ // if (Math.abs(rect.x - prevRect.x) > Math.abs(rect.y - prevRect.y)) {
64
+ // vertical = false;
65
+ // icon = 'fa fa-arrow-left';
66
+ // tooltip = '向前移动';
67
+ // }
68
+
69
+ // toolbars.push({
70
+ // icon: icon,
71
+ // // tooltip: '向前移动(⌘ + ←)',
72
+ // tooltip: tooltip,
73
+ // onClick: () => this.manager.moveUp()
74
+ // });
75
+ // }
76
+ // }
77
+
78
+ // if (idx < parent.length - 1 && node.moveable) {
79
+ // let icon = 'fa fa-arrow-down';
80
+ // let tooltip = '向下移动';
81
+
82
+ // const dom = this.manager.store
83
+ // .getDoc()
84
+ // .querySelector(`[data-editor-id="${id}"]`);
85
+ // const nextDom = this.manager.store
86
+ // .getDoc()
87
+ // .querySelector(`[data-editor-id="${parent[idx + 1]?.$$id}"]`);
88
+
89
+ // if (dom && nextDom) {
90
+ // const nextRect = nextDom.getBoundingClientRect();
91
+ // const rect = dom.getBoundingClientRect();
92
+
93
+ // if (Math.abs(rect.x - nextRect.x) > Math.abs(rect.y - nextRect.y)) {
94
+ // vertical = false;
95
+ // icon = 'fa fa-arrow-right';
96
+ // tooltip = '向后移动';
97
+ // }
98
+
99
+ // toolbars.push({
100
+ // icon: icon,
101
+ // // tooltip: '向后移动(⌘ + →)',
102
+ // tooltip: tooltip,
103
+ // onClick: () => this.manager.moveDown()
104
+ // });
105
+ // }
106
+ // }
107
+
108
+ // if (node.removable) {
109
+ // toolbars.push({
110
+ // icon: 'fa fa-trash-o',
111
+ // // tooltip: '删除(Del)',
112
+ // tooltip: '删除',
113
+ // onClick: () => this.manager.del(id)
114
+ // });
115
+ // }
116
+
117
+ if (
118
+ !host?.memberImmutable(regionNode.region) &&
119
+ store.panels.some(Panel => Panel.key === 'renderers')
120
+ ) {
121
+ const nextId = parent[idx + 1]?.$$id;
122
+
123
+ toolbars.push(
124
+ {
125
+ iconSvg: 'left-arrow-to-left',
126
+ tooltip: '向前插入组件',
127
+ // level: 'special',
128
+ placement: 'bottom',
129
+ // placement: vertical ? 'bottom' : 'right',
130
+ // className: vertical
131
+ // ? 'ae-InsertBefore is-vertical'
132
+ // : 'ae-InsertBefore',
133
+ onClick: () =>
134
+ this.manager.showInsertPanel(
135
+ regionNode.region,
136
+ regionNode.id,
137
+ regionNode.preferTag,
138
+ 'insert',
139
+ undefined,
140
+ id
141
+ )
142
+ },
143
+ {
144
+ iconSvg: 'arrow-to-right',
145
+ tooltip: '向后插入组件',
146
+ // level: 'special',
147
+ placement: 'bottom',
148
+ // placement: vertical ? 'top' : 'left',
149
+ // className: vertical
150
+ // ? 'ae-InsertAfter is-vertical'
151
+ // : 'ae-InsertAfter',
152
+ onClick: () =>
153
+ this.manager.showInsertPanel(
154
+ regionNode.region,
155
+ regionNode.id,
156
+ regionNode.preferTag,
157
+ 'insert',
158
+ undefined,
159
+ nextId
160
+ )
161
+ }
162
+ );
163
+ }
164
+ }
165
+
166
+ if (
167
+ !node.isVitualRenderer &&
168
+ (node.info.plugin.popOverBody || node.info.plugin.popOverBodyCreator)
169
+ ) {
170
+ toolbars.push({
171
+ icon: 'fa fa-pencil',
172
+ tooltip: '编辑',
173
+ placement: 'bottom',
174
+ onClick: e => this.manager.openNodePopOverForm(node.id)
175
+ });
176
+ }
177
+
178
+ // if (node.duplicatable || node.duplicatable === undefined) {
179
+ // toolbars.push({
180
+ // iconSvg: 'copy-btn',
181
+ // icon: 'fa',
182
+ // tooltip: '复制',
183
+ // placement: 'bottom',
184
+ // order: 999,
185
+ // onClick: this.manager.duplicate.bind(this.manager, id)
186
+ // });
187
+ // }
188
+
189
+ if (node.removable || node.removable === undefined) {
190
+ toolbars.push({
191
+ iconSvg: 'delete-btn',
192
+ icon: 'fa',
193
+ tooltip: '删除',
194
+ placement: 'bottom',
195
+ order: 999,
196
+ onClick: this.manager.del.bind(this.manager, id)
197
+ });
198
+ }
199
+
200
+ toolbars.push({
201
+ iconSvg: 'more-btn',
202
+ icon: 'fa fa-cog',
203
+ tooltip: '更多',
204
+ placement: 'bottom',
205
+ order: 1000,
206
+ onClick: e => {
207
+ if (!e.defaultPrevented) {
208
+ const info = (e.target as HTMLElement).parentElement!.getBoundingClientRect();
209
+ this.manager.openContextMenu(id, '', {
210
+ x: window.scrollX + info.left + info.width - 155,
211
+ y: window.scrollY + info.top + info.height + 8
212
+ });
213
+ }
214
+ }
215
+ });
216
+ }
217
+
218
+ buildEditorContextMenu(
219
+ {id, schema, region, selections}: ContextMenuEventContext,
220
+ menus: Array<ContextMenuItem>
221
+ ) {
222
+ const manager = this.manager;
223
+ const store = manager.store;
224
+ const parent = store.getSchemaParentById(id);
225
+ const node = store.getNodeById(id)!;
226
+ const paths = store.getNodePathById(id);
227
+ const first = paths.pop()!;
228
+ const host = node.host as EditorNodeType;
229
+ const regionNode = node.parent as EditorNodeType;
230
+
231
+ if (selections.length) {
232
+ // 多选时的右键菜单
233
+ menus.push({
234
+ label: '重复一份',
235
+ icon: 'copy-icon',
236
+ disabled: selections.some(item => !item.node.duplicatable),
237
+ onSelect: () => manager.duplicate(selections.map(item => item.id))
238
+ });
239
+
240
+ menus.push({
241
+ label: '取消多选',
242
+ icon: 'cancel-icon',
243
+ onSelect: () => store.setActiveId(id)
244
+ });
245
+
246
+ menus.push({
247
+ label: '删除',
248
+ icon: 'delete-icon',
249
+ disabled: selections.some(item => !item.node.removable),
250
+ className: 'text-danger',
251
+ onSelect: () => manager.del(selections.map(item => item.id))
252
+ });
253
+ } else if (region) {
254
+ const renderersPanel = store.panels.find(
255
+ item => item.key === 'renderers'
256
+ );
257
+
258
+ if (renderersPanel) {
259
+ // region增加点选后就不需要'插入组件'了
260
+ /*
261
+ menus.push({
262
+ label: '插入组件',
263
+ onHighlight: (isOn: boolean) => isOn && store.setHoverId(id, region),
264
+ onSelect: () => manager.showInsertPanel(region, id)
265
+ });
266
+ */
267
+ menus.push({
268
+ label: '插入组件',
269
+ onHighlight: (isOn: boolean) => isOn && store.setHoverId(id, region),
270
+ onSelect: () => store.showInsertRendererPanel()
271
+ });
272
+
273
+ menus.push({
274
+ label: '清空',
275
+ onSelect: () => manager.emptyRegion(id, region)
276
+ });
277
+
278
+ menus.push({
279
+ label: '粘贴',
280
+ onSelect: () => manager.paste(id, region)
281
+ });
282
+ }
283
+ } else {
284
+ menus.push({
285
+ label: `选中${first.label}`,
286
+ disabled: store.activeId === first.id,
287
+ data: id,
288
+ onSelect: (id: string) => store.setActiveId(id),
289
+ onHighlight: (isHiglight: boolean, id: string) =>
290
+ isHiglight && store.setHoverId(id)
291
+ });
292
+
293
+ if (paths.length) {
294
+ const children = paths
295
+ .filter(node => !node.isRegion && node.info?.editable !== false)
296
+ .reverse()
297
+ .map(node => ({
298
+ label: node.label,
299
+ data: node.id,
300
+ onSelect: (id: string) => store.setActiveId(id),
301
+ onHighlight: (isHiglight: boolean, currentId: string) =>
302
+ isHiglight && store.setHoverId(currentId)
303
+ }));
304
+
305
+ children.length &&
306
+ menus.push({
307
+ label: '选中层级',
308
+ children: children
309
+ });
310
+ }
311
+
312
+ menus.push({
313
+ label: '取消选中',
314
+ disabled: !store.activeId || store.activeId !== id,
315
+ onSelect: () => store.setActiveId('')
316
+ });
317
+
318
+ menus.push('|');
319
+
320
+ menus.push({
321
+ label: '重复一份',
322
+ disabled: !node.duplicatable,
323
+ onSelect: () => manager.duplicate(id)
324
+ });
325
+
326
+ menus.push({
327
+ label: '复制配置',
328
+ onSelect: () => manager.copy(id)
329
+ });
330
+
331
+ menus.push({
332
+ label: '剪切配置',
333
+ disabled: !node.removable,
334
+ onSelect: () => manager.cut(id)
335
+ });
336
+
337
+ menus.push({
338
+ label: '粘贴配置',
339
+ disabled:
340
+ !Array.isArray(parent) ||
341
+ !node.parent ||
342
+ node.info?.typeMutable === false ||
343
+ !node.replaceable,
344
+ onSelect: () => manager.paste(id)
345
+ });
346
+
347
+ menus.push({
348
+ label: '删除',
349
+ disabled: !node.removable,
350
+ className: 'text-danger',
351
+ onSelect: () => manager.del(id)
352
+ });
353
+
354
+ menus.push('|');
355
+
356
+ const idx = Array.isArray(parent) ? parent.indexOf(schema) : -1;
357
+
358
+ menus.push({
359
+ label: '向前移动',
360
+ disabled:
361
+ !(Array.isArray(parent) && idx > 0) ||
362
+ !node.moveable ||
363
+ !node.prevSibling,
364
+ onSelect: () => manager.moveUp()
365
+ });
366
+
367
+ menus.push({
368
+ label: '向后移动',
369
+ disabled:
370
+ !(Array.isArray(parent) && idx < parent.length - 1) ||
371
+ !node.moveable ||
372
+ !node.nextSibling,
373
+ onSelect: () => manager.moveDown()
374
+ });
375
+
376
+ /** 「点选(默认向后插入)」+ 「向前移动」可以替换 「前面插入节点」 */
377
+ /*
378
+ menus.push({
379
+ label: '前面插入节点',
380
+ disabled:
381
+ !Array.isArray(parent) ||
382
+ !regionNode ||
383
+ !regionNode.isRegion ||
384
+ !host ||
385
+ host.memberImmutable(regionNode.region) ||
386
+ !store.panels.some(Panel => Panel.key === 'renderers'),
387
+ onSelect: () =>
388
+ this.manager.showInsertPanel(
389
+ regionNode.region,
390
+ regionNode.id,
391
+ regionNode.preferTag,
392
+ 'insert',
393
+ undefined,
394
+ id
395
+ )
396
+ });
397
+ */
398
+
399
+ /** 「点选(默认向后插入)」可以替换 「后面插入节点」 */
400
+ /*
401
+ menus.push({
402
+ label: '后面插入节点',
403
+ disabled:
404
+ !Array.isArray(parent) ||
405
+ !regionNode ||
406
+ !regionNode.isRegion ||
407
+ !host ||
408
+ host.memberImmutable(regionNode.region) ||
409
+ !store.panels.some(Panel => Panel.key === 'renderers'),
410
+ onSelect: () =>
411
+ this.manager.showInsertPanel(
412
+ regionNode.region,
413
+ regionNode.id,
414
+ regionNode.preferTag,
415
+ 'insert',
416
+ undefined,
417
+ parent[idx + 1]?.$$id
418
+ )
419
+ });
420
+ */
421
+
422
+ menus.push('|');
423
+
424
+ // const configPanel = store.panels.find(item => item.key === 'config');
425
+ // menus.push({
426
+ // label: '设置',
427
+ // onSelect: () => manager.openConfigPanel(id),
428
+ // disabled: !configPanel
429
+ // });
430
+
431
+ // const codePanel = store.panels.find(item => item.key === 'code');
432
+ // menus.push({
433
+ // label: '编辑代码',
434
+ // onSelect: () => manager.openCodePanel(id),
435
+ // disabled:
436
+ // !codePanel || (store.activeId === id && store.getPanelKey() === 'code')
437
+ // });
438
+
439
+ menus.push({
440
+ label: '撤销(Undo)',
441
+ disabled: !store.canUndo,
442
+ onSelect: () => store.undo()
443
+ });
444
+
445
+ menus.push({
446
+ label: '重做(Redo)',
447
+ disabled: !store.canRedo,
448
+ onSelect: () => store.redo()
449
+ });
450
+
451
+ // menus.push('|');
452
+
453
+ /** 可使用「点选(默认向后插入)」替代 */
454
+ /*
455
+ const renderersPanel = store.panels.find(
456
+ item => item.key === 'renderers'
457
+ );
458
+ if (first.childRegions.length && renderersPanel) {
459
+ if (first.childRegions.length > 1) {
460
+ menus.push({
461
+ label: '插入组件',
462
+ children: first.childRegions.map(region => ({
463
+ label: `${region.label}`,
464
+ data: region.region,
465
+ onHighlight: (isOn: boolean, region: string) =>
466
+ isOn ? store.setHoverId(id, region) : store.setHoverId(''),
467
+ onSelect: (region: string) => manager.showInsertPanel(region, id)
468
+ }))
469
+ });
470
+ } else {
471
+ menus.push({
472
+ label: '插入组件',
473
+ data: first.childRegions[0].region,
474
+ onHighlight: (isOn: boolean, region: string) =>
475
+ isOn ? store.setHoverId(id, region) : store.setHoverId(''),
476
+ onSelect: (region: string) => manager.showInsertPanel(region, id)
477
+ });
478
+ }
479
+ }
480
+ */
481
+
482
+ // 使用新版插入组件面板(抽屉弹出式)
483
+ const renderersPanel = store.panels.find(
484
+ item => item.key === 'renderers'
485
+ );
486
+ if (first.childRegions.length && renderersPanel) {
487
+ if (first.childRegions.length > 1) {
488
+ menus.push({
489
+ label: '插入组件',
490
+ children: first.childRegions.map(region => ({
491
+ label: `${region.label}`,
492
+ data: region.region,
493
+ onHighlight: (isOn: boolean, region: string) =>
494
+ isOn ? store.setHoverId(id, region) : store.setHoverId(''),
495
+ onSelect: () => store.showInsertRendererPanel()
496
+ }))
497
+ });
498
+ } else {
499
+ menus.push({
500
+ label: '插入组件',
501
+ data: first.childRegions[0].region,
502
+ onHighlight: (isOn: boolean, region: string) =>
503
+ isOn ? store.setHoverId(id, region) : store.setHoverId(''),
504
+ onSelect: () => store.showInsertRendererPanel()
505
+ });
506
+ }
507
+ }
508
+
509
+ /** 「点选(默认向后插入)」+ 「删除」可以替换 「更改类型」 */
510
+ /*
511
+ menus.push({
512
+ label: '更改类型',
513
+ disabled:
514
+ !node.host ||
515
+ node.info?.typeMutable === false ||
516
+ !node.parent.isRegion ||
517
+ !store.panels.some(Panel => Panel.key === 'renderers') ||
518
+ !node.replaceable,
519
+ onSelect: () => manager.showReplacePanel(id)
520
+ });
521
+ */
522
+ }
523
+ }
524
+
525
+ buildEditorPanel(
526
+ context: BuildPanelEventContext,
527
+ panels: Array<BasicPanelItem>
528
+ ) {
529
+ if (!context.selections.length) {
530
+ return;
531
+ }
532
+
533
+ let menus: Array<ContextMenuItem> = [];
534
+ const contextMenuContext: ContextMenuEventContext = {
535
+ ...context,
536
+ data: menus,
537
+ region: ''
538
+ };
539
+
540
+ menus = this.manager.buildContextMenus(contextMenuContext);
541
+
542
+ if (menus.length) {
543
+ panels.push({
544
+ key: 'contextmenu',
545
+ icon: 'fa fa-cog',
546
+ title: '操作',
547
+ menus: menus,
548
+ render: this.manager.makeSchemaFormRender({
549
+ body: [
550
+ {
551
+ type: 'button-group',
552
+ block: true,
553
+ buttons: menus
554
+ .filter(item => item !== '|')
555
+ .map(menu => ({
556
+ ...(menu as MenuItem),
557
+ type: 'button',
558
+ onClick: (menu as MenuItem).onSelect
559
+ }))
560
+ }
561
+ ]
562
+ })
563
+ });
564
+ }
565
+ }
566
+
567
+ afterInsert(event: PluginEvent<InsertEventContext>) {
568
+ const context = event.context;
569
+
570
+ if (context.node && context.subRenderer?.plugin?.popOverBody) {
571
+ const id = context.data.$$id;
572
+
573
+ if (id) {
574
+ setTimeout(() => {
575
+ this.manager.setActiveId(id);
576
+ requestAnimationFrame(() => {
577
+ this.manager.openNodePopOverForm(id);
578
+ });
579
+ }, 200);
580
+ }
581
+ }
582
+ }
583
+ }
584
+
585
+ registerEditorPlugin(BasicToolbarPlugin);
@@ -0,0 +1,134 @@
1
+ import {registerEditorPlugin} from '../../manager';
2
+ import {BaseEventContext, BasePlugin, BasicToolbarItem} from '../../plugin';
3
+ import React from 'react';
4
+ import JsonView, {InteractionProps} from 'react-json-view';
5
+
6
+ /**
7
+ * 添加调试功能
8
+ */
9
+ export class DataDebugPlugin extends BasePlugin {
10
+ buildEditorToolbar(
11
+ {id, schema, node}: BaseEventContext,
12
+ toolbars: Array<BasicToolbarItem>
13
+ ) {
14
+ const comp = node.getComponent();
15
+ if (!comp || !comp.props.data || !comp.props.store) {
16
+ return;
17
+ }
18
+
19
+ // const renderers = getRenderers();
20
+ // const renderInfo = find(
21
+ // renderers,
22
+ // renderer => renderer.Renderer && comp instanceof renderer.Renderer
23
+ // ) as RendererConfig;
24
+
25
+ // if (!renderInfo || !renderInfo.storeType) {
26
+ // return;
27
+ // }
28
+ const store = comp.props.store;
29
+
30
+ toolbars.push({
31
+ icon: 'fa fa-bug',
32
+ order: -1000,
33
+ placement: 'bottom',
34
+ tooltip: '上下文数据',
35
+ onClick: () =>
36
+ this.openDebugForm(
37
+ comp.props.data,
38
+ store.updateData && store.data === comp.props.data
39
+ ? values => store.updateData(values)
40
+ : undefined
41
+ )
42
+ });
43
+ }
44
+
45
+ dataViewer = {
46
+ type: 'json',
47
+ name: 'ctx',
48
+ asFormItem: true,
49
+ className: 'm-b-none',
50
+ component: ({
51
+ value,
52
+ onChange,
53
+ readOnly
54
+ }: {
55
+ value: any;
56
+ onChange: (value: any) => void;
57
+ readOnly?: boolean;
58
+ }) => {
59
+ const [index, setIndex] = React.useState(0);
60
+ let start = value || {};
61
+ const stacks = [start];
62
+
63
+ while (Object.getPrototypeOf(start) !== Object.prototype) {
64
+ const superData = Object.getPrototypeOf(start);
65
+
66
+ if (Object.prototype.toString.call(superData) !== '[object Object]') {
67
+ break;
68
+ }
69
+
70
+ stacks.push(superData);
71
+ start = superData;
72
+ }
73
+
74
+ function emitChange(e: InteractionProps) {
75
+ const obj = Object.create(stacks[1] || Object.prototype);
76
+ Object.keys(e.updated_src).forEach(
77
+ key => (obj[key] = (e.updated_src as any)[key])
78
+ );
79
+ onChange(obj);
80
+ }
81
+
82
+ return (
83
+ <div className="aeDataChain">
84
+ <div className="aeDataChain-aside">
85
+ <ul>
86
+ {stacks.map((_, i) => (
87
+ <li
88
+ className={i === index ? 'is-active' : ''}
89
+ key={i}
90
+ onClick={() => setIndex(i)}
91
+ >
92
+ {i === 0 ? '当前' : i === 1 ? '上层' : `上${i}层`}
93
+ </li>
94
+ ))}
95
+ </ul>
96
+ </div>
97
+ <div className="aeDataChain-main">
98
+ <JsonView
99
+ name={false}
100
+ src={stacks[index]}
101
+ enableClipboard={false}
102
+ iconStyle="square"
103
+ onAdd={index === 0 && !readOnly ? emitChange : false}
104
+ onEdit={index === 0 && !readOnly ? emitChange : false}
105
+ onDelete={index === 0 && !readOnly ? emitChange : false}
106
+ collapsed={2}
107
+ />
108
+ </div>
109
+ </div>
110
+ );
111
+ }
112
+ };
113
+
114
+ async openDebugForm(data: any, callback?: (values: any) => void) {
115
+ const result = await this.manager.scaffold(
116
+ {
117
+ title: '上下文数据',
118
+ body: [
119
+ {
120
+ ...this.dataViewer,
121
+ readOnly: callback ? false : true
122
+ }
123
+ ]
124
+ },
125
+ {
126
+ ctx: data
127
+ }
128
+ );
129
+
130
+ callback?.((result as any).ctx);
131
+ }
132
+ }
133
+
134
+ registerEditorPlugin(DataDebugPlugin);