@plait/mind 0.9.0 → 0.11.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (54) hide show
  1. package/README.md +132 -13
  2. package/base/base.drawer.d.ts +4 -3
  3. package/constants/default.d.ts +1 -1
  4. package/constants/node-topic-style.d.ts +1 -0
  5. package/drawer/node-active.drawer.d.ts +13 -0
  6. package/drawer/node-collapse.drawer.d.ts +8 -0
  7. package/drawer/{emojis.drawer.d.ts → node-emojis.drawer.d.ts} +1 -1
  8. package/drawer/{quick-insert.drawer.d.ts → node-insert.drawer.d.ts} +2 -2
  9. package/esm2020/base/base.drawer.mjs +13 -1
  10. package/esm2020/constants/default.mjs +2 -2
  11. package/esm2020/constants/node-rule.mjs +1 -1
  12. package/esm2020/constants/node-topic-style.mjs +2 -1
  13. package/esm2020/drawer/node-active.drawer.mjs +43 -0
  14. package/esm2020/drawer/node-collapse.drawer.mjs +108 -0
  15. package/esm2020/drawer/node-emojis.drawer.mjs +72 -0
  16. package/esm2020/drawer/node-insert.drawer.mjs +98 -0
  17. package/esm2020/interfaces/element.mjs +5 -2
  18. package/esm2020/mind.component.mjs +2 -1
  19. package/esm2020/mind.module.mjs +5 -5
  20. package/esm2020/node.component.mjs +57 -424
  21. package/esm2020/plugins/with-abstract-resize.mjs +3 -3
  22. package/esm2020/plugins/with-mind-create.mjs +22 -15
  23. package/esm2020/plugins/with-mind.mjs +9 -8
  24. package/esm2020/plugins/with-node-dnd.mjs +2 -2
  25. package/esm2020/plugins/with-node-hover.mjs +65 -0
  26. package/esm2020/transforms/node.mjs +10 -7
  27. package/esm2020/utils/abstract/resize.mjs +4 -6
  28. package/esm2020/utils/dnd/common.mjs +8 -2
  29. package/esm2020/utils/draw/abstract-outline.mjs +7 -7
  30. package/esm2020/utils/draw/node-dnd.mjs +3 -3
  31. package/esm2020/utils/mind.mjs +4 -4
  32. package/esm2020/utils/node/adjust-node.mjs +4 -5
  33. package/esm2020/utils/node/common.mjs +3 -3
  34. package/esm2020/utils/node/create-node.mjs +4 -3
  35. package/fesm2015/plait-mind.mjs +547 -726
  36. package/fesm2015/plait-mind.mjs.map +1 -1
  37. package/fesm2020/plait-mind.mjs +547 -714
  38. package/fesm2020/plait-mind.mjs.map +1 -1
  39. package/interfaces/element.d.ts +1 -0
  40. package/mind.module.d.ts +2 -2
  41. package/node.component.d.ts +15 -31
  42. package/package.json +1 -1
  43. package/plugins/with-mind-create.d.ts +3 -5
  44. package/plugins/with-node-dnd.d.ts +1 -1
  45. package/plugins/with-node-hover.d.ts +5 -0
  46. package/styles/mixins.scss +13 -15
  47. package/styles/styles.scss +11 -10
  48. package/utils/abstract/resize.d.ts +2 -2
  49. package/utils/draw/abstract-outline.d.ts +0 -1
  50. package/utils/node/common.d.ts +1 -1
  51. package/esm2020/drawer/emojis.drawer.mjs +0 -72
  52. package/esm2020/drawer/quick-insert.drawer.mjs +0 -98
  53. package/esm2020/utils/draw/node-topic.mjs +0 -32
  54. package/utils/draw/node-topic.d.ts +0 -16
@@ -1,97 +1,96 @@
1
1
  import { ChangeDetectionStrategy, Component } from '@angular/core';
2
- import { PlaitPointerType, createG, createText, IS_TEXT_EDITABLE, MERGING, PlaitBoard, toPoint, transformPoint, Transforms, drawRoundRectangle, PlaitPluginElementComponent, PlaitElement, NODE_TO_INDEX, RectangleClient } from '@plait/core';
3
- import { isBottomLayout, isHorizontalLayout, isIndentedLayout, isLeftLayout, AbstractNode, isRightLayout, isTopLayout } from '@plait/layouts';
4
- import { hasEditableTarget, setFullSelectionAndFocus, updateRichText } from '@plait/richtext';
5
- import { fromEvent, Subject, timer } from 'rxjs';
6
- import { debounceTime, filter, take, takeUntil } from 'rxjs/operators';
7
- import { Editor, Operation } from 'slate';
8
- import { EXTEND_OFFSET, EXTEND_RADIUS, PRIMARY_COLOR } from './constants';
9
- import { NODE_MIN_WIDTH } from './constants/node-rule';
10
- import { drawTopicByNode, updateMindNodeTopicSize } from './utils/draw/node-topic';
2
+ import { createG, PlaitBoard, PlaitPluginElementComponent, PlaitElement, NODE_TO_INDEX, RectangleClient } from '@plait/core';
3
+ import { isHorizontalLayout, AbstractNode } from '@plait/layouts';
4
+ import { TextManage } from '@plait/text';
5
+ import { Subject } from 'rxjs';
11
6
  import { drawRoundRectangleByNode } from './utils/draw/node-shape';
12
7
  import { MindElement, PlaitMind } from './interfaces/element';
13
8
  import { MindQueries } from './queries';
14
- import { getRectangleByNode, isHitMindElement } from './utils/position/node';
15
- import { getChildrenCount } from './utils/mind';
9
+ import { isHitMindElement } from './utils/position/node';
16
10
  import { getShapeByElement } from './utils/node-style/shape';
17
11
  import { ELEMENT_TO_NODE } from './utils/weak-maps';
18
- import { getRichtextContentSize } from '@plait/richtext';
19
12
  import { drawAbstractLink } from './utils/draw/node-link/abstract-link';
20
- import { EmojisDrawer } from './drawer/emojis.drawer';
13
+ import { NodeEmojisDrawer } from './drawer/node-emojis.drawer';
21
14
  import { MindTransforms } from './transforms';
22
- import { drawAbstractIncludedOutline } from './utils/draw/abstract-outline';
23
15
  import { MindElementShape } from './interfaces';
24
- import { QuickInsertDrawer } from './drawer/quick-insert.drawer';
25
- import { hasAfterDraw } from './base/base.drawer';
26
- import { getBranchColorByMindElement, getBranchWidthByMindElement } from './utils/node-style/branch';
16
+ import { NodeInsertDrawer } from './drawer/node-insert.drawer';
27
17
  import { drawLink } from './utils/draw/node-link/draw-link';
18
+ import { getTopicRectangleByNode } from './utils/position/topic';
19
+ import { NodeActiveDrawer } from './drawer/node-active.drawer';
20
+ import { CollapseDrawer } from './drawer/node-collapse.drawer';
28
21
  import * as i0 from "@angular/core";
29
22
  import * as i1 from "@angular/common";
30
23
  import * as i2 from "@plait/core";
