mobx-keystone-yjs 1.5.4 → 1.6.0
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 +57 -45
- package/dist/mobx-keystone-yjs.esm.js +475 -299
- package/dist/mobx-keystone-yjs.esm.mjs +475 -299
- package/dist/mobx-keystone-yjs.umd.js +475 -299
- package/dist/types/binding/YjsTextModel.d.ts +5 -4
- package/dist/types/binding/applyMobxChangeToYjsObject.d.ts +3 -0
- package/dist/types/binding/applyYjsEventToMobx.d.ts +8 -0
- package/dist/types/binding/bindYjsToMobxKeystone.d.ts +1 -1
- package/dist/types/binding/convertJsonToYjsData.d.ts +23 -4
- package/dist/types/binding/convertYjsDataToJson.d.ts +1 -1
- package/dist/types/binding/resolveYjsPath.d.ts +14 -1
- package/dist/types/binding/yjsBindingContext.d.ts +2 -2
- package/dist/types/binding/yjsSnapshotTracking.d.ts +24 -0
- package/dist/types/index.d.ts +7 -6
- package/dist/types/utils/isYjsValueDeleted.d.ts +7 -0
- package/package.json +90 -78
- package/src/binding/YjsTextModel.ts +280 -247
- package/src/binding/applyMobxChangeToYjsObject.ts +77 -0
- package/src/binding/applyYjsEventToMobx.ts +173 -0
- package/src/binding/bindYjsToMobxKeystone.ts +300 -192
- package/src/binding/convertJsonToYjsData.ts +218 -76
- package/src/binding/convertYjsDataToJson.ts +1 -1
- package/src/binding/resolveYjsPath.ts +51 -27
- package/src/binding/yjsSnapshotTracking.ts +40 -0
- package/src/index.ts +11 -10
- package/src/utils/getOrCreateYjsCollectionAtom.ts +27 -27
- package/src/utils/isYjsValueDeleted.ts +14 -0
- package/dist/types/binding/applyMobxKeystonePatchToYjsObject.d.ts +0 -2
- package/dist/types/binding/convertYjsEventToPatches.d.ts +0 -3
- package/src/binding/applyMobxKeystonePatchToYjsObject.ts +0 -98
- package/src/binding/convertYjsEventToPatches.ts +0 -92
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
(function(global, factory) {
|
|
2
2
|
typeof exports === "object" && typeof module !== "undefined" ? factory(exports, require("mobx"), require("mobx-keystone"), require("yjs")) : typeof define === "function" && define.amd ? define(["exports", "mobx", "mobx-keystone", "yjs"], factory) : (global = typeof globalThis !== "undefined" ? globalThis : global || self, factory(global["mobx-keystone-yjs"] = {}, global.mobx, global["mobx-keystone"], global.yjs));
|
|
3
|
-
})(this, function(exports2, mobx, mobxKeystone, Y) {
|
|
3
|
+
})(this, (function(exports2, mobx, mobxKeystone, Y) {
|
|
4
4
|
"use strict";var __defProp = Object.defineProperty;
|
|
5
5
|
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
|
|
6
6
|
var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
|
|
@@ -22,16 +22,6 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
22
22
|
return Object.freeze(n);
|
|
23
23
|
}
|
|
24
24
|
const Y__namespace = /* @__PURE__ */ _interopNamespaceDefault(Y);
|
|
25
|
-
function __decorate(decorators, target, key, desc) {
|
|
26
|
-
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
|
27
|
-
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
|
28
|
-
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;
|
|
29
|
-
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
|
30
|
-
}
|
|
31
|
-
typeof SuppressedError === "function" ? SuppressedError : function(error, suppressed, message) {
|
|
32
|
-
var e = new Error(message);
|
|
33
|
-
return e.name = "SuppressedError", e.error = error, e.suppressed = suppressed, e;
|
|
34
|
-
};
|
|
35
25
|
class MobxKeystoneYjsError extends Error {
|
|
36
26
|
constructor(msg) {
|
|
37
27
|
super(msg);
|
|
@@ -41,7 +31,6 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
41
31
|
function failure(msg) {
|
|
42
32
|
return new MobxKeystoneYjsError(msg);
|
|
43
33
|
}
|
|
44
|
-
const yjsBindingContext = mobxKeystone.createContext(void 0);
|
|
45
34
|
const yjsCollectionAtoms = /* @__PURE__ */ new WeakMap();
|
|
46
35
|
const getYjsCollectionAtom = (yjsCollection) => {
|
|
47
36
|
return yjsCollectionAtoms.get(yjsCollection);
|
|
@@ -54,9 +43,21 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
54
43
|
}
|
|
55
44
|
return atom;
|
|
56
45
|
};
|
|
46
|
+
function isYjsValueDeleted(yjsValue) {
|
|
47
|
+
var _a, _b;
|
|
48
|
+
if (yjsValue instanceof Y__namespace.AbstractType) {
|
|
49
|
+
return !!((_a = yjsValue._item) == null ? void 0 : _a.deleted) || !!((_b = yjsValue.doc) == null ? void 0 : _b.isDestroyed);
|
|
50
|
+
}
|
|
51
|
+
return false;
|
|
52
|
+
}
|
|
57
53
|
function resolveYjsPath(yjsObject, path) {
|
|
58
54
|
let currentYjsObject = yjsObject;
|
|
59
|
-
|
|
55
|
+
let i = -1;
|
|
56
|
+
for (const pathPart of path) {
|
|
57
|
+
i++;
|
|
58
|
+
if (currentYjsObject instanceof Y__namespace.Text) {
|
|
59
|
+
return currentYjsObject;
|
|
60
|
+
}
|
|
60
61
|
if (currentYjsObject instanceof Y__namespace.Map) {
|
|
61
62
|
getOrCreateYjsCollectionAtom(currentYjsObject).reportObserved();
|
|
62
63
|
const key = String(pathPart);
|
|
@@ -66,14 +67,29 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
66
67
|
const key = Number(pathPart);
|
|
67
68
|
currentYjsObject = currentYjsObject.get(key);
|
|
68
69
|
} else {
|
|
69
|
-
throw failure(
|
|
70
|
+
throw failure(
|
|
71
|
+
`Y.Map or Y.Array was expected at path ${JSON.stringify(
|
|
72
|
+
path.slice(0, i)
|
|
73
|
+
)} in order to resolve path ${JSON.stringify(path)}, but got ${currentYjsObject} instead`
|
|
74
|
+
);
|
|
70
75
|
}
|
|
71
|
-
}
|
|
76
|
+
}
|
|
72
77
|
return currentYjsObject;
|
|
73
78
|
}
|
|
79
|
+
const yjsBindingContext = mobxKeystone.createContext(void 0);
|
|
80
|
+
var __defProp2 = Object.defineProperty;
|
|
81
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
82
|
+
var __decorateClass = (decorators, target, key, kind) => {
|
|
83
|
+
var result = kind > 1 ? void 0 : kind ? __getOwnPropDesc(target, key) : target;
|
|
84
|
+
for (var i = decorators.length - 1, decorator; i >= 0; i--)
|
|
85
|
+
if (decorator = decorators[i])
|
|
86
|
+
result = (kind ? decorator(target, key, result) : decorator(result)) || result;
|
|
87
|
+
if (kind && result) __defProp2(target, key, result);
|
|
88
|
+
return result;
|
|
89
|
+
};
|
|
74
90
|
const deltaListType = mobxKeystone.types.array(mobxKeystone.types.frozen(mobxKeystone.types.unchecked()));
|
|
75
91
|
const yjsTextModelId = "mobx-keystone-yjs/YjsTextModel";
|
|
76
|
-
exports2.YjsTextModel = class
|
|
92
|
+
exports2.YjsTextModel = class YjsTextModel extends mobxKeystone.Model({
|
|
77
93
|
deltaList: mobxKeystone.tProp(deltaListType, () => [])
|
|
78
94
|
}) {
|
|
79
95
|
constructor() {
|
|
@@ -97,13 +113,12 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
97
113
|
]
|
|
98
114
|
});
|
|
99
115
|
}
|
|
100
|
-
/**
|
|
101
|
-
* The Y.js path from the bound object to the YjsTextModel instance.
|
|
102
|
-
*/
|
|
103
116
|
get _yjsObjectPath() {
|
|
104
117
|
const ctx = yjsBindingContext.get(this);
|
|
105
118
|
if ((ctx == null ? void 0 : ctx.boundObject) == null) {
|
|
106
|
-
throw failure(
|
|
119
|
+
throw failure(
|
|
120
|
+
"the YjsTextModel instance must be part of a bound object before it can be accessed"
|
|
121
|
+
);
|
|
107
122
|
}
|
|
108
123
|
const path = mobxKeystone.getParentToChildPath(ctx.boundObject, this);
|
|
109
124
|
if (!path) {
|
|
@@ -111,17 +126,11 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
111
126
|
}
|
|
112
127
|
return path;
|
|
113
128
|
}
|
|
114
|
-
/**
|
|
115
|
-
* The Yjs.Text object present at this mobx-keystone node's path.
|
|
116
|
-
*/
|
|
117
129
|
get _yjsObjectAtPath() {
|
|
118
130
|
const path = this._yjsObjectPath;
|
|
119
131
|
const ctx = yjsBindingContext.get(this);
|
|
120
132
|
return resolveYjsPath(ctx.yjsObject, path);
|
|
121
133
|
}
|
|
122
|
-
/**
|
|
123
|
-
* The Yjs.Text object represented by this mobx-keystone node.
|
|
124
|
-
*/
|
|
125
134
|
get yjsText() {
|
|
126
135
|
const yjsObject = this._yjsObjectAtPath;
|
|
127
136
|
if (!(yjsObject instanceof Y__namespace.Text)) {
|
|
@@ -129,13 +138,27 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
129
138
|
}
|
|
130
139
|
return yjsObject;
|
|
131
140
|
}
|
|
132
|
-
/**
|
|
133
|
-
* The text value of the Yjs.Text object.
|
|
134
|
-
* Shortcut for `yjsText.toString()`, but computed.
|
|
135
|
-
*/
|
|
136
141
|
get text() {
|
|
137
142
|
this.yjsTextChangedAtom.reportObserved();
|
|
138
|
-
|
|
143
|
+
const ctx = yjsBindingContext.get(this);
|
|
144
|
+
if ((ctx == null ? void 0 : ctx.boundObject) != null) {
|
|
145
|
+
try {
|
|
146
|
+
const yjsTextString = this.yjsText.toString();
|
|
147
|
+
if (yjsTextString !== "" || this.deltaList.length === 0) {
|
|
148
|
+
return yjsTextString;
|
|
149
|
+
}
|
|
150
|
+
} catch {
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
return this.deltaListToText();
|
|
154
|
+
}
|
|
155
|
+
deltaListToText() {
|
|
156
|
+
const doc = new Y__namespace.Doc();
|
|
157
|
+
const text = doc.getText();
|
|
158
|
+
this.deltaList.forEach((d) => {
|
|
159
|
+
text.applyDelta(d.data);
|
|
160
|
+
});
|
|
161
|
+
return text.toString();
|
|
139
162
|
}
|
|
140
163
|
onInit() {
|
|
141
164
|
const shouldReplicateToYjs = (ctx) => {
|
|
@@ -144,29 +167,36 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
144
167
|
let reapplyDeltasToYjsText = false;
|
|
145
168
|
const newDeltas = [];
|
|
146
169
|
let disposeObserveDeltaList;
|
|
147
|
-
const disposeReactionToDeltaListRefChange = mobx.reaction(
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
170
|
+
const disposeReactionToDeltaListRefChange = mobx.reaction(
|
|
171
|
+
() => this.$.deltaList,
|
|
172
|
+
(deltaList) => {
|
|
173
|
+
disposeObserveDeltaList == null ? void 0 : disposeObserveDeltaList();
|
|
174
|
+
disposeObserveDeltaList = void 0;
|
|
175
|
+
disposeObserveDeltaList = mobx.observe(deltaList, (change) => {
|
|
176
|
+
if (reapplyDeltasToYjsText) {
|
|
177
|
+
return;
|
|
178
|
+
}
|
|
179
|
+
if (!shouldReplicateToYjs(yjsBindingContext.get(this))) {
|
|
180
|
+
return;
|
|
181
|
+
}
|
|
182
|
+
if (change.type === "splice" && change.removedCount === 0 && change.addedCount > 0 && change.index === this.deltaList.length) {
|
|
183
|
+
newDeltas.push(...change.added);
|
|
184
|
+
} else {
|
|
185
|
+
reapplyDeltasToYjsText = true;
|
|
186
|
+
}
|
|
187
|
+
});
|
|
188
|
+
},
|
|
189
|
+
{ fireImmediately: true }
|
|
190
|
+
);
|
|
164
191
|
const disposeOnSnapshot = mobxKeystone.onSnapshot(this, () => {
|
|
165
192
|
try {
|
|
166
193
|
if (reapplyDeltasToYjsText) {
|
|
167
194
|
const ctx = yjsBindingContext.get(this);
|
|
168
195
|
if (shouldReplicateToYjs(ctx)) {
|
|
169
196
|
const { yjsText } = this;
|
|
197
|
+
if (isYjsValueDeleted(yjsText)) {
|
|
198
|
+
throw failure("cannot reapply deltas to deleted Yjs.Text");
|
|
199
|
+
}
|
|
170
200
|
ctx.yjsDoc.transact(() => {
|
|
171
201
|
if (yjsText.length > 0) {
|
|
172
202
|
yjsText.delete(0, yjsText.length);
|
|
@@ -180,6 +210,9 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
180
210
|
const ctx = yjsBindingContext.get(this);
|
|
181
211
|
if (shouldReplicateToYjs(ctx)) {
|
|
182
212
|
const { yjsText } = this;
|
|
213
|
+
if (isYjsValueDeleted(yjsText)) {
|
|
214
|
+
throw failure("cannot reapply deltas to deleted Yjs.Text");
|
|
215
|
+
}
|
|
183
216
|
ctx.yjsDoc.transact(() => {
|
|
184
217
|
newDeltas.forEach((frozenDeltas) => {
|
|
185
218
|
yjsText.applyDelta(frozenDeltas.data);
|
|
@@ -192,7 +225,10 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
192
225
|
newDeltas.length = 0;
|
|
193
226
|
}
|
|
194
227
|
});
|
|
195
|
-
const diposeYjsTextChangedAtom = hookYjsTextChangedAtom(
|
|
228
|
+
const diposeYjsTextChangedAtom = hookYjsTextChangedAtom(
|
|
229
|
+
() => this.yjsText,
|
|
230
|
+
this.yjsTextChangedAtom
|
|
231
|
+
);
|
|
196
232
|
return () => {
|
|
197
233
|
disposeOnSnapshot();
|
|
198
234
|
disposeReactionToDeltaListRefChange();
|
|
@@ -202,19 +238,19 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
202
238
|
};
|
|
203
239
|
}
|
|
204
240
|
};
|
|
205
|
-
|
|
241
|
+
__decorateClass([
|
|
206
242
|
mobx.computed
|
|
207
|
-
], exports2.YjsTextModel.prototype, "_yjsObjectPath",
|
|
208
|
-
|
|
243
|
+
], exports2.YjsTextModel.prototype, "_yjsObjectPath", 1);
|
|
244
|
+
__decorateClass([
|
|
209
245
|
mobx.computed
|
|
210
|
-
], exports2.YjsTextModel.prototype, "_yjsObjectAtPath",
|
|
211
|
-
|
|
246
|
+
], exports2.YjsTextModel.prototype, "_yjsObjectAtPath", 1);
|
|
247
|
+
__decorateClass([
|
|
212
248
|
mobx.computed
|
|
213
|
-
], exports2.YjsTextModel.prototype, "yjsText",
|
|
214
|
-
|
|
249
|
+
], exports2.YjsTextModel.prototype, "yjsText", 1);
|
|
250
|
+
__decorateClass([
|
|
215
251
|
mobx.computed
|
|
216
|
-
], exports2.YjsTextModel.prototype, "text",
|
|
217
|
-
exports2.YjsTextModel =
|
|
252
|
+
], exports2.YjsTextModel.prototype, "text", 1);
|
|
253
|
+
exports2.YjsTextModel = __decorateClass([
|
|
218
254
|
mobxKeystone.model(yjsTextModelId)
|
|
219
255
|
], exports2.YjsTextModel);
|
|
220
256
|
const DecoratedYjsTextModel = exports2.YjsTextModel;
|
|
@@ -223,31 +259,43 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
223
259
|
const observeFn = () => {
|
|
224
260
|
textChangedAtom.reportChanged();
|
|
225
261
|
};
|
|
226
|
-
const disposeReactionToYTextChange = mobx.reaction(
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
disposeObserveYjsText =
|
|
238
|
-
|
|
239
|
-
|
|
262
|
+
const disposeReactionToYTextChange = mobx.reaction(
|
|
263
|
+
() => {
|
|
264
|
+
try {
|
|
265
|
+
const yjsText = getYjsText();
|
|
266
|
+
return isYjsValueDeleted(yjsText) ? void 0 : yjsText;
|
|
267
|
+
} catch {
|
|
268
|
+
return void 0;
|
|
269
|
+
}
|
|
270
|
+
},
|
|
271
|
+
(yjsText) => {
|
|
272
|
+
disposeObserveYjsText == null ? void 0 : disposeObserveYjsText();
|
|
273
|
+
disposeObserveYjsText = void 0;
|
|
274
|
+
if (yjsText) {
|
|
275
|
+
yjsText.observe(observeFn);
|
|
276
|
+
disposeObserveYjsText = () => {
|
|
277
|
+
yjsText.unobserve(observeFn);
|
|
278
|
+
};
|
|
279
|
+
}
|
|
280
|
+
textChangedAtom.reportChanged();
|
|
281
|
+
},
|
|
282
|
+
{
|
|
283
|
+
fireImmediately: true
|
|
240
284
|
}
|
|
241
|
-
|
|
242
|
-
}, {
|
|
243
|
-
fireImmediately: true
|
|
244
|
-
});
|
|
285
|
+
);
|
|
245
286
|
return () => {
|
|
246
287
|
disposeReactionToYTextChange();
|
|
247
288
|
disposeObserveYjsText == null ? void 0 : disposeObserveYjsText();
|
|
248
289
|
disposeObserveYjsText = void 0;
|
|
249
290
|
};
|
|
250
291
|
}
|
|
292
|
+
const yjsContainerToSnapshot = /* @__PURE__ */ new WeakMap();
|
|
293
|
+
function setYjsContainerSnapshot(container, snapshot) {
|
|
294
|
+
yjsContainerToSnapshot.set(container, snapshot);
|
|
295
|
+
}
|
|
296
|
+
function isYjsContainerUpToDate(container, snapshot) {
|
|
297
|
+
return yjsContainerToSnapshot.get(container) === snapshot;
|
|
298
|
+
}
|
|
251
299
|
function isPlainPrimitive(v) {
|
|
252
300
|
const t = typeof v;
|
|
253
301
|
return t === "string" || t === "number" || t === "boolean" || v === null || v === void 0;
|
|
@@ -256,118 +304,163 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
256
304
|
return Array.isArray(v);
|
|
257
305
|
}
|
|
258
306
|
function isPlainObject(v) {
|
|
259
|
-
return
|
|
307
|
+
return typeof v === "object" && v !== null && !Array.isArray(v);
|
|
260
308
|
}
|
|
261
309
|
function convertJsonToYjsData(v) {
|
|
262
|
-
|
|
263
|
-
|
|
310
|
+
if (isPlainPrimitive(v)) {
|
|
311
|
+
return v;
|
|
312
|
+
}
|
|
313
|
+
if (isPlainArray(v)) {
|
|
314
|
+
const arr = new Y__namespace.Array();
|
|
315
|
+
applyJsonArrayToYArray(arr, v);
|
|
316
|
+
return arr;
|
|
317
|
+
}
|
|
318
|
+
if (isPlainObject(v)) {
|
|
319
|
+
if (v[mobxKeystone.frozenKey] === true) {
|
|
264
320
|
return v;
|
|
265
321
|
}
|
|
266
|
-
if (
|
|
267
|
-
const
|
|
268
|
-
|
|
269
|
-
|
|
322
|
+
if (v[mobxKeystone.modelTypeKey] === yjsTextModelId) {
|
|
323
|
+
const text = new Y__namespace.Text();
|
|
324
|
+
const yjsTextModel = v;
|
|
325
|
+
yjsTextModel.deltaList.forEach((frozenDeltas) => {
|
|
326
|
+
text.applyDelta(frozenDeltas.data);
|
|
327
|
+
});
|
|
328
|
+
return text;
|
|
270
329
|
}
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
330
|
+
const map = new Y__namespace.Map();
|
|
331
|
+
applyJsonObjectToYMap(map, v);
|
|
332
|
+
return map;
|
|
333
|
+
}
|
|
334
|
+
throw new Error(`unsupported value type: ${v}`);
|
|
335
|
+
}
|
|
336
|
+
const applyJsonArrayToYArray = (dest, source, options = {}) => {
|
|
337
|
+
const { mode = "add" } = options;
|
|
338
|
+
if (mode === "merge" && isYjsContainerUpToDate(dest, source)) {
|
|
339
|
+
return;
|
|
340
|
+
}
|
|
341
|
+
const srcLen = source.length;
|
|
342
|
+
if (mode === "add") {
|
|
343
|
+
for (let i = 0; i < srcLen; i++) {
|
|
344
|
+
dest.push([convertJsonToYjsData(source[i])]);
|
|
345
|
+
}
|
|
346
|
+
return;
|
|
347
|
+
}
|
|
348
|
+
const destLen = dest.length;
|
|
349
|
+
if (destLen > srcLen) {
|
|
350
|
+
dest.delete(srcLen, destLen - srcLen);
|
|
351
|
+
}
|
|
352
|
+
const minLen = Math.min(destLen, srcLen);
|
|
353
|
+
for (let i = 0; i < minLen; i++) {
|
|
354
|
+
const srcItem = source[i];
|
|
355
|
+
const destItem = dest.get(i);
|
|
356
|
+
if (isPlainObject(srcItem) && destItem instanceof Y__namespace.Map) {
|
|
357
|
+
applyJsonObjectToYMap(destItem, srcItem, options);
|
|
358
|
+
continue;
|
|
359
|
+
}
|
|
360
|
+
if (isPlainArray(srcItem) && destItem instanceof Y__namespace.Array) {
|
|
361
|
+
applyJsonArrayToYArray(destItem, srcItem, options);
|
|
362
|
+
continue;
|
|
363
|
+
}
|
|
364
|
+
if (isPlainPrimitive(srcItem) && destItem === srcItem) {
|
|
365
|
+
continue;
|
|
366
|
+
}
|
|
367
|
+
dest.delete(i, 1);
|
|
368
|
+
dest.insert(i, [convertJsonToYjsData(srcItem)]);
|
|
369
|
+
}
|
|
370
|
+
for (let i = destLen; i < srcLen; i++) {
|
|
371
|
+
dest.push([convertJsonToYjsData(source[i])]);
|
|
372
|
+
}
|
|
373
|
+
setYjsContainerSnapshot(dest, source);
|
|
374
|
+
};
|
|
375
|
+
const applyJsonObjectToYMap = (dest, source, options = {}) => {
|
|
376
|
+
const { mode = "add" } = options;
|
|
377
|
+
if (mode === "merge" && isYjsContainerUpToDate(dest, source)) {
|
|
378
|
+
return;
|
|
379
|
+
}
|
|
380
|
+
if (mode === "add") {
|
|
381
|
+
for (const k of Object.keys(source)) {
|
|
382
|
+
const v = source[k];
|
|
383
|
+
if (v !== void 0) {
|
|
384
|
+
dest.set(k, convertJsonToYjsData(v));
|
|
282
385
|
}
|
|
283
|
-
const map = new Y__namespace.Map();
|
|
284
|
-
applyJsonObjectToYMap(map, v);
|
|
285
|
-
return map;
|
|
286
386
|
}
|
|
287
|
-
|
|
288
|
-
}
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
387
|
+
return;
|
|
388
|
+
}
|
|
389
|
+
const sourceKeysWithValues = new Set(Object.keys(source).filter((k) => source[k] !== void 0));
|
|
390
|
+
for (const key of dest.keys()) {
|
|
391
|
+
if (!sourceKeysWithValues.has(key)) {
|
|
392
|
+
dest.delete(key);
|
|
393
|
+
}
|
|
394
|
+
}
|
|
395
|
+
for (const k of Object.keys(source)) {
|
|
396
|
+
const v = source[k];
|
|
397
|
+
if (v === void 0) {
|
|
398
|
+
continue;
|
|
399
|
+
}
|
|
400
|
+
const existing = dest.get(k);
|
|
401
|
+
if (isPlainObject(v) && existing instanceof Y__namespace.Map) {
|
|
402
|
+
applyJsonObjectToYMap(existing, v, options);
|
|
403
|
+
continue;
|
|
404
|
+
}
|
|
405
|
+
if (isPlainArray(v) && existing instanceof Y__namespace.Array) {
|
|
406
|
+
applyJsonArrayToYArray(existing, v, options);
|
|
407
|
+
continue;
|
|
408
|
+
}
|
|
409
|
+
if (isPlainPrimitive(v) && existing === v) {
|
|
410
|
+
continue;
|
|
411
|
+
}
|
|
295
412
|
dest.set(k, convertJsonToYjsData(v));
|
|
296
|
-
}
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
case "replace": {
|
|
323
|
-
yjs.set(key, convertJsonToYjsData(patch.value));
|
|
324
|
-
break;
|
|
325
|
-
}
|
|
326
|
-
case "remove": {
|
|
327
|
-
yjs.delete(key);
|
|
328
|
-
break;
|
|
329
|
-
}
|
|
330
|
-
default: {
|
|
331
|
-
throw failure(`invalid patch operation for map`);
|
|
332
|
-
}
|
|
413
|
+
}
|
|
414
|
+
setYjsContainerSnapshot(dest, source);
|
|
415
|
+
};
|
|
416
|
+
function convertValue(v) {
|
|
417
|
+
if (v === null || v === void 0 || typeof v !== "object") {
|
|
418
|
+
return v;
|
|
419
|
+
}
|
|
420
|
+
if (Array.isArray(v) && v.length === 0) {
|
|
421
|
+
return new Y__namespace.Array();
|
|
422
|
+
}
|
|
423
|
+
return convertJsonToYjsData(v);
|
|
424
|
+
}
|
|
425
|
+
function applyMobxChangeToYjsObject(change, yjsObject) {
|
|
426
|
+
if (isYjsValueDeleted(yjsObject)) {
|
|
427
|
+
throw failure("cannot apply patch to deleted Yjs value");
|
|
428
|
+
}
|
|
429
|
+
const yjsContainer = resolveYjsPath(yjsObject, change.path);
|
|
430
|
+
if (!yjsContainer) {
|
|
431
|
+
return;
|
|
432
|
+
}
|
|
433
|
+
if (yjsContainer instanceof Y__namespace.Array) {
|
|
434
|
+
if (change.type === mobxKeystone.DeepChangeType.ArraySplice) {
|
|
435
|
+
yjsContainer.delete(change.index, change.removedValues.length);
|
|
436
|
+
if (change.addedValues.length > 0) {
|
|
437
|
+
const valuesToInsert = change.addedValues.map(convertValue);
|
|
438
|
+
yjsContainer.insert(change.index, valuesToInsert);
|
|
333
439
|
}
|
|
334
|
-
} else if (
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
} else {
|
|
348
|
-
yjs.delete(Number(key));
|
|
349
|
-
yjs.insert(Number(key), [convertJsonToYjsData(patch.value)]);
|
|
350
|
-
}
|
|
351
|
-
break;
|
|
352
|
-
}
|
|
353
|
-
case "add": {
|
|
354
|
-
yjs.insert(Number(key), [convertJsonToYjsData(patch.value)]);
|
|
355
|
-
break;
|
|
356
|
-
}
|
|
357
|
-
case "remove": {
|
|
358
|
-
yjs.delete(Number(key));
|
|
359
|
-
break;
|
|
360
|
-
}
|
|
361
|
-
default: {
|
|
362
|
-
throw failure(`invalid patch operation for array`);
|
|
363
|
-
}
|
|
440
|
+
} else if (change.type === mobxKeystone.DeepChangeType.ArrayUpdate) {
|
|
441
|
+
yjsContainer.delete(change.index, 1);
|
|
442
|
+
yjsContainer.insert(change.index, [convertValue(change.newValue)]);
|
|
443
|
+
} else {
|
|
444
|
+
throw failure(`unsupported array change type: ${change.type}`);
|
|
445
|
+
}
|
|
446
|
+
} else if (yjsContainer instanceof Y__namespace.Map) {
|
|
447
|
+
if (change.type === mobxKeystone.DeepChangeType.ObjectAdd || change.type === mobxKeystone.DeepChangeType.ObjectUpdate) {
|
|
448
|
+
const key = change.key;
|
|
449
|
+
if (change.newValue === void 0) {
|
|
450
|
+
yjsContainer.delete(key);
|
|
451
|
+
} else {
|
|
452
|
+
yjsContainer.set(key, convertValue(change.newValue));
|
|
364
453
|
}
|
|
365
|
-
} else if (
|
|
366
|
-
|
|
367
|
-
|
|
454
|
+
} else if (change.type === mobxKeystone.DeepChangeType.ObjectRemove) {
|
|
455
|
+
const key = change.key;
|
|
456
|
+
yjsContainer.delete(key);
|
|
457
|
+
} else {
|
|
458
|
+
throw failure(`unsupported object change type: ${change.type}`);
|
|
368
459
|
}
|
|
460
|
+
} else if (yjsContainer instanceof Y__namespace.Text) {
|
|
461
|
+
return;
|
|
369
462
|
} else {
|
|
370
|
-
throw failure(`
|
|
463
|
+
throw failure(`unsupported Yjs container type: ${yjsContainer}`);
|
|
371
464
|
}
|
|
372
465
|
}
|
|
373
466
|
const convertYjsDataToJson = mobx.action((yjsData) => {
|
|
@@ -389,89 +482,126 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
389
482
|
}
|
|
390
483
|
return yjsData;
|
|
391
484
|
});
|
|
392
|
-
function
|
|
393
|
-
const
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
|
|
420
|
-
|
|
421
|
-
|
|
422
|
-
|
|
423
|
-
}
|
|
424
|
-
|
|
425
|
-
|
|
426
|
-
|
|
427
|
-
|
|
485
|
+
function applyYjsEventToMobx(event, boundObject, reconciliationMap) {
|
|
486
|
+
const path = event.path;
|
|
487
|
+
const { value: target } = mobxKeystone.resolvePath(boundObject, path);
|
|
488
|
+
if (!target) {
|
|
489
|
+
throw failure(`cannot resolve path ${JSON.stringify(path)}`);
|
|
490
|
+
}
|
|
491
|
+
mobxKeystone.runUnprotected(() => {
|
|
492
|
+
if (event instanceof Y__namespace.YMapEvent) {
|
|
493
|
+
applyYMapEventToMobx(event, target, reconciliationMap);
|
|
494
|
+
} else if (event instanceof Y__namespace.YArrayEvent) {
|
|
495
|
+
applyYArrayEventToMobx(event, target, reconciliationMap);
|
|
496
|
+
} else if (event instanceof Y__namespace.YTextEvent) {
|
|
497
|
+
applyYTextEventToMobx(event, target);
|
|
498
|
+
}
|
|
499
|
+
});
|
|
500
|
+
}
|
|
501
|
+
function processDeletedValue(val, reconciliationMap) {
|
|
502
|
+
if (val && typeof val === "object" && mobxKeystone.isModel(val)) {
|
|
503
|
+
const sn = mobxKeystone.getSnapshot(val);
|
|
504
|
+
const id = mobxKeystone.getSnapshotModelId(sn);
|
|
505
|
+
if (id) {
|
|
506
|
+
reconciliationMap.set(id, val);
|
|
507
|
+
}
|
|
508
|
+
}
|
|
509
|
+
}
|
|
510
|
+
function reviveValue(jsonValue, reconciliationMap) {
|
|
511
|
+
if (jsonValue === null || typeof jsonValue !== "object") {
|
|
512
|
+
return jsonValue;
|
|
513
|
+
}
|
|
514
|
+
if (mobxKeystone.isFrozenSnapshot(jsonValue)) {
|
|
515
|
+
return mobxKeystone.frozen(jsonValue.data);
|
|
516
|
+
}
|
|
517
|
+
if (reconciliationMap && jsonValue && typeof jsonValue === "object") {
|
|
518
|
+
const modelId = mobxKeystone.getSnapshotModelId(jsonValue);
|
|
519
|
+
if (modelId) {
|
|
520
|
+
const existing = reconciliationMap.get(modelId);
|
|
521
|
+
if (existing) {
|
|
522
|
+
reconciliationMap.delete(modelId);
|
|
523
|
+
return existing;
|
|
428
524
|
}
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
|
|
433
|
-
|
|
434
|
-
|
|
435
|
-
|
|
525
|
+
}
|
|
526
|
+
}
|
|
527
|
+
return mobxKeystone.fromSnapshot(jsonValue);
|
|
528
|
+
}
|
|
529
|
+
function applyYMapEventToMobx(event, target, reconciliationMap) {
|
|
530
|
+
const source = event.target;
|
|
531
|
+
event.changes.keys.forEach((change, key) => {
|
|
532
|
+
switch (change.action) {
|
|
533
|
+
case "add":
|
|
534
|
+
case "update": {
|
|
535
|
+
const yjsValue = source.get(key);
|
|
536
|
+
const jsonValue = convertYjsDataToJson(yjsValue);
|
|
537
|
+
if (change.action === "update") {
|
|
538
|
+
processDeletedValue(target[key], reconciliationMap);
|
|
436
539
|
}
|
|
540
|
+
target[key] = reviveValue(jsonValue, reconciliationMap);
|
|
541
|
+
break;
|
|
437
542
|
}
|
|
438
|
-
|
|
439
|
-
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
|
|
443
|
-
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
});
|
|
447
|
-
retain++;
|
|
448
|
-
});
|
|
543
|
+
case "delete": {
|
|
544
|
+
processDeletedValue(target[key], reconciliationMap);
|
|
545
|
+
if (mobxKeystone.isModel(target)) {
|
|
546
|
+
mobx.remove(target.$, key);
|
|
547
|
+
} else {
|
|
548
|
+
mobx.remove(target, key);
|
|
549
|
+
}
|
|
550
|
+
break;
|
|
449
551
|
}
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
460
|
-
|
|
461
|
-
|
|
462
|
-
|
|
552
|
+
default:
|
|
553
|
+
throw failure(`unsupported Yjs map event action: ${change.action}`);
|
|
554
|
+
}
|
|
555
|
+
});
|
|
556
|
+
}
|
|
557
|
+
function applyYArrayEventToMobx(event, target, reconciliationMap) {
|
|
558
|
+
let currentIndex = 0;
|
|
559
|
+
for (const change of event.changes.delta) {
|
|
560
|
+
if (change.retain) {
|
|
561
|
+
currentIndex += change.retain;
|
|
562
|
+
}
|
|
563
|
+
if (change.delete) {
|
|
564
|
+
const deletedItems = target.slice(currentIndex, currentIndex + change.delete);
|
|
565
|
+
deletedItems.forEach((item) => {
|
|
566
|
+
processDeletedValue(item, reconciliationMap);
|
|
567
|
+
});
|
|
568
|
+
target.splice(currentIndex, change.delete);
|
|
569
|
+
}
|
|
570
|
+
if (change.insert) {
|
|
571
|
+
const insertedItems = Array.isArray(change.insert) ? change.insert : [change.insert];
|
|
572
|
+
const values = insertedItems.map((yjsValue) => {
|
|
573
|
+
const jsonValue = convertYjsDataToJson(yjsValue);
|
|
574
|
+
return reviveValue(jsonValue, reconciliationMap);
|
|
575
|
+
});
|
|
576
|
+
target.splice(currentIndex, 0, ...values);
|
|
577
|
+
currentIndex += values.length;
|
|
578
|
+
}
|
|
463
579
|
}
|
|
464
|
-
return patches;
|
|
465
580
|
}
|
|
466
|
-
function
|
|
467
|
-
if (
|
|
468
|
-
|
|
469
|
-
}
|
|
470
|
-
|
|
581
|
+
function applyYTextEventToMobx(event, target) {
|
|
582
|
+
if (target == null ? void 0 : target.deltaList) {
|
|
583
|
+
target.deltaList.push(mobxKeystone.frozen(event.delta));
|
|
584
|
+
}
|
|
585
|
+
}
|
|
586
|
+
function captureChangeSnapshots(change) {
|
|
587
|
+
if (change.type === mobxKeystone.DeepChangeType.ArraySplice && change.addedValues.length > 0) {
|
|
588
|
+
const snapshots = change.addedValues.map((v) => mobxKeystone.isTreeNode(v) ? mobxKeystone.getSnapshot(v) : v);
|
|
589
|
+
return { ...change, addedValues: snapshots };
|
|
590
|
+
} else if (change.type === mobxKeystone.DeepChangeType.ArrayUpdate) {
|
|
591
|
+
const snapshot = mobxKeystone.isTreeNode(change.newValue) ? mobxKeystone.getSnapshot(change.newValue) : change.newValue;
|
|
592
|
+
return { ...change, newValue: snapshot };
|
|
593
|
+
} else if (change.type === mobxKeystone.DeepChangeType.ObjectAdd || change.type === mobxKeystone.DeepChangeType.ObjectUpdate) {
|
|
594
|
+
const snapshot = mobxKeystone.isTreeNode(change.newValue) ? mobxKeystone.getSnapshot(change.newValue) : change.newValue;
|
|
595
|
+
return { ...change, newValue: snapshot };
|
|
471
596
|
}
|
|
597
|
+
return change;
|
|
472
598
|
}
|
|
473
|
-
function bindYjsToMobxKeystone({
|
|
474
|
-
|
|
599
|
+
function bindYjsToMobxKeystone({
|
|
600
|
+
yjsDoc,
|
|
601
|
+
yjsObject,
|
|
602
|
+
mobxKeystoneType
|
|
603
|
+
}) {
|
|
604
|
+
const yjsOrigin = /* @__PURE__ */ Symbol("bindYjsToMobxKeystoneTransactionOrigin");
|
|
475
605
|
let applyingYjsChangesToMobxKeystone = 0;
|
|
476
606
|
const bindingContext = {
|
|
477
607
|
yjsDoc,
|
|
@@ -484,90 +614,136 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
484
614
|
return applyingYjsChangesToMobxKeystone > 0;
|
|
485
615
|
}
|
|
486
616
|
};
|
|
617
|
+
if (isYjsValueDeleted(yjsObject)) {
|
|
618
|
+
throw failure("cannot apply patch to deleted Yjs value");
|
|
619
|
+
}
|
|
487
620
|
const yjsJson = convertYjsDataToJson(yjsObject);
|
|
488
|
-
|
|
621
|
+
let boundObject;
|
|
622
|
+
let hasInitChanges = false;
|
|
489
623
|
const createBoundObject = () => {
|
|
490
|
-
const
|
|
491
|
-
|
|
624
|
+
const disposeGlobalListener = mobxKeystone.onGlobalDeepChange((_target, change) => {
|
|
625
|
+
if (change.isInit) {
|
|
626
|
+
hasInitChanges = true;
|
|
627
|
+
}
|
|
492
628
|
});
|
|
493
629
|
try {
|
|
494
|
-
const
|
|
495
|
-
|
|
496
|
-
|
|
630
|
+
const result = yjsBindingContext.apply(
|
|
631
|
+
() => mobxKeystone.fromSnapshot(mobxKeystoneType, yjsJson),
|
|
632
|
+
bindingContext
|
|
633
|
+
);
|
|
634
|
+
yjsBindingContext.set(result, { ...bindingContext, boundObject: result });
|
|
635
|
+
return result;
|
|
497
636
|
} finally {
|
|
498
|
-
|
|
637
|
+
disposeGlobalListener();
|
|
499
638
|
}
|
|
500
639
|
};
|
|
501
|
-
|
|
640
|
+
boundObject = createBoundObject();
|
|
502
641
|
const observeDeepCb = mobx.action((events) => {
|
|
503
|
-
const
|
|
642
|
+
const eventsToApply = [];
|
|
504
643
|
events.forEach((event) => {
|
|
505
644
|
var _a;
|
|
506
645
|
if (event.transaction.origin !== yjsOrigin) {
|
|
507
|
-
|
|
646
|
+
eventsToApply.push(event);
|
|
508
647
|
}
|
|
509
648
|
if (event.target instanceof Y__namespace.Map || event.target instanceof Y__namespace.Array) {
|
|
510
649
|
(_a = getYjsCollectionAtom(event.target)) == null ? void 0 : _a.reportChanged();
|
|
511
650
|
}
|
|
512
651
|
});
|
|
513
|
-
if (
|
|
652
|
+
if (eventsToApply.length > 0) {
|
|
514
653
|
applyingYjsChangesToMobxKeystone++;
|
|
515
654
|
try {
|
|
516
|
-
|
|
655
|
+
const reconciliationMap = /* @__PURE__ */ new Map();
|
|
656
|
+
const initChanges = [];
|
|
657
|
+
const disposeGlobalListener = mobxKeystone.onGlobalDeepChange((target, change) => {
|
|
658
|
+
if (change.isInit) {
|
|
659
|
+
initChanges.push({ target, change: captureChangeSnapshots(change) });
|
|
660
|
+
}
|
|
661
|
+
});
|
|
662
|
+
try {
|
|
663
|
+
eventsToApply.forEach((event) => {
|
|
664
|
+
applyYjsEventToMobx(event, boundObject, reconciliationMap);
|
|
665
|
+
});
|
|
666
|
+
} finally {
|
|
667
|
+
disposeGlobalListener();
|
|
668
|
+
}
|
|
669
|
+
if (initChanges.length > 0 && !isYjsValueDeleted(yjsObject)) {
|
|
670
|
+
yjsDoc.transact(() => {
|
|
671
|
+
for (const { target, change } of initChanges) {
|
|
672
|
+
const pathToTarget = mobxKeystone.getParentToChildPath(boundObject, target);
|
|
673
|
+
if (pathToTarget !== void 0) {
|
|
674
|
+
const changeWithCorrectPath = {
|
|
675
|
+
...change,
|
|
676
|
+
path: [...pathToTarget, ...change.path]
|
|
677
|
+
};
|
|
678
|
+
applyMobxChangeToYjsObject(changeWithCorrectPath, yjsObject);
|
|
679
|
+
}
|
|
680
|
+
}
|
|
681
|
+
}, yjsOrigin);
|
|
682
|
+
}
|
|
683
|
+
if (yjsObject instanceof Y__namespace.Map || yjsObject instanceof Y__namespace.Array) {
|
|
684
|
+
setYjsContainerSnapshot(yjsObject, mobxKeystone.getSnapshot(boundObject));
|
|
685
|
+
}
|
|
517
686
|
} finally {
|
|
518
687
|
applyingYjsChangesToMobxKeystone--;
|
|
519
688
|
}
|
|
520
689
|
}
|
|
521
690
|
});
|
|
522
691
|
yjsObject.observeDeep(observeDeepCb);
|
|
523
|
-
let
|
|
524
|
-
const
|
|
692
|
+
let pendingChanges = [];
|
|
693
|
+
const disposeOnDeepChange = mobxKeystone.onDeepChange(boundObject, (change) => {
|
|
525
694
|
if (applyingYjsChangesToMobxKeystone > 0) {
|
|
526
695
|
return;
|
|
527
696
|
}
|
|
528
|
-
|
|
697
|
+
if (change.isInit) {
|
|
698
|
+
return;
|
|
699
|
+
}
|
|
700
|
+
pendingChanges.push(captureChangeSnapshots(change));
|
|
529
701
|
});
|
|
530
|
-
const disposeOnSnapshot = mobxKeystone.onSnapshot(boundObject, () => {
|
|
531
|
-
if (
|
|
702
|
+
const disposeOnSnapshot = mobxKeystone.onSnapshot(boundObject, (boundObjectSnapshot) => {
|
|
703
|
+
if (pendingChanges.length === 0) {
|
|
704
|
+
return;
|
|
705
|
+
}
|
|
706
|
+
const changesToApply = pendingChanges;
|
|
707
|
+
pendingChanges = [];
|
|
708
|
+
if (isYjsValueDeleted(yjsObject)) {
|
|
532
709
|
return;
|
|
533
710
|
}
|
|
534
|
-
const arrayOfArrayOfPatches = pendingArrayOfArrayOfPatches;
|
|
535
|
-
pendingArrayOfArrayOfPatches = [];
|
|
536
711
|
yjsDoc.transact(() => {
|
|
537
|
-
|
|
538
|
-
|
|
539
|
-
applyMobxKeystonePatchToYjsObject(patch, yjsObject);
|
|
540
|
-
});
|
|
712
|
+
changesToApply.forEach((change) => {
|
|
713
|
+
applyMobxChangeToYjsObject(change, yjsObject);
|
|
541
714
|
});
|
|
542
715
|
}, yjsOrigin);
|
|
716
|
+
if (yjsObject instanceof Y__namespace.Map || yjsObject instanceof Y__namespace.Array) {
|
|
717
|
+
setYjsContainerSnapshot(yjsObject, boundObjectSnapshot);
|
|
718
|
+
}
|
|
543
719
|
});
|
|
544
|
-
|
|
545
|
-
|
|
546
|
-
|
|
547
|
-
if (
|
|
548
|
-
|
|
549
|
-
|
|
550
|
-
}
|
|
551
|
-
|
|
552
|
-
|
|
553
|
-
|
|
554
|
-
if (parentToChildPath !== void 0) {
|
|
555
|
-
patches.forEach((patch) => {
|
|
556
|
-
applyMobxKeystonePatchToYjsObject({
|
|
557
|
-
...patch,
|
|
558
|
-
path: [...parentToChildPath, ...patch.path]
|
|
559
|
-
}, yjsObject);
|
|
720
|
+
const finalSnapshot = mobxKeystone.getSnapshot(boundObject);
|
|
721
|
+
if (hasInitChanges) {
|
|
722
|
+
yjsDoc.transact(() => {
|
|
723
|
+
if (yjsObject instanceof Y__namespace.Map) {
|
|
724
|
+
applyJsonObjectToYMap(yjsObject, finalSnapshot, {
|
|
725
|
+
mode: "merge"
|
|
726
|
+
});
|
|
727
|
+
} else if (yjsObject instanceof Y__namespace.Array) {
|
|
728
|
+
applyJsonArrayToYArray(yjsObject, finalSnapshot, {
|
|
729
|
+
mode: "merge"
|
|
560
730
|
});
|
|
561
731
|
}
|
|
562
|
-
});
|
|
563
|
-
}
|
|
732
|
+
}, yjsOrigin);
|
|
733
|
+
}
|
|
734
|
+
if (yjsObject instanceof Y__namespace.Map || yjsObject instanceof Y__namespace.Array) {
|
|
735
|
+
setYjsContainerSnapshot(yjsObject, finalSnapshot);
|
|
736
|
+
}
|
|
737
|
+
const dispose = () => {
|
|
738
|
+
yjsDoc.off("destroy", dispose);
|
|
739
|
+
disposeOnDeepChange();
|
|
740
|
+
disposeOnSnapshot();
|
|
741
|
+
yjsObject.unobserveDeep(observeDeepCb);
|
|
742
|
+
};
|
|
743
|
+
yjsDoc.on("destroy", dispose);
|
|
564
744
|
return {
|
|
565
745
|
boundObject,
|
|
566
|
-
dispose
|
|
567
|
-
disposeOnPatches();
|
|
568
|
-
disposeOnSnapshot();
|
|
569
|
-
yjsObject.unobserveDeep(observeDeepCb);
|
|
570
|
-
},
|
|
746
|
+
dispose,
|
|
571
747
|
yjsOrigin
|
|
572
748
|
};
|
|
573
749
|
}
|
|
@@ -579,5 +755,5 @@ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "sy
|
|
|
579
755
|
exports2.yjsBindingContext = yjsBindingContext;
|
|
580
756
|
exports2.yjsTextModelId = yjsTextModelId;
|
|
581
757
|
Object.defineProperty(exports2, Symbol.toStringTag, { value: "Module" });
|
|
582
|
-
});
|
|
583
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"mobx-keystone-yjs.umd.js","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":["createContext","createAtom","Y","types","YjsTextModel","Model","tProp","frozen","getParentToChildPath","reaction","observe","onSnapshot","computed","exports","model","runInAction","action","modelSnapshotOutWithMetadata","onGlobalPatches","boundObject","fromSnapshot","applyPatches","onPatches"],"mappings":"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"}
|
|
758
|
+
}));
|
|
759
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"mobx-keystone-yjs.umd.js","sources":["../src/utils/error.ts","../src/utils/getOrCreateYjsCollectionAtom.ts","../src/utils/isYjsValueDeleted.ts","../src/binding/resolveYjsPath.ts","../src/binding/yjsBindingContext.ts","../src/binding/YjsTextModel.ts","../src/binding/yjsSnapshotTracking.ts","../src/binding/convertJsonToYjsData.ts","../src/binding/applyMobxChangeToYjsObject.ts","../src/binding/convertYjsDataToJson.ts","../src/binding/applyYjsEventToMobx.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 { createAtom, IAtom } 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","import * as Y from \"yjs\"\nimport { failure } from \"../utils/error\"\nimport { getOrCreateYjsCollectionAtom } from \"../utils/getOrCreateYjsCollectionAtom\"\n\n/**\n * Resolves a path within a Yjs object structure.\n * Returns the Yjs container at the specified path.\n *\n * When a Y.Text is encountered during path resolution (either at the start\n * or mid-path), it is returned immediately since Y.Text doesn't support\n * nested path traversal.\n *\n * @param yjsObject The root Yjs object\n * @param path Array of keys/indices to traverse\n * @returns The Yjs container at the path, or Y.Text if encountered during traversal\n */\nexport function resolveYjsPath(\n  yjsObject: Y.Map<unknown> | Y.Array<unknown> | Y.Text,\n  path: readonly (string | number)[]\n): unknown {\n  let currentYjsObject: unknown = yjsObject\n\n  let i = -1\n  for (const pathPart of path) {\n    i++\n    // If we encounter a Y.Text during path resolution, return it immediately.\n    // Y.Text objects don't support nested path traversal, and their updates\n    // are handled separately by YjsTextModel's own synchronization mechanism.\n    if (currentYjsObject instanceof Y.Text) {\n      return currentYjsObject\n    }\n\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\"\n\n/**\n * WeakMap that tracks which snapshot each Y.js container was last synced from.\n * This is used during reconciliation to skip containers that are already up-to-date.\n *\n * The key is the Y.js container (Y.Map or Y.Array).\n * The value is the snapshot (plain object or array) that was last synced to it.\n */\nexport const yjsContainerToSnapshot = new WeakMap<Y.Map<any> | Y.Array<any>, unknown>()\n\n/**\n * Updates the snapshot tracking for a Y.js container.\n * Call this after syncing a snapshot to a Y.js container.\n */\nexport function setYjsContainerSnapshot(\n  container: Y.Map<any> | Y.Array<any>,\n  snapshot: unknown\n): void {\n  yjsContainerToSnapshot.set(container, snapshot)\n}\n\n/**\n * Gets the last synced snapshot for a Y.js container.\n * Returns undefined if the container has never been synced.\n */\nexport function getYjsContainerSnapshot(container: Y.Map<any> | Y.Array<any>): unknown {\n  return yjsContainerToSnapshot.get(container)\n}\n\n/**\n * Checks if a Y.js container is up-to-date with the given snapshot.\n * Uses reference equality to check if the snapshot is the same.\n */\nexport function isYjsContainerUpToDate(\n  container: Y.Map<any> | Y.Array<any>,\n  snapshot: unknown\n): boolean {\n  return yjsContainerToSnapshot.get(container) === snapshot\n}\n","import { frozenKey, modelTypeKey, SnapshotOutOf } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { PlainArray, PlainObject, PlainPrimitive, PlainValue } from \"../plainTypes\"\nimport { YjsData } from \"./convertYjsDataToJson\"\nimport { YjsTextModel, yjsTextModelId } from \"./YjsTextModel\"\nimport { isYjsContainerUpToDate, setYjsContainerSnapshot } from \"./yjsSnapshotTracking\"\n\n/**\n * Options for applying JSON data to Y.js data structures.\n */\nexport interface ApplyJsonToYjsOptions {\n  /**\n   * The mode to use when applying JSON data to Y.js data structures.\n   * - `add`: Creates new Y.js containers for objects/arrays (default, backwards compatible)\n   * - `merge`: Recursively merges values, preserving existing container references where possible\n   */\n  mode?: \"add\" | \"merge\"\n}\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 typeof v === \"object\" && v !== null && !Array.isArray(v)\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[frozenKey] === true) {\n      // frozen value, save as immutable object\n      return v\n    }\n\n    if (v[modelTypeKey] === 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 *\n * @param dest The destination Y.Array.\n * @param source The source JSON array.\n * @param options Options for applying the JSON data.\n */\nexport const applyJsonArrayToYArray = (\n  dest: Y.Array<any>,\n  source: PlainArray,\n  options: ApplyJsonToYjsOptions = {}\n) => {\n  const { mode = \"add\" } = options\n\n  // In merge mode, check if the container is already up-to-date with this snapshot\n  if (mode === \"merge\" && isYjsContainerUpToDate(dest, source)) {\n    return\n  }\n\n  const srcLen = source.length\n\n  if (mode === \"add\") {\n    // Add mode: just push all items to the end\n    for (let i = 0; i < srcLen; i++) {\n      dest.push([convertJsonToYjsData(source[i])])\n    }\n    return\n  }\n\n  // Merge mode: recursively merge values, preserving existing container references\n  const destLen = dest.length\n\n  // Remove extra items from the end\n  if (destLen > srcLen) {\n    dest.delete(srcLen, destLen - srcLen)\n  }\n\n  // Update existing items\n  const minLen = Math.min(destLen, srcLen)\n  for (let i = 0; i < minLen; i++) {\n    const srcItem = source[i]\n    const destItem = dest.get(i)\n\n    // If both are objects, merge recursively\n    if (isPlainObject(srcItem) && destItem instanceof Y.Map) {\n      applyJsonObjectToYMap(destItem, srcItem, options)\n      continue\n    }\n\n    // If both are arrays, merge recursively\n    if (isPlainArray(srcItem) && destItem instanceof Y.Array) {\n      applyJsonArrayToYArray(destItem, srcItem, options)\n      continue\n    }\n\n    // Skip if primitive value is unchanged (optimization)\n    if (isPlainPrimitive(srcItem) && destItem === srcItem) {\n      continue\n    }\n\n    // Otherwise, replace the item\n    dest.delete(i, 1)\n    dest.insert(i, [convertJsonToYjsData(srcItem)])\n  }\n\n  // Add new items at the end\n  for (let i = destLen; i < srcLen; i++) {\n    dest.push([convertJsonToYjsData(source[i])])\n  }\n\n  // Update snapshot tracking after successful merge\n  setYjsContainerSnapshot(dest, source)\n}\n\n/**\n * Applies a JSON object to a Y.Map, using the convertJsonToYjsData to convert the values.\n *\n * @param dest The destination Y.Map.\n * @param source The source JSON object.\n * @param options Options for applying the JSON data.\n */\nexport const applyJsonObjectToYMap = (\n  dest: Y.Map<any>,\n  source: PlainObject,\n  options: ApplyJsonToYjsOptions = {}\n) => {\n  const { mode = \"add\" } = options\n\n  // In merge mode, check if the container is already up-to-date with this snapshot\n  if (mode === \"merge\" && isYjsContainerUpToDate(dest, source)) {\n    return\n  }\n\n  if (mode === \"add\") {\n    // Add mode: just set all values\n    for (const k of Object.keys(source)) {\n      const v = source[k]\n      if (v !== undefined) {\n        dest.set(k, convertJsonToYjsData(v))\n      }\n    }\n    return\n  }\n\n  // Merge mode: recursively merge values, preserving existing container references\n\n  // Delete keys that are not present in source (or have undefined value)\n  const sourceKeysWithValues = new Set(Object.keys(source).filter((k) => source[k] !== undefined))\n  for (const key of dest.keys()) {\n    if (!sourceKeysWithValues.has(key)) {\n      dest.delete(key)\n    }\n  }\n\n  for (const k of Object.keys(source)) {\n    const v = source[k]\n    // Skip undefined values - Y.js maps cannot store undefined\n    if (v === undefined) {\n      continue\n    }\n\n    const existing = dest.get(k)\n\n    // If source is an object and dest has a Y.Map, merge recursively\n    if (isPlainObject(v) && existing instanceof Y.Map) {\n      applyJsonObjectToYMap(existing, v, options)\n      continue\n    }\n\n    // If source is an array and dest has a Y.Array, merge recursively\n    if (isPlainArray(v) && existing instanceof Y.Array) {\n      applyJsonArrayToYArray(existing, v, options)\n      continue\n    }\n\n    // Skip if primitive value is unchanged (optimization)\n    if (isPlainPrimitive(v) && existing === v) {\n      continue\n    }\n\n    // Otherwise, convert and set the value (this creates new containers if needed)\n    dest.set(k, convertJsonToYjsData(v))\n  }\n\n  // Update snapshot tracking after successful merge\n  setYjsContainerSnapshot(dest, source)\n}\n","import { DeepChange, DeepChangeType } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { failure } from \"../utils/error\"\nimport { isYjsValueDeleted } from \"../utils/isYjsValueDeleted\"\nimport { convertJsonToYjsData } from \"./convertJsonToYjsData\"\nimport { resolveYjsPath } from \"./resolveYjsPath\"\n\n/**\n * Converts a snapshot value to a Yjs-compatible value.\n * Note: All values passed here are already snapshots (captured at change time).\n */\nfunction convertValue(v: unknown): any {\n  // Handle primitives directly\n  if (v === null || v === undefined || typeof v !== \"object\") {\n    return v\n  }\n  // Handle plain arrays - used for empty array init\n  if (Array.isArray(v) && v.length === 0) {\n    return new Y.Array()\n  }\n  // Value is already a snapshot, convert to Yjs data\n  return convertJsonToYjsData(v as any)\n}\n\nexport function applyMobxChangeToYjsObject(\n  change: DeepChange,\n  yjsObject: Y.Map<any> | Y.Array<any> | Y.Text\n): void {\n  // Check if the YJS object is deleted\n  if (isYjsValueDeleted(yjsObject)) {\n    throw failure(\"cannot apply patch to deleted Yjs value\")\n  }\n\n  const yjsContainer = resolveYjsPath(yjsObject, change.path)\n\n  if (!yjsContainer) {\n    // Container not found, skip this change\n    return\n  }\n\n  if (yjsContainer instanceof Y.Array) {\n    if (change.type === DeepChangeType.ArraySplice) {\n      // splice\n      yjsContainer.delete(change.index, change.removedValues.length)\n      if (change.addedValues.length > 0) {\n        const valuesToInsert = change.addedValues.map(convertValue)\n        yjsContainer.insert(change.index, valuesToInsert)\n      }\n    } else if (change.type === DeepChangeType.ArrayUpdate) {\n      // update\n      yjsContainer.delete(change.index, 1)\n      yjsContainer.insert(change.index, [convertValue(change.newValue)])\n    } else {\n      throw failure(`unsupported array change type: ${change.type}`)\n    }\n  } else if (yjsContainer instanceof Y.Map) {\n    if (change.type === DeepChangeType.ObjectAdd || change.type === DeepChangeType.ObjectUpdate) {\n      const key = change.key\n      if (change.newValue === undefined) {\n        yjsContainer.delete(key)\n      } else {\n        yjsContainer.set(key, convertValue(change.newValue))\n      }\n    } else if (change.type === DeepChangeType.ObjectRemove) {\n      const key = change.key\n      yjsContainer.delete(key)\n    } else {\n      throw failure(`unsupported object change type: ${change.type}`)\n    }\n  } else if (yjsContainer instanceof Y.Text) {\n    // Y.Text is handled differently - init changes for text are managed by YjsTextModel\n    // Skip init changes for Y.Text containers\n    return\n  } else {\n    throw failure(`unsupported Yjs container type: ${yjsContainer}`)\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 { remove } from \"mobx\"\nimport {\n  Frozen,\n  fromSnapshot,\n  frozen,\n  getSnapshot,\n  getSnapshotModelId,\n  isFrozenSnapshot,\n  isModel,\n  Path,\n  resolvePath,\n  runUnprotected,\n} from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { failure } from \"../utils/error\"\nimport { convertYjsDataToJson } from \"./convertYjsDataToJson\"\n\n// Represents the map of potential objects to reconcile (ID -> Object)\nexport type ReconciliationMap = Map<string, object>\n\n/**\n * Applies a Y.js event directly to the MobX model tree using proper mutations\n * (splice for arrays, property assignment for objects).\n * This is more efficient than converting to patches first.\n */\nexport function applyYjsEventToMobx(\n  event: Y.YEvent<any>,\n  boundObject: object,\n  reconciliationMap: ReconciliationMap\n): void {\n  const path = event.path as Path\n  const { value: target } = resolvePath(boundObject, path)\n\n  if (!target) {\n    throw failure(`cannot resolve path ${JSON.stringify(path)}`)\n  }\n\n  // Wrap in runUnprotected since we're modifying the tree from outside a model action\n  runUnprotected(() => {\n    if (event instanceof Y.YMapEvent) {\n      applyYMapEventToMobx(event, target, reconciliationMap)\n    } else if (event instanceof Y.YArrayEvent) {\n      applyYArrayEventToMobx(event, target, reconciliationMap)\n    } else if (event instanceof Y.YTextEvent) {\n      applyYTextEventToMobx(event, target)\n    }\n  })\n}\n\nfunction processDeletedValue(val: unknown, reconciliationMap: ReconciliationMap) {\n  if (val && typeof val === \"object\" && isModel(val)) {\n    const sn = getSnapshot(val)\n    const id = getSnapshotModelId(sn)\n    if (id) {\n      reconciliationMap.set(id, val)\n    }\n  }\n}\n\nfunction reviveValue(jsonValue: any, reconciliationMap: ReconciliationMap): any {\n  // Handle primitives\n  if (jsonValue === null || typeof jsonValue !== \"object\") {\n    return jsonValue\n  }\n\n  // Handle frozen\n  if (isFrozenSnapshot(jsonValue)) {\n    return frozen(jsonValue.data)\n  }\n\n  // If we have a reconciliation map and the value looks like a model with an ID, check if we have it\n  if (reconciliationMap && jsonValue && typeof jsonValue === \"object\") {\n    const modelId = getSnapshotModelId(jsonValue)\n    if (modelId) {\n      const existing = reconciliationMap.get(modelId)\n      if (existing) {\n        reconciliationMap.delete(modelId)\n        return existing\n      }\n    }\n  }\n\n  return fromSnapshot(jsonValue)\n}\n\nfunction applyYMapEventToMobx(\n  event: Y.YMapEvent<any>,\n  target: Record<string, any>,\n  reconciliationMap: ReconciliationMap\n): void {\n  const source = event.target\n\n  event.changes.keys.forEach((change, key) => {\n    switch (change.action) {\n      case \"add\":\n      case \"update\": {\n        const yjsValue = source.get(key)\n        const jsonValue = convertYjsDataToJson(yjsValue)\n\n        // If updating, the old value is overwritten (deleted conceptually)\n        if (change.action === \"update\") {\n          processDeletedValue(target[key], reconciliationMap)\n        }\n\n        target[key] = reviveValue(jsonValue, reconciliationMap)\n        break\n      }\n\n      case \"delete\": {\n        processDeletedValue(target[key], reconciliationMap)\n        // Use MobX's remove to properly delete the key from the observable object\n        // This triggers the \"remove\" interceptor in mobx-keystone's tweaker\n        if (isModel(target)) {\n          remove(target.$, key)\n        } else {\n          remove(target, key)\n        }\n        break\n      }\n\n      default:\n        throw failure(`unsupported Yjs map event action: ${change.action}`)\n    }\n  })\n}\n\nfunction applyYArrayEventToMobx(\n  event: Y.YArrayEvent<any>,\n  target: any[],\n  reconciliationMap: ReconciliationMap\n): void {\n  // Process delta operations in order\n  let currentIndex = 0\n\n  for (const change of event.changes.delta) {\n    if (change.retain) {\n      currentIndex += change.retain\n    }\n\n    if (change.delete) {\n      // Capture deleted items for reconciliation\n      const deletedItems = target.slice(currentIndex, currentIndex + change.delete)\n      deletedItems.forEach((item) => {\n        processDeletedValue(item, reconciliationMap)\n      })\n\n      // Delete items at current position\n      target.splice(currentIndex, change.delete)\n    }\n\n    if (change.insert) {\n      // Insert items at current position\n      const insertedItems = Array.isArray(change.insert) ? change.insert : [change.insert]\n      const values = insertedItems.map((yjsValue) => {\n        const jsonValue = convertYjsDataToJson(yjsValue)\n        return reviveValue(jsonValue, reconciliationMap)\n      })\n\n      target.splice(currentIndex, 0, ...values)\n      currentIndex += values.length\n    }\n  }\n}\n\nfunction applyYTextEventToMobx(\n  event: Y.YTextEvent,\n  target: { deltaList?: Frozen<unknown[]>[] }\n): void {\n  // YjsTextModel handles text events by appending delta to deltaList\n  if (target?.deltaList) {\n    target.deltaList.push(frozen(event.delta))\n  }\n}\n","import { action } from \"mobx\"\nimport {\n  AnyDataModel,\n  AnyModel,\n  AnyStandardType,\n  DeepChange,\n  DeepChangeType,\n  fromSnapshot,\n  getParentToChildPath,\n  getSnapshot,\n  isTreeNode,\n  ModelClass,\n  onDeepChange,\n  onGlobalDeepChange,\n  onSnapshot,\n  SnapshotInOf,\n  TypeToData,\n} from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport type { PlainArray, PlainObject } from \"../plainTypes\"\nimport { failure } from \"../utils/error\"\nimport { getYjsCollectionAtom } from \"../utils/getOrCreateYjsCollectionAtom\"\nimport { isYjsValueDeleted } from \"../utils/isYjsValueDeleted\"\nimport { applyMobxChangeToYjsObject } from \"./applyMobxChangeToYjsObject\"\nimport { applyYjsEventToMobx, ReconciliationMap } from \"./applyYjsEventToMobx\"\nimport { applyJsonArrayToYArray, applyJsonObjectToYMap } from \"./convertJsonToYjsData\"\nimport { convertYjsDataToJson } from \"./convertYjsDataToJson\"\nimport { YjsBindingContext, yjsBindingContext } from \"./yjsBindingContext\"\nimport { setYjsContainerSnapshot } from \"./yjsSnapshotTracking\"\n\n/**\n * Captures snapshots of tree nodes in a DeepChange.\n * This ensures values are captured at change time, not at apply time,\n * preventing issues when values are mutated after being added to a collection.\n */\nfunction captureChangeSnapshots(change: DeepChange): DeepChange {\n  if (change.type === DeepChangeType.ArraySplice && change.addedValues.length > 0) {\n    const snapshots = change.addedValues.map((v) => (isTreeNode(v) ? getSnapshot(v) : v))\n    return { ...change, addedValues: snapshots }\n  } else if (change.type === DeepChangeType.ArrayUpdate) {\n    const snapshot = isTreeNode(change.newValue) ? getSnapshot(change.newValue) : change.newValue\n    return { ...change, newValue: snapshot }\n  } else if (\n    change.type === DeepChangeType.ObjectAdd ||\n    change.type === DeepChangeType.ObjectUpdate\n  ) {\n    const snapshot = isTreeNode(change.newValue) ? getSnapshot(change.newValue) : change.newValue\n    return { ...change, newValue: snapshot }\n  }\n  return change\n}\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  if (isYjsValueDeleted(yjsObject)) {\n    throw failure(\"cannot apply patch to deleted Yjs value\")\n  }\n\n  const yjsJson = convertYjsDataToJson(yjsObject)\n\n  let boundObject: TypeToData<TType>\n\n  // Track if any init changes occur during fromSnapshot\n  // (e.g., defaults being applied, onInit hooks mutating the model)\n  let hasInitChanges = false\n\n  const createBoundObject = () => {\n    // Set up a temporary global listener to detect if any init changes occur during fromSnapshot\n    const disposeGlobalListener = onGlobalDeepChange((_target, change) => {\n      if (change.isInit) {\n        hasInitChanges = true\n      }\n    })\n\n    try {\n      const result = yjsBindingContext.apply(\n        () => fromSnapshot(mobxKeystoneType, yjsJson as unknown as SnapshotInOf<TypeToData<TType>>),\n        bindingContext\n      )\n      yjsBindingContext.set(result, { ...bindingContext, boundObject: result })\n      return result\n    } finally {\n      disposeGlobalListener()\n    }\n  }\n\n  boundObject = createBoundObject()\n\n  // bind any changes from yjs to mobx-keystone\n  const observeDeepCb = action((events: Y.YEvent<any>[]) => {\n    const eventsToApply: Y.YEvent<any>[] = []\n\n    events.forEach((event) => {\n      if (event.transaction.origin !== yjsOrigin) {\n        eventsToApply.push(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 (eventsToApply.length > 0) {\n      applyingYjsChangesToMobxKeystone++\n      try {\n        const reconciliationMap: ReconciliationMap = new Map()\n\n        // Collect init changes that occur during event application\n        // (e.g., fromSnapshot calls that trigger onInit hooks)\n        // We store both target and change so we can compute the correct path later\n        // Snapshots are captured immediately to preserve values at init time\n        const initChanges: { target: object; change: DeepChange }[] = []\n        const disposeGlobalListener = onGlobalDeepChange((target, change) => {\n          if (change.isInit) {\n            initChanges.push({ target, change: captureChangeSnapshots(change) })\n          }\n        })\n\n        try {\n          eventsToApply.forEach((event) => {\n            applyYjsEventToMobx(event, boundObject, reconciliationMap)\n          })\n        } finally {\n          disposeGlobalListener()\n        }\n\n        // Sync back any init-time mutations from fromSnapshot calls\n        // (e.g., onInit hooks that modify the model)\n        // This is needed because init changes during Yjs event handling are not\n        // captured by the main onDeepChange (it skips changes when applyingYjsChangesToMobxKeystone > 0)\n        if (initChanges.length > 0 && !isYjsValueDeleted(yjsObject)) {\n          yjsDoc.transact(() => {\n            for (const { target, change } of initChanges) {\n              // Compute the path from boundObject to the target\n              const pathToTarget = getParentToChildPath(boundObject, target)\n              if (pathToTarget !== undefined) {\n                // Create a new change with the correct path from the root\n                const changeWithCorrectPath: DeepChange = {\n                  ...change,\n                  path: [...pathToTarget, ...change.path],\n                }\n                applyMobxChangeToYjsObject(changeWithCorrectPath, yjsObject)\n              }\n            }\n          }, yjsOrigin)\n        }\n\n        // Update snapshot tracking: the Y.js container is now in sync with the current MobX snapshot\n        // This enables the merge optimization to skip unchanged subtrees during reconciliation\n        if (yjsObject instanceof Y.Map || yjsObject instanceof Y.Array) {\n          setYjsContainerSnapshot(yjsObject, getSnapshot(boundObject))\n        }\n      } finally {\n        applyingYjsChangesToMobxKeystone--\n      }\n    }\n  })\n\n  yjsObject.observeDeep(observeDeepCb)\n\n  // bind any changes from mobx-keystone to yjs using deep change observation\n  // This provides proper splice detection for array operations\n  let pendingChanges: DeepChange[] = []\n\n  const disposeOnDeepChange = onDeepChange(boundObject, (change) => {\n    if (applyingYjsChangesToMobxKeystone > 0) {\n      return\n    }\n\n    // Skip init changes - they are handled by the getSnapshot + merge at the end of binding\n    if (change.isInit) {\n      return\n    }\n\n    // Capture snapshots now before the values can be mutated within the same transaction.\n    // This is necessary because changes are collected and applied after the action completes,\n    // by which time the original values may have been modified.\n    // Example: `obj.items = [a, b]; obj.items.splice(0, 1)` - without early capture,\n    // the ObjectUpdate for `items` would get the post-splice array state.\n    pendingChanges.push(captureChangeSnapshots(change))\n  })\n\n  // this is only used so we can transact all changes to the snapshot boundary\n  const disposeOnSnapshot = onSnapshot(boundObject, (boundObjectSnapshot) => {\n    if (pendingChanges.length === 0) {\n      return\n    }\n\n    const changesToApply = pendingChanges\n    pendingChanges = []\n\n    // Skip syncing to Yjs if the Yjs object has been deleted/detached\n    if (isYjsValueDeleted(yjsObject)) {\n      return\n    }\n\n    yjsDoc.transact(() => {\n      changesToApply.forEach((change) => {\n        applyMobxChangeToYjsObject(change, yjsObject)\n      })\n    }, yjsOrigin)\n\n    // Update snapshot tracking: the Y.js container is now in sync with the current MobX snapshot\n    if (yjsObject instanceof Y.Map || yjsObject instanceof Y.Array) {\n      setYjsContainerSnapshot(yjsObject, boundObjectSnapshot)\n    }\n  })\n\n  // Sync the init changes to the CRDT.\n  // Init changes include: defaults being applied, onInit hooks mutating the model.\n  // We use getSnapshot + merge because the per-change approach has issues with reference mutation\n  // (values captured in DeepChange can be mutated before we apply them).\n  // The snapshot tracking optimization ensures unchanged subtrees are skipped during merge.\n  const finalSnapshot = getSnapshot(boundObject)\n\n  if (hasInitChanges) {\n    yjsDoc.transact(() => {\n      if (yjsObject instanceof Y.Map) {\n        applyJsonObjectToYMap(yjsObject, finalSnapshot as unknown as PlainObject, {\n          mode: \"merge\",\n        })\n      } else if (yjsObject instanceof Y.Array) {\n        applyJsonArrayToYArray(yjsObject, finalSnapshot as unknown as PlainArray, {\n          mode: \"merge\",\n        })\n      }\n    }, yjsOrigin)\n  }\n\n  // Always update snapshot tracking after binding initialization\n  // This ensures the merge optimization can skip unchanged subtrees in future reconciliations\n  if (yjsObject instanceof Y.Map || yjsObject instanceof Y.Array) {\n    setYjsContainerSnapshot(yjsObject, finalSnapshot)\n  }\n\n  const dispose = () => {\n    yjsDoc.off(\"destroy\", dispose)\n    disposeOnDeepChange()\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":["createAtom","Y","createContext","types","YjsTextModel","Model","tProp","frozen","getParentToChildPath","reaction","observe","onSnapshot","computed","model","frozenKey","modelTypeKey","DeepChangeType","action","modelSnapshotOutWithMetadata","resolvePath","runUnprotected","isModel","getSnapshot","getSnapshotModelId","isFrozenSnapshot","fromSnapshot","remove","isTreeNode","onGlobalDeepChange","onDeepChange"],"mappings":"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"}
|