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