mobx-keystone-yjs 1.5.3 → 1.5.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +11 -0
- package/dist/mobx-keystone-yjs.esm.js +176 -116
- package/dist/mobx-keystone-yjs.esm.mjs +176 -116
- package/dist/mobx-keystone-yjs.umd.js +177 -117
- package/dist/types/binding/YjsTextModel.d.ts +5 -4
- package/dist/types/binding/applyMobxKeystonePatchToYjsObject.d.ts +1 -1
- package/dist/types/binding/bindYjsToMobxKeystone.d.ts +2 -2
- package/dist/types/binding/convertJsonToYjsData.d.ts +4 -4
- package/dist/types/binding/convertYjsDataToJson.d.ts +3 -3
- package/dist/types/binding/convertYjsEventToPatches.d.ts +1 -1
- package/dist/types/binding/yjsBindingContext.d.ts +2 -2
- package/dist/types/index.d.ts +6 -6
- package/dist/types/utils/isYjsValueDeleted.d.ts +7 -0
- package/package.json +12 -14
- package/src/binding/YjsTextModel.ts +280 -247
- package/src/binding/applyMobxKeystonePatchToYjsObject.ts +6 -1
- package/src/binding/bindYjsToMobxKeystone.ts +22 -12
- package/src/binding/convertJsonToYjsData.ts +6 -6
- package/src/binding/convertYjsDataToJson.ts +4 -3
- package/src/binding/convertYjsEventToPatches.ts +4 -12
- package/src/index.ts +2 -2
- package/src/utils/isYjsValueDeleted.ts +14 -0
|
@@ -2,28 +2,8 @@ var __defProp = Object.defineProperty;
|
|
|
2
2
|
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
|
|
3
3
|
var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
|
|
4
4
|
import { createAtom, reaction, observe, computed, action } from "mobx";
|
|
5
|
-
import { createContext,
|
|
5
|
+
import { createContext, Model, tProp, types, frozen, getParentToChildPath, onSnapshot, model, modelSnapshotOutWithMetadata, applyPatches, onPatches, onGlobalPatches, fromSnapshot } from "mobx-keystone";
|
|
6
6
|
import * as Y from "yjs";
|
|
7
|
-
function __decorate(decorators, target, key, desc) {
|
|
8
|
-
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
|
9
|
-
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
|
10
|
-
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;
|
|
11
|
-
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
|
12
|
-
}
|
|
13
|
-
typeof SuppressedError === "function" ? SuppressedError : function(error, suppressed, message) {
|
|
14
|
-
var e = new Error(message);
|
|
15
|
-
return e.name = "SuppressedError", e.error = error, e.suppressed = suppressed, e;
|
|
16
|
-
};
|
|
17
|
-
class MobxKeystoneYjsError extends Error {
|
|
18
|
-
constructor(msg) {
|
|
19
|
-
super(msg);
|
|
20
|
-
Object.setPrototypeOf(this, MobxKeystoneYjsError.prototype);
|
|
21
|
-
}
|
|
22
|
-
}
|
|
23
|
-
function failure(msg) {
|
|
24
|
-
return new MobxKeystoneYjsError(msg);
|
|
25
|
-
}
|
|
26
|
-
const yjsBindingContext = createContext(void 0);
|
|
27
7
|
const yjsCollectionAtoms = /* @__PURE__ */ new WeakMap();
|
|
28
8
|
const getYjsCollectionAtom = (yjsCollection) => {
|
|
29
9
|
return yjsCollectionAtoms.get(yjsCollection);
|
|
@@ -36,6 +16,22 @@ const getOrCreateYjsCollectionAtom = (yjsCollection) => {
|
|
|
36
16
|
}
|
|
37
17
|
return atom;
|
|
38
18
|
};
|
|
19
|
+
function isYjsValueDeleted(yjsValue) {
|
|
20
|
+
var _a, _b;
|
|
21
|
+
if (yjsValue instanceof Y.AbstractType) {
|
|
22
|
+
return !!((_a = yjsValue._item) == null ? void 0 : _a.deleted) || !!((_b = yjsValue.doc) == null ? void 0 : _b.isDestroyed);
|
|
23
|
+
}
|
|
24
|
+
return false;
|
|
25
|
+
}
|
|
26
|
+
class MobxKeystoneYjsError extends Error {
|
|
27
|
+
constructor(msg) {
|
|
28
|
+
super(msg);
|
|
29
|
+
Object.setPrototypeOf(this, MobxKeystoneYjsError.prototype);
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
function failure(msg) {
|
|
33
|
+
return new MobxKeystoneYjsError(msg);
|
|
34
|
+
}
|
|
39
35
|
function resolveYjsPath(yjsObject, path) {
|
|
40
36
|
let currentYjsObject = yjsObject;
|
|
41
37
|
path.forEach((pathPart, i) => {
|
|
@@ -48,14 +44,29 @@ function resolveYjsPath(yjsObject, path) {
|
|
|
48
44
|
const key = Number(pathPart);
|
|
49
45
|
currentYjsObject = currentYjsObject.get(key);
|
|
50
46
|
} else {
|
|
51
|
-
throw failure(
|
|
47
|
+
throw failure(
|
|
48
|
+
`Y.Map or Y.Array was expected at path ${JSON.stringify(
|
|
49
|
+
path.slice(0, i)
|
|
50
|
+
)} in order to resolve path ${JSON.stringify(path)}, but got ${currentYjsObject} instead`
|
|
51
|
+
);
|
|
52
52
|
}
|
|
53
53
|
});
|
|
54
54
|
return currentYjsObject;
|
|
55
55
|
}
|
|
56
|
+
const yjsBindingContext = createContext(void 0);
|
|
57
|
+
var __defProp2 = Object.defineProperty;
|
|
58
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
59
|
+
var __decorateClass = (decorators, target, key, kind) => {
|
|
60
|
+
var result = kind > 1 ? void 0 : kind ? __getOwnPropDesc(target, key) : target;
|
|
61
|
+
for (var i = decorators.length - 1, decorator; i >= 0; i--)
|
|
62
|
+
if (decorator = decorators[i])
|
|
63
|
+
result = (kind ? decorator(target, key, result) : decorator(result)) || result;
|
|
64
|
+
if (kind && result) __defProp2(target, key, result);
|
|
65
|
+
return result;
|
|
66
|
+
};
|
|
56
67
|
const deltaListType = types.array(types.frozen(types.unchecked()));
|
|
57
68
|
const yjsTextModelId = "mobx-keystone-yjs/YjsTextModel";
|
|
58
|
-
let YjsTextModel = class
|
|
69
|
+
let YjsTextModel = class extends Model({
|
|
59
70
|
deltaList: tProp(deltaListType, () => [])
|
|
60
71
|
}) {
|
|
61
72
|
constructor() {
|
|
@@ -79,13 +90,12 @@ let YjsTextModel = class YjsTextModel2 extends Model({
|
|
|
79
90
|
]
|
|
80
91
|
});
|
|
81
92
|
}
|
|
82
|
-
/**
|
|
83
|
-
* The Y.js path from the bound object to the YjsTextModel instance.
|
|
84
|
-
*/
|
|
85
93
|
get _yjsObjectPath() {
|
|
86
94
|
const ctx = yjsBindingContext.get(this);
|
|
87
95
|
if ((ctx == null ? void 0 : ctx.boundObject) == null) {
|
|
88
|
-
throw failure(
|
|
96
|
+
throw failure(
|
|
97
|
+
"the YjsTextModel instance must be part of a bound object before it can be accessed"
|
|
98
|
+
);
|
|
89
99
|
}
|
|
90
100
|
const path = getParentToChildPath(ctx.boundObject, this);
|
|
91
101
|
if (!path) {
|
|
@@ -93,17 +103,11 @@ let YjsTextModel = class YjsTextModel2 extends Model({
|
|
|
93
103
|
}
|
|
94
104
|
return path;
|
|
95
105
|
}
|
|
96
|
-
/**
|
|
97
|
-
* The Yjs.Text object present at this mobx-keystone node's path.
|
|
98
|
-
*/
|
|
99
106
|
get _yjsObjectAtPath() {
|
|
100
107
|
const path = this._yjsObjectPath;
|
|
101
108
|
const ctx = yjsBindingContext.get(this);
|
|
102
109
|
return resolveYjsPath(ctx.yjsObject, path);
|
|
103
110
|
}
|
|
104
|
-
/**
|
|
105
|
-
* The Yjs.Text object represented by this mobx-keystone node.
|
|
106
|
-
*/
|
|
107
111
|
get yjsText() {
|
|
108
112
|
const yjsObject = this._yjsObjectAtPath;
|
|
109
113
|
if (!(yjsObject instanceof Y.Text)) {
|
|
@@ -111,13 +115,27 @@ let YjsTextModel = class YjsTextModel2 extends Model({
|
|
|
111
115
|
}
|
|
112
116
|
return yjsObject;
|
|
113
117
|
}
|
|
114
|
-
/**
|
|
115
|
-
* The text value of the Yjs.Text object.
|
|
116
|
-
* Shortcut for `yjsText.toString()`, but computed.
|
|
117
|
-
*/
|
|
118
118
|
get text() {
|
|
119
119
|
this.yjsTextChangedAtom.reportObserved();
|
|
120
|
-
|
|
120
|
+
const ctx = yjsBindingContext.get(this);
|
|
121
|
+
if ((ctx == null ? void 0 : ctx.boundObject) != null) {
|
|
122
|
+
try {
|
|
123
|
+
const yjsTextString = this.yjsText.toString();
|
|
124
|
+
if (yjsTextString !== "" || this.deltaList.length === 0) {
|
|
125
|
+
return yjsTextString;
|
|
126
|
+
}
|
|
127
|
+
} catch {
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
return this.deltaListToText();
|
|
131
|
+
}
|
|
132
|
+
deltaListToText() {
|
|
133
|
+
const doc = new Y.Doc();
|
|
134
|
+
const text = doc.getText();
|
|
135
|
+
this.deltaList.forEach((d) => {
|
|
136
|
+
text.applyDelta(d.data);
|
|
137
|
+
});
|
|
138
|
+
return text.toString();
|
|
121
139
|
}
|
|
122
140
|
onInit() {
|
|
123
141
|
const shouldReplicateToYjs = (ctx) => {
|
|
@@ -126,29 +144,36 @@ let YjsTextModel = class YjsTextModel2 extends Model({
|
|
|
126
144
|
let reapplyDeltasToYjsText = false;
|
|
127
145
|
const newDeltas = [];
|
|
128
146
|
let disposeObserveDeltaList;
|
|
129
|
-
const disposeReactionToDeltaListRefChange = reaction(
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
147
|
+
const disposeReactionToDeltaListRefChange = reaction(
|
|
148
|
+
() => this.$.deltaList,
|
|
149
|
+
(deltaList) => {
|
|
150
|
+
disposeObserveDeltaList == null ? void 0 : disposeObserveDeltaList();
|
|
151
|
+
disposeObserveDeltaList = void 0;
|
|
152
|
+
disposeObserveDeltaList = observe(deltaList, (change) => {
|
|
153
|
+
if (reapplyDeltasToYjsText) {
|
|
154
|
+
return;
|
|
155
|
+
}
|
|
156
|
+
if (!shouldReplicateToYjs(yjsBindingContext.get(this))) {
|
|
157
|
+
return;
|
|
158
|
+
}
|
|
159
|
+
if (change.type === "splice" && change.removedCount === 0 && change.addedCount > 0 && change.index === this.deltaList.length) {
|
|
160
|
+
newDeltas.push(...change.added);
|
|
161
|
+
} else {
|
|
162
|
+
reapplyDeltasToYjsText = true;
|
|
163
|
+
}
|
|
164
|
+
});
|
|
165
|
+
},
|
|
166
|
+
{ fireImmediately: true }
|
|
167
|
+
);
|
|
146
168
|
const disposeOnSnapshot = onSnapshot(this, () => {
|
|
147
169
|
try {
|
|
148
170
|
if (reapplyDeltasToYjsText) {
|
|
149
171
|
const ctx = yjsBindingContext.get(this);
|
|
150
172
|
if (shouldReplicateToYjs(ctx)) {
|
|
151
173
|
const { yjsText } = this;
|
|
174
|
+
if (isYjsValueDeleted(yjsText)) {
|
|
175
|
+
throw failure("cannot reapply deltas to deleted Yjs.Text");
|
|
176
|
+
}
|
|
152
177
|
ctx.yjsDoc.transact(() => {
|
|
153
178
|
if (yjsText.length > 0) {
|
|
154
179
|
yjsText.delete(0, yjsText.length);
|
|
@@ -162,6 +187,9 @@ let YjsTextModel = class YjsTextModel2 extends Model({
|
|
|
162
187
|
const ctx = yjsBindingContext.get(this);
|
|
163
188
|
if (shouldReplicateToYjs(ctx)) {
|
|
164
189
|
const { yjsText } = this;
|
|
190
|
+
if (isYjsValueDeleted(yjsText)) {
|
|
191
|
+
throw failure("cannot reapply deltas to deleted Yjs.Text");
|
|
192
|
+
}
|
|
165
193
|
ctx.yjsDoc.transact(() => {
|
|
166
194
|
newDeltas.forEach((frozenDeltas) => {
|
|
167
195
|
yjsText.applyDelta(frozenDeltas.data);
|
|
@@ -174,7 +202,10 @@ let YjsTextModel = class YjsTextModel2 extends Model({
|
|
|
174
202
|
newDeltas.length = 0;
|
|
175
203
|
}
|
|
176
204
|
});
|
|
177
|
-
const diposeYjsTextChangedAtom = hookYjsTextChangedAtom(
|
|
205
|
+
const diposeYjsTextChangedAtom = hookYjsTextChangedAtom(
|
|
206
|
+
() => this.yjsText,
|
|
207
|
+
this.yjsTextChangedAtom
|
|
208
|
+
);
|
|
178
209
|
return () => {
|
|
179
210
|
disposeOnSnapshot();
|
|
180
211
|
disposeReactionToDeltaListRefChange();
|
|
@@ -184,19 +215,19 @@ let YjsTextModel = class YjsTextModel2 extends Model({
|
|
|
184
215
|
};
|
|
185
216
|
}
|
|
186
217
|
};
|
|
187
|
-
|
|
218
|
+
__decorateClass([
|
|
188
219
|
computed
|
|
189
|
-
], YjsTextModel.prototype, "_yjsObjectPath",
|
|
190
|
-
|
|
220
|
+
], YjsTextModel.prototype, "_yjsObjectPath", 1);
|
|
221
|
+
__decorateClass([
|
|
191
222
|
computed
|
|
192
|
-
], YjsTextModel.prototype, "_yjsObjectAtPath",
|
|
193
|
-
|
|
223
|
+
], YjsTextModel.prototype, "_yjsObjectAtPath", 1);
|
|
224
|
+
__decorateClass([
|
|
194
225
|
computed
|
|
195
|
-
], YjsTextModel.prototype, "yjsText",
|
|
196
|
-
|
|
226
|
+
], YjsTextModel.prototype, "yjsText", 1);
|
|
227
|
+
__decorateClass([
|
|
197
228
|
computed
|
|
198
|
-
], YjsTextModel.prototype, "text",
|
|
199
|
-
YjsTextModel =
|
|
229
|
+
], YjsTextModel.prototype, "text", 1);
|
|
230
|
+
YjsTextModel = __decorateClass([
|
|
200
231
|
model(yjsTextModelId)
|
|
201
232
|
], YjsTextModel);
|
|
202
233
|
const DecoratedYjsTextModel = YjsTextModel;
|
|
@@ -205,25 +236,30 @@ function hookYjsTextChangedAtom(getYjsText, textChangedAtom) {
|
|
|
205
236
|
const observeFn = () => {
|
|
206
237
|
textChangedAtom.reportChanged();
|
|
207
238
|
};
|
|
208
|
-
const disposeReactionToYTextChange = reaction(
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
disposeObserveYjsText =
|
|
220
|
-
|
|
221
|
-
|
|
239
|
+
const disposeReactionToYTextChange = reaction(
|
|
240
|
+
() => {
|
|
241
|
+
try {
|
|
242
|
+
const yjsText = getYjsText();
|
|
243
|
+
return isYjsValueDeleted(yjsText) ? void 0 : yjsText;
|
|
244
|
+
} catch {
|
|
245
|
+
return void 0;
|
|
246
|
+
}
|
|
247
|
+
},
|
|
248
|
+
(yjsText) => {
|
|
249
|
+
disposeObserveYjsText == null ? void 0 : disposeObserveYjsText();
|
|
250
|
+
disposeObserveYjsText = void 0;
|
|
251
|
+
if (yjsText) {
|
|
252
|
+
yjsText.observe(observeFn);
|
|
253
|
+
disposeObserveYjsText = () => {
|
|
254
|
+
yjsText.unobserve(observeFn);
|
|
255
|
+
};
|
|
256
|
+
}
|
|
257
|
+
textChangedAtom.reportChanged();
|
|
258
|
+
},
|
|
259
|
+
{
|
|
260
|
+
fireImmediately: true
|
|
222
261
|
}
|
|
223
|
-
|
|
224
|
-
}, {
|
|
225
|
-
fireImmediately: true
|
|
226
|
-
});
|
|
262
|
+
);
|
|
227
263
|
return () => {
|
|
228
264
|
disposeReactionToYTextChange();
|
|
229
265
|
disposeObserveYjsText == null ? void 0 : disposeObserveYjsText();
|
|
@@ -267,32 +303,45 @@ function convertJsonToYjsData(v) {
|
|
|
267
303
|
}
|
|
268
304
|
throw new Error(`unsupported value type: ${v}`);
|
|
269
305
|
}
|
|
270
|
-
|
|
306
|
+
const applyJsonArrayToYArray = (dest, source) => {
|
|
271
307
|
dest.push(source.map(convertJsonToYjsData));
|
|
272
|
-
}
|
|
273
|
-
|
|
308
|
+
};
|
|
309
|
+
const applyJsonObjectToYMap = (dest, source) => {
|
|
274
310
|
Object.entries(source).forEach(([k, v]) => {
|
|
275
311
|
dest.set(k, convertJsonToYjsData(v));
|
|
276
312
|
});
|
|
277
|
-
}
|
|
313
|
+
};
|
|
278
314
|
function applyMobxKeystonePatchToYjsObject(patch, yjs) {
|
|
315
|
+
if (isYjsValueDeleted(yjs)) {
|
|
316
|
+
throw failure("cannot apply patch to deleted Yjs value");
|
|
317
|
+
}
|
|
279
318
|
if (patch.path.length > 1) {
|
|
280
319
|
const [key, ...rest] = patch.path;
|
|
281
320
|
if (yjs instanceof Y.Map) {
|
|
282
321
|
const child = yjs.get(String(key));
|
|
283
322
|
if (child === void 0) {
|
|
284
|
-
throw failure(
|
|
323
|
+
throw failure(
|
|
324
|
+
`invalid patch path, key "${key}" not found in Yjs map - patch: ${JSON.stringify(patch)}`
|
|
325
|
+
);
|
|
285
326
|
}
|
|
286
327
|
applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child);
|
|
287
328
|
} else if (yjs instanceof Y.Array) {
|
|
288
329
|
const child = yjs.get(Number(key));
|
|
289
330
|
if (child === void 0) {
|
|
290
|
-
throw failure(
|
|
331
|
+
throw failure(
|
|
332
|
+
`invalid patch path, key "${key}" not found in Yjs array - patch: ${JSON.stringify(
|
|
333
|
+
patch
|
|
334
|
+
)}`
|
|
335
|
+
);
|
|
291
336
|
}
|
|
292
337
|
applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child);
|
|
293
338
|
} else if (yjs instanceof Y.Text) ;
|
|
294
339
|
else {
|
|
295
|
-
throw failure(
|
|
340
|
+
throw failure(
|
|
341
|
+
`invalid patch path, key "${key}" not found in unknown Yjs object - patch: ${JSON.stringify(
|
|
342
|
+
patch
|
|
343
|
+
)}`
|
|
344
|
+
);
|
|
296
345
|
}
|
|
297
346
|
} else if (patch.path.length === 1) {
|
|
298
347
|
if (yjs instanceof Y.Map) {
|
|
@@ -344,13 +393,15 @@ function applyMobxKeystonePatchToYjsObject(patch, yjs) {
|
|
|
344
393
|
}
|
|
345
394
|
} else if (yjs instanceof Y.Text) ;
|
|
346
395
|
else {
|
|
347
|
-
throw failure(
|
|
396
|
+
throw failure(
|
|
397
|
+
`invalid patch path, the Yjs object is of an unkown type, so key "${String(patch.path[0])}" cannot be found in it`
|
|
398
|
+
);
|
|
348
399
|
}
|
|
349
400
|
} else {
|
|
350
401
|
throw failure(`invalid patch path, it cannot be empty`);
|
|
351
402
|
}
|
|
352
403
|
}
|
|
353
|
-
|
|
404
|
+
const convertYjsDataToJson = action((yjsData) => {
|
|
354
405
|
if (yjsData instanceof Y.Array) {
|
|
355
406
|
return yjsData.map((v) => convertYjsDataToJson(v));
|
|
356
407
|
}
|
|
@@ -368,7 +419,7 @@ function convertYjsDataToJson(yjsData) {
|
|
|
368
419
|
});
|
|
369
420
|
}
|
|
370
421
|
return yjsData;
|
|
371
|
-
}
|
|
422
|
+
});
|
|
372
423
|
function convertYjsEventToPatches(event) {
|
|
373
424
|
const patches = [];
|
|
374
425
|
if (event instanceof Y.YMapEvent) {
|
|
@@ -380,14 +431,14 @@ function convertYjsEventToPatches(event) {
|
|
|
380
431
|
patches.push({
|
|
381
432
|
op: "add",
|
|
382
433
|
path,
|
|
383
|
-
value:
|
|
434
|
+
value: convertYjsDataToJson(source.get(key))
|
|
384
435
|
});
|
|
385
436
|
break;
|
|
386
437
|
case "update":
|
|
387
438
|
patches.push({
|
|
388
439
|
op: "replace",
|
|
389
440
|
path,
|
|
390
|
-
value:
|
|
441
|
+
value: convertYjsDataToJson(source.get(key))
|
|
391
442
|
});
|
|
392
443
|
break;
|
|
393
444
|
case "delete":
|
|
@@ -422,7 +473,7 @@ function convertYjsEventToPatches(event) {
|
|
|
422
473
|
patches.push({
|
|
423
474
|
op: "add",
|
|
424
475
|
path,
|
|
425
|
-
value:
|
|
476
|
+
value: convertYjsDataToJson(v)
|
|
426
477
|
});
|
|
427
478
|
retain++;
|
|
428
479
|
});
|
|
@@ -443,15 +494,12 @@ function convertYjsEventToPatches(event) {
|
|
|
443
494
|
}
|
|
444
495
|
return patches;
|
|
445
496
|
}
|
|
446
|
-
function
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
}
|
|
453
|
-
function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
|
|
454
|
-
const yjsOrigin = Symbol("bindYjsToMobxKeystoneTransactionOrigin");
|
|
497
|
+
function bindYjsToMobxKeystone({
|
|
498
|
+
yjsDoc,
|
|
499
|
+
yjsObject,
|
|
500
|
+
mobxKeystoneType
|
|
501
|
+
}) {
|
|
502
|
+
const yjsOrigin = /* @__PURE__ */ Symbol("bindYjsToMobxKeystoneTransactionOrigin");
|
|
455
503
|
let applyingYjsChangesToMobxKeystone = 0;
|
|
456
504
|
const bindingContext = {
|
|
457
505
|
yjsDoc,
|
|
@@ -471,7 +519,10 @@ function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
|
|
|
471
519
|
initializationGlobalPatches.push({ target, patches });
|
|
472
520
|
});
|
|
473
521
|
try {
|
|
474
|
-
const boundObject2 = yjsBindingContext.apply(
|
|
522
|
+
const boundObject2 = yjsBindingContext.apply(
|
|
523
|
+
() => fromSnapshot(mobxKeystoneType, yjsJson),
|
|
524
|
+
bindingContext
|
|
525
|
+
);
|
|
475
526
|
yjsBindingContext.set(boundObject2, { ...bindingContext, boundObject: boundObject2 });
|
|
476
527
|
return boundObject2;
|
|
477
528
|
} finally {
|
|
@@ -513,6 +564,9 @@ function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
|
|
|
513
564
|
}
|
|
514
565
|
const arrayOfArrayOfPatches = pendingArrayOfArrayOfPatches;
|
|
515
566
|
pendingArrayOfArrayOfPatches = [];
|
|
567
|
+
if (isYjsValueDeleted(yjsObject)) {
|
|
568
|
+
return;
|
|
569
|
+
}
|
|
516
570
|
yjsDoc.transact(() => {
|
|
517
571
|
arrayOfArrayOfPatches.forEach((arrayOfPatches) => {
|
|
518
572
|
arrayOfPatches.forEach((patch) => {
|
|
@@ -533,21 +587,27 @@ function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
|
|
|
533
587
|
const parentToChildPath = getParentToChildPath(boundObject, target);
|
|
534
588
|
if (parentToChildPath !== void 0) {
|
|
535
589
|
patches.forEach((patch) => {
|
|
536
|
-
applyMobxKeystonePatchToYjsObject(
|
|
537
|
-
|
|
538
|
-
|
|
539
|
-
|
|
590
|
+
applyMobxKeystonePatchToYjsObject(
|
|
591
|
+
{
|
|
592
|
+
...patch,
|
|
593
|
+
path: [...parentToChildPath, ...patch.path]
|
|
594
|
+
},
|
|
595
|
+
yjsObject
|
|
596
|
+
);
|
|
540
597
|
});
|
|
541
598
|
}
|
|
542
599
|
});
|
|
543
600
|
}, yjsOrigin);
|
|
601
|
+
const dispose = () => {
|
|
602
|
+
yjsDoc.off("destroy", dispose);
|
|
603
|
+
disposeOnPatches();
|
|
604
|
+
disposeOnSnapshot();
|
|
605
|
+
yjsObject.unobserveDeep(observeDeepCb);
|
|
606
|
+
};
|
|
607
|
+
yjsDoc.on("destroy", dispose);
|
|
544
608
|
return {
|
|
545
609
|
boundObject,
|
|
546
|
-
dispose
|
|
547
|
-
disposeOnPatches();
|
|
548
|
-
disposeOnSnapshot();
|
|
549
|
-
yjsObject.unobserveDeep(observeDeepCb);
|
|
550
|
-
},
|
|
610
|
+
dispose,
|
|
551
611
|
yjsOrigin
|
|
552
612
|
};
|
|
553
613
|
}
|
|
@@ -561,4 +621,4 @@ export {
|
|
|
561
621
|
yjsBindingContext,
|
|
562
622
|
yjsTextModelId
|
|
563
623
|
};
|
|
564
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"mobx-keystone-yjs.esm.mjs","sources":["../src/utils/error.ts","../src/binding/yjsBindingContext.ts","../src/utils/getOrCreateYjsCollectionAtom.ts","../src/binding/resolveYjsPath.ts","../src/binding/YjsTextModel.ts","../src/binding/convertJsonToYjsData.ts","../src/binding/applyMobxKeystonePatchToYjsObject.ts","../src/binding/convertYjsDataToJson.ts","../src/binding/convertYjsEventToPatches.ts","../src/binding/bindYjsToMobxKeystone.ts"],"sourcesContent":["/**\r\n * A mobx-keystone-yjs error.\r\n */\r\nexport class MobxKeystoneYjsError extends Error {\r\n  constructor(msg: string) {\r\n    super(msg)\r\n\r\n    // Set the prototype explicitly.\r\n    Object.setPrototypeOf(this, MobxKeystoneYjsError.prototype)\r\n  }\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nexport function failure(msg: string) {\r\n  return new MobxKeystoneYjsError(msg)\r\n}\r\n","import { AnyType, createContext } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\n\n/**\n * Context with info on how a mobx-keystone model is bound to a Y.js data structure.\n */\nexport interface YjsBindingContext {\n  /**\n   * The Y.js document.\n   */\n  yjsDoc: Y.Doc\n\n  /**\n   * The bound Y.js data structure.\n   */\n  yjsObject: Y.Map<unknown> | Y.Array<unknown> | Y.Text\n\n  /**\n   * The mobx-keystone model type.\n   */\n  mobxKeystoneType: AnyType\n\n  /**\n   * The origin symbol used for transactions.\n   */\n  yjsOrigin: symbol\n\n  /**\n   * The bound mobx-keystone instance.\n   */\n  boundObject: unknown\n\n  /**\n   * Whether we are currently applying Y.js changes to the mobx-keystone model.\n   */\n  isApplyingYjsChangesToMobxKeystone: boolean\n}\n\n/**\n * Context with info on how a mobx-keystone model is bound to a Y.js data structure.\n */\nexport const yjsBindingContext = createContext<YjsBindingContext | undefined>(undefined)\n","import { IAtom, createAtom } from \"mobx\"\nimport * as Y from \"yjs\"\n\nconst yjsCollectionAtoms = new WeakMap<Y.Map<unknown> | Y.Array<unknown>, IAtom>()\n\n/**\n * @internal\n */\nexport const getYjsCollectionAtom = (\n  yjsCollection: Y.Map<unknown> | Y.Array<unknown>\n): IAtom | undefined => {\n  return yjsCollectionAtoms.get(yjsCollection)\n}\n\n/**\n * @internal\n */\nexport const getOrCreateYjsCollectionAtom = (\n  yjsCollection: Y.Map<unknown> | Y.Array<unknown>\n): IAtom => {\n  let atom = yjsCollectionAtoms.get(yjsCollection)\n  if (!atom) {\n    atom = createAtom(`yjsCollectionAtom`)\n    yjsCollectionAtoms.set(yjsCollection, atom)\n  }\n  return atom\n}\n","import * as Y from \"yjs\"\nimport { failure } from \"../utils/error\"\nimport { getOrCreateYjsCollectionAtom } from \"../utils/getOrCreateYjsCollectionAtom\"\n\nexport function resolveYjsPath(yjsObject: unknown, path: readonly (string | number)[]): unknown {\n  let currentYjsObject: unknown = yjsObject\n\n  path.forEach((pathPart, i) => {\n    if (currentYjsObject instanceof Y.Map) {\n      getOrCreateYjsCollectionAtom(currentYjsObject).reportObserved()\n      const key = String(pathPart)\n      currentYjsObject = currentYjsObject.get(key)\n    } else if (currentYjsObject instanceof Y.Array) {\n      getOrCreateYjsCollectionAtom(currentYjsObject).reportObserved()\n      const key = Number(pathPart)\n      currentYjsObject = currentYjsObject.get(key)\n    } else {\n      throw failure(\n        `Y.Map or Y.Array was expected at path ${JSON.stringify(\n          path.slice(0, i)\n        )} in order to resolve path ${JSON.stringify(path)}, but got ${currentYjsObject} instead`\n      )\n    }\n  })\n\n  return currentYjsObject\n}\n","import { IAtom, computed, createAtom, observe, reaction } from \"mobx\"\r\nimport {\r\n  Frozen,\r\n  Model,\r\n  frozen,\r\n  getParentToChildPath,\r\n  model,\r\n  onSnapshot,\r\n  tProp,\r\n  types,\r\n} from \"mobx-keystone\"\r\nimport * as Y from \"yjs\"\r\nimport { failure } from \"../utils/error\"\r\nimport { YjsBindingContext, yjsBindingContext } from \"./yjsBindingContext\"\r\nimport { resolveYjsPath } from \"./resolveYjsPath\"\r\n\r\n// Delta[][], since each single change is a Delta[]\r\n// we use frozen so that we can reuse each delta change\r\nconst deltaListType = types.array(types.frozen(types.unchecked<unknown[]>()))\r\n\r\nexport const yjsTextModelId = \"mobx-keystone-yjs/YjsTextModel\"\r\n\r\n/**\r\n * A mobx-keystone model that represents a Yjs.Text object.\r\n */\r\n@model(yjsTextModelId)\r\nexport class YjsTextModel extends Model({\r\n  deltaList: tProp(deltaListType, () => []),\r\n}) {\r\n  /**\r\n   * Helper function to create a YjsTextModel instance with a simple text.\r\n   */\r\n  static withText(text: string): YjsTextModel {\r\n    return new DecoratedYjsTextModel({\r\n      deltaList: [\r\n        frozen([\r\n          {\r\n            insert: text,\r\n          },\r\n        ]),\r\n      ],\r\n    })\r\n  }\r\n\r\n  /**\r\n   * The Y.js path from the bound object to the YjsTextModel instance.\r\n   */\r\n  @computed\r\n  private get _yjsObjectPath() {\r\n    const ctx = yjsBindingContext.get(this)\r\n    if (ctx?.boundObject == null) {\r\n      throw failure(\r\n        \"the YjsTextModel instance must be part of a bound object before it can be accessed\"\r\n      )\r\n    }\r\n\r\n    const path = getParentToChildPath(ctx.boundObject, this)\r\n    if (!path) {\r\n      throw failure(\"a path from the bound object to the YjsTextModel instance is not available\")\r\n    }\r\n\r\n    return path\r\n  }\r\n\r\n  /**\r\n   * The Yjs.Text object present at this mobx-keystone node's path.\r\n   */\r\n  @computed\r\n  private get _yjsObjectAtPath(): unknown {\r\n    const path = this._yjsObjectPath\r\n\r\n    const ctx = yjsBindingContext.get(this)!\r\n\r\n    return resolveYjsPath(ctx.yjsObject, path)\r\n  }\r\n\r\n  /**\r\n   * The Yjs.Text object represented by this mobx-keystone node.\r\n   */\r\n  @computed\r\n  get yjsText(): Y.Text {\r\n    const yjsObject = this._yjsObjectAtPath\r\n\r\n    if (!(yjsObject instanceof Y.Text)) {\r\n      throw failure(`Y.Text was expected at path ${JSON.stringify(this._yjsObjectPath)}`)\r\n    }\r\n\r\n    return yjsObject\r\n  }\r\n\r\n  /**\r\n   * Atom that gets changed when the associated Y.js text changes.\r\n   */\r\n  yjsTextChangedAtom = createAtom(\"yjsTextChangedAtom\")\r\n\r\n  /**\r\n   * The text value of the Yjs.Text object.\r\n   * Shortcut for `yjsText.toString()`, but computed.\r\n   */\r\n  @computed\r\n  get text(): string {\r\n    this.yjsTextChangedAtom.reportObserved()\r\n    return this.yjsText.toString()\r\n  }\r\n\r\n  protected onInit() {\r\n    const shouldReplicateToYjs = (ctx: YjsBindingContext | undefined): ctx is YjsBindingContext => {\r\n      return !!ctx && !!ctx.boundObject && !ctx.isApplyingYjsChangesToMobxKeystone\r\n    }\r\n\r\n    let reapplyDeltasToYjsText = false\r\n    const newDeltas: Frozen<unknown[]>[] = []\r\n\r\n    let disposeObserveDeltaList: (() => void) | undefined\r\n\r\n    const disposeReactionToDeltaListRefChange = reaction(\r\n      () => this.$.deltaList,\r\n      (deltaList) => {\r\n        disposeObserveDeltaList?.()\r\n        disposeObserveDeltaList = undefined\r\n\r\n        disposeObserveDeltaList = observe(deltaList, (change) => {\r\n          if (reapplyDeltasToYjsText) {\r\n            // already gonna replace them all\r\n            return\r\n          }\r\n          if (!shouldReplicateToYjs(yjsBindingContext.get(this))) {\r\n            // yjs text is already up to date with these changes\r\n            return\r\n          }\r\n\r\n          if (\r\n            change.type === \"splice\" &&\r\n            change.removedCount === 0 &&\r\n            change.addedCount > 0 &&\r\n            change.index === this.deltaList.length\r\n          ) {\r\n            // optimization, just adding new ones to the end\r\n            newDeltas.push(...change.added)\r\n          } else {\r\n            // any other change, we need to reapply all deltas\r\n            reapplyDeltasToYjsText = true\r\n          }\r\n        })\r\n      },\r\n      { fireImmediately: true }\r\n    )\r\n\r\n    const disposeOnSnapshot = onSnapshot(this, () => {\r\n      try {\r\n        if (reapplyDeltasToYjsText) {\r\n          const ctx = yjsBindingContext.get(this)\r\n\r\n          if (shouldReplicateToYjs(ctx)) {\r\n            const { yjsText } = this\r\n\r\n            ctx.yjsDoc.transact(() => {\r\n              // didn't find a better way than this to reapply all deltas\r\n              // without having to re-create the Y.Text object\r\n              if (yjsText.length > 0) {\r\n                yjsText.delete(0, yjsText.length)\r\n              }\r\n\r\n              this.deltaList.forEach((frozenDeltas) => {\r\n                yjsText.applyDelta(frozenDeltas.data)\r\n              })\r\n            }, ctx.yjsOrigin)\r\n          }\r\n        } else if (newDeltas.length > 0) {\r\n          const ctx = yjsBindingContext.get(this)\r\n\r\n          if (shouldReplicateToYjs(ctx)) {\r\n            const { yjsText } = this\r\n\r\n            ctx.yjsDoc.transact(() => {\r\n              newDeltas.forEach((frozenDeltas) => {\r\n                yjsText.applyDelta(frozenDeltas.data)\r\n              })\r\n            }, ctx.yjsOrigin)\r\n          }\r\n        }\r\n      } finally {\r\n        reapplyDeltasToYjsText = false\r\n        newDeltas.length = 0\r\n      }\r\n    })\r\n\r\n    const diposeYjsTextChangedAtom = hookYjsTextChangedAtom(\r\n      () => this.yjsText,\r\n      this.yjsTextChangedAtom\r\n    )\r\n\r\n    return () => {\r\n      disposeOnSnapshot()\r\n      disposeReactionToDeltaListRefChange()\r\n      disposeObserveDeltaList?.()\r\n      disposeObserveDeltaList = undefined\r\n\r\n      diposeYjsTextChangedAtom()\r\n    }\r\n  }\r\n}\r\n\r\n// we use this trick just to avoid a babel bug that causes classes used inside classes not to be overriden\r\n// by the decorator\r\nconst DecoratedYjsTextModel = YjsTextModel\r\n\r\nfunction hookYjsTextChangedAtom(getYjsText: () => Y.Text, textChangedAtom: IAtom) {\r\n  let disposeObserveYjsText: (() => void) | undefined\r\n\r\n  const observeFn = () => {\r\n    textChangedAtom.reportChanged()\r\n  }\r\n\r\n  const disposeReactionToYTextChange = reaction(\r\n    () => {\r\n      try {\r\n        return getYjsText()\r\n      } catch {\r\n        return undefined\r\n      }\r\n    },\r\n    (yjsText) => {\r\n      disposeObserveYjsText?.()\r\n      disposeObserveYjsText = undefined\r\n\r\n      if (yjsText) {\r\n        yjsText.observe(observeFn)\r\n\r\n        disposeObserveYjsText = () => {\r\n          yjsText.unobserve(observeFn)\r\n        }\r\n      }\r\n\r\n      textChangedAtom.reportChanged()\r\n    },\r\n    {\r\n      fireImmediately: true,\r\n    }\r\n  )\r\n\r\n  return () => {\r\n    disposeReactionToYTextChange()\r\n    disposeObserveYjsText?.()\r\n    disposeObserveYjsText = undefined\r\n  }\r\n}\r\n","import * as Y from \"yjs\"\nimport { YjsTextModel, yjsTextModelId } from \"./YjsTextModel\"\nimport { SnapshotOutOf } from \"mobx-keystone\"\nimport { YjsData } from \"./convertYjsDataToJson\"\nimport { PlainArray, PlainObject, PlainPrimitive, PlainValue } from \"../plainTypes\"\n\nfunction isPlainPrimitive(v: PlainValue): v is PlainPrimitive {\n  const t = typeof v\n  return t === \"string\" || t === \"number\" || t === \"boolean\" || v === null || v === undefined\n}\n\nfunction isPlainArray(v: PlainValue): v is PlainArray {\n  return Array.isArray(v)\n}\n\nfunction isPlainObject(v: PlainValue): v is PlainObject {\n  return !isPlainArray(v) && typeof v === \"object\" && v !== null\n}\n\n/**\n * Converts a plain value to a Y.js data structure.\n * Objects are converted to Y.Maps, arrays to Y.Arrays, primitives are untouched.\n * Frozen values are a special case and they are kept as immutable plain values.\n */\nexport function convertJsonToYjsData(v: PlainValue): YjsData {\n  if (isPlainPrimitive(v)) {\n    return v\n  }\n\n  if (isPlainArray(v)) {\n    const arr = new Y.Array<YjsData>()\n    applyJsonArrayToYArray(arr, v)\n    return arr as YjsData\n  }\n\n  if (isPlainObject(v)) {\n    if (v.$frozen === true) {\n      // frozen value, save as immutable object\n      return v\n    }\n\n    if (v.$modelType === yjsTextModelId) {\n      const text = new Y.Text()\n      const yjsTextModel = v as unknown as SnapshotOutOf<YjsTextModel>\n      yjsTextModel.deltaList.forEach((frozenDeltas) => {\n        text.applyDelta(frozenDeltas.data)\n      })\n      return text\n    }\n\n    const map = new Y.Map<YjsData>()\n    applyJsonObjectToYMap(map, v)\n    return map as YjsData\n  }\n\n  throw new Error(`unsupported value type: ${v}`)\n}\n\n/**\n * Applies a JSON array to a Y.Array, using the convertJsonToYjsData to convert the values.\n */\nexport function applyJsonArrayToYArray(dest: Y.Array<YjsData>, source: PlainArray) {\n  dest.push(source.map(convertJsonToYjsData))\n}\n\n/**\n * Applies a JSON object to a Y.Map, using the convertJsonToYjsData to convert the values.\n */\nexport function applyJsonObjectToYMap(dest: Y.Map<YjsData>, source: PlainObject) {\n  Object.entries(source).forEach(([k, v]) => {\n    dest.set(k, convertJsonToYjsData(v))\n  })\n}\n","import { Patch } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { failure } from \"../utils/error\"\nimport { convertJsonToYjsData } from \"./convertJsonToYjsData\"\nimport { PlainValue } from \"../plainTypes\"\n\nexport function applyMobxKeystonePatchToYjsObject(patch: Patch, yjs: unknown): void {\n  if (patch.path.length > 1) {\n    const [key, ...rest] = patch.path\n\n    if (yjs instanceof Y.Map) {\n      const child = yjs.get(String(key)) as unknown\n      if (child === undefined) {\n        throw failure(\n          `invalid patch path, key \"${key}\" not found in Yjs map - patch: ${JSON.stringify(patch)}`\n        )\n      }\n      applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)\n    } else if (yjs instanceof Y.Array) {\n      const child = yjs.get(Number(key)) as unknown\n      if (child === undefined) {\n        throw failure(\n          `invalid patch path, key \"${key}\" not found in Yjs array - patch: ${JSON.stringify(\n            patch\n          )}`\n        )\n      }\n      applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)\n    } else if (yjs instanceof Y.Text) {\n      // changes to deltaList will be handled by the array observe in the YjsTextModel class\n    } else {\n      throw failure(\n        `invalid patch path, key \"${key}\" not found in unknown Yjs object - patch: ${JSON.stringify(\n          patch\n        )}`\n      )\n    }\n  } else if (patch.path.length === 1) {\n    if (yjs instanceof Y.Map) {\n      const key = String(patch.path[0])\n\n      switch (patch.op) {\n        case \"add\":\n        case \"replace\": {\n          yjs.set(key, convertJsonToYjsData(patch.value as PlainValue))\n          break\n        }\n        case \"remove\": {\n          yjs.delete(key)\n          break\n        }\n        default: {\n          throw failure(`invalid patch operation for map`)\n        }\n      }\n    } else if (yjs instanceof Y.Array) {\n      const key = patch.path[0]\n\n      switch (patch.op) {\n        case \"replace\": {\n          if (key === \"length\") {\n            const newLength = patch.value as number\n            if (yjs.length > newLength) {\n              const toDelete = yjs.length - newLength\n              yjs.delete(newLength, toDelete)\n            } else if (yjs.length < patch.value) {\n              const toInsert = patch.value - yjs.length\n              yjs.insert(yjs.length, Array.from({ length: toInsert }).fill(undefined))\n            }\n          } else {\n            yjs.delete(Number(key))\n            yjs.insert(Number(key), [convertJsonToYjsData(patch.value as PlainValue)])\n          }\n          break\n        }\n        case \"add\": {\n          yjs.insert(Number(key), [convertJsonToYjsData(patch.value as PlainValue)])\n          break\n        }\n        case \"remove\": {\n          yjs.delete(Number(key))\n          break\n        }\n        default: {\n          throw failure(`invalid patch operation for array`)\n        }\n      }\n    } else if (yjs instanceof Y.Text) {\n      // initialization of a YjsTextModel, do nothing\n    } else {\n      throw failure(\n        `invalid patch path, the Yjs object is of an unkown type, so key \"${String(patch.path[0])}\" cannot be found in it`\n      )\n    }\n  } else {\n    throw failure(`invalid patch path, it cannot be empty`)\n  }\n}\n","import { modelSnapshotOutWithMetadata } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { PlainObject, PlainValue } from \"../plainTypes\"\nimport { YjsTextModel } from \"./YjsTextModel\"\n\nexport type YjsData = Y.Array<YjsData> | Y.Map<YjsData> | Y.Text | PlainValue\n\nexport function convertYjsDataToJson(yjsData: YjsData): PlainValue {\n  if (yjsData instanceof Y.Array) {\n    return yjsData.map((v) => convertYjsDataToJson(v))\n  }\n\n  if (yjsData instanceof Y.Map) {\n    const obj: PlainObject = {}\n    yjsData.forEach((v, k) => {\n      obj[k] = convertYjsDataToJson(v)\n    })\n    return obj\n  }\n\n  if (yjsData instanceof Y.Text) {\n    const deltas = yjsData.toDelta() as unknown[]\n\n    return modelSnapshotOutWithMetadata(YjsTextModel, {\n      deltaList: deltas.length > 0 ? [{ $frozen: true, data: deltas }] : [],\n    }) as unknown as PlainValue\n  }\n\n  // assume it's a primitive\n  return yjsData\n}\n","import { Patch } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { PlainArray, PlainObject, PlainValue } from \"../plainTypes\"\nimport { failure } from \"../utils/error\"\n\nexport function convertYjsEventToPatches(event: Y.YEvent<any>): Patch[] {\n  const patches: Patch[] = []\n\n  if (event instanceof Y.YMapEvent) {\n    const source = event.target\n\n    event.changes.keys.forEach((change, key) => {\n      const path = [...event.path, key]\n\n      switch (change.action) {\n        case \"add\":\n          patches.push({\n            op: \"add\",\n            path,\n            value: toPlainValue(source.get(key)),\n          })\n          break\n\n        case \"update\":\n          patches.push({\n            op: \"replace\",\n            path,\n            value: toPlainValue(source.get(key)),\n          })\n          break\n\n        case \"delete\":\n          patches.push({\n            op: \"remove\",\n            path,\n          })\n          break\n\n        default:\n          throw failure(`unsupported Yjs map event action: ${change.action}`)\n      }\n    })\n  } else if (event instanceof Y.YArrayEvent) {\n    let retain = 0\n    event.changes.delta.forEach((change) => {\n      if (change.retain) {\n        retain += change.retain\n      }\n\n      if (change.delete) {\n        // remove X items at retain position\n        const path = [...event.path, retain]\n        for (let i = 0; i < change.delete; i++) {\n          patches.push({\n            op: \"remove\",\n            path,\n          })\n        }\n      }\n\n      if (change.insert) {\n        const newValues = Array.isArray(change.insert) ? change.insert : [change.insert]\n        newValues.forEach((v) => {\n          const path = [...event.path, retain]\n          patches.push({\n            op: \"add\",\n            path,\n            value: toPlainValue(v),\n          })\n          retain++\n        })\n      }\n    })\n  } else if (event instanceof Y.YTextEvent) {\n    const path = [...event.path, \"deltaList\", -1 /* last item */]\n    patches.push({\n      op: \"add\",\n      path,\n      value: { $frozen: true, data: event.delta },\n    })\n  }\n\n  return patches\n}\n\nfunction toPlainValue(v: Y.Map<any> | Y.Array<any> | PlainValue) {\n  if (v instanceof Y.Map || v instanceof Y.Array) {\n    return v.toJSON() as PlainObject | PlainArray\n  } else {\n    return v\n  }\n}\n","import { action } from \"mobx\"\nimport {\n  AnyDataModel,\n  AnyModel,\n  AnyStandardType,\n  ModelClass,\n  Patch,\n  SnapshotInOf,\n  TypeToData,\n  applyPatches,\n  fromSnapshot,\n  getParentToChildPath,\n  onGlobalPatches,\n  onPatches,\n  onSnapshot,\n} from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { getYjsCollectionAtom } from \"../utils/getOrCreateYjsCollectionAtom\"\nimport { applyMobxKeystonePatchToYjsObject } from \"./applyMobxKeystonePatchToYjsObject\"\nimport { YjsData, convertYjsDataToJson } from \"./convertYjsDataToJson\"\nimport { convertYjsEventToPatches } from \"./convertYjsEventToPatches\"\nimport { YjsBindingContext, yjsBindingContext } from \"./yjsBindingContext\"\n\n/**\n * Creates a bidirectional binding between a Y.js data structure and a mobx-keystone model.\n */\nexport function bindYjsToMobxKeystone<\n  TType extends AnyStandardType | ModelClass<AnyModel> | ModelClass<AnyDataModel>,\n>({\n  yjsDoc,\n  yjsObject,\n  mobxKeystoneType,\n}: {\n  /**\n   * The Y.js document.\n   */\n  yjsDoc: Y.Doc\n  /**\n   * The bound Y.js data structure.\n   */\n  yjsObject: Y.Map<unknown> | Y.Array<unknown> | Y.Text\n  /**\n   * The mobx-keystone model type.\n   */\n  mobxKeystoneType: TType\n}): {\n  /**\n   * The bound mobx-keystone instance.\n   */\n  boundObject: TypeToData<TType>\n  /**\n   * Disposes the binding.\n   */\n  dispose: () => void\n  /**\n   * The Y.js origin symbol used for binding transactions.\n   */\n  yjsOrigin: symbol\n} {\n  const yjsOrigin = Symbol(\"bindYjsToMobxKeystoneTransactionOrigin\")\n\n  let applyingYjsChangesToMobxKeystone = 0\n\n  const bindingContext: YjsBindingContext = {\n    yjsDoc,\n    yjsObject,\n    mobxKeystoneType,\n    yjsOrigin,\n    boundObject: undefined, // not yet created\n\n    get isApplyingYjsChangesToMobxKeystone() {\n      return applyingYjsChangesToMobxKeystone > 0\n    },\n  }\n\n  const yjsJson = convertYjsDataToJson(yjsObject as YjsData)\n\n  const initializationGlobalPatches: { target: object; patches: Patch[] }[] = []\n\n  const createBoundObject = () => {\n    const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {\n      initializationGlobalPatches.push({ target, patches })\n    })\n\n    try {\n      const boundObject = yjsBindingContext.apply(\n        () => fromSnapshot(mobxKeystoneType, yjsJson as unknown as SnapshotInOf<TypeToData<TType>>),\n        bindingContext\n      )\n      yjsBindingContext.set(boundObject, { ...bindingContext, boundObject })\n      return boundObject\n    } finally {\n      disposeOnGlobalPatches()\n    }\n  }\n\n  const boundObject = createBoundObject()\n\n  // bind any changes from yjs to mobx-keystone\n  const observeDeepCb = action((events: Y.YEvent<any>[]) => {\n    const patches: Patch[] = []\n    events.forEach((event) => {\n      if (event.transaction.origin !== yjsOrigin) {\n        patches.push(...convertYjsEventToPatches(event))\n      }\n\n      if (event.target instanceof Y.Map || event.target instanceof Y.Array) {\n        getYjsCollectionAtom(event.target)?.reportChanged()\n      }\n    })\n\n    if (patches.length > 0) {\n      applyingYjsChangesToMobxKeystone++\n      try {\n        applyPatches(boundObject, patches)\n      } finally {\n        applyingYjsChangesToMobxKeystone--\n      }\n    }\n  })\n\n  yjsObject.observeDeep(observeDeepCb)\n\n  // bind any changes from mobx-keystone to yjs\n  let pendingArrayOfArrayOfPatches: Patch[][] = []\n  const disposeOnPatches = onPatches(boundObject, (patches) => {\n    if (applyingYjsChangesToMobxKeystone > 0) {\n      return\n    }\n\n    pendingArrayOfArrayOfPatches.push(patches)\n  })\n\n  // this is only used so we can transact all patches to the snapshot boundary\n  const disposeOnSnapshot = onSnapshot(boundObject, () => {\n    if (pendingArrayOfArrayOfPatches.length === 0) {\n      return\n    }\n\n    const arrayOfArrayOfPatches = pendingArrayOfArrayOfPatches\n    pendingArrayOfArrayOfPatches = []\n\n    yjsDoc.transact(() => {\n      arrayOfArrayOfPatches.forEach((arrayOfPatches) => {\n        arrayOfPatches.forEach((patch) => {\n          applyMobxKeystonePatchToYjsObject(patch, yjsObject)\n        })\n      })\n    }, yjsOrigin)\n  })\n\n  // sync initial patches, that might include setting defaults, IDs, etc\n  yjsDoc.transact(() => {\n    // we need to skip initializations until we hit the initialization of the bound object\n    // this is because default objects might be created and initialized before the main object\n    // but we just need to catch when those are actually assigned to the bound object\n    let boundObjectFound = false\n\n    initializationGlobalPatches.forEach(({ target, patches }) => {\n      if (!boundObjectFound) {\n        if (target !== boundObject) {\n          return // skip\n        }\n        boundObjectFound = true\n      }\n\n      const parentToChildPath = getParentToChildPath(boundObject, target)\n      // this is undefined only if target is not a child of boundModel\n      if (parentToChildPath !== undefined) {\n        patches.forEach((patch) => {\n          applyMobxKeystonePatchToYjsObject(\n            {\n              ...patch,\n              path: [...parentToChildPath, ...patch.path],\n            },\n            yjsObject\n          )\n        })\n      }\n    })\n  }, yjsOrigin)\n\n  return {\n    boundObject,\n    dispose: () => {\n      disposeOnPatches()\n      disposeOnSnapshot()\n      yjsObject.unobserveDeep(observeDeepCb)\n    },\n    yjsOrigin,\n  }\n}\n"],"names":["YjsTextModel","boundObject"],"mappings":"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"}
|
|
624
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"mobx-keystone-yjs.esm.mjs","sources":["../src/utils/getOrCreateYjsCollectionAtom.ts","../src/utils/isYjsValueDeleted.ts","../src/utils/error.ts","../src/binding/resolveYjsPath.ts","../src/binding/yjsBindingContext.ts","../src/binding/YjsTextModel.ts","../src/binding/convertJsonToYjsData.ts","../src/binding/applyMobxKeystonePatchToYjsObject.ts","../src/binding/convertYjsDataToJson.ts","../src/binding/convertYjsEventToPatches.ts","../src/binding/bindYjsToMobxKeystone.ts"],"sourcesContent":["import { IAtom, createAtom } from \"mobx\"\nimport * as Y from \"yjs\"\n\nconst yjsCollectionAtoms = new WeakMap<Y.Map<unknown> | Y.Array<unknown>, IAtom>()\n\n/**\n * @internal\n */\nexport const getYjsCollectionAtom = (\n  yjsCollection: Y.Map<unknown> | Y.Array<unknown>\n): IAtom | undefined => {\n  return yjsCollectionAtoms.get(yjsCollection)\n}\n\n/**\n * @internal\n */\nexport const getOrCreateYjsCollectionAtom = (\n  yjsCollection: Y.Map<unknown> | Y.Array<unknown>\n): IAtom => {\n  let atom = yjsCollectionAtoms.get(yjsCollection)\n  if (!atom) {\n    atom = createAtom(`yjsCollectionAtom`)\n    yjsCollectionAtoms.set(yjsCollection, atom)\n  }\n  return atom\n}\n","import * as Y from \"yjs\"\n\n/**\n * Checks if a Y.js value has been deleted or its document destroyed.\n *\n * @param yjsValue The Y.js value to check.\n * @returns `true` if the value is deleted or destroyed, `false` otherwise.\n */\nexport function isYjsValueDeleted(yjsValue: unknown): boolean {\n  if (yjsValue instanceof Y.AbstractType) {\n    return !!(yjsValue as any)._item?.deleted || !!yjsValue.doc?.isDestroyed\n  }\n  return false\n}\n","/**\r\n * A mobx-keystone-yjs error.\r\n */\r\nexport class MobxKeystoneYjsError extends Error {\r\n  constructor(msg: string) {\r\n    super(msg)\r\n\r\n    // Set the prototype explicitly.\r\n    Object.setPrototypeOf(this, MobxKeystoneYjsError.prototype)\r\n  }\r\n}\r\n\r\n/**\r\n * @internal\r\n */\r\nexport function failure(msg: string) {\r\n  return new MobxKeystoneYjsError(msg)\r\n}\r\n","import * as Y from \"yjs\"\nimport { failure } from \"../utils/error\"\nimport { getOrCreateYjsCollectionAtom } from \"../utils/getOrCreateYjsCollectionAtom\"\n\nexport function resolveYjsPath(yjsObject: unknown, path: readonly (string | number)[]): unknown {\n  let currentYjsObject: unknown = yjsObject\n\n  path.forEach((pathPart, i) => {\n    if (currentYjsObject instanceof Y.Map) {\n      getOrCreateYjsCollectionAtom(currentYjsObject).reportObserved()\n      const key = String(pathPart)\n      currentYjsObject = currentYjsObject.get(key)\n    } else if (currentYjsObject instanceof Y.Array) {\n      getOrCreateYjsCollectionAtom(currentYjsObject).reportObserved()\n      const key = Number(pathPart)\n      currentYjsObject = currentYjsObject.get(key)\n    } else {\n      throw failure(\n        `Y.Map or Y.Array was expected at path ${JSON.stringify(\n          path.slice(0, i)\n        )} in order to resolve path ${JSON.stringify(path)}, but got ${currentYjsObject} instead`\n      )\n    }\n  })\n\n  return currentYjsObject\n}\n","import { AnyType, createContext } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\n\n/**\n * Context with info on how a mobx-keystone model is bound to a Y.js data structure.\n */\nexport interface YjsBindingContext {\n  /**\n   * The Y.js document.\n   */\n  yjsDoc: Y.Doc\n\n  /**\n   * The bound Y.js data structure.\n   */\n  yjsObject: Y.Map<unknown> | Y.Array<unknown> | Y.Text\n\n  /**\n   * The mobx-keystone model type.\n   */\n  mobxKeystoneType: AnyType\n\n  /**\n   * The origin symbol used for transactions.\n   */\n  yjsOrigin: symbol\n\n  /**\n   * The bound mobx-keystone instance.\n   */\n  boundObject: unknown\n\n  /**\n   * Whether we are currently applying Y.js changes to the mobx-keystone model.\n   */\n  isApplyingYjsChangesToMobxKeystone: boolean\n}\n\n/**\n * Context with info on how a mobx-keystone model is bound to a Y.js data structure.\n */\nexport const yjsBindingContext = createContext<YjsBindingContext | undefined>(undefined)\n","import { computed, createAtom, IAtom, observe, reaction } from \"mobx\"\nimport {\n  Frozen,\n  frozen,\n  getParentToChildPath,\n  Model,\n  model,\n  onSnapshot,\n  tProp,\n  types,\n} from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { failure } from \"../utils/error\"\nimport { isYjsValueDeleted } from \"../utils/isYjsValueDeleted\"\nimport { resolveYjsPath } from \"./resolveYjsPath\"\nimport { YjsBindingContext, yjsBindingContext } from \"./yjsBindingContext\"\n\n// Delta[][], since each single change is a Delta[]\n// we use frozen so that we can reuse each delta change\nconst deltaListType = types.array(types.frozen(types.unchecked<unknown[]>()))\n\nexport const yjsTextModelId = \"mobx-keystone-yjs/YjsTextModel\"\n\n/**\n * A mobx-keystone model that represents a Yjs.Text object.\n */\n@model(yjsTextModelId)\nexport class YjsTextModel extends Model({\n  deltaList: tProp(deltaListType, () => []),\n}) {\n  /**\n   * Helper function to create a YjsTextModel instance with a simple text.\n   */\n  static withText(text: string): YjsTextModel {\n    return new DecoratedYjsTextModel({\n      deltaList: [\n        frozen([\n          {\n            insert: text,\n          },\n        ]),\n      ],\n    })\n  }\n\n  /**\n   * The Y.js path from the bound object to the YjsTextModel instance.\n   */\n  @computed\n  private get _yjsObjectPath() {\n    const ctx = yjsBindingContext.get(this)\n    if (ctx?.boundObject == null) {\n      throw failure(\n        \"the YjsTextModel instance must be part of a bound object before it can be accessed\"\n      )\n    }\n\n    const path = getParentToChildPath(ctx.boundObject, this)\n    if (!path) {\n      throw failure(\"a path from the bound object to the YjsTextModel instance is not available\")\n    }\n\n    return path\n  }\n\n  /**\n   * The Yjs.Text object present at this mobx-keystone node's path.\n   */\n  @computed\n  private get _yjsObjectAtPath(): unknown {\n    const path = this._yjsObjectPath\n\n    const ctx = yjsBindingContext.get(this)!\n\n    return resolveYjsPath(ctx.yjsObject, path)\n  }\n\n  /**\n   * The Yjs.Text object represented by this mobx-keystone node.\n   */\n  @computed\n  get yjsText(): Y.Text {\n    const yjsObject = this._yjsObjectAtPath\n\n    if (!(yjsObject instanceof Y.Text)) {\n      throw failure(`Y.Text was expected at path ${JSON.stringify(this._yjsObjectPath)}`)\n    }\n\n    return yjsObject\n  }\n\n  /**\n   * Atom that gets changed when the associated Y.js text changes.\n   */\n  yjsTextChangedAtom = createAtom(\"yjsTextChangedAtom\")\n\n  /**\n   * The text value of the Yjs.Text object.\n   * Shortcut for `yjsText.toString()`, but computed.\n   */\n  @computed\n  get text(): string {\n    this.yjsTextChangedAtom.reportObserved()\n\n    const ctx = yjsBindingContext.get(this)\n    if (ctx?.boundObject != null) {\n      try {\n        const yjsTextString = this.yjsText.toString()\n        // if the yjsText is detached, toString() returns an empty string\n        // in that case we should use the deltaList as a fallback\n        if (yjsTextString !== \"\" || this.deltaList.length === 0) {\n          return yjsTextString\n        }\n      } catch {\n        // fall back\n      }\n    }\n\n    // fall back to deltaList\n    return this.deltaListToText()\n  }\n\n  private deltaListToText(): string {\n    const doc = new Y.Doc()\n    const text = doc.getText()\n    this.deltaList.forEach((d) => {\n      text.applyDelta(d.data)\n    })\n    return text.toString()\n  }\n\n  protected onInit() {\n    const shouldReplicateToYjs = (ctx: YjsBindingContext | undefined): ctx is YjsBindingContext => {\n      return !!ctx && !!ctx.boundObject && !ctx.isApplyingYjsChangesToMobxKeystone\n    }\n\n    let reapplyDeltasToYjsText = false\n    const newDeltas: Frozen<unknown[]>[] = []\n\n    let disposeObserveDeltaList: (() => void) | undefined\n\n    const disposeReactionToDeltaListRefChange = reaction(\n      () => this.$.deltaList,\n      (deltaList) => {\n        disposeObserveDeltaList?.()\n        disposeObserveDeltaList = undefined\n\n        disposeObserveDeltaList = observe(deltaList, (change) => {\n          if (reapplyDeltasToYjsText) {\n            // already gonna replace them all\n            return\n          }\n          if (!shouldReplicateToYjs(yjsBindingContext.get(this))) {\n            // yjs text is already up to date with these changes\n            return\n          }\n\n          if (\n            change.type === \"splice\" &&\n            change.removedCount === 0 &&\n            change.addedCount > 0 &&\n            change.index === this.deltaList.length\n          ) {\n            // optimization, just adding new ones to the end\n            newDeltas.push(...change.added)\n          } else {\n            // any other change, we need to reapply all deltas\n            reapplyDeltasToYjsText = true\n          }\n        })\n      },\n      { fireImmediately: true }\n    )\n\n    const disposeOnSnapshot = onSnapshot(this, () => {\n      try {\n        if (reapplyDeltasToYjsText) {\n          const ctx = yjsBindingContext.get(this)\n\n          if (shouldReplicateToYjs(ctx)) {\n            const { yjsText } = this\n            if (isYjsValueDeleted(yjsText)) {\n              throw failure(\"cannot reapply deltas to deleted Yjs.Text\")\n            }\n\n            ctx.yjsDoc.transact(() => {\n              // didn't find a better way than this to reapply all deltas\n              // without having to re-create the Y.Text object\n              if (yjsText.length > 0) {\n                yjsText.delete(0, yjsText.length)\n              }\n\n              this.deltaList.forEach((frozenDeltas) => {\n                yjsText.applyDelta(frozenDeltas.data)\n              })\n            }, ctx.yjsOrigin)\n          }\n        } else if (newDeltas.length > 0) {\n          const ctx = yjsBindingContext.get(this)\n\n          if (shouldReplicateToYjs(ctx)) {\n            const { yjsText } = this\n            if (isYjsValueDeleted(yjsText)) {\n              throw failure(\"cannot reapply deltas to deleted Yjs.Text\")\n            }\n\n            ctx.yjsDoc.transact(() => {\n              newDeltas.forEach((frozenDeltas) => {\n                yjsText.applyDelta(frozenDeltas.data)\n              })\n            }, ctx.yjsOrigin)\n          }\n        }\n      } finally {\n        reapplyDeltasToYjsText = false\n        newDeltas.length = 0\n      }\n    })\n\n    const diposeYjsTextChangedAtom = hookYjsTextChangedAtom(\n      () => this.yjsText,\n      this.yjsTextChangedAtom\n    )\n\n    return () => {\n      disposeOnSnapshot()\n      disposeReactionToDeltaListRefChange()\n      disposeObserveDeltaList?.()\n      disposeObserveDeltaList = undefined\n\n      diposeYjsTextChangedAtom()\n    }\n  }\n}\n\n// we use this trick just to avoid a babel bug that causes classes used inside classes not to be overriden\n// by the decorator\nconst DecoratedYjsTextModel = YjsTextModel\n\nfunction hookYjsTextChangedAtom(getYjsText: () => Y.Text, textChangedAtom: IAtom) {\n  let disposeObserveYjsText: (() => void) | undefined\n\n  const observeFn = () => {\n    textChangedAtom.reportChanged()\n  }\n\n  const disposeReactionToYTextChange = reaction(\n    () => {\n      try {\n        const yjsText = getYjsText()\n        return isYjsValueDeleted(yjsText) ? undefined : yjsText\n      } catch {\n        return undefined\n      }\n    },\n    (yjsText) => {\n      disposeObserveYjsText?.()\n      disposeObserveYjsText = undefined\n\n      if (yjsText) {\n        yjsText.observe(observeFn)\n\n        disposeObserveYjsText = () => {\n          yjsText.unobserve(observeFn)\n        }\n      }\n\n      textChangedAtom.reportChanged()\n    },\n    {\n      fireImmediately: true,\n    }\n  )\n\n  return () => {\n    disposeReactionToYTextChange()\n    disposeObserveYjsText?.()\n    disposeObserveYjsText = undefined\n  }\n}\n","import * as Y from \"yjs\"\nimport { YjsTextModel, yjsTextModelId } from \"./YjsTextModel\"\nimport { SnapshotOutOf } from \"mobx-keystone\"\nimport { YjsData } from \"./convertYjsDataToJson\"\nimport { PlainArray, PlainObject, PlainPrimitive, PlainValue } from \"../plainTypes\"\n\nfunction isPlainPrimitive(v: PlainValue): v is PlainPrimitive {\n  const t = typeof v\n  return t === \"string\" || t === \"number\" || t === \"boolean\" || v === null || v === undefined\n}\n\nfunction isPlainArray(v: PlainValue): v is PlainArray {\n  return Array.isArray(v)\n}\n\nfunction isPlainObject(v: PlainValue): v is PlainObject {\n  return !isPlainArray(v) && typeof v === \"object\" && v !== null\n}\n\n/**\n * Converts a plain value to a Y.js data structure.\n * Objects are converted to Y.Maps, arrays to Y.Arrays, primitives are untouched.\n * Frozen values are a special case and they are kept as immutable plain values.\n */\nexport function convertJsonToYjsData(v: PlainValue): YjsData {\n  if (isPlainPrimitive(v)) {\n    return v\n  }\n\n  if (isPlainArray(v)) {\n    const arr = new Y.Array()\n    applyJsonArrayToYArray(arr, v)\n    return arr\n  }\n\n  if (isPlainObject(v)) {\n    if (v.$frozen === true) {\n      // frozen value, save as immutable object\n      return v\n    }\n\n    if (v.$modelType === yjsTextModelId) {\n      const text = new Y.Text()\n      const yjsTextModel = v as unknown as SnapshotOutOf<YjsTextModel>\n      yjsTextModel.deltaList.forEach((frozenDeltas) => {\n        text.applyDelta(frozenDeltas.data)\n      })\n      return text\n    }\n\n    const map = new Y.Map()\n    applyJsonObjectToYMap(map, v)\n    return map\n  }\n\n  throw new Error(`unsupported value type: ${v}`)\n}\n\n/**\n * Applies a JSON array to a Y.Array, using the convertJsonToYjsData to convert the values.\n */\nexport const applyJsonArrayToYArray = (dest: Y.Array<any>, source: PlainArray) => {\n  dest.push(source.map(convertJsonToYjsData))\n}\n\n/**\n * Applies a JSON object to a Y.Map, using the convertJsonToYjsData to convert the values.\n */\nexport const applyJsonObjectToYMap = (dest: Y.Map<any>, source: PlainObject) => {\n  Object.entries(source).forEach(([k, v]) => {\n    dest.set(k, convertJsonToYjsData(v))\n  })\n}\n","import { Patch } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { PlainValue } from \"../plainTypes\"\nimport { failure } from \"../utils/error\"\nimport { isYjsValueDeleted } from \"../utils/isYjsValueDeleted\"\nimport { convertJsonToYjsData } from \"./convertJsonToYjsData\"\n\nexport function applyMobxKeystonePatchToYjsObject(patch: Patch, yjs: unknown): void {\n  if (isYjsValueDeleted(yjs)) {\n    throw failure(\"cannot apply patch to deleted Yjs value\")\n  }\n\n  if (patch.path.length > 1) {\n    const [key, ...rest] = patch.path\n\n    if (yjs instanceof Y.Map) {\n      const child = yjs.get(String(key)) as unknown\n      if (child === undefined) {\n        throw failure(\n          `invalid patch path, key \"${key}\" not found in Yjs map - patch: ${JSON.stringify(patch)}`\n        )\n      }\n      applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)\n    } else if (yjs instanceof Y.Array) {\n      const child = yjs.get(Number(key)) as unknown\n      if (child === undefined) {\n        throw failure(\n          `invalid patch path, key \"${key}\" not found in Yjs array - patch: ${JSON.stringify(\n            patch\n          )}`\n        )\n      }\n      applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)\n    } else if (yjs instanceof Y.Text) {\n      // changes to deltaList will be handled by the array observe in the YjsTextModel class\n    } else {\n      throw failure(\n        `invalid patch path, key \"${key}\" not found in unknown Yjs object - patch: ${JSON.stringify(\n          patch\n        )}`\n      )\n    }\n  } else if (patch.path.length === 1) {\n    if (yjs instanceof Y.Map) {\n      const key = String(patch.path[0])\n\n      switch (patch.op) {\n        case \"add\":\n        case \"replace\": {\n          yjs.set(key, convertJsonToYjsData(patch.value as PlainValue))\n          break\n        }\n        case \"remove\": {\n          yjs.delete(key)\n          break\n        }\n        default: {\n          throw failure(`invalid patch operation for map`)\n        }\n      }\n    } else if (yjs instanceof Y.Array) {\n      const key = patch.path[0]\n\n      switch (patch.op) {\n        case \"replace\": {\n          if (key === \"length\") {\n            const newLength = patch.value as number\n            if (yjs.length > newLength) {\n              const toDelete = yjs.length - newLength\n              yjs.delete(newLength, toDelete)\n            } else if (yjs.length < patch.value) {\n              const toInsert = patch.value - yjs.length\n              yjs.insert(yjs.length, Array.from({ length: toInsert }).fill(undefined))\n            }\n          } else {\n            yjs.delete(Number(key))\n            yjs.insert(Number(key), [convertJsonToYjsData(patch.value as PlainValue)])\n          }\n          break\n        }\n        case \"add\": {\n          yjs.insert(Number(key), [convertJsonToYjsData(patch.value as PlainValue)])\n          break\n        }\n        case \"remove\": {\n          yjs.delete(Number(key))\n          break\n        }\n        default: {\n          throw failure(`invalid patch operation for array`)\n        }\n      }\n    } else if (yjs instanceof Y.Text) {\n      // initialization of a YjsTextModel, do nothing\n    } else {\n      throw failure(\n        `invalid patch path, the Yjs object is of an unkown type, so key \"${String(patch.path[0])}\" cannot be found in it`\n      )\n    }\n  } else {\n    throw failure(`invalid patch path, it cannot be empty`)\n  }\n}\n","import { action } from \"mobx\"\nimport { modelSnapshotOutWithMetadata } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { PlainObject, PlainValue } from \"../plainTypes\"\nimport { YjsTextModel } from \"./YjsTextModel\"\n\nexport type YjsData = Y.Array<any> | Y.Map<any> | Y.Text | PlainValue\n\nexport const convertYjsDataToJson = action((yjsData: YjsData): PlainValue => {\n  if (yjsData instanceof Y.Array) {\n    return yjsData.map((v) => convertYjsDataToJson(v))\n  }\n\n  if (yjsData instanceof Y.Map) {\n    const obj: PlainObject = {}\n    yjsData.forEach((v, k) => {\n      obj[k] = convertYjsDataToJson(v)\n    })\n    return obj\n  }\n\n  if (yjsData instanceof Y.Text) {\n    const deltas = yjsData.toDelta() as unknown[]\n\n    return modelSnapshotOutWithMetadata(YjsTextModel, {\n      deltaList: deltas.length > 0 ? [{ $frozen: true, data: deltas }] : [],\n    }) as unknown as PlainValue\n  }\n\n  // assume it's a primitive\n  return yjsData\n})\n","import { Patch } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { failure } from \"../utils/error\"\nimport { convertYjsDataToJson } from \"./convertYjsDataToJson\"\n\nexport function convertYjsEventToPatches(event: Y.YEvent<any>): Patch[] {\n  const patches: Patch[] = []\n\n  if (event instanceof Y.YMapEvent) {\n    const source = event.target\n\n    event.changes.keys.forEach((change, key) => {\n      const path = [...event.path, key]\n\n      switch (change.action) {\n        case \"add\":\n          patches.push({\n            op: \"add\",\n            path,\n            value: convertYjsDataToJson(source.get(key)),\n          })\n          break\n\n        case \"update\":\n          patches.push({\n            op: \"replace\",\n            path,\n            value: convertYjsDataToJson(source.get(key)),\n          })\n          break\n\n        case \"delete\":\n          patches.push({\n            op: \"remove\",\n            path,\n          })\n          break\n\n        default:\n          throw failure(`unsupported Yjs map event action: ${change.action}`)\n      }\n    })\n  } else if (event instanceof Y.YArrayEvent) {\n    let retain = 0\n    event.changes.delta.forEach((change) => {\n      if (change.retain) {\n        retain += change.retain\n      }\n\n      if (change.delete) {\n        // remove X items at retain position\n        const path = [...event.path, retain]\n        for (let i = 0; i < change.delete; i++) {\n          patches.push({\n            op: \"remove\",\n            path,\n          })\n        }\n      }\n\n      if (change.insert) {\n        const newValues = Array.isArray(change.insert) ? change.insert : [change.insert]\n        newValues.forEach((v) => {\n          const path = [...event.path, retain]\n          patches.push({\n            op: \"add\",\n            path,\n            value: convertYjsDataToJson(v),\n          })\n          retain++\n        })\n      }\n    })\n  } else if (event instanceof Y.YTextEvent) {\n    const path = [...event.path, \"deltaList\", -1 /* last item */]\n    patches.push({\n      op: \"add\",\n      path,\n      value: { $frozen: true, data: event.delta },\n    })\n  }\n\n  return patches\n}\n","import { action } from \"mobx\"\nimport {\n  AnyDataModel,\n  AnyModel,\n  AnyStandardType,\n  applyPatches,\n  fromSnapshot,\n  getParentToChildPath,\n  ModelClass,\n  onGlobalPatches,\n  onPatches,\n  onSnapshot,\n  Patch,\n  SnapshotInOf,\n  TypeToData,\n} from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { getYjsCollectionAtom } from \"../utils/getOrCreateYjsCollectionAtom\"\nimport { isYjsValueDeleted } from \"../utils/isYjsValueDeleted\"\nimport { applyMobxKeystonePatchToYjsObject } from \"./applyMobxKeystonePatchToYjsObject\"\nimport { convertYjsDataToJson } from \"./convertYjsDataToJson\"\nimport { convertYjsEventToPatches } from \"./convertYjsEventToPatches\"\nimport { YjsBindingContext, yjsBindingContext } from \"./yjsBindingContext\"\n\n/**\n * Creates a bidirectional binding between a Y.js data structure and a mobx-keystone model.\n */\nexport function bindYjsToMobxKeystone<\n  TType extends AnyStandardType | ModelClass<AnyModel> | ModelClass<AnyDataModel>,\n>({\n  yjsDoc,\n  yjsObject,\n  mobxKeystoneType,\n}: {\n  /**\n   * The Y.js document.\n   */\n  yjsDoc: Y.Doc\n  /**\n   * The bound Y.js data structure.\n   */\n  yjsObject: Y.Map<any> | Y.Array<any> | Y.Text\n  /**\n   * The mobx-keystone model type.\n   */\n  mobxKeystoneType: TType\n}): {\n  /**\n   * The bound mobx-keystone instance.\n   */\n  boundObject: TypeToData<TType>\n  /**\n   * Disposes the binding.\n   */\n  dispose: () => void\n  /**\n   * The Y.js origin symbol used for binding transactions.\n   */\n  yjsOrigin: symbol\n} {\n  const yjsOrigin = Symbol(\"bindYjsToMobxKeystoneTransactionOrigin\")\n\n  let applyingYjsChangesToMobxKeystone = 0\n\n  const bindingContext: YjsBindingContext = {\n    yjsDoc,\n    yjsObject,\n    mobxKeystoneType,\n    yjsOrigin,\n    boundObject: undefined, // not yet created\n\n    get isApplyingYjsChangesToMobxKeystone() {\n      return applyingYjsChangesToMobxKeystone > 0\n    },\n  }\n\n  const yjsJson = convertYjsDataToJson(yjsObject)\n\n  const initializationGlobalPatches: { target: object; patches: Patch[] }[] = []\n\n  const createBoundObject = () => {\n    const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {\n      initializationGlobalPatches.push({ target, patches })\n    })\n\n    try {\n      const boundObject = yjsBindingContext.apply(\n        () => fromSnapshot(mobxKeystoneType, yjsJson as unknown as SnapshotInOf<TypeToData<TType>>),\n        bindingContext\n      )\n      yjsBindingContext.set(boundObject, { ...bindingContext, boundObject })\n      return boundObject\n    } finally {\n      disposeOnGlobalPatches()\n    }\n  }\n\n  const boundObject = createBoundObject()\n\n  // bind any changes from yjs to mobx-keystone\n  const observeDeepCb = action((events: Y.YEvent<any>[]) => {\n    const patches: Patch[] = []\n    events.forEach((event) => {\n      if (event.transaction.origin !== yjsOrigin) {\n        patches.push(...convertYjsEventToPatches(event))\n      }\n\n      if (event.target instanceof Y.Map || event.target instanceof Y.Array) {\n        getYjsCollectionAtom(event.target)?.reportChanged()\n      }\n    })\n\n    if (patches.length > 0) {\n      applyingYjsChangesToMobxKeystone++\n      try {\n        applyPatches(boundObject, patches)\n      } finally {\n        applyingYjsChangesToMobxKeystone--\n      }\n    }\n  })\n\n  yjsObject.observeDeep(observeDeepCb)\n\n  // bind any changes from mobx-keystone to yjs\n  let pendingArrayOfArrayOfPatches: Patch[][] = []\n  const disposeOnPatches = onPatches(boundObject, (patches) => {\n    if (applyingYjsChangesToMobxKeystone > 0) {\n      return\n    }\n\n    pendingArrayOfArrayOfPatches.push(patches)\n  })\n\n  // this is only used so we can transact all patches to the snapshot boundary\n  const disposeOnSnapshot = onSnapshot(boundObject, () => {\n    if (pendingArrayOfArrayOfPatches.length === 0) {\n      return\n    }\n\n    const arrayOfArrayOfPatches = pendingArrayOfArrayOfPatches\n    pendingArrayOfArrayOfPatches = []\n\n    if (isYjsValueDeleted(yjsObject)) {\n      return\n    }\n\n    yjsDoc.transact(() => {\n      arrayOfArrayOfPatches.forEach((arrayOfPatches) => {\n        arrayOfPatches.forEach((patch) => {\n          applyMobxKeystonePatchToYjsObject(patch, yjsObject)\n        })\n      })\n    }, yjsOrigin)\n  })\n\n  // sync initial patches, that might include setting defaults, IDs, etc\n  yjsDoc.transact(() => {\n    // we need to skip initializations until we hit the initialization of the bound object\n    // this is because default objects might be created and initialized before the main object\n    // but we just need to catch when those are actually assigned to the bound object\n    let boundObjectFound = false\n\n    initializationGlobalPatches.forEach(({ target, patches }) => {\n      if (!boundObjectFound) {\n        if (target !== boundObject) {\n          return // skip\n        }\n        boundObjectFound = true\n      }\n\n      const parentToChildPath = getParentToChildPath(boundObject, target)\n      // this is undefined only if target is not a child of boundModel\n      if (parentToChildPath !== undefined) {\n        patches.forEach((patch) => {\n          applyMobxKeystonePatchToYjsObject(\n            {\n              ...patch,\n              path: [...parentToChildPath, ...patch.path],\n            },\n            yjsObject\n          )\n        })\n      }\n    })\n  }, yjsOrigin)\n\n  const dispose = () => {\n    yjsDoc.off(\"destroy\", dispose)\n    disposeOnPatches()\n    disposeOnSnapshot()\n    yjsObject.unobserveDeep(observeDeepCb)\n  }\n\n  yjsDoc.on(\"destroy\", dispose)\n\n  return {\n    boundObject,\n    dispose,\n    yjsOrigin,\n  }\n}\n"],"names":["boundObject"],"mappings":"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"}
|