@textbus/collaborate 3.0.0-alpha.4 → 3.0.0-alpha.44

Sign up to get free protection for your applications and to get access to all the features.
@@ -1,4 +1,3 @@
1
- import { Module } from '@textbus/core';
2
- export * from './collaborate';
3
- export * from './collaborate-cursor';
4
- export declare const collaborateModule: Module;
1
+ import { Module } from '@textbus/core';
2
+ export * from './collaborate';
3
+ export declare const collaborateModule: Module;
package/package.json CHANGED
@@ -1,15 +1,15 @@
1
1
  {
2
2
  "name": "@textbus/collaborate",
3
- "version": "3.0.0-alpha.4",
3
+ "version": "3.0.0-alpha.44",
4
4
  "description": "Textbus is a rich text editor and framework that is highly customizable and extensible to achieve rich wysiwyg effects.",
5
- "main": "./bundles/public-api.js",
6
- "module": "./bundles/public-api.js",
5
+ "main": "./bundles/index.js",
6
+ "module": "./bundles/index.esm.js",
7
7
  "typings": "./bundles/public-api.d.ts",
8
8
  "scripts": {
9
9
  "start": "webpack-dev-server",
10
10
  "test": "cross-env env=test jest",
11
11
  "test-c": "cross-env env=test jest --coverage",
12
- "build:lib": "rimraf bundles && tsc -p tsconfig-build.json",
12
+ "build:lib": "rimraf bundles && rollup --config rollup.config.ts --configPlugin @rollup/plugin-typescript",
13
13
  "publish:lib": "npm run build:lib && npm publish --access=public"
14
14
  },
15
15
  "license": "GPL-3.0-only",
@@ -25,14 +25,20 @@
25
25
  "typescript editor"
26
26
  ],
27
27
  "dependencies": {
28
- "@tanbo/di": "^1.1.3",
28
+ "@tanbo/di": "^1.1.4",
29
29
  "@tanbo/stream": "^1.1.8",
30
- "@textbus/browser": "^3.0.0-alpha.4",
31
- "@textbus/core": "^3.0.0-alpha.3",
30
+ "@textbus/core": "^3.0.0-alpha.44",
32
31
  "reflect-metadata": "^0.1.13",
33
32
  "y-protocols": "^1.0.5",
34
33
  "yjs": "^13.5.39"
35
34
  },
35
+ "devDependencies": {
36
+ "@rollup/plugin-commonjs": "^23.0.2",
37
+ "@rollup/plugin-typescript": "^9.0.2",
38
+ "rimraf": "^3.0.2",
39
+ "rollup": "^3.2.5",
40
+ "tslib": "^2.4.1"
41
+ },
36
42
  "author": {
37
43
  "name": "Tanbo",
38
44
  "email": "tanbohb@qq.com"
@@ -44,5 +50,5 @@
44
50
  "bugs": {
45
51
  "url": "https://github.com/textbus/textbus.git/issues"
46
52
  },
47
- "gitHead": "95e61178adaf447e8f55cf6c8b241c02a89f1db3"
53
+ "gitHead": "3abc943158b03b675002df947d95ce14eee9b31d"
48
54
  }
