@textbus/collaborate 2.0.0-beta.4 → 2.0.0-beta.5
Sign up to get free protection for your applications and to get access to all the features.
- package/bundles/{collab/collaborate-cursor.d.ts → collaborate-cursor.d.ts} +0 -0
- package/bundles/collaborate-cursor.js +245 -0
- package/bundles/collaborate.d.ts +1 -1
- package/bundles/collaborate.js +2 -2
- package/bundles/public-api.d.ts +1 -1
- package/bundles/public-api.js +2 -2
- package/package.json +3 -3
- package/src/{collab/collaborate-cursor.ts → collaborate-cursor.ts} +0 -0
- package/src/collaborate.ts +1 -1
- package/src/public-api.ts +1 -1
- package/bundles/collab/_api.d.ts +0 -1
- package/bundles/collab/_api.js +0 -2
- package/bundles/collab/collaborate-cursor.js +0 -245
- package/src/collab/_api.ts +0 -1
File without changes
|
@@ -0,0 +1,245 @@
|
|
1
|
+
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
|
2
|
+
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
3
|
+
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
4
|
+
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
|
5
|
+
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
6
|
+
};
|
7
|
+
var __metadata = (this && this.__metadata) || function (k, v) {
|
8
|
+
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
|
9
|
+
};
|
10
|
+
var __param = (this && this.__param) || function (paramIndex, decorator) {
|
11
|
+
return function (target, key) { decorator(target, key, paramIndex); }
|
12
|
+
};
|
13
|
+
import { Inject, Injectable } from '@tanbo/di';
|
14
|
+
import { createElement, EDITABLE_DOCUMENT, EDITOR_CONTAINER, getLayoutRectByRange, SelectionBridge } from '@textbus/browser';
|
15
|
+
import { Selection } from '@textbus/core';
|
16
|
+
import { Subject } from '@tanbo/stream';
|
17
|
+
let CollaborateCursor = class CollaborateCursor {
|
18
|
+
constructor(container, document, nativeSelection, selection) {
|
19
|
+
Object.defineProperty(this, "container", {
|
20
|
+
enumerable: true,
|
21
|
+
configurable: true,
|
22
|
+
writable: true,
|
23
|
+
value: container
|
24
|
+
});
|
25
|
+
Object.defineProperty(this, "document", {
|
26
|
+
enumerable: true,
|
27
|
+
configurable: true,
|
28
|
+
writable: true,
|
29
|
+
value: document
|
30
|
+
});
|
31
|
+
Object.defineProperty(this, "nativeSelection", {
|
32
|
+
enumerable: true,
|
33
|
+
configurable: true,
|
34
|
+
writable: true,
|
35
|
+
value: nativeSelection
|
36
|
+
});
|
37
|
+
Object.defineProperty(this, "selection", {
|
38
|
+
enumerable: true,
|
39
|
+
configurable: true,
|
40
|
+
writable: true,
|
41
|
+
value: selection
|
42
|
+
});
|
43
|
+
Object.defineProperty(this, "canvas", {
|
44
|
+
enumerable: true,
|
45
|
+
configurable: true,
|
46
|
+
writable: true,
|
47
|
+
value: createElement('canvas', {
|
48
|
+
styles: {
|
49
|
+
position: 'absolute',
|
50
|
+
opacity: 0.5,
|
51
|
+
left: 0,
|
52
|
+
top: 0,
|
53
|
+
width: '100%',
|
54
|
+
height: '100%',
|
55
|
+
pointerEvents: 'none'
|
56
|
+
}
|
57
|
+
})
|
58
|
+
});
|
59
|
+
Object.defineProperty(this, "context", {
|
60
|
+
enumerable: true,
|
61
|
+
configurable: true,
|
62
|
+
writable: true,
|
63
|
+
value: this.canvas.getContext('2d')
|
64
|
+
});
|
65
|
+
Object.defineProperty(this, "tooltips", {
|
66
|
+
enumerable: true,
|
67
|
+
configurable: true,
|
68
|
+
writable: true,
|
69
|
+
value: createElement('div', {
|
70
|
+
styles: {
|
71
|
+
position: 'absolute',
|
72
|
+
left: 0,
|
73
|
+
top: 0,
|
74
|
+
width: '100%',
|
75
|
+
height: '100%',
|
76
|
+
pointerEvents: 'none',
|
77
|
+
fontSize: '12px',
|
78
|
+
zIndex: 10
|
79
|
+
}
|
80
|
+
})
|
81
|
+
});
|
82
|
+
Object.defineProperty(this, "onRectsChange", {
|
83
|
+
enumerable: true,
|
84
|
+
configurable: true,
|
85
|
+
writable: true,
|
86
|
+
value: new Subject()
|
87
|
+
});
|
88
|
+
container.prepend(this.canvas, this.tooltips);
|
89
|
+
this.onRectsChange.subscribe(rects => {
|
90
|
+
for (const rect of rects) {
|
91
|
+
this.context.fillStyle = rect.color;
|
92
|
+
this.context.beginPath();
|
93
|
+
this.context.rect(rect.x, rect.y, rect.width, rect.height);
|
94
|
+
this.context.fill();
|
95
|
+
this.context.closePath();
|
96
|
+
}
|
97
|
+
});
|
98
|
+
}
|
99
|
+
draw(paths) {
|
100
|
+
const containerRect = this.container.getBoundingClientRect();
|
101
|
+
this.canvas.width = containerRect.width;
|
102
|
+
this.canvas.height = containerRect.height;
|
103
|
+
this.context.clearRect(0, 0, this.canvas.width, this.canvas.height);
|
104
|
+
const users = [];
|
105
|
+
paths.filter(i => {
|
106
|
+
return i.paths.start.length && i.paths.end.length;
|
107
|
+
}).forEach(item => {
|
108
|
+
const startOffset = item.paths.start.pop();
|
109
|
+
const startSlot = this.selection.findSlotByPaths(item.paths.start);
|
110
|
+
const endOffset = item.paths.end.pop();
|
111
|
+
const endSlot = this.selection.findSlotByPaths(item.paths.end);
|
112
|
+
if (startSlot && endSlot) {
|
113
|
+
const position = this.nativeSelection.getPositionByRange({
|
114
|
+
startOffset,
|
115
|
+
endOffset,
|
116
|
+
startSlot,
|
117
|
+
endSlot
|
118
|
+
});
|
119
|
+
if (position.start && position.end) {
|
120
|
+
const nativeRange = this.document.createRange();
|
121
|
+
nativeRange.setStart(position.start.node, position.start.offset);
|
122
|
+
nativeRange.setEnd(position.end.node, position.end.offset);
|
123
|
+
const rects = nativeRange.getClientRects();
|
124
|
+
const selectionRects = [];
|
125
|
+
for (let i = rects.length - 1; i >= 0; i--) {
|
126
|
+
const rect = rects[i];
|
127
|
+
selectionRects.push({
|
128
|
+
color: item.color,
|
129
|
+
username: item.username,
|
130
|
+
x: rect.x - containerRect.x,
|
131
|
+
y: rect.y - containerRect.y,
|
132
|
+
width: rect.width,
|
133
|
+
height: rect.height,
|
134
|
+
});
|
135
|
+
}
|
136
|
+
this.onRectsChange.next(selectionRects);
|
137
|
+
const cursorRange = nativeRange.cloneRange();
|
138
|
+
cursorRange.collapse(!item.paths.focusEnd);
|
139
|
+
const cursorRect = getLayoutRectByRange(cursorRange);
|
140
|
+
users.push({
|
141
|
+
username: item.username,
|
142
|
+
color: item.color,
|
143
|
+
x: cursorRect.x - containerRect.x,
|
144
|
+
y: cursorRect.y - containerRect.y,
|
145
|
+
width: 2,
|
146
|
+
height: cursorRect.height
|
147
|
+
});
|
148
|
+
}
|
149
|
+
}
|
150
|
+
});
|
151
|
+
this.drawUserCursor(users);
|
152
|
+
}
|
153
|
+
drawUserCursor(rects) {
|
154
|
+
for (let i = 0; i < rects.length; i++) {
|
155
|
+
const rect = rects[i];
|
156
|
+
const { cursor, userTip, anchor } = this.getUserCursor(i);
|
157
|
+
Object.assign(cursor.style, {
|
158
|
+
left: rect.x + 'px',
|
159
|
+
top: rect.y + 'px',
|
160
|
+
width: rect.width + 'px',
|
161
|
+
height: rect.height + 'px',
|
162
|
+
background: rect.color,
|
163
|
+
display: 'block'
|
164
|
+
});
|
165
|
+
anchor.style.background = rect.color;
|
166
|
+
userTip.innerText = rect.username;
|
167
|
+
userTip.style.background = rect.color;
|
168
|
+
}
|
169
|
+
for (let i = rects.length; i < this.tooltips.children.length; i++) {
|
170
|
+
this.tooltips.removeChild(this.tooltips.children[i]);
|
171
|
+
}
|
172
|
+
}
|
173
|
+
getUserCursor(index) {
|
174
|
+
let child = this.tooltips.children[index];
|
175
|
+
if (child) {
|
176
|
+
const anchor = child.children[0];
|
177
|
+
return {
|
178
|
+
cursor: child,
|
179
|
+
anchor,
|
180
|
+
userTip: anchor.children[0]
|
181
|
+
};
|
182
|
+
}
|
183
|
+
const userTip = createElement('span', {
|
184
|
+
styles: {
|
185
|
+
position: 'absolute',
|
186
|
+
display: 'none',
|
187
|
+
left: '50%',
|
188
|
+
transform: 'translateX(-50%)',
|
189
|
+
marginBottom: '2px',
|
190
|
+
bottom: '100%',
|
191
|
+
whiteSpace: 'nowrap',
|
192
|
+
color: '#fff',
|
193
|
+
boxShadow: '0 1px 2px rgba(0,0,0,.1)',
|
194
|
+
borderRadius: '3px',
|
195
|
+
padding: '3px 5px',
|
196
|
+
pointerEvents: 'none',
|
197
|
+
}
|
198
|
+
});
|
199
|
+
const anchor = createElement('span', {
|
200
|
+
styles: {
|
201
|
+
position: 'absolute',
|
202
|
+
top: '-2px',
|
203
|
+
left: '-2px',
|
204
|
+
width: '6px',
|
205
|
+
height: '6px',
|
206
|
+
pointerEvents: 'auto',
|
207
|
+
pointer: 'cursor',
|
208
|
+
},
|
209
|
+
children: [userTip],
|
210
|
+
on: {
|
211
|
+
mouseenter() {
|
212
|
+
userTip.style.display = 'block';
|
213
|
+
},
|
214
|
+
mouseleave() {
|
215
|
+
userTip.style.display = 'none';
|
216
|
+
}
|
217
|
+
}
|
218
|
+
});
|
219
|
+
child = createElement('span', {
|
220
|
+
styles: {
|
221
|
+
position: 'absolute',
|
222
|
+
},
|
223
|
+
children: [
|
224
|
+
anchor
|
225
|
+
]
|
226
|
+
});
|
227
|
+
this.tooltips.append(child);
|
228
|
+
return {
|
229
|
+
cursor: child,
|
230
|
+
anchor,
|
231
|
+
userTip
|
232
|
+
};
|
233
|
+
}
|
234
|
+
};
|
235
|
+
CollaborateCursor = __decorate([
|
236
|
+
Injectable(),
|
237
|
+
__param(0, Inject(EDITOR_CONTAINER)),
|
238
|
+
__param(1, Inject(EDITABLE_DOCUMENT)),
|
239
|
+
__metadata("design:paramtypes", [HTMLElement,
|
240
|
+
Document,
|
241
|
+
SelectionBridge,
|
242
|
+
Selection])
|
243
|
+
], CollaborateCursor);
|
244
|
+
export { CollaborateCursor };
|
245
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"collaborate-cursor.js","sourceRoot":"","sources":["../src/collaborate-cursor.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable } from '@tanbo/di'\nimport {\n  createElement,\n  EDITABLE_DOCUMENT,\n  EDITOR_CONTAINER,\n  getLayoutRectByRange,\n  SelectionBridge\n} from '@textbus/browser'\nimport { Selection, SelectionPaths } from '@textbus/core'\nimport { Subject } from '@tanbo/stream'\n\nexport interface RemoteSelection {\n  color: string\n  username: string\n  paths: SelectionPaths\n}\n\nexport interface SelectionRect {\n  color: string\n  username: string\n  x: number\n  y: number\n  width: number\n  height: number\n}\n\nexport interface RemoteSelectionCursor {\n  cursor: HTMLElement\n  anchor: HTMLElement\n  userTip: HTMLElement\n}\n\n@Injectable()\nexport class CollaborateCursor {\n  private canvas = createElement('canvas', {\n    styles: {\n      position: 'absolute',\n      opacity: 0.5,\n      left: 0,\n      top: 0,\n      width: '100%',\n      height: '100%',\n      pointerEvents: 'none'\n    }\n  }) as HTMLCanvasElement\n  private context = this.canvas.getContext('2d')!\n  private tooltips = createElement('div', {\n    styles: {\n      position: 'absolute',\n      left: 0,\n      top: 0,\n      width: '100%',\n      height: '100%',\n      pointerEvents: 'none',\n      fontSize: '12px',\n      zIndex: 10\n    }\n  })\n\n  private onRectsChange = new Subject<SelectionRect[]>()\n\n  constructor(@Inject(EDITOR_CONTAINER) private container: HTMLElement,\n              @Inject(EDITABLE_DOCUMENT) private document: Document,\n              private nativeSelection: SelectionBridge,\n              private selection: Selection) {\n    container.prepend(this.canvas, this.tooltips)\n    this.onRectsChange.subscribe(rects => {\n      for (const rect of rects) {\n        this.context.fillStyle = rect.color\n        this.context.beginPath()\n        this.context.rect(rect.x, rect.y, rect.width, rect.height)\n        this.context.fill()\n        this.context.closePath()\n      }\n    })\n  }\n\n  draw(paths: RemoteSelection[]) {\n    const containerRect = this.container.getBoundingClientRect()\n    this.canvas.width = containerRect.width\n    this.canvas.height = containerRect.height\n    this.context.clearRect(0, 0, this.canvas.width, this.canvas.height)\n\n    const users: SelectionRect[] = []\n\n\n    paths.filter(i => {\n      return i.paths.start.length && i.paths.end.length\n    }).forEach(item => {\n      const startOffset = item.paths.start.pop()!\n      const startSlot = this.selection.findSlotByPaths(item.paths.start)\n      const endOffset = item.paths.end.pop()!\n      const endSlot = this.selection.findSlotByPaths(item.paths.end)\n\n      if (startSlot && endSlot) {\n        const position = this.nativeSelection.getPositionByRange({\n          startOffset,\n          endOffset,\n          startSlot,\n          endSlot\n        })\n        if (position.start && position.end) {\n          const nativeRange = this.document.createRange()\n          nativeRange.setStart(position.start.node, position.start.offset)\n          nativeRange.setEnd(position.end.node, position.end.offset)\n\n          const rects = nativeRange.getClientRects()\n          const selectionRects: SelectionRect[] = []\n          for (let i = rects.length - 1; i >= 0; i--) {\n            const rect = rects[i]\n            selectionRects.push({\n              color: item.color,\n              username: item.username,\n              x: rect.x - containerRect.x,\n              y: rect.y - containerRect.y,\n              width: rect.width,\n              height: rect.height,\n            })\n          }\n          this.onRectsChange.next(selectionRects)\n\n          const cursorRange = nativeRange.cloneRange()\n          cursorRange.collapse(!item.paths.focusEnd)\n\n          const cursorRect = getLayoutRectByRange(cursorRange)\n\n          users.push({\n            username: item.username,\n            color: item.color,\n            x: cursorRect.x - containerRect.x,\n            y: cursorRect.y - containerRect.y,\n            width: 2,\n            height: cursorRect.height\n          })\n        }\n      }\n    })\n    this.drawUserCursor(users)\n  }\n\n  private drawUserCursor(rects: SelectionRect[]) {\n    for (let i = 0; i < rects.length; i++) {\n      const rect = rects[i]\n      const {cursor, userTip, anchor} = this.getUserCursor(i)\n      Object.assign(cursor.style, {\n        left: rect.x + 'px',\n        top: rect.y + 'px',\n        width: rect.width + 'px',\n        height: rect.height + 'px',\n        background: rect.color,\n        display: 'block'\n      })\n      anchor.style.background = rect.color\n      userTip.innerText = rect.username\n      userTip.style.background = rect.color\n    }\n\n    for (let i = rects.length; i < this.tooltips.children.length; i++) {\n      this.tooltips.removeChild(this.tooltips.children[i])\n    }\n  }\n\n  private getUserCursor(index: number): RemoteSelectionCursor {\n    let child: HTMLElement = this.tooltips.children[index] as HTMLElement\n    if (child) {\n      const anchor = child.children[0] as HTMLElement\n      return {\n        cursor: child,\n        anchor,\n        userTip: anchor.children[0] as HTMLElement\n      }\n    }\n    const userTip = createElement('span', {\n      styles: {\n        position: 'absolute',\n        display: 'none',\n        left: '50%',\n        transform: 'translateX(-50%)',\n        marginBottom: '2px',\n        bottom: '100%',\n        whiteSpace: 'nowrap',\n        color: '#fff',\n        boxShadow: '0 1px 2px rgba(0,0,0,.1)',\n        borderRadius: '3px',\n        padding: '3px 5px',\n        pointerEvents: 'none',\n      }\n    })\n\n    const anchor = createElement('span', {\n      styles: {\n        position: 'absolute',\n        top: '-2px',\n        left: '-2px',\n        width: '6px',\n        height: '6px',\n        pointerEvents: 'auto',\n        pointer: 'cursor',\n      },\n      children: [userTip],\n      on: {\n        mouseenter() {\n          userTip.style.display = 'block'\n        },\n        mouseleave() {\n          userTip.style.display = 'none'\n        }\n      }\n    })\n    child = createElement('span', {\n      styles: {\n        position: 'absolute',\n      },\n      children: [\n        anchor\n      ]\n    })\n    this.tooltips.append(child)\n    return {\n      cursor: child,\n      anchor,\n      userTip\n    }\n  }\n}\n"]}
|
package/bundles/collaborate.d.ts
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
import { Observable } from '@tanbo/stream';
|
2
2
|
import { RootComponentRef, Starter, Translator, Registry, Selection, SelectionPaths, History, Renderer } from '@textbus/core';
|
3
3
|
import { Doc as YDoc } from 'yjs';
|
4
|
-
import { CollaborateCursor, RemoteSelection } from './
|
4
|
+
import { CollaborateCursor, RemoteSelection } from './collaborate-cursor';
|
5
5
|
export declare class Collaborate implements History {
|
6
6
|
private rootComponentRef;
|
7
7
|
private collaborateCursor;
|
package/bundles/collaborate.js
CHANGED
@@ -11,7 +11,7 @@ import { Injectable } from '@tanbo/di';
|
|
11
11
|
import { merge, microTask, Subject } from '@tanbo/stream';
|
12
12
|
import { RootComponentRef, Starter, Translator, Registry, Selection, Renderer, Slot, makeError } from '@textbus/core';
|
13
13
|
import { Doc as YDoc, Map as YMap, Text as YText, Array as YArray, UndoManager } from 'yjs';
|
14
|
-
import { CollaborateCursor } from './
|
14
|
+
import { CollaborateCursor } from './collaborate-cursor';
|
15
15
|
const collaborateErrorFn = makeError('Collaborate');
|
16
16
|
let Collaborate = class Collaborate {
|
17
17
|
constructor(rootComponentRef, collaborateCursor, translator, renderer, registry, selection, starter) {
|
@@ -608,4 +608,4 @@ function makeFormats(registry, attrs) {
|
|
608
608
|
}
|
609
609
|
return formats;
|
610
610
|
}
|
611
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"collaborate.js","sourceRoot":"","sources":["../src/collaborate.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@tanbo/di'\nimport { merge, microTask, Observable, Subject, Subscription } from '@tanbo/stream'\nimport {\n  RootComponentRef,\n  Starter,\n  Translator,\n  Registry,\n  Selection,\n  SelectionPaths,\n  History, Renderer, Slot, ComponentInstance, makeError, Formats\n} from '@textbus/core'\nimport {\n  Doc as YDoc,\n  Map as YMap,\n  Text as YText,\n  Array as YArray,\n  UndoManager,\n  Transaction\n} from 'yjs'\n\nimport { CollaborateCursor, RemoteSelection } from './collab/_api'\n\nconst collaborateErrorFn = makeError('Collaborate')\n\n@Injectable()\nexport class Collaborate implements History {\n  onSelectionChange: Observable<SelectionPaths>\n  yDoc = new YDoc()\n  onBack: Observable<void>\n  onForward: Observable<void>\n  onChange: Observable<any>\n  onPush: Observable<void>\n\n  get canBack() {\n    return this.manager?.canUndo()\n  }\n\n  get canForward() {\n    return this.manager?.canRedo()\n  }\n\n  private backEvent = new Subject<void>()\n  private forwardEvent = new Subject<void>()\n  private changeEvent = new Subject<void>()\n  private pushEvent = new Subject<void>()\n\n  private manager!: UndoManager\n\n  private subscriptions: Subscription[] = []\n  private updateFromRemote = false\n\n  private contentSyncCaches = new WeakMap<Slot, () => void>()\n  private slotStateSyncCaches = new WeakMap<Slot, () => void>()\n  private slotsSyncCaches = new WeakMap<ComponentInstance, () => void>()\n  private componentStateSyncCaches = new WeakMap<ComponentInstance, () => void>()\n\n  private selectionChangeEvent = new Subject<SelectionPaths>()\n\n  private updateRemoteActions: Array<() => void> = []\n\n  constructor(private rootComponentRef: RootComponentRef,\n              private collaborateCursor: CollaborateCursor,\n              private translator: Translator,\n              private renderer: Renderer,\n              private registry: Registry,\n              private selection: Selection,\n              private starter: Starter) {\n    this.onSelectionChange = this.selectionChangeEvent.asObservable()\n    this.onBack = this.backEvent.asObservable()\n    this.onForward = this.forwardEvent.asObservable()\n    this.onChange = this.changeEvent.asObservable()\n    this.onPush = this.pushEvent.asObservable()\n  }\n\n  setup() {\n    this.subscriptions.push(\n      this.starter.onReady.subscribe(() => {\n        this.listen2()\n      }),\n      this.selection.onChange.subscribe(() => {\n        const paths = this.selection.getPaths()\n        this.selectionChangeEvent.next(paths)\n      })\n    )\n  }\n\n  updateRemoteSelection(paths: RemoteSelection[]) {\n    this.collaborateCursor.draw(paths)\n  }\n\n  listen() {\n    //\n  }\n\n  back() {\n    if (this.canBack) {\n      this.manager.undo()\n    }\n  }\n\n  forward() {\n    if (this.canForward) {\n      this.manager.redo()\n    }\n  }\n\n  destroy() {\n    this.subscriptions.forEach(i => i.unsubscribe())\n  }\n\n  private listen2() {\n    const root = this.yDoc.getText('content')\n    const rootComponent = this.rootComponentRef.component!\n    this.manager = new UndoManager(root, {\n      trackedOrigins: new Set<any>([this.yDoc])\n    })\n    this.syncContent(root, rootComponent.slots.get(0)!)\n\n    this.subscriptions.push(\n      merge(\n        rootComponent.changeMarker.onForceChange,\n        rootComponent.changeMarker.onChange\n      ).pipe(\n        microTask()\n      ).subscribe(() => {\n        this.yDoc.transact(() => {\n          this.updateRemoteActions.forEach(fn => {\n            fn()\n          })\n          this.updateRemoteActions = []\n        }, this.yDoc)\n        this.renderer.render()\n        this.selection.restore()\n      })\n    )\n  }\n\n  private syncContent(content: YText, slot: Slot) {\n    const syncRemote = (ev, tr) => {\n      this.runRemoteUpdate(tr, () => {\n        slot.retain(0)\n        ev.delta.forEach(action => {\n          if (Reflect.has(action, 'retain')) {\n            if (action.attributes) {\n              const formats = makeFormats(this.registry, action.attributes)\n              if (formats.length) {\n                slot.retain(action.retain!, formats)\n              }\n            } else {\n              slot.retain(action.retain)\n            }\n          } else if (action.insert) {\n            const index = slot.index\n            let length = 1\n            if (typeof action.insert === 'string') {\n              length = action.insert.length\n              slot.insert(action.insert, makeFormats(this.registry, action.attributes))\n            } else {\n              const sharedComponent = action.insert as YMap<any>\n              const component = this.createComponentBySharedComponent(sharedComponent)\n              this.syncSlots(sharedComponent.get('slots'), component)\n              this.syncComponent(sharedComponent, component)\n              slot.insert(component)\n            }\n            if (this.selection.isSelected) {\n              if (slot === this.selection.startSlot && this.selection.startOffset! >= index) {\n                this.selection.setStart(slot, this.selection.startOffset! + length)\n              }\n              if (slot === this.selection.endSlot && this.selection.endOffset! >= index) {\n                this.selection.setEnd(slot, this.selection.endOffset! + length)\n              }\n            }\n          } else if (action.delete) {\n            const index = slot.index\n            slot.retain(slot.index)\n            slot.delete(action.delete)\n            if (this.selection.isSelected) {\n              if (slot === this.selection.startSlot && this.selection.startOffset! >= index) {\n                this.selection.setStart(slot, this.selection.startOffset! - action.delete)\n              }\n              if (slot === this.selection.endSlot && this.selection.endOffset! >= index) {\n                this.selection.setEnd(slot, this.selection.endOffset! - action.delete)\n              }\n            }\n          } else if (action.attributes) {\n            slot.updateState(draft => {\n              Object.assign(draft, action.attributes)\n            })\n          }\n        })\n      })\n    }\n    content.observe(syncRemote)\n\n    const sub = slot.onContentChange.subscribe(actions => {\n      this.runLocalUpdate(() => {\n        let offset = 0\n        let length = 0\n        for (const action of actions) {\n          if (action.type === 'retain') {\n            const formats = action.formats\n            if (formats) {\n              const keys = Object.keys(formats)\n              let length = keys.length\n              keys.forEach(key => {\n                if (!this.registry.getFormatter(key)) {\n                  length--\n                  Reflect.deleteProperty(formats, key)\n                }\n              })\n              if (length) {\n                content.format(offset, action.offset, formats)\n              }\n            } else {\n              offset = action.offset\n            }\n          } else if (action.type === 'insert') {\n            const delta = content.toDelta()\n            const isEmpty = delta.length === 1 && delta[0].insert === Slot.emptyPlaceholder\n            if (typeof action.content === 'string') {\n              length = action.content.length\n              content.insert(offset, action.content)\n            } else {\n              length = 1\n              const component = slot.getContentAtIndex(offset) as ComponentInstance\n              const sharedComponent = this.createSharedComponentByComponent(component)\n              content.insertEmbed(offset, sharedComponent)\n            }\n            if (action.formats) {\n              content.format(offset, length, action.formats)\n            }\n            if (isEmpty && offset === 0) {\n              content.delete(content.length - 1, 1)\n            }\n            offset += length\n          } else if (action.type === 'delete') {\n            const delta = content.toDelta()\n            content.delete(offset, action.count)\n            if (content.length === 0) {\n              content.insert(0, '\\n', delta[0]?.attributes)\n            }\n          }\n        }\n      })\n    })\n    this.contentSyncCaches.set(slot, () => {\n      content.unobserve(syncRemote)\n      sub.unsubscribe()\n    })\n  }\n\n  private syncSlot(remoteSlot: YMap<any>, slot: Slot) {\n    const syncRemote = (ev, tr) => {\n      this.runRemoteUpdate(tr, () => {\n        ev.keysChanged.forEach(key => {\n          if (key === 'state') {\n            const state = (ev.target as YMap<any>).get('state')\n            slot.updateState(draft => {\n              Object.assign(draft, state)\n            })\n          }\n        })\n      })\n    }\n    remoteSlot.observe(syncRemote)\n\n    const sub = slot.onStateChange.subscribe(actions => {\n      this.runLocalUpdate(() => {\n        actions.forEach(action => {\n          remoteSlot.set('state', action.value)\n        })\n      })\n    })\n    this.slotStateSyncCaches.set(slot, () => {\n      remoteSlot.unobserve(syncRemote)\n      sub.unsubscribe()\n    })\n  }\n\n  private syncSlots(remoteSlots: YArray<any>, component: ComponentInstance) {\n    const slots = component.slots\n    const syncRemote = (ev, tr) => {\n      this.runRemoteUpdate(tr, () => {\n        ev.delta.forEach(action => {\n          if (Reflect.has(action, 'retain')) {\n            slots.retain(action.retain!)\n          } else if (action.insert) {\n            (action.insert as Array<YMap<any>>).forEach(item => {\n              const slot = this.createSlotBySharedSlot(item)\n              slots.insert(slot)\n              this.syncContent(item.get('content'), slot)\n              this.syncSlot(item, slot)\n            })\n          } else if (action.delete) {\n            slots.retain(slots.index)\n            slots.delete(action.delete)\n          }\n        })\n      })\n    }\n    remoteSlots.observe(syncRemote)\n\n    const sub = slots.onChange.subscribe(operations => {\n      this.runLocalUpdate(() => {\n        const applyActions = operations.apply\n        let index: number\n        applyActions.forEach(action => {\n          if (action.type === 'retain') {\n            index = action.offset\n          } else if (action.type === 'insertSlot') {\n            const slot = slots.get(index)!\n            const sharedSlot = this.createSharedSlotBySlot(slot)\n            remoteSlots.insert(index, [sharedSlot])\n            index++\n          } else if (action.type === 'delete') {\n            slots.slice(index, index + action.count).forEach(slot => {\n              this.cleanSubscriptionsBySlot(slot)\n            })\n            remoteSlots.delete(index, action.count)\n          }\n        })\n      })\n    })\n\n    this.slotsSyncCaches.set(component, () => {\n      remoteSlots.unobserve(syncRemote)\n      sub.unsubscribe()\n    })\n  }\n\n  private syncComponent(remoteComponent: YMap<any>, component: ComponentInstance) {\n    const syncRemote = (ev, tr) => {\n      this.runRemoteUpdate(tr, () => {\n        ev.keysChanged.forEach(key => {\n          if (key === 'state') {\n            const state = (ev.target as YMap<any>).get('state')\n            component.updateState(draft => {\n              Object.assign(draft, state)\n            })\n          }\n        })\n      })\n    }\n    remoteComponent.observe(syncRemote)\n\n    const sub = component.onStateChange.subscribe(newState => {\n      this.runLocalUpdate(() => {\n        remoteComponent.set('state', newState)\n      })\n    })\n    this.componentStateSyncCaches.set(component, () => {\n      remoteComponent.unobserve(syncRemote)\n      sub.unsubscribe()\n    })\n  }\n\n  private runLocalUpdate(fn: () => void) {\n    if (this.updateFromRemote) {\n      return\n    }\n    this.updateRemoteActions.push(fn)\n  }\n\n  private runRemoteUpdate(tr: Transaction, fn: () => void) {\n    if (tr.origin === this.yDoc) {\n      return\n    }\n    this.updateFromRemote = true\n    fn()\n    this.updateFromRemote = false\n  }\n\n  private createSharedComponentByComponent(component: ComponentInstance): YMap<any> {\n    const sharedComponent = new YMap()\n    sharedComponent.set('state', component.state)\n    sharedComponent.set('name', component.name)\n    const sharedSlots = new YArray()\n    sharedComponent.set('slots', sharedSlots)\n    component.slots.toArray().forEach(slot => {\n      const sharedSlot = this.createSharedSlotBySlot(slot)\n      sharedSlots.push([sharedSlot])\n    })\n    this.syncSlots(sharedSlots, component)\n    this.syncComponent(sharedComponent, component)\n    return sharedComponent\n  }\n\n  private createSharedSlotBySlot(slot: Slot): YMap<any> {\n    const sharedSlot = new YMap()\n    sharedSlot.set('schema', slot.schema)\n    sharedSlot.set('state', slot.state)\n    const sharedContent = new YText()\n    sharedSlot.set('content', sharedContent)\n    let offset = 0\n    slot.toDelta().forEach(i => {\n      let formats: any = {}\n      if (i.formats) {\n        i.formats.forEach(item => {\n          formats[item[0].name] = item[1]\n        })\n      } else {\n        formats = null\n      }\n      if (typeof i.insert === 'string') {\n        sharedContent.insert(offset, i.insert, formats)\n      } else {\n        const sharedComponent = this.createSharedComponentByComponent(i.insert)\n        sharedContent.insertEmbed(offset, sharedComponent, formats)\n      }\n      offset += i.insert.length\n    })\n    this.syncContent(sharedContent, slot)\n    this.syncSlot(sharedSlot, slot)\n    return sharedSlot\n  }\n\n  private createComponentBySharedComponent(yMap: YMap<any>): ComponentInstance {\n    const sharedSlots = yMap.get('slots') as YArray<YMap<any>>\n    const slots: Slot[] = []\n    sharedSlots.forEach(sharedSlot => {\n      const slot = this.createSlotBySharedSlot(sharedSlot)\n      slots.push(slot)\n    })\n    const name = yMap.get('name')\n    const instance = this.translator.createComponentByData(name, {\n      state: yMap.get('state'),\n      slots\n    })\n    if (instance) {\n      instance.slots.toArray().forEach((slot, index) => {\n        const sharedSlot = sharedSlots.get(index)\n        this.syncSlot(sharedSlot, slot)\n        this.syncContent(sharedSlot.get('content'), slot)\n      })\n      return instance\n    }\n    throw collaborateErrorFn(`cannot find component factory \\`${name}\\`.`)\n  }\n\n  private createSlotBySharedSlot(sharedSlot: YMap<any>): Slot {\n    const content = sharedSlot.get('content') as YText\n    const delta = content.toDelta()\n\n    const slot = this.translator.createSlot({\n      schema: sharedSlot.get('schema'),\n      state: sharedSlot.get('state'),\n      formats: {},\n      content: []\n    })\n\n    for (const action of delta) {\n      if (action.insert) {\n        if (typeof action.insert === 'string') {\n          slot.insert(action.insert, makeFormats(this.registry, action.attributes))\n        } else {\n          const sharedComponent = action.insert as YMap<any>\n          const component = this.createComponentBySharedComponent(sharedComponent)\n          slot.insert(component)\n          this.syncSlots(sharedComponent.get('slots'), component)\n          this.syncComponent(sharedComponent, component)\n        }\n      } else {\n        throw collaborateErrorFn('unexpected delta action.')\n      }\n    }\n    return slot\n  }\n\n  private cleanSubscriptionsBySlot(slot: Slot) {\n    [this.contentSyncCaches.get(slot), this.slotStateSyncCaches.get(slot)].forEach(fn => {\n      if (fn) {\n        fn()\n      }\n    })\n    slot.sliceContent().forEach(i => {\n      if (typeof i !== 'string') {\n        this.cleanSubscriptionsByComponent(i)\n      }\n    })\n  }\n\n  private cleanSubscriptionsByComponent(component: ComponentInstance) {\n    [this.slotsSyncCaches.get(component), this.componentStateSyncCaches.get(component)].forEach(fn => {\n      if (fn) {\n        fn()\n      }\n    })\n    component.slots.toArray().forEach(slot => {\n      this.cleanSubscriptionsBySlot(slot)\n    })\n  }\n}\n\nfunction makeFormats(registry: Registry, attrs?: any) {\n  const formats: Formats = []\n  if (attrs) {\n    Object.keys(attrs).map(key => {\n      const formatter = registry.getFormatter(key)\n      if (formatter) {\n        formats.push([formatter, attrs[key]])\n      }\n    })\n  }\n  return formats\n}\n"]}
|
611
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"collaborate.js","sourceRoot":"","sources":["../src/collaborate.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@tanbo/di'\nimport { merge, microTask, Observable, Subject, Subscription } from '@tanbo/stream'\nimport {\n  RootComponentRef,\n  Starter,\n  Translator,\n  Registry,\n  Selection,\n  SelectionPaths,\n  History, Renderer, Slot, ComponentInstance, makeError, Formats\n} from '@textbus/core'\nimport {\n  Doc as YDoc,\n  Map as YMap,\n  Text as YText,\n  Array as YArray,\n  UndoManager,\n  Transaction\n} from 'yjs'\n\nimport { CollaborateCursor, RemoteSelection } from './collaborate-cursor'\n\nconst collaborateErrorFn = makeError('Collaborate')\n\n@Injectable()\nexport class Collaborate implements History {\n  onSelectionChange: Observable<SelectionPaths>\n  yDoc = new YDoc()\n  onBack: Observable<void>\n  onForward: Observable<void>\n  onChange: Observable<any>\n  onPush: Observable<void>\n\n  get canBack() {\n    return this.manager?.canUndo()\n  }\n\n  get canForward() {\n    return this.manager?.canRedo()\n  }\n\n  private backEvent = new Subject<void>()\n  private forwardEvent = new Subject<void>()\n  private changeEvent = new Subject<void>()\n  private pushEvent = new Subject<void>()\n\n  private manager!: UndoManager\n\n  private subscriptions: Subscription[] = []\n  private updateFromRemote = false\n\n  private contentSyncCaches = new WeakMap<Slot, () => void>()\n  private slotStateSyncCaches = new WeakMap<Slot, () => void>()\n  private slotsSyncCaches = new WeakMap<ComponentInstance, () => void>()\n  private componentStateSyncCaches = new WeakMap<ComponentInstance, () => void>()\n\n  private selectionChangeEvent = new Subject<SelectionPaths>()\n\n  private updateRemoteActions: Array<() => void> = []\n\n  constructor(private rootComponentRef: RootComponentRef,\n              private collaborateCursor: CollaborateCursor,\n              private translator: Translator,\n              private renderer: Renderer,\n              private registry: Registry,\n              private selection: Selection,\n              private starter: Starter) {\n    this.onSelectionChange = this.selectionChangeEvent.asObservable()\n    this.onBack = this.backEvent.asObservable()\n    this.onForward = this.forwardEvent.asObservable()\n    this.onChange = this.changeEvent.asObservable()\n    this.onPush = this.pushEvent.asObservable()\n  }\n\n  setup() {\n    this.subscriptions.push(\n      this.starter.onReady.subscribe(() => {\n        this.listen2()\n      }),\n      this.selection.onChange.subscribe(() => {\n        const paths = this.selection.getPaths()\n        this.selectionChangeEvent.next(paths)\n      })\n    )\n  }\n\n  updateRemoteSelection(paths: RemoteSelection[]) {\n    this.collaborateCursor.draw(paths)\n  }\n\n  listen() {\n    //\n  }\n\n  back() {\n    if (this.canBack) {\n      this.manager.undo()\n    }\n  }\n\n  forward() {\n    if (this.canForward) {\n      this.manager.redo()\n    }\n  }\n\n  destroy() {\n    this.subscriptions.forEach(i => i.unsubscribe())\n  }\n\n  private listen2() {\n    const root = this.yDoc.getText('content')\n    const rootComponent = this.rootComponentRef.component!\n    this.manager = new UndoManager(root, {\n      trackedOrigins: new Set<any>([this.yDoc])\n    })\n    this.syncContent(root, rootComponent.slots.get(0)!)\n\n    this.subscriptions.push(\n      merge(\n        rootComponent.changeMarker.onForceChange,\n        rootComponent.changeMarker.onChange\n      ).pipe(\n        microTask()\n      ).subscribe(() => {\n        this.yDoc.transact(() => {\n          this.updateRemoteActions.forEach(fn => {\n            fn()\n          })\n          this.updateRemoteActions = []\n        }, this.yDoc)\n        this.renderer.render()\n        this.selection.restore()\n      })\n    )\n  }\n\n  private syncContent(content: YText, slot: Slot) {\n    const syncRemote = (ev, tr) => {\n      this.runRemoteUpdate(tr, () => {\n        slot.retain(0)\n        ev.delta.forEach(action => {\n          if (Reflect.has(action, 'retain')) {\n            if (action.attributes) {\n              const formats = makeFormats(this.registry, action.attributes)\n              if (formats.length) {\n                slot.retain(action.retain!, formats)\n              }\n            } else {\n              slot.retain(action.retain)\n            }\n          } else if (action.insert) {\n            const index = slot.index\n            let length = 1\n            if (typeof action.insert === 'string') {\n              length = action.insert.length\n              slot.insert(action.insert, makeFormats(this.registry, action.attributes))\n            } else {\n              const sharedComponent = action.insert as YMap<any>\n              const component = this.createComponentBySharedComponent(sharedComponent)\n              this.syncSlots(sharedComponent.get('slots'), component)\n              this.syncComponent(sharedComponent, component)\n              slot.insert(component)\n            }\n            if (this.selection.isSelected) {\n              if (slot === this.selection.startSlot && this.selection.startOffset! >= index) {\n                this.selection.setStart(slot, this.selection.startOffset! + length)\n              }\n              if (slot === this.selection.endSlot && this.selection.endOffset! >= index) {\n                this.selection.setEnd(slot, this.selection.endOffset! + length)\n              }\n            }\n          } else if (action.delete) {\n            const index = slot.index\n            slot.retain(slot.index)\n            slot.delete(action.delete)\n            if (this.selection.isSelected) {\n              if (slot === this.selection.startSlot && this.selection.startOffset! >= index) {\n                this.selection.setStart(slot, this.selection.startOffset! - action.delete)\n              }\n              if (slot === this.selection.endSlot && this.selection.endOffset! >= index) {\n                this.selection.setEnd(slot, this.selection.endOffset! - action.delete)\n              }\n            }\n          } else if (action.attributes) {\n            slot.updateState(draft => {\n              Object.assign(draft, action.attributes)\n            })\n          }\n        })\n      })\n    }\n    content.observe(syncRemote)\n\n    const sub = slot.onContentChange.subscribe(actions => {\n      this.runLocalUpdate(() => {\n        let offset = 0\n        let length = 0\n        for (const action of actions) {\n          if (action.type === 'retain') {\n            const formats = action.formats\n            if (formats) {\n              const keys = Object.keys(formats)\n              let length = keys.length\n              keys.forEach(key => {\n                if (!this.registry.getFormatter(key)) {\n                  length--\n                  Reflect.deleteProperty(formats, key)\n                }\n              })\n              if (length) {\n                content.format(offset, action.offset, formats)\n              }\n            } else {\n              offset = action.offset\n            }\n          } else if (action.type === 'insert') {\n            const delta = content.toDelta()\n            const isEmpty = delta.length === 1 && delta[0].insert === Slot.emptyPlaceholder\n            if (typeof action.content === 'string') {\n              length = action.content.length\n              content.insert(offset, action.content)\n            } else {\n              length = 1\n              const component = slot.getContentAtIndex(offset) as ComponentInstance\n              const sharedComponent = this.createSharedComponentByComponent(component)\n              content.insertEmbed(offset, sharedComponent)\n            }\n            if (action.formats) {\n              content.format(offset, length, action.formats)\n            }\n            if (isEmpty && offset === 0) {\n              content.delete(content.length - 1, 1)\n            }\n            offset += length\n          } else if (action.type === 'delete') {\n            const delta = content.toDelta()\n            content.delete(offset, action.count)\n            if (content.length === 0) {\n              content.insert(0, '\\n', delta[0]?.attributes)\n            }\n          }\n        }\n      })\n    })\n    this.contentSyncCaches.set(slot, () => {\n      content.unobserve(syncRemote)\n      sub.unsubscribe()\n    })\n  }\n\n  private syncSlot(remoteSlot: YMap<any>, slot: Slot) {\n    const syncRemote = (ev, tr) => {\n      this.runRemoteUpdate(tr, () => {\n        ev.keysChanged.forEach(key => {\n          if (key === 'state') {\n            const state = (ev.target as YMap<any>).get('state')\n            slot.updateState(draft => {\n              Object.assign(draft, state)\n            })\n          }\n        })\n      })\n    }\n    remoteSlot.observe(syncRemote)\n\n    const sub = slot.onStateChange.subscribe(actions => {\n      this.runLocalUpdate(() => {\n        actions.forEach(action => {\n          remoteSlot.set('state', action.value)\n        })\n      })\n    })\n    this.slotStateSyncCaches.set(slot, () => {\n      remoteSlot.unobserve(syncRemote)\n      sub.unsubscribe()\n    })\n  }\n\n  private syncSlots(remoteSlots: YArray<any>, component: ComponentInstance) {\n    const slots = component.slots\n    const syncRemote = (ev, tr) => {\n      this.runRemoteUpdate(tr, () => {\n        ev.delta.forEach(action => {\n          if (Reflect.has(action, 'retain')) {\n            slots.retain(action.retain!)\n          } else if (action.insert) {\n            (action.insert as Array<YMap<any>>).forEach(item => {\n              const slot = this.createSlotBySharedSlot(item)\n              slots.insert(slot)\n              this.syncContent(item.get('content'), slot)\n              this.syncSlot(item, slot)\n            })\n          } else if (action.delete) {\n            slots.retain(slots.index)\n            slots.delete(action.delete)\n          }\n        })\n      })\n    }\n    remoteSlots.observe(syncRemote)\n\n    const sub = slots.onChange.subscribe(operations => {\n      this.runLocalUpdate(() => {\n        const applyActions = operations.apply\n        let index: number\n        applyActions.forEach(action => {\n          if (action.type === 'retain') {\n            index = action.offset\n          } else if (action.type === 'insertSlot') {\n            const slot = slots.get(index)!\n            const sharedSlot = this.createSharedSlotBySlot(slot)\n            remoteSlots.insert(index, [sharedSlot])\n            index++\n          } else if (action.type === 'delete') {\n            slots.slice(index, index + action.count).forEach(slot => {\n              this.cleanSubscriptionsBySlot(slot)\n            })\n            remoteSlots.delete(index, action.count)\n          }\n        })\n      })\n    })\n\n    this.slotsSyncCaches.set(component, () => {\n      remoteSlots.unobserve(syncRemote)\n      sub.unsubscribe()\n    })\n  }\n\n  private syncComponent(remoteComponent: YMap<any>, component: ComponentInstance) {\n    const syncRemote = (ev, tr) => {\n      this.runRemoteUpdate(tr, () => {\n        ev.keysChanged.forEach(key => {\n          if (key === 'state') {\n            const state = (ev.target as YMap<any>).get('state')\n            component.updateState(draft => {\n              Object.assign(draft, state)\n            })\n          }\n        })\n      })\n    }\n    remoteComponent.observe(syncRemote)\n\n    const sub = component.onStateChange.subscribe(newState => {\n      this.runLocalUpdate(() => {\n        remoteComponent.set('state', newState)\n      })\n    })\n    this.componentStateSyncCaches.set(component, () => {\n      remoteComponent.unobserve(syncRemote)\n      sub.unsubscribe()\n    })\n  }\n\n  private runLocalUpdate(fn: () => void) {\n    if (this.updateFromRemote) {\n      return\n    }\n    this.updateRemoteActions.push(fn)\n  }\n\n  private runRemoteUpdate(tr: Transaction, fn: () => void) {\n    if (tr.origin === this.yDoc) {\n      return\n    }\n    this.updateFromRemote = true\n    fn()\n    this.updateFromRemote = false\n  }\n\n  private createSharedComponentByComponent(component: ComponentInstance): YMap<any> {\n    const sharedComponent = new YMap()\n    sharedComponent.set('state', component.state)\n    sharedComponent.set('name', component.name)\n    const sharedSlots = new YArray()\n    sharedComponent.set('slots', sharedSlots)\n    component.slots.toArray().forEach(slot => {\n      const sharedSlot = this.createSharedSlotBySlot(slot)\n      sharedSlots.push([sharedSlot])\n    })\n    this.syncSlots(sharedSlots, component)\n    this.syncComponent(sharedComponent, component)\n    return sharedComponent\n  }\n\n  private createSharedSlotBySlot(slot: Slot): YMap<any> {\n    const sharedSlot = new YMap()\n    sharedSlot.set('schema', slot.schema)\n    sharedSlot.set('state', slot.state)\n    const sharedContent = new YText()\n    sharedSlot.set('content', sharedContent)\n    let offset = 0\n    slot.toDelta().forEach(i => {\n      let formats: any = {}\n      if (i.formats) {\n        i.formats.forEach(item => {\n          formats[item[0].name] = item[1]\n        })\n      } else {\n        formats = null\n      }\n      if (typeof i.insert === 'string') {\n        sharedContent.insert(offset, i.insert, formats)\n      } else {\n        const sharedComponent = this.createSharedComponentByComponent(i.insert)\n        sharedContent.insertEmbed(offset, sharedComponent, formats)\n      }\n      offset += i.insert.length\n    })\n    this.syncContent(sharedContent, slot)\n    this.syncSlot(sharedSlot, slot)\n    return sharedSlot\n  }\n\n  private createComponentBySharedComponent(yMap: YMap<any>): ComponentInstance {\n    const sharedSlots = yMap.get('slots') as YArray<YMap<any>>\n    const slots: Slot[] = []\n    sharedSlots.forEach(sharedSlot => {\n      const slot = this.createSlotBySharedSlot(sharedSlot)\n      slots.push(slot)\n    })\n    const name = yMap.get('name')\n    const instance = this.translator.createComponentByData(name, {\n      state: yMap.get('state'),\n      slots\n    })\n    if (instance) {\n      instance.slots.toArray().forEach((slot, index) => {\n        const sharedSlot = sharedSlots.get(index)\n        this.syncSlot(sharedSlot, slot)\n        this.syncContent(sharedSlot.get('content'), slot)\n      })\n      return instance\n    }\n    throw collaborateErrorFn(`cannot find component factory \\`${name}\\`.`)\n  }\n\n  private createSlotBySharedSlot(sharedSlot: YMap<any>): Slot {\n    const content = sharedSlot.get('content') as YText\n    const delta = content.toDelta()\n\n    const slot = this.translator.createSlot({\n      schema: sharedSlot.get('schema'),\n      state: sharedSlot.get('state'),\n      formats: {},\n      content: []\n    })\n\n    for (const action of delta) {\n      if (action.insert) {\n        if (typeof action.insert === 'string') {\n          slot.insert(action.insert, makeFormats(this.registry, action.attributes))\n        } else {\n          const sharedComponent = action.insert as YMap<any>\n          const component = this.createComponentBySharedComponent(sharedComponent)\n          slot.insert(component)\n          this.syncSlots(sharedComponent.get('slots'), component)\n          this.syncComponent(sharedComponent, component)\n        }\n      } else {\n        throw collaborateErrorFn('unexpected delta action.')\n      }\n    }\n    return slot\n  }\n\n  private cleanSubscriptionsBySlot(slot: Slot) {\n    [this.contentSyncCaches.get(slot), this.slotStateSyncCaches.get(slot)].forEach(fn => {\n      if (fn) {\n        fn()\n      }\n    })\n    slot.sliceContent().forEach(i => {\n      if (typeof i !== 'string') {\n        this.cleanSubscriptionsByComponent(i)\n      }\n    })\n  }\n\n  private cleanSubscriptionsByComponent(component: ComponentInstance) {\n    [this.slotsSyncCaches.get(component), this.componentStateSyncCaches.get(component)].forEach(fn => {\n      if (fn) {\n        fn()\n      }\n    })\n    component.slots.toArray().forEach(slot => {\n      this.cleanSubscriptionsBySlot(slot)\n    })\n  }\n}\n\nfunction makeFormats(registry: Registry, attrs?: any) {\n  const formats: Formats = []\n  if (attrs) {\n    Object.keys(attrs).map(key => {\n      const formatter = registry.getFormatter(key)\n      if (formatter) {\n        formats.push([formatter, attrs[key]])\n      }\n    })\n  }\n  return formats\n}\n"]}
|
package/bundles/public-api.d.ts
CHANGED
package/bundles/public-api.js
CHANGED
@@ -1,3 +1,3 @@
|
|
1
|
-
export * from './collab/_api';
|
2
1
|
export * from './collaborate';
|
3
|
-
|
2
|
+
export * from './collaborate-cursor';
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsZUFBZSxDQUFBO0FBQzdCLGNBQWMsc0JBQXNCLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2NvbGxhYm9yYXRlJ1xuZXhwb3J0ICogZnJvbSAnLi9jb2xsYWJvcmF0ZS1jdXJzb3InXG4iXX0=
|
package/package.json
CHANGED
@@ -1,6 +1,6 @@
|
|
1
1
|
{
|
2
2
|
"name": "@textbus/collaborate",
|
3
|
-
"version": "2.0.0-beta.
|
3
|
+
"version": "2.0.0-beta.5",
|
4
4
|
"description": "Textbus is a rich text editor and framework that is highly customizable and extensible to achieve rich wysiwyg effects.",
|
5
5
|
"main": "./bundles/public-api.js",
|
6
6
|
"module": "./bundles/public-api.js",
|
@@ -27,7 +27,7 @@
|
|
27
27
|
"dependencies": {
|
28
28
|
"@tanbo/di": "^1.1.0",
|
29
29
|
"@tanbo/stream": "^1.0.0",
|
30
|
-
"@textbus/browser": "^2.0.0-beta.
|
30
|
+
"@textbus/browser": "^2.0.0-beta.5",
|
31
31
|
"@textbus/core": "^2.0.0-beta.4",
|
32
32
|
"reflect-metadata": "^0.1.13",
|
33
33
|
"y-protocols": "^1.0.5",
|
@@ -44,5 +44,5 @@
|
|
44
44
|
"bugs": {
|
45
45
|
"url": "https://github.com/textbus/textbus.git/issues"
|
46
46
|
},
|
47
|
-
"gitHead": "
|
47
|
+
"gitHead": "c2d4326a24e8e2acf969737b315cfef462b16b2f"
|
48
48
|
}
|
File without changes
|
package/src/collaborate.ts
CHANGED
package/src/public-api.ts
CHANGED
package/bundles/collab/_api.d.ts
DELETED
@@ -1 +0,0 @@
|
|
1
|
-
export * from './collaborate-cursor';
|
package/bundles/collab/_api.js
DELETED
@@ -1,2 +0,0 @@
|
|
1
|
-
export * from './collaborate-cursor';
|
2
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiX2FwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb2xsYWIvX2FwaS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLHNCQUFzQixDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9jb2xsYWJvcmF0ZS1jdXJzb3InXG4iXX0=
|
@@ -1,245 +0,0 @@
|
|
1
|
-
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
|
2
|
-
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
3
|
-
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
4
|
-
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
|
5
|
-
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
6
|
-
};
|
7
|
-
var __metadata = (this && this.__metadata) || function (k, v) {
|
8
|
-
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
|
9
|
-
};
|
10
|
-
var __param = (this && this.__param) || function (paramIndex, decorator) {
|
11
|
-
return function (target, key) { decorator(target, key, paramIndex); }
|
12
|
-
};
|
13
|
-
import { Inject, Injectable } from '@tanbo/di';
|
14
|
-
import { createElement, EDITABLE_DOCUMENT, EDITOR_CONTAINER, getLayoutRectByRange, SelectionBridge } from '@textbus/browser';
|
15
|
-
import { Selection } from '@textbus/core';
|
16
|
-
import { Subject } from '@tanbo/stream';
|
17
|
-
let CollaborateCursor = class CollaborateCursor {
|
18
|
-
constructor(container, document, nativeSelection, selection) {
|
19
|
-
Object.defineProperty(this, "container", {
|
20
|
-
enumerable: true,
|
21
|
-
configurable: true,
|
22
|
-
writable: true,
|
23
|
-
value: container
|
24
|
-
});
|
25
|
-
Object.defineProperty(this, "document", {
|
26
|
-
enumerable: true,
|
27
|
-
configurable: true,
|
28
|
-
writable: true,
|
29
|
-
value: document
|
30
|
-
});
|
31
|
-
Object.defineProperty(this, "nativeSelection", {
|
32
|
-
enumerable: true,
|
33
|
-
configurable: true,
|
34
|
-
writable: true,
|
35
|
-
value: nativeSelection
|
36
|
-
});
|
37
|
-
Object.defineProperty(this, "selection", {
|
38
|
-
enumerable: true,
|
39
|
-
configurable: true,
|
40
|
-
writable: true,
|
41
|
-
value: selection
|
42
|
-
});
|
43
|
-
Object.defineProperty(this, "canvas", {
|
44
|
-
enumerable: true,
|
45
|
-
configurable: true,
|
46
|
-
writable: true,
|
47
|
-
value: createElement('canvas', {
|
48
|
-
styles: {
|
49
|
-
position: 'absolute',
|
50
|
-
opacity: 0.5,
|
51
|
-
left: 0,
|
52
|
-
top: 0,
|
53
|
-
width: '100%',
|
54
|
-
height: '100%',
|
55
|
-
pointerEvents: 'none'
|
56
|
-
}
|
57
|
-
})
|
58
|
-
});
|
59
|
-
Object.defineProperty(this, "context", {
|
60
|
-
enumerable: true,
|
61
|
-
configurable: true,
|
62
|
-
writable: true,
|
63
|
-
value: this.canvas.getContext('2d')
|
64
|
-
});
|
65
|
-
Object.defineProperty(this, "tooltips", {
|
66
|
-
enumerable: true,
|
67
|
-
configurable: true,
|
68
|
-
writable: true,
|
69
|
-
value: createElement('div', {
|
70
|
-
styles: {
|
71
|
-
position: 'absolute',
|
72
|
-
left: 0,
|
73
|
-
top: 0,
|
74
|
-
width: '100%',
|
75
|
-
height: '100%',
|
76
|
-
pointerEvents: 'none',
|
77
|
-
fontSize: '12px',
|
78
|
-
zIndex: 10
|
79
|
-
}
|
80
|
-
})
|
81
|
-
});
|
82
|
-
Object.defineProperty(this, "onRectsChange", {
|
83
|
-
enumerable: true,
|
84
|
-
configurable: true,
|
85
|
-
writable: true,
|
86
|
-
value: new Subject()
|
87
|
-
});
|
88
|
-
container.prepend(this.canvas, this.tooltips);
|
89
|
-
this.onRectsChange.subscribe(rects => {
|
90
|
-
for (const rect of rects) {
|
91
|
-
this.context.fillStyle = rect.color;
|
92
|
-
this.context.beginPath();
|
93
|
-
this.context.rect(rect.x, rect.y, rect.width, rect.height);
|
94
|
-
this.context.fill();
|
95
|
-
this.context.closePath();
|
96
|
-
}
|
97
|
-
});
|
98
|
-
}
|
99
|
-
draw(paths) {
|
100
|
-
const containerRect = this.container.getBoundingClientRect();
|
101
|
-
this.canvas.width = containerRect.width;
|
102
|
-
this.canvas.height = containerRect.height;
|
103
|
-
this.context.clearRect(0, 0, this.canvas.width, this.canvas.height);
|
104
|
-
const users = [];
|
105
|
-
paths.filter(i => {
|
106
|
-
return i.paths.start.length && i.paths.end.length;
|
107
|
-
}).forEach(item => {
|
108
|
-
const startOffset = item.paths.start.pop();
|
109
|
-
const startSlot = this.selection.findSlotByPaths(item.paths.start);
|
110
|
-
const endOffset = item.paths.end.pop();
|
111
|
-
const endSlot = this.selection.findSlotByPaths(item.paths.end);
|
112
|
-
if (startSlot && endSlot) {
|
113
|
-
const position = this.nativeSelection.getPositionByRange({
|
114
|
-
startOffset,
|
115
|
-
endOffset,
|
116
|
-
startSlot,
|
117
|
-
endSlot
|
118
|
-
});
|
119
|
-
if (position.start && position.end) {
|
120
|
-
const nativeRange = this.document.createRange();
|
121
|
-
nativeRange.setStart(position.start.node, position.start.offset);
|
122
|
-
nativeRange.setEnd(position.end.node, position.end.offset);
|
123
|
-
const rects = nativeRange.getClientRects();
|
124
|
-
const selectionRects = [];
|
125
|
-
for (let i = rects.length - 1; i >= 0; i--) {
|
126
|
-
const rect = rects[i];
|
127
|
-
selectionRects.push({
|
128
|
-
color: item.color,
|
129
|
-
username: item.username,
|
130
|
-
x: rect.x - containerRect.x,
|
131
|
-
y: rect.y - containerRect.y,
|
132
|
-
width: rect.width,
|
133
|
-
height: rect.height,
|
134
|
-
});
|
135
|
-
}
|
136
|
-
this.onRectsChange.next(selectionRects);
|
137
|
-
const cursorRange = nativeRange.cloneRange();
|
138
|
-
cursorRange.collapse(!item.paths.focusEnd);
|
139
|
-
const cursorRect = getLayoutRectByRange(cursorRange);
|
140
|
-
users.push({
|
141
|
-
username: item.username,
|
142
|
-
color: item.color,
|
143
|
-
x: cursorRect.x - containerRect.x,
|
144
|
-
y: cursorRect.y - containerRect.y,
|
145
|
-
width: 2,
|
146
|
-
height: cursorRect.height
|
147
|
-
});
|
148
|
-
}
|
149
|
-
}
|
150
|
-
});
|
151
|
-
this.drawUserCursor(users);
|
152
|
-
}
|
153
|
-
drawUserCursor(rects) {
|
154
|
-
for (let i = 0; i < rects.length; i++) {
|
155
|
-
const rect = rects[i];
|
156
|
-
const { cursor, userTip, anchor } = this.getUserCursor(i);
|
157
|
-
Object.assign(cursor.style, {
|
158
|
-
left: rect.x + 'px',
|
159
|
-
top: rect.y + 'px',
|
160
|
-
width: rect.width + 'px',
|
161
|
-
height: rect.height + 'px',
|
162
|
-
background: rect.color,
|
163
|
-
display: 'block'
|
164
|
-
});
|
165
|
-
anchor.style.background = rect.color;
|
166
|
-
userTip.innerText = rect.username;
|
167
|
-
userTip.style.background = rect.color;
|
168
|
-
}
|
169
|
-
for (let i = rects.length; i < this.tooltips.children.length; i++) {
|
170
|
-
this.tooltips.removeChild(this.tooltips.children[i]);
|
171
|
-
}
|
172
|
-
}
|
173
|
-
getUserCursor(index) {
|
174
|
-
let child = this.tooltips.children[index];
|
175
|
-
if (child) {
|
176
|
-
const anchor = child.children[0];
|
177
|
-
return {
|
178
|
-
cursor: child,
|
179
|
-
anchor,
|
180
|
-
userTip: anchor.children[0]
|
181
|
-
};
|
182
|
-
}
|
183
|
-
const userTip = createElement('span', {
|
184
|
-
styles: {
|
185
|
-
position: 'absolute',
|
186
|
-
display: 'none',
|
187
|
-
left: '50%',
|
188
|
-
transform: 'translateX(-50%)',
|
189
|
-
marginBottom: '2px',
|
190
|
-
bottom: '100%',
|
191
|
-
whiteSpace: 'nowrap',
|
192
|
-
color: '#fff',
|
193
|
-
boxShadow: '0 1px 2px rgba(0,0,0,.1)',
|
194
|
-
borderRadius: '3px',
|
195
|
-
padding: '3px 5px',
|
196
|
-
pointerEvents: 'none',
|
197
|
-
}
|
198
|
-
});
|
199
|
-
const anchor = createElement('span', {
|
200
|
-
styles: {
|
201
|
-
position: 'absolute',
|
202
|
-
top: '-2px',
|
203
|
-
left: '-2px',
|
204
|
-
width: '6px',
|
205
|
-
height: '6px',
|
206
|
-
pointerEvents: 'auto',
|
207
|
-
pointer: 'cursor',
|
208
|
-
},
|
209
|
-
children: [userTip],
|
210
|
-
on: {
|
211
|
-
mouseenter() {
|
212
|
-
userTip.style.display = 'block';
|
213
|
-
},
|
214
|
-
mouseleave() {
|
215
|
-
userTip.style.display = 'none';
|
216
|
-
}
|
217
|
-
}
|
218
|
-
});
|
219
|
-
child = createElement('span', {
|
220
|
-
styles: {
|
221
|
-
position: 'absolute',
|
222
|
-
},
|
223
|
-
children: [
|
224
|
-
anchor
|
225
|
-
]
|
226
|
-
});
|
227
|
-
this.tooltips.append(child);
|
228
|
-
return {
|
229
|
-
cursor: child,
|
230
|
-
anchor,
|
231
|
-
userTip
|
232
|
-
};
|
233
|
-
}
|
234
|
-
};
|
235
|
-
CollaborateCursor = __decorate([
|
236
|
-
Injectable(),
|
237
|
-
__param(0, Inject(EDITOR_CONTAINER)),
|
238
|
-
__param(1, Inject(EDITABLE_DOCUMENT)),
|
239
|
-
__metadata("design:paramtypes", [HTMLElement,
|
240
|
-
Document,
|
241
|
-
SelectionBridge,
|
242
|
-
Selection])
|
243
|
-
], CollaborateCursor);
|
244
|
-
export { CollaborateCursor };
|
245
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"collaborate-cursor.js","sourceRoot":"","sources":["../../src/collab/collaborate-cursor.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable } from '@tanbo/di'\nimport {\n  createElement,\n  EDITABLE_DOCUMENT,\n  EDITOR_CONTAINER,\n  getLayoutRectByRange,\n  SelectionBridge\n} from '@textbus/browser'\nimport { Selection, SelectionPaths } from '@textbus/core'\nimport { Subject } from '@tanbo/stream'\n\nexport interface RemoteSelection {\n  color: string\n  username: string\n  paths: SelectionPaths\n}\n\nexport interface SelectionRect {\n  color: string\n  username: string\n  x: number\n  y: number\n  width: number\n  height: number\n}\n\nexport interface RemoteSelectionCursor {\n  cursor: HTMLElement\n  anchor: HTMLElement\n  userTip: HTMLElement\n}\n\n@Injectable()\nexport class CollaborateCursor {\n  private canvas = createElement('canvas', {\n    styles: {\n      position: 'absolute',\n      opacity: 0.5,\n      left: 0,\n      top: 0,\n      width: '100%',\n      height: '100%',\n      pointerEvents: 'none'\n    }\n  }) as HTMLCanvasElement\n  private context = this.canvas.getContext('2d')!\n  private tooltips = createElement('div', {\n    styles: {\n      position: 'absolute',\n      left: 0,\n      top: 0,\n      width: '100%',\n      height: '100%',\n      pointerEvents: 'none',\n      fontSize: '12px',\n      zIndex: 10\n    }\n  })\n\n  private onRectsChange = new Subject<SelectionRect[]>()\n\n  constructor(@Inject(EDITOR_CONTAINER) private container: HTMLElement,\n              @Inject(EDITABLE_DOCUMENT) private document: Document,\n              private nativeSelection: SelectionBridge,\n              private selection: Selection) {\n    container.prepend(this.canvas, this.tooltips)\n    this.onRectsChange.subscribe(rects => {\n      for (const rect of rects) {\n        this.context.fillStyle = rect.color\n        this.context.beginPath()\n        this.context.rect(rect.x, rect.y, rect.width, rect.height)\n        this.context.fill()\n        this.context.closePath()\n      }\n    })\n  }\n\n  draw(paths: RemoteSelection[]) {\n    const containerRect = this.container.getBoundingClientRect()\n    this.canvas.width = containerRect.width\n    this.canvas.height = containerRect.height\n    this.context.clearRect(0, 0, this.canvas.width, this.canvas.height)\n\n    const users: SelectionRect[] = []\n\n\n    paths.filter(i => {\n      return i.paths.start.length && i.paths.end.length\n    }).forEach(item => {\n      const startOffset = item.paths.start.pop()!\n      const startSlot = this.selection.findSlotByPaths(item.paths.start)\n      const endOffset = item.paths.end.pop()!\n      const endSlot = this.selection.findSlotByPaths(item.paths.end)\n\n      if (startSlot && endSlot) {\n        const position = this.nativeSelection.getPositionByRange({\n          startOffset,\n          endOffset,\n          startSlot,\n          endSlot\n        })\n        if (position.start && position.end) {\n          const nativeRange = this.document.createRange()\n          nativeRange.setStart(position.start.node, position.start.offset)\n          nativeRange.setEnd(position.end.node, position.end.offset)\n\n          const rects = nativeRange.getClientRects()\n          const selectionRects: SelectionRect[] = []\n          for (let i = rects.length - 1; i >= 0; i--) {\n            const rect = rects[i]\n            selectionRects.push({\n              color: item.color,\n              username: item.username,\n              x: rect.x - containerRect.x,\n              y: rect.y - containerRect.y,\n              width: rect.width,\n              height: rect.height,\n            })\n          }\n          this.onRectsChange.next(selectionRects)\n\n          const cursorRange = nativeRange.cloneRange()\n          cursorRange.collapse(!item.paths.focusEnd)\n\n          const cursorRect = getLayoutRectByRange(cursorRange)\n\n          users.push({\n            username: item.username,\n            color: item.color,\n            x: cursorRect.x - containerRect.x,\n            y: cursorRect.y - containerRect.y,\n            width: 2,\n            height: cursorRect.height\n          })\n        }\n      }\n    })\n    this.drawUserCursor(users)\n  }\n\n  private drawUserCursor(rects: SelectionRect[]) {\n    for (let i = 0; i < rects.length; i++) {\n      const rect = rects[i]\n      const {cursor, userTip, anchor} = this.getUserCursor(i)\n      Object.assign(cursor.style, {\n        left: rect.x + 'px',\n        top: rect.y + 'px',\n        width: rect.width + 'px',\n        height: rect.height + 'px',\n        background: rect.color,\n        display: 'block'\n      })\n      anchor.style.background = rect.color\n      userTip.innerText = rect.username\n      userTip.style.background = rect.color\n    }\n\n    for (let i = rects.length; i < this.tooltips.children.length; i++) {\n      this.tooltips.removeChild(this.tooltips.children[i])\n    }\n  }\n\n  private getUserCursor(index: number): RemoteSelectionCursor {\n    let child: HTMLElement = this.tooltips.children[index] as HTMLElement\n    if (child) {\n      const anchor = child.children[0] as HTMLElement\n      return {\n        cursor: child,\n        anchor,\n        userTip: anchor.children[0] as HTMLElement\n      }\n    }\n    const userTip = createElement('span', {\n      styles: {\n        position: 'absolute',\n        display: 'none',\n        left: '50%',\n        transform: 'translateX(-50%)',\n        marginBottom: '2px',\n        bottom: '100%',\n        whiteSpace: 'nowrap',\n        color: '#fff',\n        boxShadow: '0 1px 2px rgba(0,0,0,.1)',\n        borderRadius: '3px',\n        padding: '3px 5px',\n        pointerEvents: 'none',\n      }\n    })\n\n    const anchor = createElement('span', {\n      styles: {\n        position: 'absolute',\n        top: '-2px',\n        left: '-2px',\n        width: '6px',\n        height: '6px',\n        pointerEvents: 'auto',\n        pointer: 'cursor',\n      },\n      children: [userTip],\n      on: {\n        mouseenter() {\n          userTip.style.display = 'block'\n        },\n        mouseleave() {\n          userTip.style.display = 'none'\n        }\n      }\n    })\n    child = createElement('span', {\n      styles: {\n        position: 'absolute',\n      },\n      children: [\n        anchor\n      ]\n    })\n    this.tooltips.append(child)\n    return {\n      cursor: child,\n      anchor,\n      userTip\n    }\n  }\n}\n"]}
|
package/src/collab/_api.ts
DELETED
@@ -1 +0,0 @@
|
|
1
|
-
export * from './collaborate-cursor'
|