-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.js
110 lines (102 loc) · 3.69 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
/** @license Apache-2.0 */
// @ts-check
// TODO refactor: rename the file, I guess? It's not `init` anymore.
import {
WebxdcSyncProvider as WebxdcSyncProviderGeneric
} from "./WebxdcSyncProviderGeneric"
import { fromUint8Array, toUint8Array } from "js-base64"
// Why not `lodash.throttle`? This one's ligther.
import throttle from "just-throttle"
/** @typedef {import("yjs").applyUpdate} YApplyUpdate */
/** @typedef {Parameters<YApplyUpdate>[1]} YUpdate */
// TODO perf: the js-base64 library not the lightest one.
/**
* @param {YUpdate} update
*/
export function serializeUpdate(update) {
return fromUint8Array(update);
}
/**
* @param {ReturnType<typeof serializeUpdate>} serializedUpdate
*/
export function deserializeUpdate(serializedUpdate) {
return toUint8Array(serializedUpdate);
}
/**
* A version of {@link WebxdcSyncProviderGeneric} with reasonable defaults.
*/
export class WebxdcSyncProvider extends WebxdcSyncProviderGeneric {
/**
* @param {ConstructorParameters<typeof WebxdcSyncProviderGeneric>[0]} ydoc
* @param {ConstructorParameters<typeof WebxdcSyncProviderGeneric>[3]} [transactionOrigin]
*/
constructor(ydoc, transactionOrigin) {
super(
ydoc,
serializeUpdate,
deserializeUpdate,
sendUpdate,
transactionOrigin,
);
// TODO refactor: make `onIncomingYjsUpdate` private for the inheriting class?
/**
* Resolves when the stored state of the webxdc app has been applied to the {@link ydoc}.
* @public
* @readonly
* @type {Promise<void>}
*/
this.initialStateRestored = webxdc.setUpdateListener(
(webxdcUpdate) => {
// Keep in mind that this is also called for the updates that we send.
// Re-applying an update is fine in Yjs.
// TODO perf: don't update one by one. Batch updates? ydoc.transact?
this.onIncomingYjsUpdate(webxdcUpdate.payload)
},
// TODO perf: utilize local cache, e.g. `y-indexeddb`. Although make sure that
// the update is actually stored. See https://github.com/yjs/y-indexeddb/issues/28
0,
);
/**
* Delta Chat Core throttles updates pretty heavily:
* https://github.com/deltachat/deltachat-core-rust/blob/b96028cd87f02a83f8f0a5282da4b4bb88cdc05c/src/context.rs#L379
* https://codeberg.org/webxdc/editor/pulls/23#issuecomment-996164
* So we better spend the quota wisely, i.e. not spend all of it immediately
* and then get throttled by the messenger implementation.
*/
const throttlePeriodMs = 2000;
/** @type {WebxdcSyncProviderGeneric['onNeedToSendLocalUpdates']} */
this.onNeedToSendLocalUpdates = throttle(
this.sendUnsentLocalUpdates,
throttlePeriodMs,
// TODO maybe use both leading and trailing.
// But it doesn't work properly in `just-throttle` IMO:
// https://github.com/angus-c/just/issues/207#issuecomment-1621879811
{ leading: false, trailing: true }
);
// TODO refactor: turn this into a constructor parameter so it's easier
// to remember it?
/**
* @public
* @see https://docs.webxdc.org/spec.html#sendupdate
* @type {string}
*/
this.sendUpdateDescr = 'Document updated';
/**
* @param {ReturnType<typeof serializeUpdate>} outgoingSerializedYjsUpdate
* @returns {void}
*/
function sendUpdate(outgoingSerializedYjsUpdate) {
webxdc.sendUpdate(
{
payload: outgoingSerializedYjsUpdate,
},
this.sendUpdateDescr
)
}
// TODO fix: need to ensure that the updates get send when the page gets closed.
// See https://developer.chrome.com/blog/page-lifecycle-api/#the-beforeunload-event
// https://stackoverflow.com/questions/7317273/warn-user-before-leaving-web-page-with-unsaved-changes
// Also maybe need to add info about this to the docs of `WebxdcSyncProviderGeneric`.
// Or should it just implement it by default itself?
}
}