Ei kuvausta

index.ts 5.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. import { MarkerType } from '../types';
  2. import { RectangularMarker } from '../RectangularMarker';
  3. import { SvgHelper } from '../../renderer/SvgHelper';
  4. import { WhitePage } from '../../whiteboard/WhitePage';
  5. import { PositionType } from '../../utils/layout';
  6. import { MarkerSnap } from '../../whiteboard/AbstractWhiteboard/snap';
  7. const OkIcon = require('../../assets/check.svg');
  8. const CancelIcon = require('../../assets/times.svg');
  9. export class TextMarker extends RectangularMarker {
  10. type: MarkerType = 'text';
  11. public static createMarker = (page?: WhitePage): TextMarker => {
  12. const marker = new TextMarker();
  13. marker.page = page;
  14. marker.setup();
  15. return marker;
  16. };
  17. /** UI Options */
  18. protected readonly MIN_SIZE = 50;
  19. private readonly DEFAULT_TEXT = 'Double-click to edit text';
  20. private text: string = this.DEFAULT_TEXT;
  21. private inDoubleTap = false;
  22. /** UI Handlers */
  23. private textElement: SVGTextElement;
  24. private editor: HTMLDivElement;
  25. private editorTextArea: HTMLTextAreaElement;
  26. /** Getter & Setter */
  27. public setText(text: string) {
  28. this.text = text;
  29. this.renderText();
  30. }
  31. public captureSnap(): MarkerSnap {
  32. const baseSnap = super.captureSnap();
  33. baseSnap.textSnap = { text: this.text };
  34. return baseSnap;
  35. }
  36. public applySnap(snap: MarkerSnap) {
  37. super.applySnap(snap);
  38. if (snap.textSnap && snap.textSnap.text !== this.text) {
  39. this.setText(snap.textSnap.text);
  40. }
  41. }
  42. protected setup() {
  43. super.setup();
  44. this.textElement = SvgHelper.createText();
  45. this.addToRenderVisual(this.textElement);
  46. SvgHelper.setAttributes(this.visual, [['class', 'text-marker']]);
  47. this.textElement.transform.baseVal.appendItem(SvgHelper.createTransform()); // translate transorm
  48. this.textElement.transform.baseVal.appendItem(SvgHelper.createTransform()); // scale transorm
  49. this.renderText();
  50. this.visual.addEventListener('dblclick', this.onDblClick);
  51. this.visual.addEventListener('touchstart', this.onTap);
  52. }
  53. protected resize(x: number, y: number, onPosition?: (pos: PositionType) => void) {
  54. super.resize(x, y, onPosition);
  55. this.sizeText();
  56. }
  57. private renderText = () => {
  58. const LINE_SIZE = '1.2em';
  59. while (this.textElement.lastChild) {
  60. this.textElement.removeChild(this.textElement.lastChild);
  61. }
  62. const lines = this.text.split(/\r\n|[\n\v\f\r\x85\u2028\u2029]/);
  63. for (let line of lines) {
  64. if (line.trim() === '') {
  65. line = ' '; // workaround for swallowed empty lines
  66. }
  67. this.textElement.appendChild(SvgHelper.createTSpan(line, [['x', '0'], ['dy', LINE_SIZE]]));
  68. }
  69. setTimeout(this.sizeText, 10);
  70. };
  71. private sizeText = () => {
  72. const textSize = this.textElement.getBBox();
  73. let x = 0;
  74. let y = 0;
  75. let scale = 1.0;
  76. if (textSize.width > 0 && textSize.height > 0) {
  77. const xScale = (this.width * 1.0) / textSize.width;
  78. const yScale = (this.height * 1.0) / textSize.height;
  79. scale = Math.min(xScale, yScale);
  80. x = (this.width - textSize.width * scale) / 2;
  81. y = (this.height - textSize.height * scale) / 2;
  82. }
  83. this.textElement.transform.baseVal.getItem(0).setTranslate(x, y);
  84. this.textElement.transform.baseVal.getItem(1).setScale(scale, scale);
  85. };
  86. private onDblClick = (ev: MouseEvent) => {
  87. this.showEditor();
  88. };
  89. private onTap = (ev: TouchEvent) => {
  90. if (this.inDoubleTap) {
  91. this.inDoubleTap = false;
  92. this.showEditor();
  93. } else {
  94. this.inDoubleTap = true;
  95. setTimeout(() => {
  96. this.inDoubleTap = false;
  97. }, 300);
  98. }
  99. };
  100. private showEditor = () => {
  101. this.editor = document.createElement('div');
  102. this.editor.className = 'fc-whiteboard-text-editor';
  103. this.editorTextArea = document.createElement('textarea');
  104. if (this.text !== this.DEFAULT_TEXT) {
  105. this.editorTextArea.value = this.text;
  106. }
  107. this.editorTextArea.addEventListener('keydown', this.onEditorKeyDown);
  108. this.editor.appendChild(this.editorTextArea);
  109. document.body.appendChild(this.editor);
  110. const buttons = document.createElement('div');
  111. buttons.className = 'fc-whiteboard-text-editor-button-bar';
  112. this.editor.appendChild(buttons);
  113. const okButton = document.createElement('div');
  114. okButton.className = 'fc-whiteboard-text-editor-button';
  115. okButton.innerHTML = OkIcon;
  116. okButton.addEventListener('click', this.onEditorOkClick);
  117. buttons.appendChild(okButton);
  118. const cancelButton = document.createElement('div');
  119. cancelButton.className = 'fc-whiteboard-text-editor-button';
  120. cancelButton.innerHTML = CancelIcon;
  121. cancelButton.addEventListener('click', this.closeEditor);
  122. buttons.appendChild(cancelButton);
  123. };
  124. /** 响应文本输入的事件 */
  125. private onEditorOkClick = (ev: MouseEvent | null) => {
  126. if (this.editorTextArea.value.trim()) {
  127. this.text = this.editorTextArea.value;
  128. } else {
  129. this.text = this.DEFAULT_TEXT;
  130. }
  131. // 触发文本修改时间
  132. this.onChange({
  133. target: 'marker',
  134. id: this.id,
  135. event: 'inputMarker',
  136. marker: { text: this.text }
  137. });
  138. this.renderText();
  139. this.closeEditor();
  140. };
  141. private closeEditor = () => {
  142. document.body.removeChild(this.editor);
  143. };
  144. private onEditorKeyDown = (ev: KeyboardEvent) => {
  145. if (ev.key === 'Enter' && ev.ctrlKey) {
  146. ev.preventDefault();
  147. this.onEditorOkClick(null);
  148. }
  149. };
  150. }