@@ -1,42 +0,0 @@
1
- import { SelectionBridge } from '@textbus/browser';
2
- import { Selection, SelectionPaths, AbstractSelection, Scheduler, Rect } from '@textbus/core';
3
- export interface RemoteSelection {
4
- id: string;
5
- color: string;
6
- username: string;
7
- paths: SelectionPaths;
8
- }
9
- export interface SelectionRect extends Rect {
10
- color: string;
11
- username: string;
12
- id: string;
13
- }
14
- export interface RemoteSelectionCursor {
15
- cursor: HTMLElement;
16
- anchor: HTMLElement;
17
- userTip: HTMLElement;
18
- }
19
- export declare abstract class CollaborateSelectionAwarenessDelegate {
20
- abstract getRects(abstractSelection: AbstractSelection, nativeRange: Range): false | Rect[];
21
- }
22
- export declare class CollaborateCursor {
23
- private container;
24
- private awarenessDelegate;
25
- private nativeSelection;
26
- private scheduler;
27
- private selection;
28
- private host;
29
- private canvasContainer;
30
- private canvas;
31
- private context;
32
- private tooltips;
33
- private onRectsChange;
34
- private subscription;
35
- private currentSelection;
36
- constructor(container: HTMLElement, awarenessDelegate: CollaborateSelectionAwarenessDelegate, nativeSelection: SelectionBridge, scheduler: Scheduler, selection: Selection);
37
- refresh(): void;
38
- destroy(): void;
39
- draw(paths: RemoteSelection[]): void;
40
- protected drawUserCursor(rects: SelectionRect[]): void;
41
- private getUserCursor;
42
- }
@@ -1,273 +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, Optional } from '@tanbo/di';
14
- import { createElement, VIEW_CONTAINER, getLayoutRectByRange, SelectionBridge } from '@textbus/browser';
15
- import { Selection, Scheduler } from '@textbus/core';
16
- import { fromEvent, Subject, Subscription } from '@tanbo/stream';
17
- export class CollaborateSelectionAwarenessDelegate {
18
- }
19
- let CollaborateCursor = class CollaborateCursor {
20
- constructor(container, awarenessDelegate, nativeSelection, scheduler, selection) {
21
- this.container = container;
22
- this.awarenessDelegate = awarenessDelegate;
23
- this.nativeSelection = nativeSelection;
24
- this.scheduler = scheduler;
25
- this.selection = selection;
26
- this.host = createElement('div', {
27
- styles: {
28
- position: 'absolute',
29
- left: 0,
30
- top: 0,
31
- width: '100%',
32
- height: '100%',
33
- pointerEvents: 'none',
34
- zIndex: 1
35
- }
36
- });
37
- this.canvasContainer = createElement('div', {
38
- styles: {
39
- position: 'absolute',
40
- left: 0,
41
- top: 0,
42
- width: '100%',
43
- height: '100%',
44
- overflow: 'hidden'
45
- }
46
- });
47
- this.canvas = createElement('canvas', {
48
- styles: {
49
- position: 'absolute',
50
- opacity: 0.5,
51
- left: 0,
52
- top: 0,
53
- width: '100%',
54
- height: document.documentElement.clientHeight + 'px',
55
- pointerEvents: 'none',
56
- }
57
- });
58
- this.context = this.canvas.getContext('2d');
59
- this.tooltips = createElement('div', {
60
- styles: {
61
- position: 'absolute',
62
- left: 0,
63
- top: 0,
64
- width: '100%',
65
- height: '100%',
66
- pointerEvents: 'none',
67
- fontSize: '12px',
68
- zIndex: 10
69
- }
70
- });
71
- this.onRectsChange = new Subject();
72
- this.subscription = new Subscription();
73
- this.currentSelection = [];
74
- this.canvasContainer.append(this.canvas);
75
- this.host.append(this.canvasContainer, this.tooltips);
76
- container.prepend(this.host);
77
- this.subscription.add(this.onRectsChange.subscribe(rects => {
78
- for (const rect of rects) {
79
- this.context.fillStyle = rect.color;
80
- this.context.beginPath();
81
- this.context.rect(rect.left, rect.top, rect.width, rect.height);
82
- this.context.fill();
83
- this.context.closePath();
84
- }
85
- }), fromEvent(window, 'resize').subscribe(() => {
86
- this.canvas.style.height = document.documentElement.clientHeight + 'px';
87
- this.refresh();
88
- }), this.scheduler.onDocChanged.subscribe(() => {
89
- this.refresh();
90
- }));
91
- }
92
- refresh() {
93
- this.draw(this.currentSelection);
94
- }
95
- destroy() {
96
- this.subscription.unsubscribe();
97
- }
98
- draw(paths) {
99
- this.currentSelection = paths;
100
- const containerRect = this.container.getBoundingClientRect();
101
- this.canvas.style.top = containerRect.top * -1 + 'px';
102
- this.canvas.width = this.canvas.offsetWidth;
103
- this.canvas.height = this.canvas.offsetHeight;
104
- this.context.clearRect(0, 0, this.canvas.width, this.canvas.height);
105
- const users = [];
106
- paths.filter(i => {
107
- return i.paths.anchor.length && i.paths.focus.length;
108
- }).forEach(item => {
109
- const anchorPaths = [...item.paths.anchor];
110
- const focusPaths = [...item.paths.focus];
111
- const anchorOffset = anchorPaths.pop();
112
- const anchorSlot = this.selection.findSlotByPaths(anchorPaths);
113
- const focusOffset = focusPaths.pop();
114
- const focusSlot = this.selection.findSlotByPaths(focusPaths);
115
- if (!anchorSlot || !focusSlot) {
116
- return;
117
- }
118
- const { focus, anchor } = this.nativeSelection.getPositionByRange({
119
- focusOffset,
120
- anchorOffset,
121
- focusSlot,
122
- anchorSlot
123
- });
124
- if (!focus || !anchor) {
125
- return;
126
- }
127
- const nativeRange = document.createRange();
128
- nativeRange.setStart(anchor.node, anchor.offset);
129
- nativeRange.setEnd(focus.node, focus.offset);
130
- if ((anchor.node !== focus.node || anchor.offset !== focus.offset) && nativeRange.collapsed) {
131
- nativeRange.setStart(focus.node, focus.offset);
132
- nativeRange.setEnd(anchor.node, anchor.offset);
133
- }
134
- let rects = false;
135
- if (this.awarenessDelegate) {
136
- rects = this.awarenessDelegate.getRects({
137
- focusOffset,
138
- anchorOffset,
139
- focusSlot,
140
- anchorSlot
141
- }, nativeRange);
142
- }
143
- if (!rects) {
144
- rects = nativeRange.getClientRects();
145
- }
146
- const selectionRects = [];
147
- for (let i = rects.length - 1; i >= 0; i--) {
148
- const rect = rects[i];
149
- selectionRects.push({
150
- id: item.id,
151
- color: item.color,
152
- username: item.username,
153
- left: rect.left - containerRect.left,
154
- top: rect.top,
155
- width: rect.width,
156
- height: rect.height,
157
- });
158
- }
159
- this.onRectsChange.next(selectionRects);
160
- const cursorRange = nativeRange.cloneRange();
161
- cursorRange.setStart(focus.node, focus.offset);
162
- cursorRange.collapse(true);
163
- const cursorRect = getLayoutRectByRange(cursorRange);
164
- const rect = {
165
- id: item.id,
166
- username: item.username,
167
- color: item.color,
168
- left: cursorRect.left - containerRect.left,
169
- top: cursorRect.top - containerRect.top,
170
- width: 2,
171
- height: cursorRect.height
172
- };
173
- if (rect.left < 0 || rect.top < 0 || rect.left > containerRect.width) {
174
- return;
175
- }
176
- users.push(rect);
177
- });
178
- this.drawUserCursor(users);
179
- }
180
- drawUserCursor(rects) {
181
- for (let i = 0; i < rects.length; i++) {
182
- const rect = rects[i];
183
- const { cursor, userTip, anchor } = this.getUserCursor(i);
184
- Object.assign(cursor.style, {
185
- left: rect.left + 'px',
186
- top: rect.top + 'px',
187
- width: rect.width + 'px',
188
- height: rect.height + 'px',
189
- background: rect.color,
190
- display: 'block'
191
- });
192
- anchor.style.background = rect.color;
193
- userTip.innerText = rect.username;
194
- userTip.style.background = rect.color;
195
- }
196
- for (let i = rects.length; i < this.tooltips.children.length; i++) {
197
- this.tooltips.removeChild(this.tooltips.children[i]);
198
- }
199
- }
200
- getUserCursor(index) {
201
- let child = this.tooltips.children[index];
202
- if (child) {
203
- const anchor = child.children[0];
204
- return {
205
- cursor: child,
206
- anchor,
207
- userTip: anchor.children[0]
208
- };
209
- }
210
- const userTip = createElement('span', {
211
- styles: {
212
- position: 'absolute',
213
- display: 'none',
214
- left: '50%',
215
- transform: 'translateX(-50%)',
216
- marginBottom: '2px',
217
- bottom: '100%',
218
- whiteSpace: 'nowrap',
219
- color: '#fff',
220
- boxShadow: '0 1px 2px rgba(0,0,0,.1)',
221
- borderRadius: '3px',
222
- padding: '3px 5px',
223
- pointerEvents: 'none',
224
- }
225
- });
226
- const anchor = createElement('span', {
227
- styles: {
228
- position: 'absolute',
229
- top: '-2px',
230
- left: '-2px',
231
- width: '6px',
232
- height: '6px',
233
- pointerEvents: 'auto',
234
- pointer: 'cursor',
235
- },
236
- children: [userTip],
237
- on: {
238
- mouseenter() {
239
- userTip.style.display = 'block';
240
- },
241
- mouseleave() {
242
- userTip.style.display = 'none';
243
- }
244
- }
245
- });
246
- child = createElement('span', {
247
- styles: {
248
- position: 'absolute',
249
- },
250
- children: [
251
- anchor
252
- ]
253
- });
254
- this.tooltips.append(child);
255
- return {
256
- cursor: child,
257
- anchor,
258
- userTip
259
- };
260
- }
261
- };
262
- CollaborateCursor = __decorate([
263
- Injectable(),
264
- __param(0, Inject(VIEW_CONTAINER)),
265
- __param(1, Optional()),
266
- __metadata("design:paramtypes", [HTMLElement,
267
- CollaborateSelectionAwarenessDelegate,
268
- SelectionBridge,
269
- Scheduler,
270
- Selection])
271
- ], CollaborateCursor);
272
- export { CollaborateCursor };
273
- //# 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, Optional } from '@tanbo/di'\nimport {\n  createElement,\n  VIEW_CONTAINER,\n  getLayoutRectByRange,\n  SelectionBridge\n} from '@textbus/browser'\nimport { Selection, SelectionPaths, AbstractSelection, Scheduler, Rect } from '@textbus/core'\nimport { fromEvent, Subject, Subscription } from '@tanbo/stream'\n\nexport interface RemoteSelection {\n  id: string\n  color: string\n  username: string\n  paths: SelectionPaths\n}\n\nexport interface SelectionRect extends Rect {\n  color: string\n  username: string\n  id: string\n}\n\nexport interface RemoteSelectionCursor {\n  cursor: HTMLElement\n  anchor: HTMLElement\n  userTip: HTMLElement\n}\n\nexport abstract class CollaborateSelectionAwarenessDelegate {\n  abstract getRects(abstractSelection: AbstractSelection, nativeRange: Range): false | Rect[]\n}\n\n@Injectable()\nexport class CollaborateCursor {\n  private host = createElement('div', {\n    styles: {\n      position: 'absolute',\n      left: 0,\n      top: 0,\n      width: '100%',\n      height: '100%',\n      pointerEvents: 'none',\n      zIndex: 1\n    }\n  })\n  private canvasContainer = createElement('div', {\n    styles: {\n      position: 'absolute',\n      left: 0,\n      top: 0,\n      width: '100%',\n      height: '100%',\n      overflow: 'hidden'\n    }\n  })\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: document.documentElement.clientHeight + 'px',\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  private subscription = new Subscription()\n  private currentSelection: RemoteSelection[] = []\n\n  constructor(@Inject(VIEW_CONTAINER) private container: HTMLElement,\n              @Optional() private awarenessDelegate: CollaborateSelectionAwarenessDelegate,\n              private nativeSelection: SelectionBridge,\n              private scheduler: Scheduler,\n              private selection: Selection) {\n    this.canvasContainer.append(this.canvas)\n    this.host.append(this.canvasContainer, this.tooltips)\n    container.prepend(this.host)\n    this.subscription.add(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.left, rect.top, rect.width, rect.height)\n        this.context.fill()\n        this.context.closePath()\n      }\n    }), fromEvent(window, 'resize').subscribe(() => {\n      this.canvas.style.height = document.documentElement.clientHeight + 'px'\n      this.refresh()\n    }), this.scheduler.onDocChanged.subscribe(() => {\n      this.refresh()\n    }))\n  }\n\n  refresh() {\n    this.draw(this.currentSelection)\n  }\n\n  destroy() {\n    this.subscription.unsubscribe()\n  }\n\n  draw(paths: RemoteSelection[]) {\n    this.currentSelection = paths\n    const containerRect = this.container.getBoundingClientRect()\n    this.canvas.style.top = containerRect.top * -1 + 'px'\n    this.canvas.width = this.canvas.offsetWidth\n    this.canvas.height = this.canvas.offsetHeight\n    this.context.clearRect(0, 0, this.canvas.width, this.canvas.height)\n\n    const users: SelectionRect[] = []\n\n    paths.filter(i => {\n      return i.paths.anchor.length && i.paths.focus.length\n    }).forEach(item => {\n      const anchorPaths = [...item.paths.anchor]\n      const focusPaths = [...item.paths.focus]\n      const anchorOffset = anchorPaths.pop()!\n      const anchorSlot = this.selection.findSlotByPaths(anchorPaths)\n      const focusOffset = focusPaths.pop()!\n      const focusSlot = this.selection.findSlotByPaths(focusPaths)\n      if (!anchorSlot || !focusSlot) {\n        return\n      }\n\n      const { focus, anchor } = this.nativeSelection.getPositionByRange({\n        focusOffset,\n        anchorOffset,\n        focusSlot,\n        anchorSlot\n      })\n      if (!focus || !anchor) {\n        return\n      }\n      const nativeRange = document.createRange()\n      nativeRange.setStart(anchor.node, anchor.offset)\n      nativeRange.setEnd(focus.node, focus.offset)\n      if ((anchor.node !== focus.node || anchor.offset !== focus.offset) && nativeRange.collapsed) {\n        nativeRange.setStart(focus.node, focus.offset)\n        nativeRange.setEnd(anchor.node, anchor.offset)\n      }\n\n      let rects: Rect[] | DOMRectList | false = false\n      if (this.awarenessDelegate) {\n        rects = this.awarenessDelegate.getRects({\n          focusOffset,\n          anchorOffset,\n          focusSlot,\n          anchorSlot\n        }, nativeRange)\n      }\n      if (!rects) {\n        rects = nativeRange.getClientRects()\n      }\n      const selectionRects: SelectionRect[] = []\n      for (let i = rects.length - 1; i >= 0; i--) {\n        const rect = rects[i]\n        selectionRects.push({\n          id: item.id,\n          color: item.color,\n          username: item.username,\n          left: rect.left - containerRect.left,\n          top: rect.top,\n          width: rect.width,\n          height: rect.height,\n        })\n      }\n      this.onRectsChange.next(selectionRects)\n\n      const cursorRange = nativeRange.cloneRange()\n      cursorRange.setStart(focus.node, focus.offset)\n      cursorRange.collapse(true)\n\n      const cursorRect = getLayoutRectByRange(cursorRange)\n\n      const rect: SelectionRect = {\n        id: item.id,\n        username: item.username,\n        color: item.color,\n        left: cursorRect.left - containerRect.left,\n        top: cursorRect.top - containerRect.top,\n        width: 2,\n        height: cursorRect.height\n      }\n      if (rect.left < 0 || rect.top < 0 || rect.left > containerRect.width) {\n        return\n      }\n      users.push(rect)\n    })\n    this.drawUserCursor(users)\n  }\n\n  protected 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.left + 'px',\n        top: rect.top + '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"]}