31
24
  export class MindNodeComponent extends PlaitPluginElementComponent {
32
- get handActive() {
33
- return this.board.pointer === PlaitPointerType.hand;
34
- }
35
- constructor(viewContainerRef, cdr, render2, ngZone) {
25
+ constructor(viewContainerRef, cdr) {
36
26
  super(cdr);
37
27
  this.viewContainerRef = viewContainerRef;
38
28
  this.cdr = cdr;
39
- this.render2 = render2;
40
- this.ngZone = ngZone;
41
- this.isEditable = false;
42
- this.activeG = [];
43
29
  this.shapeG = null;
44
30
  this.destroy$ = new Subject();
45
31
  this.trackBy = (index, node) => {
46
32
  return node.origin.id;
47
33
  };
48
34
  }
35
+ initializeDrawer() {
36
+ this.nodeEmojisDrawer = new NodeEmojisDrawer(this.board, this.viewContainerRef);
37
+ this.nodeInsertDrawer = new NodeInsertDrawer(this.board);
38
+ this.activeDrawer = new NodeActiveDrawer(this.board);
39
+ this.collapseDrawer = new CollapseDrawer(this.board);
40
+ this.textManage = new TextManage(this.board, this.viewContainerRef, () => {
41
+ return getTopicRectangleByNode(this.board, this.node);
42
+ }, (point) => {
43
+ return isHitMindElement(this.board, point, this.element);
44
+ }, (textManageRef) => {
45
+ const width = textManageRef.width;
46
+ const height = textManageRef.height;
47
+ if (textManageRef.newValue) {
48
+ MindTransforms.setTopic(this.board, this.element, textManageRef.newValue, width, height);
49
+ }
50
+ else {
51
+ MindTransforms.setTopicSize(this.board, this.element, width, height);
52
+ }
53
+ });
54
+ }
49
55
  ngOnInit() {
50
- this.emojisDrawer = new EmojisDrawer(this.board, this.viewContainerRef);
51
- this.quickInsertDrawer = new QuickInsertDrawer(this.board);
52
56
  super.ngOnInit();
57
+ this.initializeDrawer();
53
58
  this.node = MindElement.getNode(this.element);
54
59
  this.index = NODE_TO_INDEX.get(this.element) || 0;
55
60
  this.roughSVG = PlaitBoard.getRoughSVG(this.board);
56
61
  this.parentG = PlaitElement.getComponent(MindElement.getRoot(this.board, this.element)).rootG;
57
62
  this.drawShape();
58
63
  this.drawLink();
59
- this.drawRichtext();
60
- this.drawActiveG();
61
- this.updateActiveClass();
62
- this.drawMaskG();
64
+ this.drawText();
65
+ this.activeDrawer.draw(this.element, this.g, { selected: this.selected, isEditing: this.textManage.isEditing });
63
66
  this.drawEmojis();
64
67
  this.drawExtend();
65
- this.drawQuickInsert();
68
+ }
69
+ editTopic() {
70
+ this.activeDrawer.draw(this.element, this.g, { selected: this.selected, isEditing: true });
71
+ this.textManage.edit(() => {
72
+ this.activeDrawer.draw(this.element, this.g, { selected: this.selected, isEditing: false });
73
+ });
66
74
  }
67
75
  onContextChanged(value, previous) {
68
76
  const newNode = MindElement.getNode(value.element);
69
- // resolve move node richtext lose issue
70
- if (this.node !== newNode) {
71
- if (this.foreignObject && this.foreignObject.children.length <= 0) {
72
- this.foreignObject?.appendChild(this.richtextComponentRef?.instance.editable);
73
- }
74
- }
75
77
  const isEqualNode = RectangleClient.isEqual(this.node, newNode);
76
78
  this.node = newNode;
77
79
  const isChangeTheme = this.board.operations.find(op => op.type === 'set_theme');
78
80
  if (!isEqualNode || value.element !== previous.element || isChangeTheme) {
79
- this.drawActiveG();
80
- this.updateActiveClass();
81
+ this.activeDrawer.draw(this.element, this.g, { selected: this.selected, isEditing: this.textManage.isEditing });
81
82
  this.drawShape();
82
83
  this.drawLink();
83
- this.updateRichtext();
84
- this.drawMaskG();
85
- this.drawExtend();
86
- this.drawQuickInsert();
87
84
  this.drawEmojis();
85
+ this.drawExtend();
86
+ this.textManage.updateText(this.element.data.topic);
87
+ this.textManage.updateRectangle();
88
88
  }
89
89
  else {
90
90
  const hasSameSelected = value.selected === previous.selected;
91
91
  const hasSameParent = value.parent === previous.parent;
92
92
  if (!hasSameSelected) {
93
- this.drawActiveG();
94
- this.updateActiveClass();
93
+ this.activeDrawer.draw(this.element, this.g, { selected: this.selected, isEditing: this.textManage.isEditing });
95
94
  }
96
95
  if (!hasSameParent) {
97
96
  this.drawLink();
@@ -99,21 +98,11 @@ export class MindNodeComponent extends PlaitPluginElementComponent {
99
98
  }
100
99
  }
101
100
  drawEmojis() {
102
- const g = this.emojisDrawer.drawEmojis(this.element);
101
+ const g = this.nodeEmojisDrawer.drawEmojis(this.element);
103
102
  if (g) {
104
103
  this.g.append(g);
105
104
  }
106
105
  }
107
- drawQuickInsert() {
108
- this.quickInsertDrawer.destroy();
109
- if (this.quickInsertDrawer.canDraw(this.element)) {
110
- const g = this.quickInsertDrawer.draw(this.element);
111
- if (hasAfterDraw(this.quickInsertDrawer)) {
112
- this.quickInsertDrawer.afterDraw(this.element);
113
- }
114
- this.extendG?.appendChild(g);
115
- }
116
- }
117
106
  drawShape() {
118
107
  this.destroyShape();
119
108
  const shape = getShapeByElement(this.board, this.node.origin);
@@ -155,389 +144,33 @@ export class MindNodeComponent extends PlaitPluginElementComponent {
155
144
  this.linkG.remove();
156
145
  }
157
146
  }
158
- drawMaskG() {
159
- this.destroyMaskG();
160
- const lineWidthOffset = 2;
161
- const extendOffset = 15;
162
- const nodeLayout = MindQueries.getLayoutByElement(this.node.origin);
163
- const isTop = isTopLayout(nodeLayout);
164
- const isRight = isRightLayout(nodeLayout);
165
- const isBottom = isBottomLayout(nodeLayout);
166
- const isLeft = isLeftLayout(nodeLayout);
167
- const { x, y, width, height } = getRectangleByNode(this.node);
168
- let drawX = x;
169
- let drawY = y;
170
- let drawWidth = x + width;
171
- let drawHeight = y + height;
172
- switch (true) {
173
- case isTop:
174
- drawX = x - lineWidthOffset;
175
- drawY = y - extendOffset;
176
- drawWidth = x + width + lineWidthOffset;
177
- drawHeight = y + height + lineWidthOffset;
178
- break;
179
- case isBottom:
180
- drawX = x - lineWidthOffset;
181
- drawY = y - lineWidthOffset;
182
- drawWidth = x + width + lineWidthOffset;
183
- drawHeight = y + height + extendOffset;
184
- break;
185
- case isLeft:
186
- drawX = x - extendOffset;
187
- drawY = y - lineWidthOffset;
188
- drawWidth = x + width + lineWidthOffset;
189
- drawHeight = y + height + lineWidthOffset;
190
- break;
191
- case isRight:
192
- drawX = x - lineWidthOffset;
193
- drawY = y - lineWidthOffset;
194
- drawWidth = x + width + extendOffset;
195
- drawHeight = y + height + lineWidthOffset;
196
- break;
197
- }
198
- this.maskG = drawRoundRectangle(this.roughSVG, drawX, drawY, drawWidth, drawHeight, { stroke: 'none', fill: 'rgba(255,255,255,0)', fillStyle: 'solid' }, true);
199
- this.maskG.classList.add('mask');
200
- this.maskG.setAttribute('visibility', 'visible');
201
- this.g.append(this.maskG);
202
- if (this.isEditable) {
203
- this.disabledMaskG();
204
- }
205
- fromEvent(this.maskG, 'mouseenter')
206
- .pipe(takeUntil(this.destroy$), filter(() => {
207
- return PlaitBoard.isFocus(this.board) && !this.element.isCollapsed && !this.handActive;
208
- }))
209
- .subscribe(() => {
210
- this.g.classList.add('hovered');
211
- });
212
- fromEvent(this.maskG, 'mouseleave')
213
- .pipe(takeUntil(this.destroy$), filter(() => {
214
- return PlaitBoard.isFocus(this.board) && !this.element.isCollapsed;
215
- }))
216
- .subscribe(() => {
217
- this.g.classList.remove('hovered');
218
- });
219
- }
220
- destroyMaskG() {
221
- if (this.maskG) {
222
- this.maskG.remove();
223
- this.g.classList.remove('hovered');
224
- }
225
- }
226
- enableMaskG() {
227
- if (this.maskG) {
228
- this.maskG.setAttribute('visibility', 'visible');
229
- }
230
- }
231
- disabledMaskG() {
232
- if (this.maskG) {
233
- this.maskG.setAttribute('visibility', 'hidden');
234
- }
235
- }
236
- drawActiveG() {
237
- this.destroyActiveG();
238
- this.abstractIncludedOutlineG?.remove();
239
- if (this.selected) {
240
- if (AbstractNode.isAbstract(this.element)) {
241
- this.updateAbstractIncludedOutline();
242
- }
243
- let { x, y, width, height } = getRectangleByNode(this.node);
244
- const selectedStrokeG = drawRoundRectangle(this.roughSVG, x - 2, y - 2, x + width + 2, y + height + 2, { stroke: PRIMARY_COLOR, strokeWidth: 2, fill: '' }, true);
245
- // 影响 mask 移入移出事件
246
- selectedStrokeG.style.pointerEvents = 'none';
247
- this.g.appendChild(selectedStrokeG);
248
- this.activeG.push(selectedStrokeG);
249
- if (this.richtextComponentRef?.instance.plaitReadonly === true) {
250
- const selectedBackgroundG = drawRoundRectangle(this.roughSVG, x - 2, y - 2, x + width + 2, y + height + 2, { stroke: PRIMARY_COLOR, fill: PRIMARY_COLOR, fillStyle: 'solid' }, true);
251
- selectedBackgroundG.style.opacity = '0.15';
252
- // 影响双击事件
253
- selectedBackgroundG.style.pointerEvents = 'none';
254
- this.g.appendChild(selectedBackgroundG);
255
- this.activeG.push(selectedBackgroundG, selectedStrokeG);
256
- }
257
- }
258
- }
259
- destroyActiveG() {
260
- this.activeG.forEach(g => g.remove());
261
- this.activeG = [];
262
- }
263
- updateActiveClass() {
264
- if (!this.g) {
265
- return;
266
- }
267
- if (this.selected) {
268
- this.render2.addClass(this.g, 'active');
269
- }
270
- else {
271
- this.render2.removeClass(this.g, 'active');
272
- }
273
- }
274
- drawRichtext() {
275
- const { richtextG, richtextComponentRef, foreignObject } = drawTopicByNode(this.board, this.node, this.viewContainerRef);
276
- this.richtextComponentRef = richtextComponentRef;
277
- this.richtextG = richtextG;
278
- this.foreignObject = foreignObject;
279
- this.render2.addClass(richtextG, 'richtext');
280
- this.g.append(richtextG);
281
- }
282
147
  drawExtend() {
283
148
  this.destroyExtend();
284
- // create extend
285
149
  this.extendG = createG();
286
- const collapseG = createG();
287
150
  this.extendG.classList.add('extend');
288
- collapseG.classList.add('collapse-container');
289
151
  this.g.append(this.extendG);
290
- this.extendG.append(collapseG);
291
- if (this.node.origin.isRoot) {
292
- return;
293
- }
294
- // interactive
295
- fromEvent(collapseG, 'mouseup')
296
- .pipe(filter(() => !this.handActive || !!PlaitBoard.isReadonly(this.board)), take(1))
297
- .subscribe(() => {
298
- const isCollapsed = !this.node.origin.isCollapsed;
299
- const newElement = { isCollapsed };
300
- const path = PlaitBoard.findPath(this.board, this.element);
301
- Transforms.setNode(this.board, newElement, path);
302
- });
303
- const { x, y, width, height } = getRectangleByNode(this.node);
304
- const stroke = getBranchColorByMindElement(this.board, this.element);
305
- const branchWidth = getBranchWidthByMindElement(this.board, this.element);
306
- const extendY = y + height / 2;
307
- const nodeLayout = MindQueries.getCorrectLayoutByElement(this.board, this.element);
308
- let extendLineXY = [
309
- [x + width, extendY],
310
- [x + width + EXTEND_OFFSET, extendY]
311
- ];
312
- let arrowYOffset = [-4, 1, -0.6, 4];
313
- let arrowXOffset = [10, 5.5, 5.5, 10];
314
- let extendLineXOffset = [0, 0];
315
- let extendLineYOffset = [0, 0];
316
- let circleOffset = [EXTEND_RADIUS / 2, 0];
317
- if (isHorizontalLayout(nodeLayout) && !isIndentedLayout(nodeLayout)) {
318
- extendLineYOffset =
319
- getShapeByElement(this.board, this.node.origin) === MindElementShape.roundRectangle
320
- ? [0, 0]
321
- : [height / 2, height / 2];
322
- if (isLeftLayout(nodeLayout)) {
323
- //左
324
- extendLineXOffset = [-width, -width - EXTEND_OFFSET * 2];
325
- circleOffset = [-EXTEND_RADIUS / 2, 0];
326
- arrowXOffset = [-10, -5.5, -5.5, -10];
327
- }
328
- }
329
- else {
330
- arrowXOffset = [-4, 0.6, -1, 4];
331
- if (isTopLayout(nodeLayout)) {
332
- //上
333
- extendLineXOffset = [-width / 2, -width / 2 - EXTEND_OFFSET];
334
- extendLineYOffset = [-height / 2, -height / 2 - EXTEND_OFFSET];
335
- arrowYOffset = [-10, -5.5, -5.5, -10];
336
- circleOffset = [0, -EXTEND_RADIUS / 2];
337
- }
338
- else {
339
- //下
340
- extendLineXOffset = [-width / 2, -width / 2 - EXTEND_OFFSET];
341
- extendLineYOffset = [height / 2, height / 2 + EXTEND_OFFSET];
342
- arrowYOffset = [10, 5.5, 5.5, 10];
343
- circleOffset = [0, EXTEND_RADIUS / 2];
344
- }
345
- }
346
- extendLineXY = [
347
- [extendLineXY[0][0] + extendLineXOffset[0], extendLineXY[0][1] + extendLineYOffset[0]],
348
- [extendLineXY[1][0] + extendLineXOffset[1], extendLineXY[1][1] + extendLineYOffset[1]]
349
- ];
350
- const extendLine = this.roughSVG.line(extendLineXY[0][0], extendLineXY[0][1], extendLineXY[1][0], extendLineXY[1][1], {
351
- strokeWidth: branchWidth,
352
- stroke
353
- });
354
- //绘制箭头
355
- const hideArrowTopLine = this.roughSVG.line(extendLineXY[1][0] + arrowXOffset[0], extendLineXY[1][1] + arrowYOffset[0], extendLineXY[1][0] + arrowXOffset[1], extendLineXY[1][1] + arrowYOffset[1], {
356
- stroke,
357
- strokeWidth: 2
358
- });
359
- const hideArrowBottomLine = this.roughSVG.line(extendLineXY[1][0] + arrowXOffset[2], extendLineXY[1][1] + arrowYOffset[2], extendLineXY[1][0] + arrowXOffset[3], extendLineXY[1][1] + arrowYOffset[3], {
360
- stroke,
361
- strokeWidth: 2
362
- });
363
- if (this.node.origin.isCollapsed) {
364
- const badge = this.roughSVG.circle(extendLineXY[1][0] + circleOffset[0], extendLineXY[1][1] + circleOffset[1], EXTEND_RADIUS, {
365
- fill: stroke,
366
- stroke,
367
- fillStyle: 'solid'
368
- });
369
- let numberOffset = 0;
370
- if (getChildrenCount(this.node.origin) >= 10)
371
- numberOffset = -2;
372
- if (getChildrenCount(this.node.origin) === 1)
373
- numberOffset = 1;
374
- const badgeText = createText(extendLineXY[1][0] + circleOffset[0] - 4 + numberOffset, extendLineXY[1][1] + circleOffset[1] + 4, stroke, `${getChildrenCount(this.node.origin)}`);
152
+ if (this.element.isCollapsed) {
375
153
  this.g.classList.add('collapsed');
376
- badge.setAttribute('style', 'opacity: 0.15');
377
- badgeText.setAttribute('style', 'font-size: 12px');
378
- collapseG.appendChild(badge);
379
- collapseG.appendChild(badgeText);
380
- collapseG.appendChild(extendLine);
381
154
  }
382
155
  else {
383
156
  this.g.classList.remove('collapsed');
384
- if (this.node.origin.children.length > 0) {
385
- const hideCircleG = this.roughSVG.circle(extendLineXY[1][0] + circleOffset[0], extendLineXY[1][1] + circleOffset[1], EXTEND_RADIUS - 1, {
386
- fill: '#fff',
387
- stroke,
388
- strokeWidth: branchWidth > 3 ? 3 : branchWidth,
389
- fillStyle: 'solid'
390
- });
391
- collapseG.appendChild(hideCircleG);
392
- collapseG.appendChild(hideArrowTopLine);
393
- collapseG.appendChild(hideArrowBottomLine);
394
- }
395
157
  }
158
+ this.nodeInsertDrawer.draw(this.element, this.extendG);
159
+ this.collapseDrawer.draw(this.element, this.extendG);
396
160
  }
397
161
  destroyExtend() {
398
162
  if (this.extendG) {
399
163
  this.extendG.remove();
400
164
  }
401
165
  }
402
- destroyRichtext() {
403
- if (this.richtextG) {
404
- this.richtextG.remove();
405
- }
406
- if (this.richtextComponentRef) {
407
- this.richtextComponentRef.destroy();
408
- }
409
- }
410
- updateAbstractIncludedOutline(activeHandlePosition, resizingLocation) {
411
- this.abstractIncludedOutlineG?.remove();
412
- this.abstractIncludedOutlineG = drawAbstractIncludedOutline(this.board, this.roughSVG, this.element, activeHandlePosition, resizingLocation);
413
- PlaitBoard.getHost(this.board).append(this.abstractIncludedOutlineG);
414
- }
415
- updateRichtext() {
416
- updateRichText(this.node.origin.data.topic, this.richtextComponentRef);
417
- updateMindNodeTopicSize(this.board, this.node, this.richtextG, this.isEditable);
418
- }
419
- startEditText(isEnd, isClear) {
420
- if (!this.richtextComponentRef) {
421
- throw new Error('undefined richtextComponentRef');
422
- }
423
- const richtextInstance = this.richtextComponentRef.instance;
424
- this.isEditable = true;
425
- IS_TEXT_EDITABLE.set(this.board, true);
426
- this.disabledMaskG();
427
- updateMindNodeTopicSize(this.board, this.node, this.richtextG, this.isEditable);
428
- if (richtextInstance.plaitReadonly) {
429
- richtextInstance.plaitReadonly = false;
430
- this.richtextComponentRef.changeDetectorRef.detectChanges();
431
- this.drawActiveG();
432
- const location = isEnd ? Editor.end(richtextInstance.editor, [0]) : [0];
433
- setFullSelectionAndFocus(richtextInstance.editor, location);
434
- if (isClear) {
435
- Editor.deleteBackward(richtextInstance.editor);
436
- }
437
- // handle invalid width and height (old data)
438
- let { width, height } = getRichtextContentSize(richtextInstance.editable);
439
- if (width !== this.element.width || height !== this.element.height) {
440
- MindTransforms.setTopicSize(this.board, this.element, width, height);
441
- }
442
- }
443
- let richtext = richtextInstance.plaitValue;
444
- // use debounceTime to wait DOM render complete
445
- const valueChange$ = richtextInstance.plaitChange
446
- .pipe(debounceTime(0), filter(event => {
447
- // 过滤掉 operations 中全是 set_selection 的操作
448
- return !event.operations.every(op => Operation.isSelectionOperation(op));
449
- }))
450
- .subscribe(event => {
451
- if (richtext === event.value) {
452
- return;
453
- }
454
- this.updateRichtext();
455
- // 更新富文本、更新宽高
456
- let { width, height } = getRichtextContentSize(richtextInstance.editable);
457
- MindTransforms.setTopic(this.board, this.element, event.value, width, height);
458
- MERGING.set(this.board, true);
459
- });
460
- const composition$ = richtextInstance.plaitComposition.pipe(debounceTime(0)).subscribe(event => {
461
- let { width, height } = getRichtextContentSize(richtextInstance.editable);
462
- if (width < NODE_MIN_WIDTH) {
463
- width = NODE_MIN_WIDTH;
464
- }
465
- if (event.isComposing && (width !== this.node.origin.width || height !== this.node.origin.height)) {
466
- const newElement = {
467
- width: width / this.board.viewport.zoom,
468
- height: height / this.board.viewport.zoom
469
- };
470
- const path = PlaitBoard.findPath(this.board, this.element);
471
- Transforms.setNode(this.board, newElement, path);
472
- MERGING.set(this.board, true);
473
- }
474
- });
475
- const mousedown$ = fromEvent(document, 'mousedown').subscribe((event) => {
476
- const point = transformPoint(this.board, toPoint(event.x, event.y, PlaitBoard.getHost(this.board)));
477
- const clickInNode = isHitMindElement(this.board, point, this.element);
478
- if (clickInNode && !hasEditableTarget(richtextInstance.editor, event.target)) {
479
- event.preventDefault();
480
- }
481
- else if (!clickInNode) {
482
- // handle composition input state, like: Chinese IME Composition Input
483
- timer(0).subscribe(() => {
484
- exitHandle();
485
- this.enableMaskG();
486
- });
487
- }
488
- });
489
- const editor = richtextInstance.editor;
490
- const { keydown } = editor;
491
- editor.keydown = (event) => {
492
- if (event.isComposing) {
493
- return;
494
- }
495
- if (event.key === 'Escape') {
496
- event.preventDefault();
497
- event.stopPropagation();
498
- exitHandle();
499
- this.drawActiveG();
500
- this.enableMaskG();
501
- return;
502
- }
503
- if (event.key === 'Enter' && !event.shiftKey) {
504
- event.preventDefault();
505
- event.stopPropagation();
506
- exitHandle();
507
- this.drawActiveG();
508
- this.enableMaskG();
509
- return;
510
- }
511
- if (event.key === 'Tab') {
512
- event.preventDefault();
513
- event.stopPropagation();
514
- exitHandle();
515
- this.drawActiveG();
516
- this.drawMaskG();
517
- }
518
- };
519
- const exitHandle = () => {
520
- this.ngZone.run(() => {
521
- // unsubscribe
522
- valueChange$.unsubscribe();
523
- composition$.unsubscribe();
524
- mousedown$.unsubscribe();
525
- editor.keydown = keydown; // reset keydown
526
- // editable status
527
- MERGING.set(this.board, false);
528
- richtextInstance.plaitReadonly = true;
529
- this.richtextComponentRef?.changeDetectorRef.markForCheck();
530
- this.isEditable = false;
531
- updateMindNodeTopicSize(this.board, this.node, this.richtextG, this.isEditable);
532
- IS_TEXT_EDITABLE.set(this.board, false);
533
- });
534
- };
166
+ drawText() {
167
+ this.textManage.draw(this.element.data.topic);
168
+ this.g.append(this.textManage.g);
535
169
  }
536
170
  ngOnDestroy() {
537
171
  super.ngOnDestroy();
538
- this.abstractIncludedOutlineG?.remove();
539
- this.destroyRichtext();
540
- this.emojisDrawer.destroy();
172
+ this.textManage.destroy();
173
+ this.nodeEmojisDrawer.destroy();
541
174
  this.destroy$.next();
542
175
  this.destroy$.complete();
543
176
  if (ELEMENT_TO_NODE.get(this.element) === this.node) {
@@ -545,7 +178,7 @@ export class MindNodeComponent extends PlaitPluginElementComponent {
545
178
  }
546
179
  }
547
180
  }
548
- MindNodeComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.5", ngImport: i0, type: MindNodeComponent, deps: [{ token: i0.ViewContainerRef }, { token: i0.ChangeDetectorRef }, { token: i0.Renderer2 }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component });
181
+ MindNodeComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.5", ngImport: i0, type: MindNodeComponent, deps: [{ token: i0.ViewContainerRef }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
549
182
  MindNodeComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.5", type: MindNodeComponent, selector: "plait-mind-node", usesInheritance: true, ngImport: i0, template: `
550
183
  <plait-children
551
184
  *ngIf="!element.isCollapsed"
@@ -570,5 +203,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.5", ngImpor
570
203
  `,
571
204
  changeDetection: ChangeDetectionStrategy.OnPush
572
205
  }]
573
- }], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i0.ChangeDetectorRef }, { type: i0.Renderer2 }, { type: i0.NgZone }]; } });
574
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node.component.js","sourceRoot":"","sources":["../../../packages/mind/src/node.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ComponentRef,\n    NgZone,\n    OnDestroy,\n    OnInit,\n    Renderer2,\n    ViewContainerRef\n} from '@angular/core';\nimport {\n    PlaitPointerType,\n    createG,\n    createText,\n    IS_TEXT_EDITABLE,\n    MERGING,\n    PlaitBoard,\n    toPoint,\n    transformPoint,\n    Transforms,\n    drawRoundRectangle,\n    PlaitPluginElementComponent,\n    PlaitElement,\n    NODE_TO_INDEX,\n    PlaitPluginElementContext,\n    OnContextChanged,\n    RectangleClient\n} from '@plait/core';\nimport {\n    isBottomLayout,\n    isHorizontalLayout,\n    isIndentedLayout,\n    isLeftLayout,\n    AbstractNode,\n    isRightLayout,\n    isTopLayout,\n    MindLayoutType\n} from '@plait/layouts';\nimport { hasEditableTarget, PlaitRichtextComponent, setFullSelectionAndFocus, updateRichText } from '@plait/richtext';\nimport { RoughSVG } from 'roughjs/bin/svg';\nimport { fromEvent, Subject, timer } from 'rxjs';\nimport { debounceTime, filter, take, takeUntil } from 'rxjs/operators';\nimport { Editor, Operation } from 'slate';\nimport { EXTEND_OFFSET, EXTEND_RADIUS, PRIMARY_COLOR } from './constants';\nimport { NODE_MIN_WIDTH } from './constants/node-rule';\nimport { drawTopicByNode, updateMindNodeTopicSize } from './utils/draw/node-topic';\nimport { drawRoundRectangleByNode } from './utils/draw/node-shape';\nimport { MindElement, PlaitMind } from './interfaces/element';\nimport { MindNode } from './interfaces/node';\nimport { MindQueries } from './queries';\nimport { getRectangleByNode, isHitMindElement } from './utils/position/node';\nimport { getChildrenCount } from './utils/mind';\nimport { getShapeByElement } from './utils/node-style/shape';\nimport { ELEMENT_TO_NODE } from './utils/weak-maps';\nimport { getRichtextContentSize } from '@plait/richtext';\nimport { drawAbstractLink } from './utils/draw/node-link/abstract-link';\nimport { EmojisDrawer } from './drawer/emojis.drawer';\nimport { MindTransforms } from './transforms';\nimport { drawAbstractIncludedOutline } from './utils/draw/abstract-outline';\nimport { MindElementShape } from './interfaces';\nimport { QuickInsertDrawer } from './drawer/quick-insert.drawer';\nimport { hasAfterDraw } from './base/base.drawer';\nimport { getBranchColorByMindElement, getBranchWidthByMindElement } from './utils/node-style/branch';\nimport { PlaitMindBoard } from './plugins/with-mind.board';\nimport { AbstractHandlePosition } from './plugins/with-abstract-resize.board';\nimport { drawLink } from './utils/draw/node-link/draw-link';\n\n@Component({\n    selector: 'plait-mind-node',\n    template: `\n        <plait-children\n            *ngIf=\"!element.isCollapsed\"\n            [board]=\"board\"\n            [parent]=\"element\"\n            [effect]=\"effect\"\n            [parentG]=\"parentG\"\n        ></plait-children>\n    `,\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class MindNodeComponent extends PlaitPluginElementComponent<MindElement, PlaitMindBoard>\n    implements OnInit, OnDestroy, OnContextChanged<MindElement, PlaitMindBoard> {\n    isEditable = false;\n\n    roughSVG!: RoughSVG;\n\n    node!: MindNode;\n\n    index!: number;\n\n    abstractIncludedOutlineG?: SVGGElement;\n\n    parentG!: SVGGElement;\n\n    activeG: SVGGElement[] = [];\n\n    shapeG: SVGGElement | null = null;\n\n    linkG?: SVGGElement;\n\n    richtextG?: SVGGElement;\n\n    foreignObject?: SVGForeignObjectElement;\n\n    extendG?: SVGGElement;\n\n    maskG!: SVGGElement;\n\n    richtextComponentRef?: ComponentRef<PlaitRichtextComponent>;\n\n    destroy$ = new Subject<void>();\n\n    emojisDrawer!: EmojisDrawer;\n\n    quickInsertDrawer!: QuickInsertDrawer;\n\n    public get handActive(): boolean {\n        return this.board.pointer === PlaitPointerType.hand;\n    }\n\n    constructor(\n        private viewContainerRef: ViewContainerRef,\n        protected cdr: ChangeDetectorRef,\n        private render2: Renderer2,\n        private ngZone: NgZone\n    ) {\n        super(cdr);\n    }\n\n    ngOnInit(): void {\n        this.emojisDrawer = new EmojisDrawer(this.board, this.viewContainerRef);\n        this.quickInsertDrawer = new QuickInsertDrawer(this.board);\n        super.ngOnInit();\n        this.node = MindElement.getNode(this.element);\n        this.index = NODE_TO_INDEX.get(this.element) || 0;\n        this.roughSVG = PlaitBoard.getRoughSVG(this.board);\n        this.parentG = PlaitElement.getComponent(MindElement.getRoot(this.board, this.element)).rootG as SVGGElement;\n        this.drawShape();\n        this.drawLink();\n        this.drawRichtext();\n        this.drawActiveG();\n        this.updateActiveClass();\n        this.drawMaskG();\n        this.drawEmojis();\n        this.drawExtend();\n        this.drawQuickInsert();\n    }\n\n    onContextChanged(\n        value: PlaitPluginElementContext<MindElement, PlaitMindBoard>,\n        previous: PlaitPluginElementContext<MindElement, PlaitMindBoard>\n    ) {\n        const newNode = MindElement.getNode(value.element);\n\n        // resolve move node richtext lose issue\n        if (this.node !== newNode) {\n            if (this.foreignObject && this.foreignObject.children.length <= 0) {\n                this.foreignObject?.appendChild(this.richtextComponentRef?.instance.editable as HTMLElement);\n            }\n        }\n\n        const isEqualNode = RectangleClient.isEqual(this.node, newNode);\n        this.node = newNode;\n\n        const isChangeTheme = this.board.operations.find(op => op.type === 'set_theme');\n\n        if (!isEqualNode || value.element !== previous.element || isChangeTheme) {\n            this.drawActiveG();\n            this.updateActiveClass();\n            this.drawShape();\n            this.drawLink();\n            this.updateRichtext();\n            this.drawMaskG();\n            this.drawExtend();\n            this.drawQuickInsert();\n            this.drawEmojis();\n        } else {\n            const hasSameSelected = value.selected === previous.selected;\n            const hasSameParent = value.parent === previous.parent;\n            if (!hasSameSelected) {\n                this.drawActiveG();\n                this.updateActiveClass();\n            }\n            if (!hasSameParent) {\n                this.drawLink();\n            }\n        }\n    }\n\n    drawEmojis() {\n        const g = this.emojisDrawer.drawEmojis(this.element);\n        if (g) {\n            this.g.append(g);\n        }\n    }\n\n    drawQuickInsert() {\n        this.quickInsertDrawer.destroy();\n        if (this.quickInsertDrawer.canDraw(this.element)) {\n            const g = this.quickInsertDrawer.draw(this.element);\n            if (hasAfterDraw(this.quickInsertDrawer)) {\n                this.quickInsertDrawer.afterDraw(this.element);\n            }\n            this.extendG?.appendChild(g);\n        }\n    }\n\n    drawShape() {\n        this.destroyShape();\n        const shape = getShapeByElement(this.board, this.node.origin) as MindElementShape;\n        switch (shape) {\n            case MindElementShape.roundRectangle:\n                this.shapeG = drawRoundRectangleByNode(this.board, this.node as MindNode);\n                this.g.prepend(this.shapeG);\n                break;\n            default:\n                break;\n        }\n    }\n\n    destroyShape() {\n        if (this.shapeG) {\n            this.shapeG.remove();\n            this.shapeG = null;\n        }\n    }\n\n    drawLink() {\n        if (PlaitMind.isMind(this.element)) {\n            return;\n        }\n\n        const parent = MindElement.getParent(this.element);\n        const parentNode = MindElement.getNode(parent);\n\n        if (this.linkG) {\n            this.linkG.remove();\n        }\n\n        const layout = MindQueries.getLayoutByElement(parent) as MindLayoutType;\n        if (AbstractNode.isAbstract(this.node.origin)) {\n            this.linkG = drawAbstractLink(this.board, this.node, isHorizontalLayout(layout));\n        } else {\n            this.linkG = drawLink(this.board, parentNode, this.node, isHorizontalLayout(layout));\n        }\n        this.g.append(this.linkG);\n    }\n\n    destroyLine() {\n        if (this.linkG) {\n            this.linkG.remove();\n        }\n    }\n\n    drawMaskG() {\n        this.destroyMaskG();\n        const lineWidthOffset = 2;\n        const extendOffset = 15;\n        const nodeLayout = MindQueries.getLayoutByElement(this.node.origin) as MindLayoutType;\n        const isTop = isTopLayout(nodeLayout);\n        const isRight = isRightLayout(nodeLayout);\n        const isBottom = isBottomLayout(nodeLayout);\n        const isLeft = isLeftLayout(nodeLayout);\n        const { x, y, width, height } = getRectangleByNode(this.node as MindNode);\n\n        let drawX = x;\n        let drawY = y;\n        let drawWidth = x + width;\n        let drawHeight = y + height;\n\n        switch (true) {\n            case isTop:\n                drawX = x - lineWidthOffset;\n                drawY = y - extendOffset;\n                drawWidth = x + width + lineWidthOffset;\n                drawHeight = y + height + lineWidthOffset;\n                break;\n            case isBottom:\n                drawX = x - lineWidthOffset;\n                drawY = y - lineWidthOffset;\n                drawWidth = x + width + lineWidthOffset;\n                drawHeight = y + height + extendOffset;\n                break;\n            case isLeft:\n                drawX = x - extendOffset;\n                drawY = y - lineWidthOffset;\n                drawWidth = x + width + lineWidthOffset;\n                drawHeight = y + height + lineWidthOffset;\n                break;\n            case isRight:\n                drawX = x - lineWidthOffset;\n                drawY = y - lineWidthOffset;\n                drawWidth = x + width + extendOffset;\n                drawHeight = y + height + lineWidthOffset;\n                break;\n        }\n        this.maskG = drawRoundRectangle(\n            this.roughSVG as RoughSVG,\n            drawX,\n            drawY,\n            drawWidth,\n            drawHeight,\n            { stroke: 'none', fill: 'rgba(255,255,255,0)', fillStyle: 'solid' },\n            true\n        );\n        this.maskG.classList.add('mask');\n        this.maskG.setAttribute('visibility', 'visible');\n        this.g.append(this.maskG);\n\n        if (this.isEditable) {\n            this.disabledMaskG();\n        }\n\n        fromEvent<MouseEvent>(this.maskG, 'mouseenter')\n            .pipe(\n                takeUntil(this.destroy$),\n                filter(() => {\n                    return PlaitBoard.isFocus(this.board) && !this.element.isCollapsed && !this.handActive;\n                })\n            )\n            .subscribe(() => {\n                this.g.classList.add('hovered');\n            });\n        fromEvent<MouseEvent>(this.maskG, 'mouseleave')\n            .pipe(\n                takeUntil(this.destroy$),\n                filter(() => {\n                    return PlaitBoard.isFocus(this.board) && !this.element.isCollapsed;\n                })\n            )\n            .subscribe(() => {\n                this.g.classList.remove('hovered');\n            });\n    }\n\n    destroyMaskG() {\n        if (this.maskG) {\n            this.maskG.remove();\n            this.g.classList.remove('hovered');\n        }\n    }\n\n    enableMaskG() {\n        if (this.maskG) {\n            this.maskG.setAttribute('visibility', 'visible');\n        }\n    }\n\n    disabledMaskG() {\n        if (this.maskG) {\n            this.maskG.setAttribute('visibility', 'hidden');\n        }\n    }\n\n    drawActiveG() {\n        this.destroyActiveG();\n        this.abstractIncludedOutlineG?.remove();\n        if (this.selected) {\n            if (AbstractNode.isAbstract(this.element)) {\n                this.updateAbstractIncludedOutline();\n            }\n            let { x, y, width, height } = getRectangleByNode(this.node as MindNode);\n            const selectedStrokeG = drawRoundRectangle(\n                this.roughSVG as RoughSVG,\n                x - 2,\n                y - 2,\n                x + width + 2,\n                y + height + 2,\n                { stroke: PRIMARY_COLOR, strokeWidth: 2, fill: '' },\n                true\n            );\n            // 影响 mask 移入移出事件\n            selectedStrokeG.style.pointerEvents = 'none';\n            this.g.appendChild(selectedStrokeG);\n            this.activeG.push(selectedStrokeG);\n            if (this.richtextComponentRef?.instance.plaitReadonly === true) {\n                const selectedBackgroundG = drawRoundRectangle(\n                    this.roughSVG as RoughSVG,\n                    x - 2,\n                    y - 2,\n                    x + width + 2,\n                    y + height + 2,\n                    { stroke: PRIMARY_COLOR, fill: PRIMARY_COLOR, fillStyle: 'solid' },\n                    true\n                );\n                selectedBackgroundG.style.opacity = '0.15';\n                // 影响双击事件\n                selectedBackgroundG.style.pointerEvents = 'none';\n                this.g.appendChild(selectedBackgroundG);\n                this.activeG.push(selectedBackgroundG, selectedStrokeG);\n            }\n        }\n    }\n\n    destroyActiveG() {\n        this.activeG.forEach(g => g.remove());\n        this.activeG = [];\n    }\n\n    updateActiveClass() {\n        if (!this.g) {\n            return;\n        }\n        if (this.selected) {\n            this.render2.addClass(this.g, 'active');\n        } else {\n            this.render2.removeClass(this.g, 'active');\n        }\n    }\n\n    drawRichtext() {\n        const { richtextG, richtextComponentRef, foreignObject } = drawTopicByNode(this.board, this.node, this.viewContainerRef);\n        this.richtextComponentRef = richtextComponentRef;\n        this.richtextG = richtextG;\n        this.foreignObject = foreignObject;\n        this.render2.addClass(richtextG, 'richtext');\n        this.g.append(richtextG);\n    }\n\n    drawExtend() {\n        this.destroyExtend();\n        // create extend\n        this.extendG = createG();\n        const collapseG = createG();\n        this.extendG.classList.add('extend');\n        collapseG.classList.add('collapse-container');\n        this.g.append(this.extendG);\n        this.extendG.append(collapseG);\n        if (this.node.origin.isRoot) {\n            return;\n        }\n        // interactive\n        fromEvent(collapseG, 'mouseup')\n            .pipe(\n                filter(() => !this.handActive || !!PlaitBoard.isReadonly(this.board)),\n                take(1)\n            )\n            .subscribe(() => {\n                const isCollapsed = !this.node.origin.isCollapsed;\n                const newElement: Partial<MindElement> = { isCollapsed };\n                const path = PlaitBoard.findPath(this.board, this.element);\n                Transforms.setNode(this.board, newElement, path);\n            });\n\n        const { x, y, width, height } = getRectangleByNode(this.node);\n        const stroke = getBranchColorByMindElement(this.board, this.element);\n        const branchWidth = getBranchWidthByMindElement(this.board, this.element);\n        const extendY = y + height / 2;\n        const nodeLayout = MindQueries.getCorrectLayoutByElement(this.board, this.element) as MindLayoutType;\n\n        let extendLineXY = [\n            [x + width, extendY],\n            [x + width + EXTEND_OFFSET, extendY]\n        ];\n\n        let arrowYOffset = [-4, 1, -0.6, 4];\n        let arrowXOffset = [10, 5.5, 5.5, 10];\n\n        let extendLineXOffset = [0, 0];\n        let extendLineYOffset = [0, 0];\n\n        let circleOffset = [EXTEND_RADIUS / 2, 0];\n\n        if (isHorizontalLayout(nodeLayout) && !isIndentedLayout(nodeLayout)) {\n            extendLineYOffset =\n                (getShapeByElement(this.board, this.node.origin) as MindElementShape) === MindElementShape.roundRectangle\n                    ? [0, 0]\n                    : [height / 2, height / 2];\n            if (isLeftLayout(nodeLayout)) {\n                //左\n                extendLineXOffset = [-width, -width - EXTEND_OFFSET * 2];\n                circleOffset = [-EXTEND_RADIUS / 2, 0];\n                arrowXOffset = [-10, -5.5, -5.5, -10];\n            }\n        } else {\n            arrowXOffset = [-4, 0.6, -1, 4];\n            if (isTopLayout(nodeLayout)) {\n                //上\n                extendLineXOffset = [-width / 2, -width / 2 - EXTEND_OFFSET];\n                extendLineYOffset = [-height / 2, -height / 2 - EXTEND_OFFSET];\n                arrowYOffset = [-10, -5.5, -5.5, -10];\n                circleOffset = [0, -EXTEND_RADIUS / 2];\n            } else {\n                //下\n                extendLineXOffset = [-width / 2, -width / 2 - EXTEND_OFFSET];\n                extendLineYOffset = [height / 2, height / 2 + EXTEND_OFFSET];\n                arrowYOffset = [10, 5.5, 5.5, 10];\n                circleOffset = [0, EXTEND_RADIUS / 2];\n            }\n        }\n\n        extendLineXY = [\n            [extendLineXY[0][0] + extendLineXOffset[0], extendLineXY[0][1] + extendLineYOffset[0]],\n            [extendLineXY[1][0] + extendLineXOffset[1], extendLineXY[1][1] + extendLineYOffset[1]]\n        ];\n\n        const extendLine = this.roughSVG.line(extendLineXY[0][0], extendLineXY[0][1], extendLineXY[1][0], extendLineXY[1][1], {\n            strokeWidth: branchWidth,\n            stroke\n        });\n\n        //绘制箭头\n        const hideArrowTopLine = this.roughSVG.line(\n            extendLineXY[1][0] + arrowXOffset[0],\n            extendLineXY[1][1] + arrowYOffset[0],\n            extendLineXY[1][0] + arrowXOffset[1],\n            extendLineXY[1][1] + arrowYOffset[1],\n            {\n                stroke,\n                strokeWidth: 2\n            }\n        );\n        const hideArrowBottomLine = this.roughSVG.line(\n            extendLineXY[1][0] + arrowXOffset[2],\n            extendLineXY[1][1] + arrowYOffset[2],\n            extendLineXY[1][0] + arrowXOffset[3],\n            extendLineXY[1][1] + arrowYOffset[3],\n            {\n                stroke,\n                strokeWidth: 2\n            }\n        );\n\n        if (this.node.origin.isCollapsed) {\n            const badge = this.roughSVG.circle(extendLineXY[1][0] + circleOffset[0], extendLineXY[1][1] + circleOffset[1], EXTEND_RADIUS, {\n                fill: stroke,\n                stroke,\n                fillStyle: 'solid'\n            });\n\n            let numberOffset = 0;\n            if (getChildrenCount(this.node.origin) >= 10) numberOffset = -2;\n            if (getChildrenCount(this.node.origin) === 1) numberOffset = 1;\n\n            const badgeText = createText(\n                extendLineXY[1][0] + circleOffset[0] - 4 + numberOffset,\n                extendLineXY[1][1] + circleOffset[1] + 4,\n                stroke,\n                `${getChildrenCount(this.node.origin)}`\n            );\n\n            this.g.classList.add('collapsed');\n            badge.setAttribute('style', 'opacity: 0.15');\n            badgeText.setAttribute('style', 'font-size: 12px');\n            collapseG.appendChild(badge);\n            collapseG.appendChild(badgeText);\n            collapseG.appendChild(extendLine);\n        } else {\n            this.g.classList.remove('collapsed');\n            if (this.node.origin.children.length > 0) {\n                const hideCircleG = this.roughSVG.circle(\n                    extendLineXY[1][0] + circleOffset[0],\n                    extendLineXY[1][1] + circleOffset[1],\n                    EXTEND_RADIUS - 1,\n                    {\n                        fill: '#fff',\n                        stroke,\n                        strokeWidth: branchWidth > 3 ? 3 : branchWidth,\n                        fillStyle: 'solid'\n                    }\n                );\n                collapseG.appendChild(hideCircleG);\n                collapseG.appendChild(hideArrowTopLine);\n                collapseG.appendChild(hideArrowBottomLine);\n            }\n        }\n    }\n\n    destroyExtend() {\n        if (this.extendG) {\n            this.extendG.remove();\n        }\n    }\n\n    destroyRichtext() {\n        if (this.richtextG) {\n            this.richtextG.remove();\n        }\n        if (this.richtextComponentRef) {\n            this.richtextComponentRef.destroy();\n        }\n    }\n\n    updateAbstractIncludedOutline(activeHandlePosition?: AbstractHandlePosition, resizingLocation?: number) {\n        this.abstractIncludedOutlineG?.remove();\n        this.abstractIncludedOutlineG = drawAbstractIncludedOutline(\n            this.board,\n            this.roughSVG,\n            this.element,\n            activeHandlePosition,\n            resizingLocation\n        );\n        PlaitBoard.getHost(this.board).append(this.abstractIncludedOutlineG);\n    }\n\n    updateRichtext() {\n        updateRichText(this.node.origin.data.topic, this.richtextComponentRef!);\n        updateMindNodeTopicSize(this.board, this.node, this.richtextG as SVGGElement, this.isEditable);\n    }\n\n    startEditText(isEnd: boolean, isClear: boolean) {\n        if (!this.richtextComponentRef) {\n            throw new Error('undefined richtextComponentRef');\n        }\n        const richtextInstance = this.richtextComponentRef.instance;\n\n        this.isEditable = true;\n        IS_TEXT_EDITABLE.set(this.board, true);\n        this.disabledMaskG();\n        updateMindNodeTopicSize(this.board, this.node, this.richtextG as SVGGElement, this.isEditable);\n        if (richtextInstance.plaitReadonly) {\n            richtextInstance.plaitReadonly = false;\n            this.richtextComponentRef.changeDetectorRef.detectChanges();\n            this.drawActiveG();\n            const location = isEnd ? Editor.end(richtextInstance.editor, [0]) : [0];\n            setFullSelectionAndFocus(richtextInstance.editor, location);\n            if (isClear) {\n                Editor.deleteBackward(richtextInstance.editor);\n            }\n            // handle invalid width and height (old data)\n            let { width, height } = getRichtextContentSize(richtextInstance.editable);\n            if (width !== this.element.width || height !== this.element.height) {\n                MindTransforms.setTopicSize(this.board, this.element, width, height);\n            }\n        }\n        let richtext = richtextInstance.plaitValue;\n        // use debounceTime to wait DOM render complete\n        const valueChange$ = richtextInstance.plaitChange\n            .pipe(\n                debounceTime(0),\n                filter(event => {\n                    // 过滤掉 operations 中全是 set_selection 的操作\n                    return !event.operations.every(op => Operation.isSelectionOperation(op));\n                })\n            )\n            .subscribe(event => {\n                if (richtext === event.value) {\n                    return;\n                }\n                this.updateRichtext();\n                // 更新富文本、更新宽高\n                let { width, height } = getRichtextContentSize(richtextInstance.editable);\n                MindTransforms.setTopic(this.board, this.element, event.value, width, height);\n                MERGING.set(this.board, true);\n            });\n        const composition$ = richtextInstance.plaitComposition.pipe(debounceTime(0)).subscribe(event => {\n            let { width, height } = getRichtextContentSize(richtextInstance.editable);\n            if (width < NODE_MIN_WIDTH) {\n                width = NODE_MIN_WIDTH;\n            }\n            if (event.isComposing && (width !== this.node.origin.width || height !== this.node.origin.height)) {\n                const newElement: Partial<MindElement> = {\n                    width: width / this.board.viewport.zoom,\n                    height: height / this.board.viewport.zoom\n                };\n                const path = PlaitBoard.findPath(this.board, this.element);\n                Transforms.setNode(this.board, newElement, path);\n                MERGING.set(this.board, true);\n            }\n        });\n        const mousedown$ = fromEvent<MouseEvent>(document, 'mousedown').subscribe((event: MouseEvent) => {\n            const point = transformPoint(this.board, toPoint(event.x, event.y, PlaitBoard.getHost(this.board)));\n            const clickInNode = isHitMindElement(this.board, point, this.element);\n            if (clickInNode && !hasEditableTarget(richtextInstance.editor, event.target)) {\n                event.preventDefault();\n            } else if (!clickInNode) {\n                // handle composition input state, like: Chinese IME Composition Input\n                timer(0).subscribe(() => {\n                    exitHandle();\n                    this.enableMaskG();\n                });\n            }\n        });\n        const editor = richtextInstance.editor;\n        const { keydown } = editor;\n        editor.keydown = (event: KeyboardEvent) => {\n            if (event.isComposing) {\n                return;\n            }\n            if (event.key === 'Escape') {\n                event.preventDefault();\n                event.stopPropagation();\n                exitHandle();\n                this.drawActiveG();\n                this.enableMaskG();\n                return;\n            }\n            if (event.key === 'Enter' && !event.shiftKey) {\n                event.preventDefault();\n                event.stopPropagation();\n                exitHandle();\n                this.drawActiveG();\n                this.enableMaskG();\n                return;\n            }\n            if (event.key === 'Tab') {\n                event.preventDefault();\n                event.stopPropagation();\n                exitHandle();\n                this.drawActiveG();\n                this.drawMaskG();\n            }\n        };\n        const exitHandle = () => {\n            this.ngZone.run(() => {\n                // unsubscribe\n                valueChange$.unsubscribe();\n                composition$.unsubscribe();\n                mousedown$.unsubscribe();\n                editor.keydown = keydown; // reset keydown\n                // editable status\n                MERGING.set(this.board, false);\n                richtextInstance.plaitReadonly = true;\n                this.richtextComponentRef?.changeDetectorRef.markForCheck();\n                this.isEditable = false;\n                updateMindNodeTopicSize(this.board, this.node, this.richtextG as SVGGElement, this.isEditable);\n                IS_TEXT_EDITABLE.set(this.board, false);\n            });\n        };\n    }\n\n    trackBy = (index: number, node: MindNode) => {\n        return node.origin.id;\n    };\n\n    ngOnDestroy(): void {\n        super.ngOnDestroy();\n        this.abstractIncludedOutlineG?.remove();\n        this.destroyRichtext();\n        this.emojisDrawer.destroy();\n        this.destroy$.next();\n        this.destroy$.complete();\n        if (ELEMENT_TO_NODE.get(this.element) === this.node) {\n            ELEMENT_TO_NODE.delete(this.element);\n        }\n    }\n}\n"]}
206
+ }], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i0.ChangeDetectorRef }]; } });
207
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node.component.js","sourceRoot":"","sources":["../../../packages/mind/src/node.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, OnDestroy, OnInit, ViewContainerRef } from '@angular/core';\nimport {\n    createG,\n    PlaitBoard,\n    PlaitPluginElementComponent,\n    PlaitElement,\n    NODE_TO_INDEX,\n    PlaitPluginElementContext,\n    OnContextChanged,\n    RectangleClient,\n    Point\n} from '@plait/core';\nimport { isHorizontalLayout, AbstractNode, MindLayoutType } from '@plait/layouts';\nimport { TextManageRef, TextManage } from '@plait/text';\nimport { RoughSVG } from 'roughjs/bin/svg';\nimport { Subject } from 'rxjs';\nimport { drawRoundRectangleByNode } from './utils/draw/node-shape';\nimport { MindElement, PlaitMind } from './interfaces/element';\nimport { MindNode } from './interfaces/node';\nimport { MindQueries } from './queries';\nimport { isHitMindElement } from './utils/position/node';\nimport { getShapeByElement } from './utils/node-style/shape';\nimport { ELEMENT_TO_NODE } from './utils/weak-maps';\nimport { drawAbstractLink } from './utils/draw/node-link/abstract-link';\nimport { NodeEmojisDrawer } from './drawer/node-emojis.drawer';\nimport { MindTransforms } from './transforms';\nimport { MindElementShape } from './interfaces';\nimport { NodeInsertDrawer } from './drawer/node-insert.drawer';\nimport { PlaitMindBoard } from './plugins/with-mind.board';\nimport { drawLink } from './utils/draw/node-link/draw-link';\nimport { getTopicRectangleByNode } from './utils/position/topic';\nimport { NodeActiveDrawer } from './drawer/node-active.drawer';\nimport { CollapseDrawer } from './drawer/node-collapse.drawer';\nimport { BaseElement } from 'slate';\n\n@Component({\n    selector: 'plait-mind-node',\n    template: `\n        <plait-children\n            *ngIf=\"!element.isCollapsed\"\n            [board]=\"board\"\n            [parent]=\"element\"\n            [effect]=\"effect\"\n            [parentG]=\"parentG\"\n        ></plait-children>\n    `,\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class MindNodeComponent extends PlaitPluginElementComponent<MindElement, PlaitMindBoard>\n    implements OnInit, OnDestroy, OnContextChanged<MindElement, PlaitMindBoard> {\n    roughSVG!: RoughSVG;\n\n    node!: MindNode;\n\n    index!: number;\n\n    parentG!: SVGGElement;\n\n    shapeG: SVGGElement | null = null;\n\n    linkG?: SVGGElement;\n\n    extendG?: SVGGElement;\n\n    destroy$ = new Subject<void>();\n\n    nodeEmojisDrawer!: NodeEmojisDrawer;\n\n    nodeInsertDrawer!: NodeInsertDrawer;\n\n    textManage!: TextManage;\n\n    activeDrawer!: NodeActiveDrawer;\n\n    collapseDrawer!: CollapseDrawer;\n\n    constructor(private viewContainerRef: ViewContainerRef, protected cdr: ChangeDetectorRef) {\n        super(cdr);\n    }\n\n    initializeDrawer() {\n        this.nodeEmojisDrawer = new NodeEmojisDrawer(this.board, this.viewContainerRef);\n        this.nodeInsertDrawer = new NodeInsertDrawer(this.board);\n        this.activeDrawer = new NodeActiveDrawer(this.board);\n        this.collapseDrawer = new CollapseDrawer(this.board);\n\n        this.textManage = new TextManage(\n            this.board,\n            this.viewContainerRef,\n            () => {\n                return getTopicRectangleByNode(this.board, this.node);\n            },\n            (point: Point) => {\n                return isHitMindElement(this.board, point, this.element);\n            },\n            (textManageRef: TextManageRef) => {\n                const width = textManageRef.width;\n                const height = textManageRef.height;\n                if (textManageRef.newValue) {\n                    MindTransforms.setTopic(this.board, this.element, textManageRef.newValue as MindElement, width, height);\n                } else {\n                    MindTransforms.setTopicSize(this.board, this.element, width, height);\n                }\n            }\n        );\n    }\n\n    ngOnInit(): void {\n        super.ngOnInit();\n        this.initializeDrawer();\n        this.node = MindElement.getNode(this.element);\n        this.index = NODE_TO_INDEX.get(this.element) || 0;\n        this.roughSVG = PlaitBoard.getRoughSVG(this.board);\n        this.parentG = PlaitElement.getComponent(MindElement.getRoot(this.board, this.element)).rootG as SVGGElement;\n        this.drawShape();\n        this.drawLink();\n        this.drawText();\n        this.activeDrawer.draw(this.element, this.g, { selected: this.selected, isEditing: this.textManage.isEditing });\n        this.drawEmojis();\n        this.drawExtend();\n    }\n\n    editTopic() {\n        this.activeDrawer.draw(this.element, this.g, { selected: this.selected, isEditing: true });\n        this.textManage.edit(() => {\n            this.activeDrawer.draw(this.element, this.g, { selected: this.selected, isEditing: false });\n        });\n    }\n\n    onContextChanged(\n        value: PlaitPluginElementContext<MindElement, PlaitMindBoard>,\n        previous: PlaitPluginElementContext<MindElement, PlaitMindBoard>\n    ) {\n        const newNode = MindElement.getNode(value.element);\n        const isEqualNode = RectangleClient.isEqual(this.node, newNode);\n        this.node = newNode;\n\n        const isChangeTheme = this.board.operations.find(op => op.type === 'set_theme');\n\n        if (!isEqualNode || value.element !== previous.element || isChangeTheme) {\n            this.activeDrawer.draw(this.element, this.g, { selected: this.selected, isEditing: this.textManage.isEditing });\n            this.drawShape();\n            this.drawLink();\n            this.drawEmojis();\n            this.drawExtend();\n            this.textManage.updateText(this.element.data.topic);\n            this.textManage.updateRectangle();\n        } else {\n            const hasSameSelected = value.selected === previous.selected;\n            const hasSameParent = value.parent === previous.parent;\n            if (!hasSameSelected) {\n                this.activeDrawer.draw(this.element, this.g, { selected: this.selected, isEditing: this.textManage.isEditing });\n            }\n            if (!hasSameParent) {\n                this.drawLink();\n            }\n        }\n    }\n\n    drawEmojis() {\n        const g = this.nodeEmojisDrawer.drawEmojis(this.element);\n        if (g) {\n            this.g.append(g);\n        }\n    }\n\n    drawShape() {\n        this.destroyShape();\n        const shape = getShapeByElement(this.board, this.node.origin) as MindElementShape;\n        switch (shape) {\n            case MindElementShape.roundRectangle:\n                this.shapeG = drawRoundRectangleByNode(this.board, this.node as MindNode);\n                this.g.prepend(this.shapeG);\n                break;\n            default:\n                break;\n        }\n    }\n\n    destroyShape() {\n        if (this.shapeG) {\n            this.shapeG.remove();\n            this.shapeG = null;\n        }\n    }\n\n    drawLink() {\n        if (PlaitMind.isMind(this.element)) {\n            return;\n        }\n\n        const parent = MindElement.getParent(this.element);\n        const parentNode = MindElement.getNode(parent);\n\n        if (this.linkG) {\n            this.linkG.remove();\n        }\n\n        const layout = MindQueries.getLayoutByElement(parent) as MindLayoutType;\n        if (AbstractNode.isAbstract(this.node.origin)) {\n            this.linkG = drawAbstractLink(this.board, this.node, isHorizontalLayout(layout));\n        } else {\n            this.linkG = drawLink(this.board, parentNode, this.node, isHorizontalLayout(layout));\n        }\n        this.g.append(this.linkG);\n    }\n\n    destroyLine() {\n        if (this.linkG) {\n            this.linkG.remove();\n        }\n    }\n\n    drawExtend() {\n        this.destroyExtend();\n\n        this.extendG = createG();\n        this.extendG.classList.add('extend');\n        this.g.append(this.extendG);\n\n        if (this.element.isCollapsed) {\n            this.g.classList.add('collapsed');\n        } else {\n            this.g.classList.remove('collapsed');\n        }\n\n        this.nodeInsertDrawer.draw(this.element, this.extendG!);\n        this.collapseDrawer.draw(this.element, this.extendG!);\n    }\n\n    destroyExtend() {\n        if (this.extendG) {\n            this.extendG.remove();\n        }\n    }\n\n    drawText() {\n        this.textManage.draw(this.element.data.topic);\n        this.g.append(this.textManage.g);\n    }\n\n    trackBy = (index: number, node: MindNode) => {\n        return node.origin.id;\n    };\n\n    ngOnDestroy(): void {\n        super.ngOnDestroy();\n        this.textManage.destroy();\n        this.nodeEmojisDrawer.destroy();\n        this.destroy$.next();\n        this.destroy$.complete();\n        if (ELEMENT_TO_NODE.get(this.element) === this.node) {\n            ELEMENT_TO_NODE.delete(this.element);\n        }\n    }\n}\n"]}