mobx-keystone-yjs 1.5.4 → 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.
@@ -1,29 +1,9 @@
1
1
  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
- import { createAtom, reaction, observe, computed, action, runInAction } from "mobx";
5
- import { createContext, types, Model, tProp, frozen, getParentToChildPath, onSnapshot, model, modelSnapshotOutWithMetadata, applyPatches, onPatches, onGlobalPatches, fromSnapshot } from "mobx-keystone";
4
+ import { createAtom, reaction, observe, computed, action } from "mobx";
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(`Y.Map or Y.Array was expected at path ${JSON.stringify(path.slice(0, i))} in order to resolve path ${JSON.stringify(path)}, but got ${currentYjsObject} instead`);
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 YjsTextModel2 extends Model({
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("the YjsTextModel instance must be part of a bound object before it can be accessed");
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
- return this.yjsText.toString();
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(() => this.$.deltaList, (deltaList) => {
130
- disposeObserveDeltaList == null ? void 0 : disposeObserveDeltaList();
131
- disposeObserveDeltaList = void 0;
132
- disposeObserveDeltaList = observe(deltaList, (change) => {
133
- if (reapplyDeltasToYjsText) {
134
- return;
135
- }
136
- if (!shouldReplicateToYjs(yjsBindingContext.get(this))) {
137
- return;
138
- }
139
- if (change.type === "splice" && change.removedCount === 0 && change.addedCount > 0 && change.index === this.deltaList.length) {
140
- newDeltas.push(...change.added);
141
- } else {
142
- reapplyDeltasToYjsText = true;
143
- }
144
- });
145
- }, { fireImmediately: true });
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(() => this.yjsText, this.yjsTextChangedAtom);
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
- __decorate([
218
+ __decorateClass([
188
219
  computed
189
- ], YjsTextModel.prototype, "_yjsObjectPath", null);
190
- __decorate([
220
+ ], YjsTextModel.prototype, "_yjsObjectPath", 1);
221
+ __decorateClass([
191
222
  computed
192
- ], YjsTextModel.prototype, "_yjsObjectAtPath", null);
193
- __decorate([
223
+ ], YjsTextModel.prototype, "_yjsObjectAtPath", 1);
224
+ __decorateClass([
194
225
  computed
195
- ], YjsTextModel.prototype, "yjsText", null);
196
- __decorate([
226
+ ], YjsTextModel.prototype, "yjsText", 1);
227
+ __decorateClass([
197
228
  computed
198
- ], YjsTextModel.prototype, "text", null);
199
- YjsTextModel = __decorate([
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
- try {
210
- return getYjsText();
211
- } catch {
212
- return void 0;
213
- }
214
- }, (yjsText) => {
215
- disposeObserveYjsText == null ? void 0 : disposeObserveYjsText();
216
- disposeObserveYjsText = void 0;
217
- if (yjsText) {
218
- yjsText.observe(observeFn);
219
- disposeObserveYjsText = () => {
220
- yjsText.unobserve(observeFn);
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
- textChangedAtom.reportChanged();
224
- }, {
225
- fireImmediately: true
226
- });
262
+ );
227
263
  return () => {
228
264
  disposeReactionToYTextChange();
229
265
  disposeObserveYjsText == null ? void 0 : disposeObserveYjsText();
@@ -241,60 +277,71 @@ function isPlainObject(v) {
241
277
  return !isPlainArray(v) && typeof v === "object" && v !== null;
242
278
  }
243
279
  function convertJsonToYjsData(v) {
244
- return runInAction(() => {
245
- if (isPlainPrimitive(v)) {
280
+ if (isPlainPrimitive(v)) {
281
+ return v;
282
+ }
283
+ if (isPlainArray(v)) {
284
+ const arr = new Y.Array();
285
+ applyJsonArrayToYArray(arr, v);
286
+ return arr;
287
+ }
288
+ if (isPlainObject(v)) {
289
+ if (v.$frozen === true) {
246
290
  return v;
247
291
  }
248
- if (isPlainArray(v)) {
249
- const arr = new Y.Array();
250
- applyJsonArrayToYArray(arr, v);
251
- return arr;
252
- }
253
- if (isPlainObject(v)) {
254
- if (v.$frozen === true) {
255
- return v;
256
- }
257
- if (v.$modelType === yjsTextModelId) {
258
- const text = new Y.Text();
259
- const yjsTextModel = v;
260
- yjsTextModel.deltaList.forEach((frozenDeltas) => {
261
- text.applyDelta(frozenDeltas.data);
262
- });
263
- return text;
264
- }
265
- const map = new Y.Map();
266
- applyJsonObjectToYMap(map, v);
267
- return map;
292
+ if (v.$modelType === yjsTextModelId) {
293
+ const text = new Y.Text();
294
+ const yjsTextModel = v;
295
+ yjsTextModel.deltaList.forEach((frozenDeltas) => {
296
+ text.applyDelta(frozenDeltas.data);
297
+ });
298
+ return text;
268
299
  }
269
- throw new Error(`unsupported value type: ${v}`);
270
- });
300
+ const map = new Y.Map();
301
+ applyJsonObjectToYMap(map, v);
302
+ return map;
303
+ }
304
+ throw new Error(`unsupported value type: ${v}`);
271
305
  }
272
- const applyJsonArrayToYArray = action((dest, source) => {
306
+ const applyJsonArrayToYArray = (dest, source) => {
273
307
  dest.push(source.map(convertJsonToYjsData));
274
- });
275
- const applyJsonObjectToYMap = action((dest, source) => {
308
+ };
309
+ const applyJsonObjectToYMap = (dest, source) => {
276
310
  Object.entries(source).forEach(([k, v]) => {
277
311
  dest.set(k, convertJsonToYjsData(v));
278
312
  });
279
- });
313
+ };
280
314
  function applyMobxKeystonePatchToYjsObject(patch, yjs) {
315
+ if (isYjsValueDeleted(yjs)) {
316
+ throw failure("cannot apply patch to deleted Yjs value");
317
+ }
281
318
  if (patch.path.length > 1) {
282
319
  const [key, ...rest] = patch.path;
283
320
  if (yjs instanceof Y.Map) {
284
321
  const child = yjs.get(String(key));
285
322
  if (child === void 0) {
286
- throw failure(`invalid patch path, key "${key}" not found in Yjs map - patch: ${JSON.stringify(patch)}`);
323
+ throw failure(
324
+ `invalid patch path, key "${key}" not found in Yjs map - patch: ${JSON.stringify(patch)}`
325
+ );
287
326
  }
288
327
  applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child);
289
328
  } else if (yjs instanceof Y.Array) {
290
329
  const child = yjs.get(Number(key));
291
330
  if (child === void 0) {
292
- throw failure(`invalid patch path, key "${key}" not found in Yjs array - patch: ${JSON.stringify(patch)}`);
331
+ throw failure(
332
+ `invalid patch path, key "${key}" not found in Yjs array - patch: ${JSON.stringify(
333
+ patch
334
+ )}`
335
+ );
293
336
  }
294
337
  applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child);
295
338
  } else if (yjs instanceof Y.Text) ;
296
339
  else {
297
- throw failure(`invalid patch path, key "${key}" not found in unknown Yjs object - patch: ${JSON.stringify(patch)}`);
340
+ throw failure(
341
+ `invalid patch path, key "${key}" not found in unknown Yjs object - patch: ${JSON.stringify(
342
+ patch
343
+ )}`
344
+ );
298
345
  }
299
346
  } else if (patch.path.length === 1) {
300
347
  if (yjs instanceof Y.Map) {
@@ -346,7 +393,9 @@ function applyMobxKeystonePatchToYjsObject(patch, yjs) {
346
393
  }
347
394
  } else if (yjs instanceof Y.Text) ;
348
395
  else {
349
- throw failure(`invalid patch path, the Yjs object is of an unkown type, so key "${String(patch.path[0])}" cannot be found in it`);
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
+ );
350
399
  }
351
400
  } else {
352
401
  throw failure(`invalid patch path, it cannot be empty`);
@@ -382,14 +431,14 @@ function convertYjsEventToPatches(event) {
382
431
  patches.push({
383
432
  op: "add",
384
433
  path,
385
- value: toPlainValue(source.get(key))
434
+ value: convertYjsDataToJson(source.get(key))
386
435
  });
387
436
  break;
388
437
  case "update":
389
438
  patches.push({
390
439
  op: "replace",
391
440
  path,
392
- value: toPlainValue(source.get(key))
441
+ value: convertYjsDataToJson(source.get(key))
393
442
  });
394
443
  break;
395
444
  case "delete":
@@ -424,7 +473,7 @@ function convertYjsEventToPatches(event) {
424
473
  patches.push({
425
474
  op: "add",
426
475
  path,
427
- value: toPlainValue(v)
476
+ value: convertYjsDataToJson(v)
428
477
  });
429
478
  retain++;
430
479
  });
@@ -445,15 +494,12 @@ function convertYjsEventToPatches(event) {
445
494
  }
446
495
  return patches;
447
496
  }
448
- function toPlainValue(v) {
449
- if (v instanceof Y.Map || v instanceof Y.Array) {
450
- return v.toJSON();
451
- } else {
452
- return v;
453
- }
454
- }
455
- function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
456
- const yjsOrigin = Symbol("bindYjsToMobxKeystoneTransactionOrigin");
497
+ function bindYjsToMobxKeystone({
498
+ yjsDoc,
499
+ yjsObject,
500
+ mobxKeystoneType
501
+ }) {
502
+ const yjsOrigin = /* @__PURE__ */ Symbol("bindYjsToMobxKeystoneTransactionOrigin");
457
503
  let applyingYjsChangesToMobxKeystone = 0;
458
504
  const bindingContext = {
459
505
  yjsDoc,
@@ -473,7 +519,10 @@ function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
473
519
  initializationGlobalPatches.push({ target, patches });
474
520
  });
475
521
  try {
476
- const boundObject2 = yjsBindingContext.apply(() => fromSnapshot(mobxKeystoneType, yjsJson), bindingContext);
522
+ const boundObject2 = yjsBindingContext.apply(
523
+ () => fromSnapshot(mobxKeystoneType, yjsJson),
524
+ bindingContext
525
+ );
477
526
  yjsBindingContext.set(boundObject2, { ...bindingContext, boundObject: boundObject2 });
478
527
  return boundObject2;
479
528
  } finally {
@@ -515,6 +564,9 @@ function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
515
564
  }
516
565
  const arrayOfArrayOfPatches = pendingArrayOfArrayOfPatches;
517
566
  pendingArrayOfArrayOfPatches = [];
567
+ if (isYjsValueDeleted(yjsObject)) {
568
+ return;
569
+ }
518
570
  yjsDoc.transact(() => {
519
571
  arrayOfArrayOfPatches.forEach((arrayOfPatches) => {
520
572
  arrayOfPatches.forEach((patch) => {
@@ -535,21 +587,27 @@ function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
535
587
  const parentToChildPath = getParentToChildPath(boundObject, target);
536
588
  if (parentToChildPath !== void 0) {
537
589
  patches.forEach((patch) => {
538
- applyMobxKeystonePatchToYjsObject({
539
- ...patch,
540
- path: [...parentToChildPath, ...patch.path]
541
- }, yjsObject);
590
+ applyMobxKeystonePatchToYjsObject(
591
+ {
592
+ ...patch,
593
+ path: [...parentToChildPath, ...patch.path]
594
+ },
595
+ yjsObject
596
+ );
542
597
  });
543
598
  }
544
599
  });
545
600
  }, yjsOrigin);
601
+ const dispose = () => {
602
+ yjsDoc.off("destroy", dispose);
603
+ disposeOnPatches();
604
+ disposeOnSnapshot();
605
+ yjsObject.unobserveDeep(observeDeepCb);
606
+ };
607
+ yjsDoc.on("destroy", dispose);
546
608
  return {
547
609
  boundObject,
548
- dispose: () => {
549
- disposeOnPatches();
550
- disposeOnSnapshot();
551
- yjsObject.unobserveDeep(observeDeepCb);
552
- },
610
+ dispose,
553
611
  yjsOrigin
554
612
  };
555
613
  }
@@ -563,4 +621,4 @@ export {
563
621
  yjsBindingContext,
564
622
  yjsTextModelId
565
623
  };
566
- //# 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\"\nimport { action, runInAction } from \"mobx\"\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  return runInAction(() => {\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/**\n * Applies a JSON array to a Y.Array, using the convertJsonToYjsData to convert the values.\n */\nexport const applyJsonArrayToYArray = action((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 = action((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 { 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\"\nimport { action } from \"mobx\"\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 { 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 { 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    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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"}