mobx-keystone-yjs 1.0.0 → 1.1.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 +9 -5
- package/dist/mobx-keystone-yjs.esm.js +25 -21
- package/dist/mobx-keystone-yjs.esm.mjs +25 -21
- package/dist/mobx-keystone-yjs.umd.js +24 -20
- package/dist/types/binding/convertJsonToYjsData.d.ts +5 -0
- package/dist/types/index.d.ts +1 -0
- package/dist/types/jsonTypes.d.ts +5 -5
- package/package.json +87 -87
- package/src/binding/applyMobxKeystonePatchToYjsObject.ts +92 -92
- package/src/binding/convertJsonToYjsData.ts +45 -0
- package/src/binding/convertYjsEventToPatches.ts +85 -85
- package/src/index.ts +7 -2
- package/src/jsonTypes.ts +4 -4
- package/dist/types/binding/toYDataType.d.ts +0 -3
- package/src/binding/toYDataType.ts +0 -41
package/CHANGELOG.md
CHANGED
|
@@ -1,5 +1,9 @@
|
|
|
1
|
-
# Change Log
|
|
2
|
-
|
|
3
|
-
## 1.
|
|
4
|
-
|
|
5
|
-
-
|
|
1
|
+
# Change Log
|
|
2
|
+
|
|
3
|
+
## 1.1.0
|
|
4
|
+
|
|
5
|
+
- Added the `convertJsonToYjsData`, `applyJsonArrayToYArray` and `applyJsonObjectToYMap` function to help with first migrations from snapshots to Y.js states.
|
|
6
|
+
|
|
7
|
+
## 1.0.0
|
|
8
|
+
|
|
9
|
+
- First public release.
|
|
@@ -9,37 +9,38 @@ class MobxKeystoneYjsError extends Error {
|
|
|
9
9
|
function failure(msg) {
|
|
10
10
|
return new MobxKeystoneYjsError(msg);
|
|
11
11
|
}
|
|
12
|
-
function
|
|
12
|
+
function isJsonPrimitive(v) {
|
|
13
13
|
const t = typeof v;
|
|
14
14
|
return t === "string" || t === "number" || t === "boolean" || v === null;
|
|
15
15
|
}
|
|
16
|
-
function
|
|
16
|
+
function isJsonArray(v) {
|
|
17
17
|
return Array.isArray(v);
|
|
18
18
|
}
|
|
19
|
-
function
|
|
20
|
-
return !
|
|
19
|
+
function isJsonObject(v) {
|
|
20
|
+
return !isJsonArray(v) && typeof v === "object";
|
|
21
21
|
}
|
|
22
|
-
function
|
|
23
|
-
if (
|
|
22
|
+
function convertJsonToYjsData(v) {
|
|
23
|
+
if (v === void 0 || isJsonPrimitive(v)) {
|
|
24
24
|
return v;
|
|
25
|
-
}
|
|
25
|
+
}
|
|
26
|
+
if (isJsonArray(v)) {
|
|
26
27
|
const arr = new Y.Array();
|
|
27
|
-
|
|
28
|
+
applyJsonArrayYArray(arr, v);
|
|
28
29
|
return arr;
|
|
29
|
-
}
|
|
30
|
+
}
|
|
31
|
+
if (isJsonObject(v)) {
|
|
30
32
|
const map = new Y.Map();
|
|
31
|
-
|
|
33
|
+
applyJsonObjectToYMap(map, v);
|
|
32
34
|
return map;
|
|
33
|
-
} else {
|
|
34
|
-
return void 0;
|
|
35
35
|
}
|
|
36
|
+
throw new Error(`unsupported value type: ${v}`);
|
|
36
37
|
}
|
|
37
|
-
function
|
|
38
|
-
dest.push(source.map(
|
|
38
|
+
function applyJsonArrayYArray(dest, source) {
|
|
39
|
+
dest.push(source.map(convertJsonToYjsData));
|
|
39
40
|
}
|
|
40
|
-
function
|
|
41
|
+
function applyJsonObjectToYMap(dest, source) {
|
|
41
42
|
Object.entries(source).forEach(([k, v]) => {
|
|
42
|
-
dest.set(k,
|
|
43
|
+
dest.set(k, convertJsonToYjsData(v));
|
|
43
44
|
});
|
|
44
45
|
}
|
|
45
46
|
function applyMobxKeystonePatchToYjsObject(patch, yjs) {
|
|
@@ -66,7 +67,7 @@ function applyMobxKeystonePatchToYjsObject(patch, yjs) {
|
|
|
66
67
|
switch (patch.op) {
|
|
67
68
|
case "add":
|
|
68
69
|
case "replace": {
|
|
69
|
-
yjs.set(key,
|
|
70
|
+
yjs.set(key, convertJsonToYjsData(patch.value));
|
|
70
71
|
break;
|
|
71
72
|
}
|
|
72
73
|
case "remove": {
|
|
@@ -91,12 +92,12 @@ function applyMobxKeystonePatchToYjsObject(patch, yjs) {
|
|
|
91
92
|
}
|
|
92
93
|
} else {
|
|
93
94
|
yjs.delete(Number(key));
|
|
94
|
-
yjs.insert(Number(key), [
|
|
95
|
+
yjs.insert(Number(key), [convertJsonToYjsData(patch.value)]);
|
|
95
96
|
}
|
|
96
97
|
break;
|
|
97
98
|
}
|
|
98
99
|
case "add": {
|
|
99
|
-
yjs.insert(Number(key), [
|
|
100
|
+
yjs.insert(Number(key), [convertJsonToYjsData(patch.value)]);
|
|
100
101
|
break;
|
|
101
102
|
}
|
|
102
103
|
case "remove": {
|
|
@@ -267,6 +268,9 @@ function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
|
|
|
267
268
|
}
|
|
268
269
|
export {
|
|
269
270
|
MobxKeystoneYjsError,
|
|
270
|
-
|
|
271
|
+
applyJsonArrayYArray,
|
|
272
|
+
applyJsonObjectToYMap,
|
|
273
|
+
bindYjsToMobxKeystone,
|
|
274
|
+
convertJsonToYjsData
|
|
271
275
|
};
|
|
272
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"mobx-keystone-yjs.esm.mjs","sources":["../src/utils/error.ts","../src/binding/toYDataType.ts","../src/binding/applyMobxKeystonePatchToYjsObject.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 * as Y from \"yjs\"\r\nimport { JSONValue, JSONArray, JSONObject, JSONPrimitive } from \"../jsonTypes\"\r\n\r\nfunction isJSONPrimitive(v: JSONValue): v is JSONPrimitive {\r\n  const t = typeof v\r\n  return t === \"string\" || t === \"number\" || t === \"boolean\" || v === null\r\n}\r\n\r\nfunction isJSONArray(v: JSONValue): v is JSONArray {\r\n  return Array.isArray(v)\r\n}\r\n\r\nfunction isJSONObject(v: JSONValue): v is JSONObject {\r\n  return !isJSONArray(v) && typeof v === \"object\"\r\n}\r\n\r\nexport function toYDataType(v: JSONValue) {\r\n  if (isJSONPrimitive(v)) {\r\n    return v\r\n  } else if (isJSONArray(v)) {\r\n    const arr = new Y.Array()\r\n    applyJsonArray(arr, v)\r\n    return arr\r\n  } else if (isJSONObject(v)) {\r\n    const map = new Y.Map()\r\n    applyJsonObject(map, v)\r\n    return map\r\n  } else {\r\n    return undefined\r\n  }\r\n}\r\n\r\nfunction applyJsonArray(dest: Y.Array<unknown>, source: JSONArray) {\r\n  dest.push(source.map(toYDataType))\r\n}\r\n\r\nfunction applyJsonObject(dest: Y.Map<unknown>, source: JSONObject) {\r\n  Object.entries(source).forEach(([k, v]) => {\r\n    dest.set(k, toYDataType(v))\r\n  })\r\n}\r\n","import { Patch } from \"mobx-keystone\"\r\nimport { failure } from \"../utils/error\"\r\nimport * as Y from \"yjs\"\r\nimport { toYDataType } from \"./toYDataType\"\r\n\r\nexport function applyMobxKeystonePatchToYjsObject(patch: Patch, yjs: unknown): void {\r\n  if (patch.path.length > 1) {\r\n    const [key, ...rest] = patch.path\r\n\r\n    if (yjs instanceof Y.Map) {\r\n      const child = yjs.get(String(key))\r\n      if (child === undefined) {\r\n        throw failure(\r\n          `invalid patch path, key \"${key}\" not found in Yjs map - patch: ${JSON.stringify(patch)}`\r\n        )\r\n      }\r\n      applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)\r\n    } else if (yjs instanceof Y.Array) {\r\n      const child = yjs.get(Number(key))\r\n      if (child === undefined) {\r\n        throw failure(\r\n          `invalid patch path, key \"${key}\" not found in Yjs array - patch: ${JSON.stringify(\r\n            patch\r\n          )}`\r\n        )\r\n      }\r\n      applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)\r\n    } else {\r\n      throw failure(\r\n        `invalid patch path, key \"${key}\" not found in unknown Yjs object - patch: ${JSON.stringify(\r\n          patch\r\n        )}`\r\n      )\r\n    }\r\n  } else if (patch.path.length === 1) {\r\n    if (yjs instanceof Y.Map) {\r\n      const key = String(patch.path[0])\r\n\r\n      switch (patch.op) {\r\n        case \"add\":\r\n        case \"replace\": {\r\n          yjs.set(key, toYDataType(patch.value))\r\n          break\r\n        }\r\n        case \"remove\": {\r\n          yjs.delete(key)\r\n          break\r\n        }\r\n        default: {\r\n          throw failure(`invalid patch operation for map`)\r\n        }\r\n      }\r\n    } else if (yjs instanceof Y.Array) {\r\n      const key = patch.path[0]\r\n\r\n      switch (patch.op) {\r\n        case \"replace\": {\r\n          if (key === \"length\") {\r\n            if (yjs.length > patch.value) {\r\n              const toDelete = yjs.length - patch.value\r\n              yjs.delete(patch.value, toDelete)\r\n            } else if (yjs.length < patch.value) {\r\n              const toInsert = patch.value - yjs.length\r\n              yjs.insert(yjs.length, Array(toInsert).fill(undefined))\r\n            }\r\n          } else {\r\n            yjs.delete(Number(key))\r\n            yjs.insert(Number(key), [toYDataType(patch.value)])\r\n          }\r\n          break\r\n        }\r\n        case \"add\": {\r\n          yjs.insert(Number(key), [toYDataType(patch.value)])\r\n          break\r\n        }\r\n        case \"remove\": {\r\n          yjs.delete(Number(key))\r\n          break\r\n        }\r\n        default: {\r\n          throw failure(`invalid patch operation for array`)\r\n        }\r\n      }\r\n    } else {\r\n      throw failure(\r\n        `invalid patch path, the Yjs object is of an unkown type, so key \"${patch.path[0]}\" cannot be found in it`\r\n      )\r\n    }\r\n  } else {\r\n    throw failure(`invalid patch path, it cannot be empty`)\r\n  }\r\n}\r\n","import { Patch } from \"mobx-keystone\"\r\nimport * as Y from \"yjs\"\r\nimport { JSONArray, JSONObject, JSONValue } from \"../jsonTypes\"\r\nimport { failure } from \"../utils/error\"\r\n\r\nexport function convertYjsEventToPatches(event: Y.YEvent<any>): Patch[] {\r\n  const patches: Patch[] = []\r\n\r\n  if (event instanceof Y.YMapEvent) {\r\n    const source = event.target as Y.Map<any>\r\n\r\n    event.changes.keys.forEach((change, key) => {\r\n      const path = [...event.path, key]\r\n\r\n      switch (change.action) {\r\n        case \"add\":\r\n          patches.push({\r\n            op: \"add\",\r\n            path,\r\n            value: toPlainValue(source.get(key)),\r\n          })\r\n          break\r\n\r\n        case \"update\":\r\n          patches.push({\r\n            op: \"replace\",\r\n            path,\r\n            value: toPlainValue(source.get(key)),\r\n          })\r\n          break\r\n\r\n        case \"delete\":\r\n          patches.push({\r\n            op: \"remove\",\r\n            path,\r\n          })\r\n          break\r\n\r\n        default:\r\n          throw failure(`unsupported Yjs map event action: ${change.action}`)\r\n      }\r\n    })\r\n  } else if (event instanceof Y.YArrayEvent) {\r\n    let retain = 0\r\n    event.changes.delta.forEach((change) => {\r\n      if (change.retain) {\r\n        retain += change.retain\r\n      }\r\n\r\n      if (change.delete) {\r\n        // remove X items at retain position\r\n        const path = [...event.path, retain]\r\n        for (let i = 0; i < change.delete; i++) {\r\n          patches.push({\r\n            op: \"remove\",\r\n            path,\r\n          })\r\n        }\r\n      }\r\n\r\n      if (change.insert) {\r\n        const newValues = Array.isArray(change.insert) ? change.insert : [change.insert]\r\n        newValues.forEach((v) => {\r\n          const path = [...event.path, retain]\r\n          patches.push({\r\n            op: \"add\",\r\n            path,\r\n            value: toPlainValue(v),\r\n          })\r\n          retain++\r\n        })\r\n      }\r\n    })\r\n  }\r\n\r\n  return patches\r\n}\r\n\r\nfunction toPlainValue(v: Y.Map<any> | Y.Array<any> | JSONValue) {\r\n  if (v instanceof Y.Map || v instanceof Y.Array) {\r\n    return v.toJSON() as JSONObject | JSONArray\r\n  } else {\r\n    return v\r\n  }\r\n}\r\n","import {\r\n  AnyDataModel,\r\n  AnyModel,\r\n  AnyStandardType,\r\n  ModelClass,\r\n  Patch,\r\n  TypeToData,\r\n  applyPatches,\r\n  fromSnapshot,\r\n  getParentToChildPath,\r\n  onGlobalPatches,\r\n  onPatches,\r\n  onSnapshot,\r\n} from \"mobx-keystone\"\r\nimport * as Y from \"yjs\"\r\nimport { applyMobxKeystonePatchToYjsObject } from \"./applyMobxKeystonePatchToYjsObject\"\r\nimport { convertYjsEventToPatches } from \"./convertYjsEventToPatches\"\r\n\r\nexport function bindYjsToMobxKeystone<\r\n  TType extends AnyStandardType | ModelClass<AnyModel> | ModelClass<AnyDataModel>,\r\n>({\r\n  yjsDoc,\r\n  yjsObject,\r\n  mobxKeystoneType,\r\n}: {\r\n  yjsDoc: Y.Doc\r\n  yjsObject: Y.Map<unknown> | Y.Array<unknown>\r\n  mobxKeystoneType: TType\r\n}): {\r\n  boundObject: TypeToData<TType>\r\n  dispose(): void\r\n  yjsOrigin: symbol\r\n} {\r\n  const yjsJson = yjsObject.toJSON()\r\n\r\n  const initializationGlobalPatches: { target: object; patches: Patch[] }[] = []\r\n\r\n  const createBoundObject = () => {\r\n    const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {\r\n      initializationGlobalPatches.push({ target, patches })\r\n    })\r\n\r\n    try {\r\n      return fromSnapshot(mobxKeystoneType, yjsJson as any)\r\n    } finally {\r\n      disposeOnGlobalPatches()\r\n    }\r\n  }\r\n\r\n  const boundObject = createBoundObject()\r\n\r\n  let applyingMobxKeystoneChanges = 0\r\n  const yjsOrigin = Symbol(\"bindYjsToMobxKeystoneTransactionOrigin\")\r\n\r\n  // bind any changes from yjs to mobx-keystone\r\n  const observeDeepCb = (events: Y.YEvent<any>[]) => {\r\n    const patches: Patch[] = []\r\n    events.forEach((event) => {\r\n      if (event.transaction.origin !== yjsOrigin) {\r\n        patches.push(...convertYjsEventToPatches(event))\r\n      }\r\n    })\r\n\r\n    if (patches.length > 0) {\r\n      applyingMobxKeystoneChanges++\r\n      try {\r\n        applyPatches(boundObject, patches)\r\n      } finally {\r\n        applyingMobxKeystoneChanges--\r\n      }\r\n    }\r\n  }\r\n\r\n  yjsObject.observeDeep(observeDeepCb)\r\n\r\n  // bind any changes from mobx-keystone to yjs\r\n  let pendingPatches: Patch[] = []\r\n  const disposeOnPatches = onPatches(boundObject, (patches) => {\r\n    if (applyingMobxKeystoneChanges > 0) {\r\n      return\r\n    }\r\n\r\n    pendingPatches.push(...patches)\r\n  })\r\n\r\n  // this is only used so we can transact all patches to the snapshot boundary\r\n  const disposeOnSnapshot = onSnapshot(boundObject, () => {\r\n    if (pendingPatches.length === 0) {\r\n      return\r\n    }\r\n\r\n    const patches = pendingPatches\r\n    pendingPatches = []\r\n\r\n    yjsDoc.transact(() => {\r\n      patches.forEach((patch) => {\r\n        applyMobxKeystonePatchToYjsObject(patch, yjsObject)\r\n      })\r\n    }, yjsOrigin)\r\n  })\r\n\r\n  // sync initial patches, that might include setting defaults, IDs, etc\r\n  yjsDoc.transact(() => {\r\n    // we need to skip initializations until we hit the initialization of the bound object\r\n    // this is because default objects might be created and initialized before the main object\r\n    // but we just need to catch when those are actually assigned to the bound object\r\n    let boundObjectFound = false\r\n\r\n    initializationGlobalPatches.forEach(({ target, patches }) => {\r\n      if (!boundObjectFound) {\r\n        if (target !== boundObject) {\r\n          return // skip\r\n        }\r\n        boundObjectFound = true\r\n      }\r\n\r\n      const parentToChildPath = getParentToChildPath(boundObject, target)\r\n      // this is undefined only if target is not a child of boundModel\r\n      if (parentToChildPath !== undefined) {\r\n        patches.forEach((patch) => {\r\n          applyMobxKeystonePatchToYjsObject(\r\n            {\r\n              ...patch,\r\n              path: [...parentToChildPath, ...patch.path],\r\n            },\r\n            yjsObject\r\n          )\r\n        })\r\n      }\r\n    })\r\n  }, yjsOrigin)\r\n\r\n  return {\r\n    boundObject,\r\n    dispose: () => {\r\n      disposeOnPatches()\r\n      disposeOnSnapshot()\r\n      yjsObject.unobserveDeep(observeDeepCb)\r\n    },\r\n    yjsOrigin,\r\n  }\r\n}\r\n"],"names":[],"mappings":";;AAGM,MAAO,6BAA6B,MAAK;AAAA,EAC7C,YAAY,KAAW;AACrB,UAAM,GAAG;AAGF,WAAA,eAAe,MAAM,qBAAqB,SAAS;AAAA,EAC5D;AACD;AAKK,SAAU,QAAQ,KAAW;AAC1B,SAAA,IAAI,qBAAqB,GAAG;AACrC;ACdA,SAAS,gBAAgB,GAAY;AACnC,QAAM,IAAI,OAAO;AACjB,SAAO,MAAM,YAAY,MAAM,YAAY,MAAM,aAAa,MAAM;AACtE;AAEA,SAAS,YAAY,GAAY;AACxB,SAAA,MAAM,QAAQ,CAAC;AACxB;AAEA,SAAS,aAAa,GAAY;AAChC,SAAO,CAAC,YAAY,CAAC,KAAK,OAAO,MAAM;AACzC;AAEM,SAAU,YAAY,GAAY;AAClC,MAAA,gBAAgB,CAAC,GAAG;AACf,WAAA;AAAA,EAAA,WACE,YAAY,CAAC,GAAG;AACnB,UAAA,MAAM,IAAI,EAAE;AAClB,mBAAe,KAAK,CAAC;AACd,WAAA;AAAA,EAAA,WACE,aAAa,CAAC,GAAG;AACpB,UAAA,MAAM,IAAI,EAAE;AAClB,oBAAgB,KAAK,CAAC;AACf,WAAA;AAAA,EAAA,OACF;AACE,WAAA;AAAA,EACT;AACF;AAEA,SAAS,eAAe,MAAwB,QAAiB;AAC/D,OAAK,KAAK,OAAO,IAAI,WAAW,CAAC;AACnC;AAEA,SAAS,gBAAgB,MAAsB,QAAkB;AACxD,SAAA,QAAQ,MAAM,EAAE,QAAQ,CAAC,CAAC,GAAG,CAAC,MAAK;AACxC,SAAK,IAAI,GAAG,YAAY,CAAC,CAAC;AAAA,EAAA,CAC3B;AACH;ACnCgB,SAAA,kCAAkC,OAAc,KAAY;AACtE,MAAA,MAAM,KAAK,SAAS,GAAG;AACzB,UAAM,CAAC,KAAK,GAAG,IAAI,IAAI,MAAM;AAEzB,QAAA,eAAe,EAAE,KAAK;AACxB,YAAM,QAAQ,IAAI,IAAI,OAAO,GAAG,CAAC;AACjC,UAAI,UAAU,QAAW;AACjB,cAAA,QACJ,4BAA4B,GAAG,mCAAmC,KAAK,UAAU,KAAK,CAAC,EAAE;AAAA,MAE7F;AACA,wCAAkC,EAAE,GAAG,OAAO,MAAM,KAAA,GAAQ,KAAK;AAAA,IAAA,WACxD,eAAe,EAAE,OAAO;AACjC,YAAM,QAAQ,IAAI,IAAI,OAAO,GAAG,CAAC;AACjC,UAAI,UAAU,QAAW;AACjB,cAAA,QACJ,4BAA4B,GAAG,qCAAqC,KAAK,UACvE,KAAK,CACN,EAAE;AAAA,MAEP;AACA,wCAAkC,EAAE,GAAG,OAAO,MAAM,KAAA,GAAQ,KAAK;AAAA,IAAA,OAC5D;AACC,YAAA,QACJ,4BAA4B,GAAG,8CAA8C,KAAK,UAChF,KAAK,CACN,EAAE;AAAA,IAEP;AAAA,EACS,WAAA,MAAM,KAAK,WAAW,GAAG;AAC9B,QAAA,eAAe,EAAE,KAAK;AACxB,YAAM,MAAM,OAAO,MAAM,KAAK,CAAC,CAAC;AAEhC,cAAQ,MAAM,IAAI;AAAA,QAChB,KAAK;AAAA,QACL,KAAK,WAAW;AACd,cAAI,IAAI,KAAK,YAAY,MAAM,KAAK,CAAC;AACrC;AAAA,QACF;AAAA,QACA,KAAK,UAAU;AACb,cAAI,OAAO,GAAG;AACd;AAAA,QACF;AAAA,QACA,SAAS;AACP,gBAAM,QAAQ,iCAAiC;AAAA,QACjD;AAAA,MACF;AAAA,IAAA,WACS,eAAe,EAAE,OAAO;AAC3B,YAAA,MAAM,MAAM,KAAK,CAAC;AAExB,cAAQ,MAAM,IAAI;AAAA,QAChB,KAAK,WAAW;AACd,cAAI,QAAQ,UAAU;AAChB,gBAAA,IAAI,SAAS,MAAM,OAAO;AACtB,oBAAA,WAAW,IAAI,SAAS,MAAM;AAChC,kBAAA,OAAO,MAAM,OAAO,QAAQ;AAAA,YACvB,WAAA,IAAI,SAAS,MAAM,OAAO;AAC7B,oBAAA,WAAW,MAAM,QAAQ,IAAI;AAC/B,kBAAA,OAAO,IAAI,QAAQ,MAAM,QAAQ,EAAE,KAAK,MAAS,CAAC;AAAA,YACxD;AAAA,UAAA,OACK;AACD,gBAAA,OAAO,OAAO,GAAG,CAAC;AAClB,gBAAA,OAAO,OAAO,GAAG,GAAG,CAAC,YAAY,MAAM,KAAK,CAAC,CAAC;AAAA,UACpD;AACA;AAAA,QACF;AAAA,QACA,KAAK,OAAO;AACN,cAAA,OAAO,OAAO,GAAG,GAAG,CAAC,YAAY,MAAM,KAAK,CAAC,CAAC;AAClD;AAAA,QACF;AAAA,QACA,KAAK,UAAU;AACT,cAAA,OAAO,OAAO,GAAG,CAAC;AACtB;AAAA,QACF;AAAA,QACA,SAAS;AACP,gBAAM,QAAQ,mCAAmC;AAAA,QACnD;AAAA,MACF;AAAA,IAAA,OACK;AACL,YAAM,QACJ,oEAAoE,MAAM,KAAK,CAAC,CAAC,yBAAyB;AAAA,IAE9G;AAAA,EAAA,OACK;AACL,UAAM,QAAQ,wCAAwC;AAAA,EACxD;AACF;ACtFM,SAAU,yBAAyB,OAAoB;AAC3D,QAAM,UAAmB,CAAA;AAErB,MAAA,iBAAiB,EAAE,WAAW;AAChC,UAAM,SAAS,MAAM;AAErB,UAAM,QAAQ,KAAK,QAAQ,CAAC,QAAQ,QAAO;AACzC,YAAM,OAAO,CAAC,GAAG,MAAM,MAAM,GAAG;AAEhC,cAAQ,OAAO,QAAQ;AAAA,QACrB,KAAK;AACH,kBAAQ,KAAK;AAAA,YACX,IAAI;AAAA,YACJ;AAAA,YACA,OAAO,aAAa,OAAO,IAAI,GAAG,CAAC;AAAA,UAAA,CACpC;AACD;AAAA,QAEF,KAAK;AACH,kBAAQ,KAAK;AAAA,YACX,IAAI;AAAA,YACJ;AAAA,YACA,OAAO,aAAa,OAAO,IAAI,GAAG,CAAC;AAAA,UAAA,CACpC;AACD;AAAA,QAEF,KAAK;AACH,kBAAQ,KAAK;AAAA,YACX,IAAI;AAAA,YACJ;AAAA,UAAA,CACD;AACD;AAAA,QAEF;AACE,gBAAM,QAAQ,qCAAqC,OAAO,MAAM,EAAE;AAAA,MACtE;AAAA,IAAA,CACD;AAAA,EAAA,WACQ,iBAAiB,EAAE,aAAa;AACzC,QAAI,SAAS;AACb,UAAM,QAAQ,MAAM,QAAQ,CAAC,WAAU;AACrC,UAAI,OAAO,QAAQ;AACjB,kBAAU,OAAO;AAAA,MACnB;AAEA,UAAI,OAAO,QAAQ;AAEjB,cAAM,OAAO,CAAC,GAAG,MAAM,MAAM,MAAM;AACnC,iBAAS,IAAI,GAAG,IAAI,OAAO,QAAQ,KAAK;AACtC,kBAAQ,KAAK;AAAA,YACX,IAAI;AAAA,YACJ;AAAA,UAAA,CACD;AAAA,QACH;AAAA,MACF;AAEA,UAAI,OAAO,QAAQ;AACX,cAAA,YAAY,MAAM,QAAQ,OAAO,MAAM,IAAI,OAAO,SAAS,CAAC,OAAO,MAAM;AACrE,kBAAA,QAAQ,CAAC,MAAK;AACtB,gBAAM,OAAO,CAAC,GAAG,MAAM,MAAM,MAAM;AACnC,kBAAQ,KAAK;AAAA,YACX,IAAI;AAAA,YACJ;AAAA,YACA,OAAO,aAAa,CAAC;AAAA,UAAA,CACtB;AACD;AAAA,QAAA,CACD;AAAA,MACH;AAAA,IAAA,CACD;AAAA,EACH;AAEO,SAAA;AACT;AAEA,SAAS,aAAa,GAAwC;AAC5D,MAAI,aAAa,EAAE,OAAO,aAAa,EAAE,OAAO;AAC9C,WAAO,EAAE;EAAM,OACV;AACE,WAAA;AAAA,EACT;AACF;AClEM,SAAU,sBAEd,EACA,QACA,WACA,oBAKD;AAKO,QAAA,UAAU,UAAU;AAE1B,QAAM,8BAAsE,CAAA;AAE5E,QAAM,oBAAoB,MAAK;AAC7B,UAAM,yBAAyB,gBAAgB,CAAC,QAAQ,YAAW;AACjE,kCAA4B,KAAK,EAAE,QAAQ,QAAS,CAAA;AAAA,IAAA,CACrD;AAEG,QAAA;AACK,aAAA,aAAa,kBAAkB,OAAc;AAAA,IAAA;;IAGtD;AAAA,EAAA;AAGF,QAAM,cAAc;AAEpB,MAAI,8BAA8B;AAC5B,QAAA,YAAY,OAAO,wCAAwC;AAG3D,QAAA,gBAAgB,CAAC,WAA2B;AAChD,UAAM,UAAmB,CAAA;AAClB,WAAA,QAAQ,CAAC,UAAS;AACnB,UAAA,MAAM,YAAY,WAAW,WAAW;AAC1C,gBAAQ,KAAK,GAAG,yBAAyB,KAAK,CAAC;AAAA,MACjD;AAAA,IAAA,CACD;AAEG,QAAA,QAAQ,SAAS,GAAG;AACtB;AACI,UAAA;AACF,qBAAa,aAAa,OAAO;AAAA,MAAA;AAEjC;AAAA,MACF;AAAA,IACF;AAAA,EAAA;AAGF,YAAU,YAAY,aAAa;AAGnC,MAAI,iBAA0B,CAAA;AAC9B,QAAM,mBAAmB,UAAU,aAAa,CAAC,YAAW;AAC1D,QAAI,8BAA8B,GAAG;AACnC;AAAA,IACF;AAEe,mBAAA,KAAK,GAAG,OAAO;AAAA,EAAA,CAC/B;AAGK,QAAA,oBAAoB,WAAW,aAAa,MAAK;AACjD,QAAA,eAAe,WAAW,GAAG;AAC/B;AAAA,IACF;AAEA,UAAM,UAAU;AAChB,qBAAiB,CAAA;AAEjB,WAAO,SAAS,MAAK;AACX,cAAA,QAAQ,CAAC,UAAS;AACxB,0CAAkC,OAAO,SAAS;AAAA,MAAA,CACnD;AAAA,OACA,SAAS;AAAA,EAAA,CACb;AAGD,SAAO,SAAS,MAAK;AAInB,QAAI,mBAAmB;AAEvB,gCAA4B,QAAQ,CAAC,EAAE,QAAQ,cAAa;AAC1D,UAAI,CAAC,kBAAkB;AACrB,YAAI,WAAW,aAAa;AAC1B;AAAA,QACF;AACmB,2BAAA;AAAA,MACrB;AAEM,YAAA,oBAAoB,qBAAqB,aAAa,MAAM;AAElE,UAAI,sBAAsB,QAAW;AAC3B,gBAAA,QAAQ,CAAC,UAAS;AAEtB,4CAAA;AAAA,YACE,GAAG;AAAA,YACH,MAAM,CAAC,GAAG,mBAAmB,GAAG,MAAM,IAAI;AAAA,aAE5C,SAAS;AAAA,QAAA,CAEZ;AAAA,MACH;AAAA,IAAA,CACD;AAAA,KACA,SAAS;AAEL,SAAA;AAAA,IACL;AAAA,IACA,SAAS,MAAK;;;AAGZ,gBAAU,cAAc,aAAa;AAAA,IACvC;AAAA,IACA;AAAA,EAAA;AAEJ;"}
|
|
276
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"mobx-keystone-yjs.esm.mjs","sources":["../src/utils/error.ts","../src/binding/convertJsonToYjsData.ts","../src/binding/applyMobxKeystonePatchToYjsObject.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 * as Y from \"yjs\"\nimport { JsonValue, JsonArray, JsonObject, JsonPrimitive } from \"../jsonTypes\"\n\nfunction isJsonPrimitive(v: JsonValue): v is JsonPrimitive {\n  const t = typeof v\n  return t === \"string\" || t === \"number\" || t === \"boolean\" || v === null\n}\n\nfunction isJsonArray(v: JsonValue): v is JsonArray {\n  return Array.isArray(v)\n}\n\nfunction isJsonObject(v: JsonValue): v is JsonObject {\n  return !isJsonArray(v) && typeof v === \"object\"\n}\n\nexport function convertJsonToYjsData(v: JsonValue) {\n  if (v === undefined || isJsonPrimitive(v)) {\n    return v\n  }\n\n  if (isJsonArray(v)) {\n    const arr = new Y.Array()\n    applyJsonArrayYArray(arr, v)\n    return arr\n  }\n\n  if (isJsonObject(v)) {\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\nexport function applyJsonArrayYArray(dest: Y.Array<unknown>, source: JsonArray) {\n  dest.push(source.map(convertJsonToYjsData))\n}\n\nexport function applyJsonObjectToYMap(dest: Y.Map<unknown>, source: JsonObject) {\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\"\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))\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))\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 {\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))\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            if (yjs.length > patch.value) {\n              const toDelete = yjs.length - patch.value\n              yjs.delete(patch.value, toDelete)\n            } else if (yjs.length < patch.value) {\n              const toInsert = patch.value - yjs.length\n              yjs.insert(yjs.length, Array(toInsert).fill(undefined))\n            }\n          } else {\n            yjs.delete(Number(key))\n            yjs.insert(Number(key), [convertJsonToYjsData(patch.value)])\n          }\n          break\n        }\n        case \"add\": {\n          yjs.insert(Number(key), [convertJsonToYjsData(patch.value)])\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 {\n      throw failure(\n        `invalid patch path, the Yjs object is of an unkown type, so key \"${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 { Patch } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { JsonArray, JsonObject, JsonValue } from \"../jsonTypes\"\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 as Y.Map<any>\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  }\n\n  return patches\n}\n\nfunction toPlainValue(v: Y.Map<any> | Y.Array<any> | JsonValue) {\n  if (v instanceof Y.Map || v instanceof Y.Array) {\n    return v.toJSON() as JsonObject | JsonArray\n  } else {\n    return v\n  }\n}\n","import {\r\n  AnyDataModel,\r\n  AnyModel,\r\n  AnyStandardType,\r\n  ModelClass,\r\n  Patch,\r\n  TypeToData,\r\n  applyPatches,\r\n  fromSnapshot,\r\n  getParentToChildPath,\r\n  onGlobalPatches,\r\n  onPatches,\r\n  onSnapshot,\r\n} from \"mobx-keystone\"\r\nimport * as Y from \"yjs\"\r\nimport { applyMobxKeystonePatchToYjsObject } from \"./applyMobxKeystonePatchToYjsObject\"\r\nimport { convertYjsEventToPatches } from \"./convertYjsEventToPatches\"\r\n\r\nexport function bindYjsToMobxKeystone<\r\n  TType extends AnyStandardType | ModelClass<AnyModel> | ModelClass<AnyDataModel>,\r\n>({\r\n  yjsDoc,\r\n  yjsObject,\r\n  mobxKeystoneType,\r\n}: {\r\n  yjsDoc: Y.Doc\r\n  yjsObject: Y.Map<unknown> | Y.Array<unknown>\r\n  mobxKeystoneType: TType\r\n}): {\r\n  boundObject: TypeToData<TType>\r\n  dispose(): void\r\n  yjsOrigin: symbol\r\n} {\r\n  const yjsJson = yjsObject.toJSON()\r\n\r\n  const initializationGlobalPatches: { target: object; patches: Patch[] }[] = []\r\n\r\n  const createBoundObject = () => {\r\n    const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {\r\n      initializationGlobalPatches.push({ target, patches })\r\n    })\r\n\r\n    try {\r\n      return fromSnapshot(mobxKeystoneType, yjsJson as any)\r\n    } finally {\r\n      disposeOnGlobalPatches()\r\n    }\r\n  }\r\n\r\n  const boundObject = createBoundObject()\r\n\r\n  let applyingMobxKeystoneChanges = 0\r\n  const yjsOrigin = Symbol(\"bindYjsToMobxKeystoneTransactionOrigin\")\r\n\r\n  // bind any changes from yjs to mobx-keystone\r\n  const observeDeepCb = (events: Y.YEvent<any>[]) => {\r\n    const patches: Patch[] = []\r\n    events.forEach((event) => {\r\n      if (event.transaction.origin !== yjsOrigin) {\r\n        patches.push(...convertYjsEventToPatches(event))\r\n      }\r\n    })\r\n\r\n    if (patches.length > 0) {\r\n      applyingMobxKeystoneChanges++\r\n      try {\r\n        applyPatches(boundObject, patches)\r\n      } finally {\r\n        applyingMobxKeystoneChanges--\r\n      }\r\n    }\r\n  }\r\n\r\n  yjsObject.observeDeep(observeDeepCb)\r\n\r\n  // bind any changes from mobx-keystone to yjs\r\n  let pendingPatches: Patch[] = []\r\n  const disposeOnPatches = onPatches(boundObject, (patches) => {\r\n    if (applyingMobxKeystoneChanges > 0) {\r\n      return\r\n    }\r\n\r\n    pendingPatches.push(...patches)\r\n  })\r\n\r\n  // this is only used so we can transact all patches to the snapshot boundary\r\n  const disposeOnSnapshot = onSnapshot(boundObject, () => {\r\n    if (pendingPatches.length === 0) {\r\n      return\r\n    }\r\n\r\n    const patches = pendingPatches\r\n    pendingPatches = []\r\n\r\n    yjsDoc.transact(() => {\r\n      patches.forEach((patch) => {\r\n        applyMobxKeystonePatchToYjsObject(patch, yjsObject)\r\n      })\r\n    }, yjsOrigin)\r\n  })\r\n\r\n  // sync initial patches, that might include setting defaults, IDs, etc\r\n  yjsDoc.transact(() => {\r\n    // we need to skip initializations until we hit the initialization of the bound object\r\n    // this is because default objects might be created and initialized before the main object\r\n    // but we just need to catch when those are actually assigned to the bound object\r\n    let boundObjectFound = false\r\n\r\n    initializationGlobalPatches.forEach(({ target, patches }) => {\r\n      if (!boundObjectFound) {\r\n        if (target !== boundObject) {\r\n          return // skip\r\n        }\r\n        boundObjectFound = true\r\n      }\r\n\r\n      const parentToChildPath = getParentToChildPath(boundObject, target)\r\n      // this is undefined only if target is not a child of boundModel\r\n      if (parentToChildPath !== undefined) {\r\n        patches.forEach((patch) => {\r\n          applyMobxKeystonePatchToYjsObject(\r\n            {\r\n              ...patch,\r\n              path: [...parentToChildPath, ...patch.path],\r\n            },\r\n            yjsObject\r\n          )\r\n        })\r\n      }\r\n    })\r\n  }, yjsOrigin)\r\n\r\n  return {\r\n    boundObject,\r\n    dispose: () => {\r\n      disposeOnPatches()\r\n      disposeOnSnapshot()\r\n      yjsObject.unobserveDeep(observeDeepCb)\r\n    },\r\n    yjsOrigin,\r\n  }\r\n}\r\n"],"names":[],"mappings":"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"}
|
|
@@ -9,37 +9,38 @@ class MobxKeystoneYjsError extends Error {
|
|
|
9
9
|
function failure(msg) {
|
|
10
10
|
return new MobxKeystoneYjsError(msg);
|
|
11
11
|
}
|
|
12
|
-
function
|
|
12
|
+
function isJsonPrimitive(v) {
|
|
13
13
|
const t = typeof v;
|
|
14
14
|
return t === "string" || t === "number" || t === "boolean" || v === null;
|
|
15
15
|
}
|
|
16
|
-
function
|
|
16
|
+
function isJsonArray(v) {
|
|
17
17
|
return Array.isArray(v);
|
|
18
18
|
}
|
|
19
|
-
function
|
|
20
|
-
return !
|
|
19
|
+
function isJsonObject(v) {
|
|
20
|
+
return !isJsonArray(v) && typeof v === "object";
|
|
21
21
|
}
|
|
22
|
-
function
|
|
23
|
-
if (
|
|
22
|
+
function convertJsonToYjsData(v) {
|
|
23
|
+
if (v === void 0 || isJsonPrimitive(v)) {
|
|
24
24
|
return v;
|
|
25
|
-
}
|
|
25
|
+
}
|
|
26
|
+
if (isJsonArray(v)) {
|
|
26
27
|
const arr = new Y.Array();
|
|
27
|
-
|
|
28
|
+
applyJsonArrayYArray(arr, v);
|
|
28
29
|
return arr;
|
|
29
|
-
}
|
|
30
|
+
}
|
|
31
|
+
if (isJsonObject(v)) {
|
|
30
32
|
const map = new Y.Map();
|
|
31
|
-
|
|
33
|
+
applyJsonObjectToYMap(map, v);
|
|
32
34
|
return map;
|
|
33
|
-
} else {
|
|
34
|
-
return void 0;
|
|
35
35
|
}
|
|
36
|
+
throw new Error(`unsupported value type: ${v}`);
|
|
36
37
|
}
|
|
37
|
-
function
|
|
38
|
-
dest.push(source.map(
|
|
38
|
+
function applyJsonArrayYArray(dest, source) {
|
|
39
|
+
dest.push(source.map(convertJsonToYjsData));
|
|
39
40
|
}
|
|
40
|
-
function
|
|
41
|
+
function applyJsonObjectToYMap(dest, source) {
|
|
41
42
|
Object.entries(source).forEach(([k, v]) => {
|
|
42
|
-
dest.set(k,
|
|
43
|
+
dest.set(k, convertJsonToYjsData(v));
|
|
43
44
|
});
|
|
44
45
|
}
|
|
45
46
|
function applyMobxKeystonePatchToYjsObject(patch, yjs) {
|
|
@@ -66,7 +67,7 @@ function applyMobxKeystonePatchToYjsObject(patch, yjs) {
|
|
|
66
67
|
switch (patch.op) {
|
|
67
68
|
case "add":
|
|
68
69
|
case "replace": {
|
|
69
|
-
yjs.set(key,
|
|
70
|
+
yjs.set(key, convertJsonToYjsData(patch.value));
|
|
70
71
|
break;
|
|
71
72
|
}
|
|
72
73
|
case "remove": {
|
|
@@ -91,12 +92,12 @@ function applyMobxKeystonePatchToYjsObject(patch, yjs) {
|
|
|
91
92
|
}
|
|
92
93
|
} else {
|
|
93
94
|
yjs.delete(Number(key));
|
|
94
|
-
yjs.insert(Number(key), [
|
|
95
|
+
yjs.insert(Number(key), [convertJsonToYjsData(patch.value)]);
|
|
95
96
|
}
|
|
96
97
|
break;
|
|
97
98
|
}
|
|
98
99
|
case "add": {
|
|
99
|
-
yjs.insert(Number(key), [
|
|
100
|
+
yjs.insert(Number(key), [convertJsonToYjsData(patch.value)]);
|
|
100
101
|
break;
|
|
101
102
|
}
|
|
102
103
|
case "remove": {
|
|
@@ -267,6 +268,9 @@ function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
|
|
|
267
268
|
}
|
|
268
269
|
export {
|
|
269
270
|
MobxKeystoneYjsError,
|
|
270
|
-
|
|
271
|
+
applyJsonArrayYArray,
|
|
272
|
+
applyJsonObjectToYMap,
|
|
273
|
+
bindYjsToMobxKeystone,
|
|
274
|
+
convertJsonToYjsData
|
|
271
275
|
};
|
|
272
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"mobx-keystone-yjs.esm.mjs","sources":["../src/utils/error.ts","../src/binding/toYDataType.ts","../src/binding/applyMobxKeystonePatchToYjsObject.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 * as Y from \"yjs\"\r\nimport { JSONValue, JSONArray, JSONObject, JSONPrimitive } from \"../jsonTypes\"\r\n\r\nfunction isJSONPrimitive(v: JSONValue): v is JSONPrimitive {\r\n  const t = typeof v\r\n  return t === \"string\" || t === \"number\" || t === \"boolean\" || v === null\r\n}\r\n\r\nfunction isJSONArray(v: JSONValue): v is JSONArray {\r\n  return Array.isArray(v)\r\n}\r\n\r\nfunction isJSONObject(v: JSONValue): v is JSONObject {\r\n  return !isJSONArray(v) && typeof v === \"object\"\r\n}\r\n\r\nexport function toYDataType(v: JSONValue) {\r\n  if (isJSONPrimitive(v)) {\r\n    return v\r\n  } else if (isJSONArray(v)) {\r\n    const arr = new Y.Array()\r\n    applyJsonArray(arr, v)\r\n    return arr\r\n  } else if (isJSONObject(v)) {\r\n    const map = new Y.Map()\r\n    applyJsonObject(map, v)\r\n    return map\r\n  } else {\r\n    return undefined\r\n  }\r\n}\r\n\r\nfunction applyJsonArray(dest: Y.Array<unknown>, source: JSONArray) {\r\n  dest.push(source.map(toYDataType))\r\n}\r\n\r\nfunction applyJsonObject(dest: Y.Map<unknown>, source: JSONObject) {\r\n  Object.entries(source).forEach(([k, v]) => {\r\n    dest.set(k, toYDataType(v))\r\n  })\r\n}\r\n","import { Patch } from \"mobx-keystone\"\r\nimport { failure } from \"../utils/error\"\r\nimport * as Y from \"yjs\"\r\nimport { toYDataType } from \"./toYDataType\"\r\n\r\nexport function applyMobxKeystonePatchToYjsObject(patch: Patch, yjs: unknown): void {\r\n  if (patch.path.length > 1) {\r\n    const [key, ...rest] = patch.path\r\n\r\n    if (yjs instanceof Y.Map) {\r\n      const child = yjs.get(String(key))\r\n      if (child === undefined) {\r\n        throw failure(\r\n          `invalid patch path, key \"${key}\" not found in Yjs map - patch: ${JSON.stringify(patch)}`\r\n        )\r\n      }\r\n      applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)\r\n    } else if (yjs instanceof Y.Array) {\r\n      const child = yjs.get(Number(key))\r\n      if (child === undefined) {\r\n        throw failure(\r\n          `invalid patch path, key \"${key}\" not found in Yjs array - patch: ${JSON.stringify(\r\n            patch\r\n          )}`\r\n        )\r\n      }\r\n      applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)\r\n    } else {\r\n      throw failure(\r\n        `invalid patch path, key \"${key}\" not found in unknown Yjs object - patch: ${JSON.stringify(\r\n          patch\r\n        )}`\r\n      )\r\n    }\r\n  } else if (patch.path.length === 1) {\r\n    if (yjs instanceof Y.Map) {\r\n      const key = String(patch.path[0])\r\n\r\n      switch (patch.op) {\r\n        case \"add\":\r\n        case \"replace\": {\r\n          yjs.set(key, toYDataType(patch.value))\r\n          break\r\n        }\r\n        case \"remove\": {\r\n          yjs.delete(key)\r\n          break\r\n        }\r\n        default: {\r\n          throw failure(`invalid patch operation for map`)\r\n        }\r\n      }\r\n    } else if (yjs instanceof Y.Array) {\r\n      const key = patch.path[0]\r\n\r\n      switch (patch.op) {\r\n        case \"replace\": {\r\n          if (key === \"length\") {\r\n            if (yjs.length > patch.value) {\r\n              const toDelete = yjs.length - patch.value\r\n              yjs.delete(patch.value, toDelete)\r\n            } else if (yjs.length < patch.value) {\r\n              const toInsert = patch.value - yjs.length\r\n              yjs.insert(yjs.length, Array(toInsert).fill(undefined))\r\n            }\r\n          } else {\r\n            yjs.delete(Number(key))\r\n            yjs.insert(Number(key), [toYDataType(patch.value)])\r\n          }\r\n          break\r\n        }\r\n        case \"add\": {\r\n          yjs.insert(Number(key), [toYDataType(patch.value)])\r\n          break\r\n        }\r\n        case \"remove\": {\r\n          yjs.delete(Number(key))\r\n          break\r\n        }\r\n        default: {\r\n          throw failure(`invalid patch operation for array`)\r\n        }\r\n      }\r\n    } else {\r\n      throw failure(\r\n        `invalid patch path, the Yjs object is of an unkown type, so key \"${patch.path[0]}\" cannot be found in it`\r\n      )\r\n    }\r\n  } else {\r\n    throw failure(`invalid patch path, it cannot be empty`)\r\n  }\r\n}\r\n","import { Patch } from \"mobx-keystone\"\r\nimport * as Y from \"yjs\"\r\nimport { JSONArray, JSONObject, JSONValue } from \"../jsonTypes\"\r\nimport { failure } from \"../utils/error\"\r\n\r\nexport function convertYjsEventToPatches(event: Y.YEvent<any>): Patch[] {\r\n  const patches: Patch[] = []\r\n\r\n  if (event instanceof Y.YMapEvent) {\r\n    const source = event.target as Y.Map<any>\r\n\r\n    event.changes.keys.forEach((change, key) => {\r\n      const path = [...event.path, key]\r\n\r\n      switch (change.action) {\r\n        case \"add\":\r\n          patches.push({\r\n            op: \"add\",\r\n            path,\r\n            value: toPlainValue(source.get(key)),\r\n          })\r\n          break\r\n\r\n        case \"update\":\r\n          patches.push({\r\n            op: \"replace\",\r\n            path,\r\n            value: toPlainValue(source.get(key)),\r\n          })\r\n          break\r\n\r\n        case \"delete\":\r\n          patches.push({\r\n            op: \"remove\",\r\n            path,\r\n          })\r\n          break\r\n\r\n        default:\r\n          throw failure(`unsupported Yjs map event action: ${change.action}`)\r\n      }\r\n    })\r\n  } else if (event instanceof Y.YArrayEvent) {\r\n    let retain = 0\r\n    event.changes.delta.forEach((change) => {\r\n      if (change.retain) {\r\n        retain += change.retain\r\n      }\r\n\r\n      if (change.delete) {\r\n        // remove X items at retain position\r\n        const path = [...event.path, retain]\r\n        for (let i = 0; i < change.delete; i++) {\r\n          patches.push({\r\n            op: \"remove\",\r\n            path,\r\n          })\r\n        }\r\n      }\r\n\r\n      if (change.insert) {\r\n        const newValues = Array.isArray(change.insert) ? change.insert : [change.insert]\r\n        newValues.forEach((v) => {\r\n          const path = [...event.path, retain]\r\n          patches.push({\r\n            op: \"add\",\r\n            path,\r\n            value: toPlainValue(v),\r\n          })\r\n          retain++\r\n        })\r\n      }\r\n    })\r\n  }\r\n\r\n  return patches\r\n}\r\n\r\nfunction toPlainValue(v: Y.Map<any> | Y.Array<any> | JSONValue) {\r\n  if (v instanceof Y.Map || v instanceof Y.Array) {\r\n    return v.toJSON() as JSONObject | JSONArray\r\n  } else {\r\n    return v\r\n  }\r\n}\r\n","import {\r\n  AnyDataModel,\r\n  AnyModel,\r\n  AnyStandardType,\r\n  ModelClass,\r\n  Patch,\r\n  TypeToData,\r\n  applyPatches,\r\n  fromSnapshot,\r\n  getParentToChildPath,\r\n  onGlobalPatches,\r\n  onPatches,\r\n  onSnapshot,\r\n} from \"mobx-keystone\"\r\nimport * as Y from \"yjs\"\r\nimport { applyMobxKeystonePatchToYjsObject } from \"./applyMobxKeystonePatchToYjsObject\"\r\nimport { convertYjsEventToPatches } from \"./convertYjsEventToPatches\"\r\n\r\nexport function bindYjsToMobxKeystone<\r\n  TType extends AnyStandardType | ModelClass<AnyModel> | ModelClass<AnyDataModel>,\r\n>({\r\n  yjsDoc,\r\n  yjsObject,\r\n  mobxKeystoneType,\r\n}: {\r\n  yjsDoc: Y.Doc\r\n  yjsObject: Y.Map<unknown> | Y.Array<unknown>\r\n  mobxKeystoneType: TType\r\n}): {\r\n  boundObject: TypeToData<TType>\r\n  dispose(): void\r\n  yjsOrigin: symbol\r\n} {\r\n  const yjsJson = yjsObject.toJSON()\r\n\r\n  const initializationGlobalPatches: { target: object; patches: Patch[] }[] = []\r\n\r\n  const createBoundObject = () => {\r\n    const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {\r\n      initializationGlobalPatches.push({ target, patches })\r\n    })\r\n\r\n    try {\r\n      return fromSnapshot(mobxKeystoneType, yjsJson as any)\r\n    } finally {\r\n      disposeOnGlobalPatches()\r\n    }\r\n  }\r\n\r\n  const boundObject = createBoundObject()\r\n\r\n  let applyingMobxKeystoneChanges = 0\r\n  const yjsOrigin = Symbol(\"bindYjsToMobxKeystoneTransactionOrigin\")\r\n\r\n  // bind any changes from yjs to mobx-keystone\r\n  const observeDeepCb = (events: Y.YEvent<any>[]) => {\r\n    const patches: Patch[] = []\r\n    events.forEach((event) => {\r\n      if (event.transaction.origin !== yjsOrigin) {\r\n        patches.push(...convertYjsEventToPatches(event))\r\n      }\r\n    })\r\n\r\n    if (patches.length > 0) {\r\n      applyingMobxKeystoneChanges++\r\n      try {\r\n        applyPatches(boundObject, patches)\r\n      } finally {\r\n        applyingMobxKeystoneChanges--\r\n      }\r\n    }\r\n  }\r\n\r\n  yjsObject.observeDeep(observeDeepCb)\r\n\r\n  // bind any changes from mobx-keystone to yjs\r\n  let pendingPatches: Patch[] = []\r\n  const disposeOnPatches = onPatches(boundObject, (patches) => {\r\n    if (applyingMobxKeystoneChanges > 0) {\r\n      return\r\n    }\r\n\r\n    pendingPatches.push(...patches)\r\n  })\r\n\r\n  // this is only used so we can transact all patches to the snapshot boundary\r\n  const disposeOnSnapshot = onSnapshot(boundObject, () => {\r\n    if (pendingPatches.length === 0) {\r\n      return\r\n    }\r\n\r\n    const patches = pendingPatches\r\n    pendingPatches = []\r\n\r\n    yjsDoc.transact(() => {\r\n      patches.forEach((patch) => {\r\n        applyMobxKeystonePatchToYjsObject(patch, yjsObject)\r\n      })\r\n    }, yjsOrigin)\r\n  })\r\n\r\n  // sync initial patches, that might include setting defaults, IDs, etc\r\n  yjsDoc.transact(() => {\r\n    // we need to skip initializations until we hit the initialization of the bound object\r\n    // this is because default objects might be created and initialized before the main object\r\n    // but we just need to catch when those are actually assigned to the bound object\r\n    let boundObjectFound = false\r\n\r\n    initializationGlobalPatches.forEach(({ target, patches }) => {\r\n      if (!boundObjectFound) {\r\n        if (target !== boundObject) {\r\n          return // skip\r\n        }\r\n        boundObjectFound = true\r\n      }\r\n\r\n      const parentToChildPath = getParentToChildPath(boundObject, target)\r\n      // this is undefined only if target is not a child of boundModel\r\n      if (parentToChildPath !== undefined) {\r\n        patches.forEach((patch) => {\r\n          applyMobxKeystonePatchToYjsObject(\r\n            {\r\n              ...patch,\r\n              path: [...parentToChildPath, ...patch.path],\r\n            },\r\n            yjsObject\r\n          )\r\n        })\r\n      }\r\n    })\r\n  }, yjsOrigin)\r\n\r\n  return {\r\n    boundObject,\r\n    dispose: () => {\r\n      disposeOnPatches()\r\n      disposeOnSnapshot()\r\n      yjsObject.unobserveDeep(observeDeepCb)\r\n    },\r\n    yjsOrigin,\r\n  }\r\n}\r\n"],"names":[],"mappings":";;AAGM,MAAO,6BAA6B,MAAK;AAAA,EAC7C,YAAY,KAAW;AACrB,UAAM,GAAG;AAGF,WAAA,eAAe,MAAM,qBAAqB,SAAS;AAAA,EAC5D;AACD;AAKK,SAAU,QAAQ,KAAW;AAC1B,SAAA,IAAI,qBAAqB,GAAG;AACrC;ACdA,SAAS,gBAAgB,GAAY;AACnC,QAAM,IAAI,OAAO;AACjB,SAAO,MAAM,YAAY,MAAM,YAAY,MAAM,aAAa,MAAM;AACtE;AAEA,SAAS,YAAY,GAAY;AACxB,SAAA,MAAM,QAAQ,CAAC;AACxB;AAEA,SAAS,aAAa,GAAY;AAChC,SAAO,CAAC,YAAY,CAAC,KAAK,OAAO,MAAM;AACzC;AAEM,SAAU,YAAY,GAAY;AAClC,MAAA,gBAAgB,CAAC,GAAG;AACf,WAAA;AAAA,EAAA,WACE,YAAY,CAAC,GAAG;AACnB,UAAA,MAAM,IAAI,EAAE;AAClB,mBAAe,KAAK,CAAC;AACd,WAAA;AAAA,EAAA,WACE,aAAa,CAAC,GAAG;AACpB,UAAA,MAAM,IAAI,EAAE;AAClB,oBAAgB,KAAK,CAAC;AACf,WAAA;AAAA,EAAA,OACF;AACE,WAAA;AAAA,EACT;AACF;AAEA,SAAS,eAAe,MAAwB,QAAiB;AAC/D,OAAK,KAAK,OAAO,IAAI,WAAW,CAAC;AACnC;AAEA,SAAS,gBAAgB,MAAsB,QAAkB;AACxD,SAAA,QAAQ,MAAM,EAAE,QAAQ,CAAC,CAAC,GAAG,CAAC,MAAK;AACxC,SAAK,IAAI,GAAG,YAAY,CAAC,CAAC;AAAA,EAAA,CAC3B;AACH;ACnCgB,SAAA,kCAAkC,OAAc,KAAY;AACtE,MAAA,MAAM,KAAK,SAAS,GAAG;AACzB,UAAM,CAAC,KAAK,GAAG,IAAI,IAAI,MAAM;AAEzB,QAAA,eAAe,EAAE,KAAK;AACxB,YAAM,QAAQ,IAAI,IAAI,OAAO,GAAG,CAAC;AACjC,UAAI,UAAU,QAAW;AACjB,cAAA,QACJ,4BAA4B,GAAG,mCAAmC,KAAK,UAAU,KAAK,CAAC,EAAE;AAAA,MAE7F;AACA,wCAAkC,EAAE,GAAG,OAAO,MAAM,KAAA,GAAQ,KAAK;AAAA,IAAA,WACxD,eAAe,EAAE,OAAO;AACjC,YAAM,QAAQ,IAAI,IAAI,OAAO,GAAG,CAAC;AACjC,UAAI,UAAU,QAAW;AACjB,cAAA,QACJ,4BAA4B,GAAG,qCAAqC,KAAK,UACvE,KAAK,CACN,EAAE;AAAA,MAEP;AACA,wCAAkC,EAAE,GAAG,OAAO,MAAM,KAAA,GAAQ,KAAK;AAAA,IAAA,OAC5D;AACC,YAAA,QACJ,4BAA4B,GAAG,8CAA8C,KAAK,UAChF,KAAK,CACN,EAAE;AAAA,IAEP;AAAA,EACS,WAAA,MAAM,KAAK,WAAW,GAAG;AAC9B,QAAA,eAAe,EAAE,KAAK;AACxB,YAAM,MAAM,OAAO,MAAM,KAAK,CAAC,CAAC;AAEhC,cAAQ,MAAM,IAAI;AAAA,QAChB,KAAK;AAAA,QACL,KAAK,WAAW;AACd,cAAI,IAAI,KAAK,YAAY,MAAM,KAAK,CAAC;AACrC;AAAA,QACF;AAAA,QACA,KAAK,UAAU;AACb,cAAI,OAAO,GAAG;AACd;AAAA,QACF;AAAA,QACA,SAAS;AACP,gBAAM,QAAQ,iCAAiC;AAAA,QACjD;AAAA,MACF;AAAA,IAAA,WACS,eAAe,EAAE,OAAO;AAC3B,YAAA,MAAM,MAAM,KAAK,CAAC;AAExB,cAAQ,MAAM,IAAI;AAAA,QAChB,KAAK,WAAW;AACd,cAAI,QAAQ,UAAU;AAChB,gBAAA,IAAI,SAAS,MAAM,OAAO;AACtB,oBAAA,WAAW,IAAI,SAAS,MAAM;AAChC,kBAAA,OAAO,MAAM,OAAO,QAAQ;AAAA,YACvB,WAAA,IAAI,SAAS,MAAM,OAAO;AAC7B,oBAAA,WAAW,MAAM,QAAQ,IAAI;AAC/B,kBAAA,OAAO,IAAI,QAAQ,MAAM,QAAQ,EAAE,KAAK,MAAS,CAAC;AAAA,YACxD;AAAA,UAAA,OACK;AACD,gBAAA,OAAO,OAAO,GAAG,CAAC;AAClB,gBAAA,OAAO,OAAO,GAAG,GAAG,CAAC,YAAY,MAAM,KAAK,CAAC,CAAC;AAAA,UACpD;AACA;AAAA,QACF;AAAA,QACA,KAAK,OAAO;AACN,cAAA,OAAO,OAAO,GAAG,GAAG,CAAC,YAAY,MAAM,KAAK,CAAC,CAAC;AAClD;AAAA,QACF;AAAA,QACA,KAAK,UAAU;AACT,cAAA,OAAO,OAAO,GAAG,CAAC;AACtB;AAAA,QACF;AAAA,QACA,SAAS;AACP,gBAAM,QAAQ,mCAAmC;AAAA,QACnD;AAAA,MACF;AAAA,IAAA,OACK;AACL,YAAM,QACJ,oEAAoE,MAAM,KAAK,CAAC,CAAC,yBAAyB;AAAA,IAE9G;AAAA,EAAA,OACK;AACL,UAAM,QAAQ,wCAAwC;AAAA,EACxD;AACF;ACtFM,SAAU,yBAAyB,OAAoB;AAC3D,QAAM,UAAmB,CAAA;AAErB,MAAA,iBAAiB,EAAE,WAAW;AAChC,UAAM,SAAS,MAAM;AAErB,UAAM,QAAQ,KAAK,QAAQ,CAAC,QAAQ,QAAO;AACzC,YAAM,OAAO,CAAC,GAAG,MAAM,MAAM,GAAG;AAEhC,cAAQ,OAAO,QAAQ;AAAA,QACrB,KAAK;AACH,kBAAQ,KAAK;AAAA,YACX,IAAI;AAAA,YACJ;AAAA,YACA,OAAO,aAAa,OAAO,IAAI,GAAG,CAAC;AAAA,UAAA,CACpC;AACD;AAAA,QAEF,KAAK;AACH,kBAAQ,KAAK;AAAA,YACX,IAAI;AAAA,YACJ;AAAA,YACA,OAAO,aAAa,OAAO,IAAI,GAAG,CAAC;AAAA,UAAA,CACpC;AACD;AAAA,QAEF,KAAK;AACH,kBAAQ,KAAK;AAAA,YACX,IAAI;AAAA,YACJ;AAAA,UAAA,CACD;AACD;AAAA,QAEF;AACE,gBAAM,QAAQ,qCAAqC,OAAO,MAAM,EAAE;AAAA,MACtE;AAAA,IAAA,CACD;AAAA,EAAA,WACQ,iBAAiB,EAAE,aAAa;AACzC,QAAI,SAAS;AACb,UAAM,QAAQ,MAAM,QAAQ,CAAC,WAAU;AACrC,UAAI,OAAO,QAAQ;AACjB,kBAAU,OAAO;AAAA,MACnB;AAEA,UAAI,OAAO,QAAQ;AAEjB,cAAM,OAAO,CAAC,GAAG,MAAM,MAAM,MAAM;AACnC,iBAAS,IAAI,GAAG,IAAI,OAAO,QAAQ,KAAK;AACtC,kBAAQ,KAAK;AAAA,YACX,IAAI;AAAA,YACJ;AAAA,UAAA,CACD;AAAA,QACH;AAAA,MACF;AAEA,UAAI,OAAO,QAAQ;AACX,cAAA,YAAY,MAAM,QAAQ,OAAO,MAAM,IAAI,OAAO,SAAS,CAAC,OAAO,MAAM;AACrE,kBAAA,QAAQ,CAAC,MAAK;AACtB,gBAAM,OAAO,CAAC,GAAG,MAAM,MAAM,MAAM;AACnC,kBAAQ,KAAK;AAAA,YACX,IAAI;AAAA,YACJ;AAAA,YACA,OAAO,aAAa,CAAC;AAAA,UAAA,CACtB;AACD;AAAA,QAAA,CACD;AAAA,MACH;AAAA,IAAA,CACD;AAAA,EACH;AAEO,SAAA;AACT;AAEA,SAAS,aAAa,GAAwC;AAC5D,MAAI,aAAa,EAAE,OAAO,aAAa,EAAE,OAAO;AAC9C,WAAO,EAAE;EAAM,OACV;AACE,WAAA;AAAA,EACT;AACF;AClEM,SAAU,sBAEd,EACA,QACA,WACA,oBAKD;AAKO,QAAA,UAAU,UAAU;AAE1B,QAAM,8BAAsE,CAAA;AAE5E,QAAM,oBAAoB,MAAK;AAC7B,UAAM,yBAAyB,gBAAgB,CAAC,QAAQ,YAAW;AACjE,kCAA4B,KAAK,EAAE,QAAQ,QAAS,CAAA;AAAA,IAAA,CACrD;AAEG,QAAA;AACK,aAAA,aAAa,kBAAkB,OAAc;AAAA,IAAA;;IAGtD;AAAA,EAAA;AAGF,QAAM,cAAc;AAEpB,MAAI,8BAA8B;AAC5B,QAAA,YAAY,OAAO,wCAAwC;AAG3D,QAAA,gBAAgB,CAAC,WAA2B;AAChD,UAAM,UAAmB,CAAA;AAClB,WAAA,QAAQ,CAAC,UAAS;AACnB,UAAA,MAAM,YAAY,WAAW,WAAW;AAC1C,gBAAQ,KAAK,GAAG,yBAAyB,KAAK,CAAC;AAAA,MACjD;AAAA,IAAA,CACD;AAEG,QAAA,QAAQ,SAAS,GAAG;AACtB;AACI,UAAA;AACF,qBAAa,aAAa,OAAO;AAAA,MAAA;AAEjC;AAAA,MACF;AAAA,IACF;AAAA,EAAA;AAGF,YAAU,YAAY,aAAa;AAGnC,MAAI,iBAA0B,CAAA;AAC9B,QAAM,mBAAmB,UAAU,aAAa,CAAC,YAAW;AAC1D,QAAI,8BAA8B,GAAG;AACnC;AAAA,IACF;AAEe,mBAAA,KAAK,GAAG,OAAO;AAAA,EAAA,CAC/B;AAGK,QAAA,oBAAoB,WAAW,aAAa,MAAK;AACjD,QAAA,eAAe,WAAW,GAAG;AAC/B;AAAA,IACF;AAEA,UAAM,UAAU;AAChB,qBAAiB,CAAA;AAEjB,WAAO,SAAS,MAAK;AACX,cAAA,QAAQ,CAAC,UAAS;AACxB,0CAAkC,OAAO,SAAS;AAAA,MAAA,CACnD;AAAA,OACA,SAAS;AAAA,EAAA,CACb;AAGD,SAAO,SAAS,MAAK;AAInB,QAAI,mBAAmB;AAEvB,gCAA4B,QAAQ,CAAC,EAAE,QAAQ,cAAa;AAC1D,UAAI,CAAC,kBAAkB;AACrB,YAAI,WAAW,aAAa;AAC1B;AAAA,QACF;AACmB,2BAAA;AAAA,MACrB;AAEM,YAAA,oBAAoB,qBAAqB,aAAa,MAAM;AAElE,UAAI,sBAAsB,QAAW;AAC3B,gBAAA,QAAQ,CAAC,UAAS;AAEtB,4CAAA;AAAA,YACE,GAAG;AAAA,YACH,MAAM,CAAC,GAAG,mBAAmB,GAAG,MAAM,IAAI;AAAA,aAE5C,SAAS;AAAA,QAAA,CAEZ;AAAA,MACH;AAAA,IAAA,CACD;AAAA,KACA,SAAS;AAEL,SAAA;AAAA,IACL;AAAA,IACA,SAAS,MAAK;;;AAGZ,gBAAU,cAAc,aAAa;AAAA,IACvC;AAAA,IACA;AAAA,EAAA;AAEJ;"}
|
|
276
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"mobx-keystone-yjs.esm.mjs","sources":["../src/utils/error.ts","../src/binding/convertJsonToYjsData.ts","../src/binding/applyMobxKeystonePatchToYjsObject.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 * as Y from \"yjs\"\nimport { JsonValue, JsonArray, JsonObject, JsonPrimitive } from \"../jsonTypes\"\n\nfunction isJsonPrimitive(v: JsonValue): v is JsonPrimitive {\n  const t = typeof v\n  return t === \"string\" || t === \"number\" || t === \"boolean\" || v === null\n}\n\nfunction isJsonArray(v: JsonValue): v is JsonArray {\n  return Array.isArray(v)\n}\n\nfunction isJsonObject(v: JsonValue): v is JsonObject {\n  return !isJsonArray(v) && typeof v === \"object\"\n}\n\nexport function convertJsonToYjsData(v: JsonValue) {\n  if (v === undefined || isJsonPrimitive(v)) {\n    return v\n  }\n\n  if (isJsonArray(v)) {\n    const arr = new Y.Array()\n    applyJsonArrayYArray(arr, v)\n    return arr\n  }\n\n  if (isJsonObject(v)) {\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\nexport function applyJsonArrayYArray(dest: Y.Array<unknown>, source: JsonArray) {\n  dest.push(source.map(convertJsonToYjsData))\n}\n\nexport function applyJsonObjectToYMap(dest: Y.Map<unknown>, source: JsonObject) {\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\"\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))\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))\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 {\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))\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            if (yjs.length > patch.value) {\n              const toDelete = yjs.length - patch.value\n              yjs.delete(patch.value, toDelete)\n            } else if (yjs.length < patch.value) {\n              const toInsert = patch.value - yjs.length\n              yjs.insert(yjs.length, Array(toInsert).fill(undefined))\n            }\n          } else {\n            yjs.delete(Number(key))\n            yjs.insert(Number(key), [convertJsonToYjsData(patch.value)])\n          }\n          break\n        }\n        case \"add\": {\n          yjs.insert(Number(key), [convertJsonToYjsData(patch.value)])\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 {\n      throw failure(\n        `invalid patch path, the Yjs object is of an unkown type, so key \"${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 { Patch } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { JsonArray, JsonObject, JsonValue } from \"../jsonTypes\"\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 as Y.Map<any>\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  }\n\n  return patches\n}\n\nfunction toPlainValue(v: Y.Map<any> | Y.Array<any> | JsonValue) {\n  if (v instanceof Y.Map || v instanceof Y.Array) {\n    return v.toJSON() as JsonObject | JsonArray\n  } else {\n    return v\n  }\n}\n","import {\r\n  AnyDataModel,\r\n  AnyModel,\r\n  AnyStandardType,\r\n  ModelClass,\r\n  Patch,\r\n  TypeToData,\r\n  applyPatches,\r\n  fromSnapshot,\r\n  getParentToChildPath,\r\n  onGlobalPatches,\r\n  onPatches,\r\n  onSnapshot,\r\n} from \"mobx-keystone\"\r\nimport * as Y from \"yjs\"\r\nimport { applyMobxKeystonePatchToYjsObject } from \"./applyMobxKeystonePatchToYjsObject\"\r\nimport { convertYjsEventToPatches } from \"./convertYjsEventToPatches\"\r\n\r\nexport function bindYjsToMobxKeystone<\r\n  TType extends AnyStandardType | ModelClass<AnyModel> | ModelClass<AnyDataModel>,\r\n>({\r\n  yjsDoc,\r\n  yjsObject,\r\n  mobxKeystoneType,\r\n}: {\r\n  yjsDoc: Y.Doc\r\n  yjsObject: Y.Map<unknown> | Y.Array<unknown>\r\n  mobxKeystoneType: TType\r\n}): {\r\n  boundObject: TypeToData<TType>\r\n  dispose(): void\r\n  yjsOrigin: symbol\r\n} {\r\n  const yjsJson = yjsObject.toJSON()\r\n\r\n  const initializationGlobalPatches: { target: object; patches: Patch[] }[] = []\r\n\r\n  const createBoundObject = () => {\r\n    const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {\r\n      initializationGlobalPatches.push({ target, patches })\r\n    })\r\n\r\n    try {\r\n      return fromSnapshot(mobxKeystoneType, yjsJson as any)\r\n    } finally {\r\n      disposeOnGlobalPatches()\r\n    }\r\n  }\r\n\r\n  const boundObject = createBoundObject()\r\n\r\n  let applyingMobxKeystoneChanges = 0\r\n  const yjsOrigin = Symbol(\"bindYjsToMobxKeystoneTransactionOrigin\")\r\n\r\n  // bind any changes from yjs to mobx-keystone\r\n  const observeDeepCb = (events: Y.YEvent<any>[]) => {\r\n    const patches: Patch[] = []\r\n    events.forEach((event) => {\r\n      if (event.transaction.origin !== yjsOrigin) {\r\n        patches.push(...convertYjsEventToPatches(event))\r\n      }\r\n    })\r\n\r\n    if (patches.length > 0) {\r\n      applyingMobxKeystoneChanges++\r\n      try {\r\n        applyPatches(boundObject, patches)\r\n      } finally {\r\n        applyingMobxKeystoneChanges--\r\n      }\r\n    }\r\n  }\r\n\r\n  yjsObject.observeDeep(observeDeepCb)\r\n\r\n  // bind any changes from mobx-keystone to yjs\r\n  let pendingPatches: Patch[] = []\r\n  const disposeOnPatches = onPatches(boundObject, (patches) => {\r\n    if (applyingMobxKeystoneChanges > 0) {\r\n      return\r\n    }\r\n\r\n    pendingPatches.push(...patches)\r\n  })\r\n\r\n  // this is only used so we can transact all patches to the snapshot boundary\r\n  const disposeOnSnapshot = onSnapshot(boundObject, () => {\r\n    if (pendingPatches.length === 0) {\r\n      return\r\n    }\r\n\r\n    const patches = pendingPatches\r\n    pendingPatches = []\r\n\r\n    yjsDoc.transact(() => {\r\n      patches.forEach((patch) => {\r\n        applyMobxKeystonePatchToYjsObject(patch, yjsObject)\r\n      })\r\n    }, yjsOrigin)\r\n  })\r\n\r\n  // sync initial patches, that might include setting defaults, IDs, etc\r\n  yjsDoc.transact(() => {\r\n    // we need to skip initializations until we hit the initialization of the bound object\r\n    // this is because default objects might be created and initialized before the main object\r\n    // but we just need to catch when those are actually assigned to the bound object\r\n    let boundObjectFound = false\r\n\r\n    initializationGlobalPatches.forEach(({ target, patches }) => {\r\n      if (!boundObjectFound) {\r\n        if (target !== boundObject) {\r\n          return // skip\r\n        }\r\n        boundObjectFound = true\r\n      }\r\n\r\n      const parentToChildPath = getParentToChildPath(boundObject, target)\r\n      // this is undefined only if target is not a child of boundModel\r\n      if (parentToChildPath !== undefined) {\r\n        patches.forEach((patch) => {\r\n          applyMobxKeystonePatchToYjsObject(\r\n            {\r\n              ...patch,\r\n              path: [...parentToChildPath, ...patch.path],\r\n            },\r\n            yjsObject\r\n          )\r\n        })\r\n      }\r\n    })\r\n  }, yjsOrigin)\r\n\r\n  return {\r\n    boundObject,\r\n    dispose: () => {\r\n      disposeOnPatches()\r\n      disposeOnSnapshot()\r\n      yjsObject.unobserveDeep(observeDeepCb)\r\n    },\r\n    yjsOrigin,\r\n  }\r\n}\r\n"],"names":[],"mappings":"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"}
|
|
@@ -28,37 +28,38 @@
|
|
|
28
28
|
function failure(msg) {
|
|
29
29
|
return new MobxKeystoneYjsError(msg);
|
|
30
30
|
}
|
|
31
|
-
function
|
|
31
|
+
function isJsonPrimitive(v) {
|
|
32
32
|
const t = typeof v;
|
|
33
33
|
return t === "string" || t === "number" || t === "boolean" || v === null;
|
|
34
34
|
}
|
|
35
|
-
function
|
|
35
|
+
function isJsonArray(v) {
|
|
36
36
|
return Array.isArray(v);
|
|
37
37
|
}
|
|
38
|
-
function
|
|
39
|
-
return !
|
|
38
|
+
function isJsonObject(v) {
|
|
39
|
+
return !isJsonArray(v) && typeof v === "object";
|
|
40
40
|
}
|
|
41
|
-
function
|
|
42
|
-
if (
|
|
41
|
+
function convertJsonToYjsData(v) {
|
|
42
|
+
if (v === void 0 || isJsonPrimitive(v)) {
|
|
43
43
|
return v;
|
|
44
|
-
}
|
|
44
|
+
}
|
|
45
|
+
if (isJsonArray(v)) {
|
|
45
46
|
const arr = new Y__namespace.Array();
|
|
46
|
-
|
|
47
|
+
applyJsonArrayYArray(arr, v);
|
|
47
48
|
return arr;
|
|
48
|
-
}
|
|
49
|
+
}
|
|
50
|
+
if (isJsonObject(v)) {
|
|
49
51
|
const map = new Y__namespace.Map();
|
|
50
|
-
|
|
52
|
+
applyJsonObjectToYMap(map, v);
|
|
51
53
|
return map;
|
|
52
|
-
} else {
|
|
53
|
-
return void 0;
|
|
54
54
|
}
|
|
55
|
+
throw new Error(`unsupported value type: ${v}`);
|
|
55
56
|
}
|
|
56
|
-
function
|
|
57
|
-
dest.push(source.map(
|
|
57
|
+
function applyJsonArrayYArray(dest, source) {
|
|
58
|
+
dest.push(source.map(convertJsonToYjsData));
|
|
58
59
|
}
|
|
59
|
-
function
|
|
60
|
+
function applyJsonObjectToYMap(dest, source) {
|
|
60
61
|
Object.entries(source).forEach(([k, v]) => {
|
|
61
|
-
dest.set(k,
|
|
62
|
+
dest.set(k, convertJsonToYjsData(v));
|
|
62
63
|
});
|
|
63
64
|
}
|
|
64
65
|
function applyMobxKeystonePatchToYjsObject(patch, yjs) {
|
|
@@ -85,7 +86,7 @@
|
|
|
85
86
|
switch (patch.op) {
|
|
86
87
|
case "add":
|
|
87
88
|
case "replace": {
|
|
88
|
-
yjs.set(key,
|
|
89
|
+
yjs.set(key, convertJsonToYjsData(patch.value));
|
|
89
90
|
break;
|
|
90
91
|
}
|
|
91
92
|
case "remove": {
|
|
@@ -110,12 +111,12 @@
|
|
|
110
111
|
}
|
|
111
112
|
} else {
|
|
112
113
|
yjs.delete(Number(key));
|
|
113
|
-
yjs.insert(Number(key), [
|
|
114
|
+
yjs.insert(Number(key), [convertJsonToYjsData(patch.value)]);
|
|
114
115
|
}
|
|
115
116
|
break;
|
|
116
117
|
}
|
|
117
118
|
case "add": {
|
|
118
|
-
yjs.insert(Number(key), [
|
|
119
|
+
yjs.insert(Number(key), [convertJsonToYjsData(patch.value)]);
|
|
119
120
|
break;
|
|
120
121
|
}
|
|
121
122
|
case "remove": {
|
|
@@ -285,7 +286,10 @@
|
|
|
285
286
|
};
|
|
286
287
|
}
|
|
287
288
|
exports2.MobxKeystoneYjsError = MobxKeystoneYjsError;
|
|
289
|
+
exports2.applyJsonArrayYArray = applyJsonArrayYArray;
|
|
290
|
+
exports2.applyJsonObjectToYMap = applyJsonObjectToYMap;
|
|
288
291
|
exports2.bindYjsToMobxKeystone = bindYjsToMobxKeystone;
|
|
292
|
+
exports2.convertJsonToYjsData = convertJsonToYjsData;
|
|
289
293
|
Object.defineProperty(exports2, Symbol.toStringTag, { value: "Module" });
|
|
290
294
|
});
|
|
291
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"mobx-keystone-yjs.umd.js","sources":["../src/utils/error.ts","../src/binding/toYDataType.ts","../src/binding/applyMobxKeystonePatchToYjsObject.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 * as Y from \"yjs\"\r\nimport { JSONValue, JSONArray, JSONObject, JSONPrimitive } from \"../jsonTypes\"\r\n\r\nfunction isJSONPrimitive(v: JSONValue): v is JSONPrimitive {\r\n  const t = typeof v\r\n  return t === \"string\" || t === \"number\" || t === \"boolean\" || v === null\r\n}\r\n\r\nfunction isJSONArray(v: JSONValue): v is JSONArray {\r\n  return Array.isArray(v)\r\n}\r\n\r\nfunction isJSONObject(v: JSONValue): v is JSONObject {\r\n  return !isJSONArray(v) && typeof v === \"object\"\r\n}\r\n\r\nexport function toYDataType(v: JSONValue) {\r\n  if (isJSONPrimitive(v)) {\r\n    return v\r\n  } else if (isJSONArray(v)) {\r\n    const arr = new Y.Array()\r\n    applyJsonArray(arr, v)\r\n    return arr\r\n  } else if (isJSONObject(v)) {\r\n    const map = new Y.Map()\r\n    applyJsonObject(map, v)\r\n    return map\r\n  } else {\r\n    return undefined\r\n  }\r\n}\r\n\r\nfunction applyJsonArray(dest: Y.Array<unknown>, source: JSONArray) {\r\n  dest.push(source.map(toYDataType))\r\n}\r\n\r\nfunction applyJsonObject(dest: Y.Map<unknown>, source: JSONObject) {\r\n  Object.entries(source).forEach(([k, v]) => {\r\n    dest.set(k, toYDataType(v))\r\n  })\r\n}\r\n","import { Patch } from \"mobx-keystone\"\r\nimport { failure } from \"../utils/error\"\r\nimport * as Y from \"yjs\"\r\nimport { toYDataType } from \"./toYDataType\"\r\n\r\nexport function applyMobxKeystonePatchToYjsObject(patch: Patch, yjs: unknown): void {\r\n  if (patch.path.length > 1) {\r\n    const [key, ...rest] = patch.path\r\n\r\n    if (yjs instanceof Y.Map) {\r\n      const child = yjs.get(String(key))\r\n      if (child === undefined) {\r\n        throw failure(\r\n          `invalid patch path, key \"${key}\" not found in Yjs map - patch: ${JSON.stringify(patch)}`\r\n        )\r\n      }\r\n      applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)\r\n    } else if (yjs instanceof Y.Array) {\r\n      const child = yjs.get(Number(key))\r\n      if (child === undefined) {\r\n        throw failure(\r\n          `invalid patch path, key \"${key}\" not found in Yjs array - patch: ${JSON.stringify(\r\n            patch\r\n          )}`\r\n        )\r\n      }\r\n      applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)\r\n    } else {\r\n      throw failure(\r\n        `invalid patch path, key \"${key}\" not found in unknown Yjs object - patch: ${JSON.stringify(\r\n          patch\r\n        )}`\r\n      )\r\n    }\r\n  } else if (patch.path.length === 1) {\r\n    if (yjs instanceof Y.Map) {\r\n      const key = String(patch.path[0])\r\n\r\n      switch (patch.op) {\r\n        case \"add\":\r\n        case \"replace\": {\r\n          yjs.set(key, toYDataType(patch.value))\r\n          break\r\n        }\r\n        case \"remove\": {\r\n          yjs.delete(key)\r\n          break\r\n        }\r\n        default: {\r\n          throw failure(`invalid patch operation for map`)\r\n        }\r\n      }\r\n    } else if (yjs instanceof Y.Array) {\r\n      const key = patch.path[0]\r\n\r\n      switch (patch.op) {\r\n        case \"replace\": {\r\n          if (key === \"length\") {\r\n            if (yjs.length > patch.value) {\r\n              const toDelete = yjs.length - patch.value\r\n              yjs.delete(patch.value, toDelete)\r\n            } else if (yjs.length < patch.value) {\r\n              const toInsert = patch.value - yjs.length\r\n              yjs.insert(yjs.length, Array(toInsert).fill(undefined))\r\n            }\r\n          } else {\r\n            yjs.delete(Number(key))\r\n            yjs.insert(Number(key), [toYDataType(patch.value)])\r\n          }\r\n          break\r\n        }\r\n        case \"add\": {\r\n          yjs.insert(Number(key), [toYDataType(patch.value)])\r\n          break\r\n        }\r\n        case \"remove\": {\r\n          yjs.delete(Number(key))\r\n          break\r\n        }\r\n        default: {\r\n          throw failure(`invalid patch operation for array`)\r\n        }\r\n      }\r\n    } else {\r\n      throw failure(\r\n        `invalid patch path, the Yjs object is of an unkown type, so key \"${patch.path[0]}\" cannot be found in it`\r\n      )\r\n    }\r\n  } else {\r\n    throw failure(`invalid patch path, it cannot be empty`)\r\n  }\r\n}\r\n","import { Patch } from \"mobx-keystone\"\r\nimport * as Y from \"yjs\"\r\nimport { JSONArray, JSONObject, JSONValue } from \"../jsonTypes\"\r\nimport { failure } from \"../utils/error\"\r\n\r\nexport function convertYjsEventToPatches(event: Y.YEvent<any>): Patch[] {\r\n  const patches: Patch[] = []\r\n\r\n  if (event instanceof Y.YMapEvent) {\r\n    const source = event.target as Y.Map<any>\r\n\r\n    event.changes.keys.forEach((change, key) => {\r\n      const path = [...event.path, key]\r\n\r\n      switch (change.action) {\r\n        case \"add\":\r\n          patches.push({\r\n            op: \"add\",\r\n            path,\r\n            value: toPlainValue(source.get(key)),\r\n          })\r\n          break\r\n\r\n        case \"update\":\r\n          patches.push({\r\n            op: \"replace\",\r\n            path,\r\n            value: toPlainValue(source.get(key)),\r\n          })\r\n          break\r\n\r\n        case \"delete\":\r\n          patches.push({\r\n            op: \"remove\",\r\n            path,\r\n          })\r\n          break\r\n\r\n        default:\r\n          throw failure(`unsupported Yjs map event action: ${change.action}`)\r\n      }\r\n    })\r\n  } else if (event instanceof Y.YArrayEvent) {\r\n    let retain = 0\r\n    event.changes.delta.forEach((change) => {\r\n      if (change.retain) {\r\n        retain += change.retain\r\n      }\r\n\r\n      if (change.delete) {\r\n        // remove X items at retain position\r\n        const path = [...event.path, retain]\r\n        for (let i = 0; i < change.delete; i++) {\r\n          patches.push({\r\n            op: \"remove\",\r\n            path,\r\n          })\r\n        }\r\n      }\r\n\r\n      if (change.insert) {\r\n        const newValues = Array.isArray(change.insert) ? change.insert : [change.insert]\r\n        newValues.forEach((v) => {\r\n          const path = [...event.path, retain]\r\n          patches.push({\r\n            op: \"add\",\r\n            path,\r\n            value: toPlainValue(v),\r\n          })\r\n          retain++\r\n        })\r\n      }\r\n    })\r\n  }\r\n\r\n  return patches\r\n}\r\n\r\nfunction toPlainValue(v: Y.Map<any> | Y.Array<any> | JSONValue) {\r\n  if (v instanceof Y.Map || v instanceof Y.Array) {\r\n    return v.toJSON() as JSONObject | JSONArray\r\n  } else {\r\n    return v\r\n  }\r\n}\r\n","import {\r\n  AnyDataModel,\r\n  AnyModel,\r\n  AnyStandardType,\r\n  ModelClass,\r\n  Patch,\r\n  TypeToData,\r\n  applyPatches,\r\n  fromSnapshot,\r\n  getParentToChildPath,\r\n  onGlobalPatches,\r\n  onPatches,\r\n  onSnapshot,\r\n} from \"mobx-keystone\"\r\nimport * as Y from \"yjs\"\r\nimport { applyMobxKeystonePatchToYjsObject } from \"./applyMobxKeystonePatchToYjsObject\"\r\nimport { convertYjsEventToPatches } from \"./convertYjsEventToPatches\"\r\n\r\nexport function bindYjsToMobxKeystone<\r\n  TType extends AnyStandardType | ModelClass<AnyModel> | ModelClass<AnyDataModel>,\r\n>({\r\n  yjsDoc,\r\n  yjsObject,\r\n  mobxKeystoneType,\r\n}: {\r\n  yjsDoc: Y.Doc\r\n  yjsObject: Y.Map<unknown> | Y.Array<unknown>\r\n  mobxKeystoneType: TType\r\n}): {\r\n  boundObject: TypeToData<TType>\r\n  dispose(): void\r\n  yjsOrigin: symbol\r\n} {\r\n  const yjsJson = yjsObject.toJSON()\r\n\r\n  const initializationGlobalPatches: { target: object; patches: Patch[] }[] = []\r\n\r\n  const createBoundObject = () => {\r\n    const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {\r\n      initializationGlobalPatches.push({ target, patches })\r\n    })\r\n\r\n    try {\r\n      return fromSnapshot(mobxKeystoneType, yjsJson as any)\r\n    } finally {\r\n      disposeOnGlobalPatches()\r\n    }\r\n  }\r\n\r\n  const boundObject = createBoundObject()\r\n\r\n  let applyingMobxKeystoneChanges = 0\r\n  const yjsOrigin = Symbol(\"bindYjsToMobxKeystoneTransactionOrigin\")\r\n\r\n  // bind any changes from yjs to mobx-keystone\r\n  const observeDeepCb = (events: Y.YEvent<any>[]) => {\r\n    const patches: Patch[] = []\r\n    events.forEach((event) => {\r\n      if (event.transaction.origin !== yjsOrigin) {\r\n        patches.push(...convertYjsEventToPatches(event))\r\n      }\r\n    })\r\n\r\n    if (patches.length > 0) {\r\n      applyingMobxKeystoneChanges++\r\n      try {\r\n        applyPatches(boundObject, patches)\r\n      } finally {\r\n        applyingMobxKeystoneChanges--\r\n      }\r\n    }\r\n  }\r\n\r\n  yjsObject.observeDeep(observeDeepCb)\r\n\r\n  // bind any changes from mobx-keystone to yjs\r\n  let pendingPatches: Patch[] = []\r\n  const disposeOnPatches = onPatches(boundObject, (patches) => {\r\n    if (applyingMobxKeystoneChanges > 0) {\r\n      return\r\n    }\r\n\r\n    pendingPatches.push(...patches)\r\n  })\r\n\r\n  // this is only used so we can transact all patches to the snapshot boundary\r\n  const disposeOnSnapshot = onSnapshot(boundObject, () => {\r\n    if (pendingPatches.length === 0) {\r\n      return\r\n    }\r\n\r\n    const patches = pendingPatches\r\n    pendingPatches = []\r\n\r\n    yjsDoc.transact(() => {\r\n      patches.forEach((patch) => {\r\n        applyMobxKeystonePatchToYjsObject(patch, yjsObject)\r\n      })\r\n    }, yjsOrigin)\r\n  })\r\n\r\n  // sync initial patches, that might include setting defaults, IDs, etc\r\n  yjsDoc.transact(() => {\r\n    // we need to skip initializations until we hit the initialization of the bound object\r\n    // this is because default objects might be created and initialized before the main object\r\n    // but we just need to catch when those are actually assigned to the bound object\r\n    let boundObjectFound = false\r\n\r\n    initializationGlobalPatches.forEach(({ target, patches }) => {\r\n      if (!boundObjectFound) {\r\n        if (target !== boundObject) {\r\n          return // skip\r\n        }\r\n        boundObjectFound = true\r\n      }\r\n\r\n      const parentToChildPath = getParentToChildPath(boundObject, target)\r\n      // this is undefined only if target is not a child of boundModel\r\n      if (parentToChildPath !== undefined) {\r\n        patches.forEach((patch) => {\r\n          applyMobxKeystonePatchToYjsObject(\r\n            {\r\n              ...patch,\r\n              path: [...parentToChildPath, ...patch.path],\r\n            },\r\n            yjsObject\r\n          )\r\n        })\r\n      }\r\n    })\r\n  }, yjsOrigin)\r\n\r\n  return {\r\n    boundObject,\r\n    dispose: () => {\r\n      disposeOnPatches()\r\n      disposeOnSnapshot()\r\n      yjsObject.unobserveDeep(observeDeepCb)\r\n    },\r\n    yjsOrigin,\r\n  }\r\n}\r\n"],"names":["Y","onGlobalPatches","fromSnapshot","applyPatches","onPatches","onSnapshot","getParentToChildPath"],"mappings":"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"}
|
|
295
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"mobx-keystone-yjs.umd.js","sources":["../src/utils/error.ts","../src/binding/convertJsonToYjsData.ts","../src/binding/applyMobxKeystonePatchToYjsObject.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 * as Y from \"yjs\"\nimport { JsonValue, JsonArray, JsonObject, JsonPrimitive } from \"../jsonTypes\"\n\nfunction isJsonPrimitive(v: JsonValue): v is JsonPrimitive {\n  const t = typeof v\n  return t === \"string\" || t === \"number\" || t === \"boolean\" || v === null\n}\n\nfunction isJsonArray(v: JsonValue): v is JsonArray {\n  return Array.isArray(v)\n}\n\nfunction isJsonObject(v: JsonValue): v is JsonObject {\n  return !isJsonArray(v) && typeof v === \"object\"\n}\n\nexport function convertJsonToYjsData(v: JsonValue) {\n  if (v === undefined || isJsonPrimitive(v)) {\n    return v\n  }\n\n  if (isJsonArray(v)) {\n    const arr = new Y.Array()\n    applyJsonArrayYArray(arr, v)\n    return arr\n  }\n\n  if (isJsonObject(v)) {\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\nexport function applyJsonArrayYArray(dest: Y.Array<unknown>, source: JsonArray) {\n  dest.push(source.map(convertJsonToYjsData))\n}\n\nexport function applyJsonObjectToYMap(dest: Y.Map<unknown>, source: JsonObject) {\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\"\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))\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))\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 {\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))\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            if (yjs.length > patch.value) {\n              const toDelete = yjs.length - patch.value\n              yjs.delete(patch.value, toDelete)\n            } else if (yjs.length < patch.value) {\n              const toInsert = patch.value - yjs.length\n              yjs.insert(yjs.length, Array(toInsert).fill(undefined))\n            }\n          } else {\n            yjs.delete(Number(key))\n            yjs.insert(Number(key), [convertJsonToYjsData(patch.value)])\n          }\n          break\n        }\n        case \"add\": {\n          yjs.insert(Number(key), [convertJsonToYjsData(patch.value)])\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 {\n      throw failure(\n        `invalid patch path, the Yjs object is of an unkown type, so key \"${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 { Patch } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { JsonArray, JsonObject, JsonValue } from \"../jsonTypes\"\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 as Y.Map<any>\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  }\n\n  return patches\n}\n\nfunction toPlainValue(v: Y.Map<any> | Y.Array<any> | JsonValue) {\n  if (v instanceof Y.Map || v instanceof Y.Array) {\n    return v.toJSON() as JsonObject | JsonArray\n  } else {\n    return v\n  }\n}\n","import {\r\n  AnyDataModel,\r\n  AnyModel,\r\n  AnyStandardType,\r\n  ModelClass,\r\n  Patch,\r\n  TypeToData,\r\n  applyPatches,\r\n  fromSnapshot,\r\n  getParentToChildPath,\r\n  onGlobalPatches,\r\n  onPatches,\r\n  onSnapshot,\r\n} from \"mobx-keystone\"\r\nimport * as Y from \"yjs\"\r\nimport { applyMobxKeystonePatchToYjsObject } from \"./applyMobxKeystonePatchToYjsObject\"\r\nimport { convertYjsEventToPatches } from \"./convertYjsEventToPatches\"\r\n\r\nexport function bindYjsToMobxKeystone<\r\n  TType extends AnyStandardType | ModelClass<AnyModel> | ModelClass<AnyDataModel>,\r\n>({\r\n  yjsDoc,\r\n  yjsObject,\r\n  mobxKeystoneType,\r\n}: {\r\n  yjsDoc: Y.Doc\r\n  yjsObject: Y.Map<unknown> | Y.Array<unknown>\r\n  mobxKeystoneType: TType\r\n}): {\r\n  boundObject: TypeToData<TType>\r\n  dispose(): void\r\n  yjsOrigin: symbol\r\n} {\r\n  const yjsJson = yjsObject.toJSON()\r\n\r\n  const initializationGlobalPatches: { target: object; patches: Patch[] }[] = []\r\n\r\n  const createBoundObject = () => {\r\n    const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {\r\n      initializationGlobalPatches.push({ target, patches })\r\n    })\r\n\r\n    try {\r\n      return fromSnapshot(mobxKeystoneType, yjsJson as any)\r\n    } finally {\r\n      disposeOnGlobalPatches()\r\n    }\r\n  }\r\n\r\n  const boundObject = createBoundObject()\r\n\r\n  let applyingMobxKeystoneChanges = 0\r\n  const yjsOrigin = Symbol(\"bindYjsToMobxKeystoneTransactionOrigin\")\r\n\r\n  // bind any changes from yjs to mobx-keystone\r\n  const observeDeepCb = (events: Y.YEvent<any>[]) => {\r\n    const patches: Patch[] = []\r\n    events.forEach((event) => {\r\n      if (event.transaction.origin !== yjsOrigin) {\r\n        patches.push(...convertYjsEventToPatches(event))\r\n      }\r\n    })\r\n\r\n    if (patches.length > 0) {\r\n      applyingMobxKeystoneChanges++\r\n      try {\r\n        applyPatches(boundObject, patches)\r\n      } finally {\r\n        applyingMobxKeystoneChanges--\r\n      }\r\n    }\r\n  }\r\n\r\n  yjsObject.observeDeep(observeDeepCb)\r\n\r\n  // bind any changes from mobx-keystone to yjs\r\n  let pendingPatches: Patch[] = []\r\n  const disposeOnPatches = onPatches(boundObject, (patches) => {\r\n    if (applyingMobxKeystoneChanges > 0) {\r\n      return\r\n    }\r\n\r\n    pendingPatches.push(...patches)\r\n  })\r\n\r\n  // this is only used so we can transact all patches to the snapshot boundary\r\n  const disposeOnSnapshot = onSnapshot(boundObject, () => {\r\n    if (pendingPatches.length === 0) {\r\n      return\r\n    }\r\n\r\n    const patches = pendingPatches\r\n    pendingPatches = []\r\n\r\n    yjsDoc.transact(() => {\r\n      patches.forEach((patch) => {\r\n        applyMobxKeystonePatchToYjsObject(patch, yjsObject)\r\n      })\r\n    }, yjsOrigin)\r\n  })\r\n\r\n  // sync initial patches, that might include setting defaults, IDs, etc\r\n  yjsDoc.transact(() => {\r\n    // we need to skip initializations until we hit the initialization of the bound object\r\n    // this is because default objects might be created and initialized before the main object\r\n    // but we just need to catch when those are actually assigned to the bound object\r\n    let boundObjectFound = false\r\n\r\n    initializationGlobalPatches.forEach(({ target, patches }) => {\r\n      if (!boundObjectFound) {\r\n        if (target !== boundObject) {\r\n          return // skip\r\n        }\r\n        boundObjectFound = true\r\n      }\r\n\r\n      const parentToChildPath = getParentToChildPath(boundObject, target)\r\n      // this is undefined only if target is not a child of boundModel\r\n      if (parentToChildPath !== undefined) {\r\n        patches.forEach((patch) => {\r\n          applyMobxKeystonePatchToYjsObject(\r\n            {\r\n              ...patch,\r\n              path: [...parentToChildPath, ...patch.path],\r\n            },\r\n            yjsObject\r\n          )\r\n        })\r\n      }\r\n    })\r\n  }, yjsOrigin)\r\n\r\n  return {\r\n    boundObject,\r\n    dispose: () => {\r\n      disposeOnPatches()\r\n      disposeOnSnapshot()\r\n      yjsObject.unobserveDeep(observeDeepCb)\r\n    },\r\n    yjsOrigin,\r\n  }\r\n}\r\n"],"names":["Y","onGlobalPatches","fromSnapshot","applyPatches","onPatches","onSnapshot","getParentToChildPath"],"mappings":"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"}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import * as Y from "yjs";
|
|
2
|
+
import { JsonValue, JsonArray, JsonObject, JsonPrimitive } from "../jsonTypes";
|
|
3
|
+
export declare function convertJsonToYjsData(v: JsonValue): JsonPrimitive | Y.Array<unknown> | Y.Map<unknown>;
|
|
4
|
+
export declare function applyJsonArrayYArray(dest: Y.Array<unknown>, source: JsonArray): void;
|
|
5
|
+
export declare function applyJsonObjectToYMap(dest: Y.Map<unknown>, source: JsonObject): void;
|
package/dist/types/index.d.ts
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
|
-
export type
|
|
2
|
-
export type
|
|
3
|
-
export type
|
|
4
|
-
[key: string]:
|
|
1
|
+
export type JsonPrimitive = string | number | boolean | null;
|
|
2
|
+
export type JsonValue = JsonPrimitive | JsonObject | JsonArray;
|
|
3
|
+
export type JsonObject = {
|
|
4
|
+
[key: string]: JsonValue;
|
|
5
5
|
};
|
|
6
|
-
export interface
|
|
6
|
+
export interface JsonArray extends Array<JsonValue> {
|
|
7
7
|
}
|
package/package.json
CHANGED
|
@@ -1,87 +1,87 @@
|
|
|
1
|
-
{
|
|
2
|
-
"name": "mobx-keystone-yjs",
|
|
3
|
-
"version": "1.
|
|
4
|
-
"description": "Yjs bindings for mobx-keystone",
|
|
5
|
-
"keywords": [
|
|
6
|
-
"mobx",
|
|
7
|
-
"mobx-keystone",
|
|
8
|
-
"yjs",
|
|
9
|
-
"crdt",
|
|
10
|
-
"state management"
|
|
11
|
-
],
|
|
12
|
-
"repository": {
|
|
13
|
-
"type": "git",
|
|
14
|
-
"url": "https://github.com/xaviergonz/mobx-keystone.git"
|
|
15
|
-
},
|
|
16
|
-
"bugs": {
|
|
17
|
-
"url": "https://github.com/xaviergonz/mobx-keystone/issues"
|
|
18
|
-
},
|
|
19
|
-
"homepage": "https://mobx-keystone.js.org",
|
|
20
|
-
"license": "MIT",
|
|
21
|
-
"author": "Javier González Garcés",
|
|
22
|
-
"source": "./src/index.ts",
|
|
23
|
-
"exports": {
|
|
24
|
-
"./package.json": "./package.json",
|
|
25
|
-
".": {
|
|
26
|
-
"import": "./dist/mobx-keystone-yjs.esm.mjs",
|
|
27
|
-
"require": "./dist/mobx-keystone-yjs.umd.js",
|
|
28
|
-
"script": "./dist/mobx-keystone-yjs.umd.js",
|
|
29
|
-
"default": "./dist/mobx-keystone-yjs.esm.mjs",
|
|
30
|
-
"types": "./dist/types/index.d.ts"
|
|
31
|
-
}
|
|
32
|
-
},
|
|
33
|
-
"esmodule": "./dist/mobx-keystone-yjs.esm.js",
|
|
34
|
-
"module": "./dist/mobx-keystone-yjs.esm.js",
|
|
35
|
-
"jsnext:main": "./dist/mobx-keystone-yjs.esm.js",
|
|
36
|
-
"react-native": "./dist/mobx-keystone-yjs.umd.js",
|
|
37
|
-
"umd:main": "./dist/mobx-keystone-yjs.umd.js",
|
|
38
|
-
"unpkg": "./dist/mobx-keystone-yjs.umd.js",
|
|
39
|
-
"jsdelivr": "./dist/mobx-keystone-yjs.umd.js",
|
|
40
|
-
"main": "./dist/mobx-keystone-yjs.umd.js",
|
|
41
|
-
"types": "./dist/types/index.d.ts",
|
|
42
|
-
"typings": "./dist/types/index.d.ts",
|
|
43
|
-
"sideEffects": false,
|
|
44
|
-
"files": [
|
|
45
|
-
"src",
|
|
46
|
-
"dist",
|
|
47
|
-
"LICENSE",
|
|
48
|
-
"CHANGELOG.md",
|
|
49
|
-
"README.md"
|
|
50
|
-
],
|
|
51
|
-
"scripts": {
|
|
52
|
-
"quick-build": "tsc",
|
|
53
|
-
"quick-build-tests": "tsc -p test",
|
|
54
|
-
"copy-root-files": "shx cp ../../LICENSE .",
|
|
55
|
-
"build": "yarn quick-build && yarn copy-root-files && shx rm -rf dist && vite build && shx cp dist/mobx-keystone-yjs.esm.mjs dist/mobx-keystone-yjs.esm.js",
|
|
56
|
-
"test": "jest",
|
|
57
|
-
"test:ci": "yarn test -i",
|
|
58
|
-
"test:perf": "yarn build && yarn test:perf:run",
|
|
59
|
-
"lint": "cd ../.. && yarn eslint \"packages/mobx-keystone-yjs/src/**/*.ts\" \"packages/mobx-keystone-yjs/test/**/*.ts\""
|
|
60
|
-
},
|
|
61
|
-
"peerDependencies": {
|
|
62
|
-
"mobx": "^6.0.0 || ^5.0.0 || ^4.0.0",
|
|
63
|
-
"mobx-keystone": "^1.8.1",
|
|
64
|
-
"yjs": "^13.0.0"
|
|
65
|
-
},
|
|
66
|
-
"devDependencies": {
|
|
67
|
-
"@types/jest": "^29.5.11",
|
|
68
|
-
"@types/node": "^20.10.5",
|
|
69
|
-
"babel-jest": "^29.7.0",
|
|
70
|
-
"jest": "^29.7.0",
|
|
71
|
-
"mobx": "^6.12.0",
|
|
72
|
-
"mobx-keystone": "workspace:packages/lib",
|
|
73
|
-
"rollup-plugin-typescript2": "^0.36.0",
|
|
74
|
-
"shx": "^0.3.4",
|
|
75
|
-
"spec.ts": "^1.1.3",
|
|
76
|
-
"ts-jest": "^29.1.1",
|
|
77
|
-
"ts-node": "^10.9.2",
|
|
78
|
-
"typescript": "^5.3.3",
|
|
79
|
-
"vite": "^5.0.10"
|
|
80
|
-
},
|
|
81
|
-
"dependencies": {
|
|
82
|
-
"tslib": "^2.6.2"
|
|
83
|
-
},
|
|
84
|
-
"directories": {
|
|
85
|
-
"test": "test"
|
|
86
|
-
}
|
|
87
|
-
}
|
|
1
|
+
{
|
|
2
|
+
"name": "mobx-keystone-yjs",
|
|
3
|
+
"version": "1.1.0",
|
|
4
|
+
"description": "Yjs bindings for mobx-keystone",
|
|
5
|
+
"keywords": [
|
|
6
|
+
"mobx",
|
|
7
|
+
"mobx-keystone",
|
|
8
|
+
"yjs",
|
|
9
|
+
"crdt",
|
|
10
|
+
"state management"
|
|
11
|
+
],
|
|
12
|
+
"repository": {
|
|
13
|
+
"type": "git",
|
|
14
|
+
"url": "https://github.com/xaviergonz/mobx-keystone.git"
|
|
15
|
+
},
|
|
16
|
+
"bugs": {
|
|
17
|
+
"url": "https://github.com/xaviergonz/mobx-keystone/issues"
|
|
18
|
+
},
|
|
19
|
+
"homepage": "https://mobx-keystone.js.org",
|
|
20
|
+
"license": "MIT",
|
|
21
|
+
"author": "Javier González Garcés",
|
|
22
|
+
"source": "./src/index.ts",
|
|
23
|
+
"exports": {
|
|
24
|
+
"./package.json": "./package.json",
|
|
25
|
+
".": {
|
|
26
|
+
"import": "./dist/mobx-keystone-yjs.esm.mjs",
|
|
27
|
+
"require": "./dist/mobx-keystone-yjs.umd.js",
|
|
28
|
+
"script": "./dist/mobx-keystone-yjs.umd.js",
|
|
29
|
+
"default": "./dist/mobx-keystone-yjs.esm.mjs",
|
|
30
|
+
"types": "./dist/types/index.d.ts"
|
|
31
|
+
}
|
|
32
|
+
},
|
|
33
|
+
"esmodule": "./dist/mobx-keystone-yjs.esm.js",
|
|
34
|
+
"module": "./dist/mobx-keystone-yjs.esm.js",
|
|
35
|
+
"jsnext:main": "./dist/mobx-keystone-yjs.esm.js",
|
|
36
|
+
"react-native": "./dist/mobx-keystone-yjs.umd.js",
|
|
37
|
+
"umd:main": "./dist/mobx-keystone-yjs.umd.js",
|
|
38
|
+
"unpkg": "./dist/mobx-keystone-yjs.umd.js",
|
|
39
|
+
"jsdelivr": "./dist/mobx-keystone-yjs.umd.js",
|
|
40
|
+
"main": "./dist/mobx-keystone-yjs.umd.js",
|
|
41
|
+
"types": "./dist/types/index.d.ts",
|
|
42
|
+
"typings": "./dist/types/index.d.ts",
|
|
43
|
+
"sideEffects": false,
|
|
44
|
+
"files": [
|
|
45
|
+
"src",
|
|
46
|
+
"dist",
|
|
47
|
+
"LICENSE",
|
|
48
|
+
"CHANGELOG.md",
|
|
49
|
+
"README.md"
|
|
50
|
+
],
|
|
51
|
+
"scripts": {
|
|
52
|
+
"quick-build": "tsc",
|
|
53
|
+
"quick-build-tests": "tsc -p test",
|
|
54
|
+
"copy-root-files": "shx cp ../../LICENSE .",
|
|
55
|
+
"build": "yarn quick-build && yarn copy-root-files && shx rm -rf dist && vite build && shx cp dist/mobx-keystone-yjs.esm.mjs dist/mobx-keystone-yjs.esm.js",
|
|
56
|
+
"test": "jest",
|
|
57
|
+
"test:ci": "yarn test -i",
|
|
58
|
+
"test:perf": "yarn build && yarn test:perf:run",
|
|
59
|
+
"lint": "cd ../.. && yarn eslint \"packages/mobx-keystone-yjs/src/**/*.ts\" \"packages/mobx-keystone-yjs/test/**/*.ts\""
|
|
60
|
+
},
|
|
61
|
+
"peerDependencies": {
|
|
62
|
+
"mobx": "^6.0.0 || ^5.0.0 || ^4.0.0",
|
|
63
|
+
"mobx-keystone": "^1.8.1",
|
|
64
|
+
"yjs": "^13.0.0"
|
|
65
|
+
},
|
|
66
|
+
"devDependencies": {
|
|
67
|
+
"@types/jest": "^29.5.11",
|
|
68
|
+
"@types/node": "^20.10.5",
|
|
69
|
+
"babel-jest": "^29.7.0",
|
|
70
|
+
"jest": "^29.7.0",
|
|
71
|
+
"mobx": "^6.12.0",
|
|
72
|
+
"mobx-keystone": "workspace:packages/lib",
|
|
73
|
+
"rollup-plugin-typescript2": "^0.36.0",
|
|
74
|
+
"shx": "^0.3.4",
|
|
75
|
+
"spec.ts": "^1.1.3",
|
|
76
|
+
"ts-jest": "^29.1.1",
|
|
77
|
+
"ts-node": "^10.9.2",
|
|
78
|
+
"typescript": "^5.3.3",
|
|
79
|
+
"vite": "^5.0.10"
|
|
80
|
+
},
|
|
81
|
+
"dependencies": {
|
|
82
|
+
"tslib": "^2.6.2"
|
|
83
|
+
},
|
|
84
|
+
"directories": {
|
|
85
|
+
"test": "test"
|
|
86
|
+
}
|
|
87
|
+
}
|
|
@@ -1,92 +1,92 @@
|
|
|
1
|
-
import { Patch } from "mobx-keystone"
|
|
2
|
-
import
|
|
3
|
-
import
|
|
4
|
-
import {
|
|
5
|
-
|
|
6
|
-
export function applyMobxKeystonePatchToYjsObject(patch: Patch, yjs: unknown): void {
|
|
7
|
-
if (patch.path.length > 1) {
|
|
8
|
-
const [key, ...rest] = patch.path
|
|
9
|
-
|
|
10
|
-
if (yjs instanceof Y.Map) {
|
|
11
|
-
const child = yjs.get(String(key))
|
|
12
|
-
if (child === undefined) {
|
|
13
|
-
throw failure(
|
|
14
|
-
`invalid patch path, key "${key}" not found in Yjs map - patch: ${JSON.stringify(patch)}`
|
|
15
|
-
)
|
|
16
|
-
}
|
|
17
|
-
applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)
|
|
18
|
-
} else if (yjs instanceof Y.Array) {
|
|
19
|
-
const child = yjs.get(Number(key))
|
|
20
|
-
if (child === undefined) {
|
|
21
|
-
throw failure(
|
|
22
|
-
`invalid patch path, key "${key}" not found in Yjs array - patch: ${JSON.stringify(
|
|
23
|
-
patch
|
|
24
|
-
)}`
|
|
25
|
-
)
|
|
26
|
-
}
|
|
27
|
-
applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)
|
|
28
|
-
} else {
|
|
29
|
-
throw failure(
|
|
30
|
-
`invalid patch path, key "${key}" not found in unknown Yjs object - patch: ${JSON.stringify(
|
|
31
|
-
patch
|
|
32
|
-
)}`
|
|
33
|
-
)
|
|
34
|
-
}
|
|
35
|
-
} else if (patch.path.length === 1) {
|
|
36
|
-
if (yjs instanceof Y.Map) {
|
|
37
|
-
const key = String(patch.path[0])
|
|
38
|
-
|
|
39
|
-
switch (patch.op) {
|
|
40
|
-
case "add":
|
|
41
|
-
case "replace": {
|
|
42
|
-
yjs.set(key,
|
|
43
|
-
break
|
|
44
|
-
}
|
|
45
|
-
case "remove": {
|
|
46
|
-
yjs.delete(key)
|
|
47
|
-
break
|
|
48
|
-
}
|
|
49
|
-
default: {
|
|
50
|
-
throw failure(`invalid patch operation for map`)
|
|
51
|
-
}
|
|
52
|
-
}
|
|
53
|
-
} else if (yjs instanceof Y.Array) {
|
|
54
|
-
const key = patch.path[0]
|
|
55
|
-
|
|
56
|
-
switch (patch.op) {
|
|
57
|
-
case "replace": {
|
|
58
|
-
if (key === "length") {
|
|
59
|
-
if (yjs.length > patch.value) {
|
|
60
|
-
const toDelete = yjs.length - patch.value
|
|
61
|
-
yjs.delete(patch.value, toDelete)
|
|
62
|
-
} else if (yjs.length < patch.value) {
|
|
63
|
-
const toInsert = patch.value - yjs.length
|
|
64
|
-
yjs.insert(yjs.length, Array(toInsert).fill(undefined))
|
|
65
|
-
}
|
|
66
|
-
} else {
|
|
67
|
-
yjs.delete(Number(key))
|
|
68
|
-
yjs.insert(Number(key), [
|
|
69
|
-
}
|
|
70
|
-
break
|
|
71
|
-
}
|
|
72
|
-
case "add": {
|
|
73
|
-
yjs.insert(Number(key), [
|
|
74
|
-
break
|
|
75
|
-
}
|
|
76
|
-
case "remove": {
|
|
77
|
-
yjs.delete(Number(key))
|
|
78
|
-
break
|
|
79
|
-
}
|
|
80
|
-
default: {
|
|
81
|
-
throw failure(`invalid patch operation for array`)
|
|
82
|
-
}
|
|
83
|
-
}
|
|
84
|
-
} else {
|
|
85
|
-
throw failure(
|
|
86
|
-
`invalid patch path, the Yjs object is of an unkown type, so key "${patch.path[0]}" cannot be found in it`
|
|
87
|
-
)
|
|
88
|
-
}
|
|
89
|
-
} else {
|
|
90
|
-
throw failure(`invalid patch path, it cannot be empty`)
|
|
91
|
-
}
|
|
92
|
-
}
|
|
1
|
+
import { Patch } from "mobx-keystone"
|
|
2
|
+
import * as Y from "yjs"
|
|
3
|
+
import { failure } from "../utils/error"
|
|
4
|
+
import { convertJsonToYjsData } from "./convertJsonToYjsData"
|
|
5
|
+
|
|
6
|
+
export function applyMobxKeystonePatchToYjsObject(patch: Patch, yjs: unknown): void {
|
|
7
|
+
if (patch.path.length > 1) {
|
|
8
|
+
const [key, ...rest] = patch.path
|
|
9
|
+
|
|
10
|
+
if (yjs instanceof Y.Map) {
|
|
11
|
+
const child = yjs.get(String(key))
|
|
12
|
+
if (child === undefined) {
|
|
13
|
+
throw failure(
|
|
14
|
+
`invalid patch path, key "${key}" not found in Yjs map - patch: ${JSON.stringify(patch)}`
|
|
15
|
+
)
|
|
16
|
+
}
|
|
17
|
+
applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)
|
|
18
|
+
} else if (yjs instanceof Y.Array) {
|
|
19
|
+
const child = yjs.get(Number(key))
|
|
20
|
+
if (child === undefined) {
|
|
21
|
+
throw failure(
|
|
22
|
+
`invalid patch path, key "${key}" not found in Yjs array - patch: ${JSON.stringify(
|
|
23
|
+
patch
|
|
24
|
+
)}`
|
|
25
|
+
)
|
|
26
|
+
}
|
|
27
|
+
applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child)
|
|
28
|
+
} else {
|
|
29
|
+
throw failure(
|
|
30
|
+
`invalid patch path, key "${key}" not found in unknown Yjs object - patch: ${JSON.stringify(
|
|
31
|
+
patch
|
|
32
|
+
)}`
|
|
33
|
+
)
|
|
34
|
+
}
|
|
35
|
+
} else if (patch.path.length === 1) {
|
|
36
|
+
if (yjs instanceof Y.Map) {
|
|
37
|
+
const key = String(patch.path[0])
|
|
38
|
+
|
|
39
|
+
switch (patch.op) {
|
|
40
|
+
case "add":
|
|
41
|
+
case "replace": {
|
|
42
|
+
yjs.set(key, convertJsonToYjsData(patch.value))
|
|
43
|
+
break
|
|
44
|
+
}
|
|
45
|
+
case "remove": {
|
|
46
|
+
yjs.delete(key)
|
|
47
|
+
break
|
|
48
|
+
}
|
|
49
|
+
default: {
|
|
50
|
+
throw failure(`invalid patch operation for map`)
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
} else if (yjs instanceof Y.Array) {
|
|
54
|
+
const key = patch.path[0]
|
|
55
|
+
|
|
56
|
+
switch (patch.op) {
|
|
57
|
+
case "replace": {
|
|
58
|
+
if (key === "length") {
|
|
59
|
+
if (yjs.length > patch.value) {
|
|
60
|
+
const toDelete = yjs.length - patch.value
|
|
61
|
+
yjs.delete(patch.value, toDelete)
|
|
62
|
+
} else if (yjs.length < patch.value) {
|
|
63
|
+
const toInsert = patch.value - yjs.length
|
|
64
|
+
yjs.insert(yjs.length, Array(toInsert).fill(undefined))
|
|
65
|
+
}
|
|
66
|
+
} else {
|
|
67
|
+
yjs.delete(Number(key))
|
|
68
|
+
yjs.insert(Number(key), [convertJsonToYjsData(patch.value)])
|
|
69
|
+
}
|
|
70
|
+
break
|
|
71
|
+
}
|
|
72
|
+
case "add": {
|
|
73
|
+
yjs.insert(Number(key), [convertJsonToYjsData(patch.value)])
|
|
74
|
+
break
|
|
75
|
+
}
|
|
76
|
+
case "remove": {
|
|
77
|
+
yjs.delete(Number(key))
|
|
78
|
+
break
|
|
79
|
+
}
|
|
80
|
+
default: {
|
|
81
|
+
throw failure(`invalid patch operation for array`)
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
} else {
|
|
85
|
+
throw failure(
|
|
86
|
+
`invalid patch path, the Yjs object is of an unkown type, so key "${patch.path[0]}" cannot be found in it`
|
|
87
|
+
)
|
|
88
|
+
}
|
|
89
|
+
} else {
|
|
90
|
+
throw failure(`invalid patch path, it cannot be empty`)
|
|
91
|
+
}
|
|
92
|
+
}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import * as Y from "yjs"
|
|
2
|
+
import { JsonValue, JsonArray, JsonObject, JsonPrimitive } from "../jsonTypes"
|
|
3
|
+
|
|
4
|
+
function isJsonPrimitive(v: JsonValue): v is JsonPrimitive {
|
|
5
|
+
const t = typeof v
|
|
6
|
+
return t === "string" || t === "number" || t === "boolean" || v === null
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
function isJsonArray(v: JsonValue): v is JsonArray {
|
|
10
|
+
return Array.isArray(v)
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
function isJsonObject(v: JsonValue): v is JsonObject {
|
|
14
|
+
return !isJsonArray(v) && typeof v === "object"
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export function convertJsonToYjsData(v: JsonValue) {
|
|
18
|
+
if (v === undefined || isJsonPrimitive(v)) {
|
|
19
|
+
return v
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
if (isJsonArray(v)) {
|
|
23
|
+
const arr = new Y.Array()
|
|
24
|
+
applyJsonArrayYArray(arr, v)
|
|
25
|
+
return arr
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
if (isJsonObject(v)) {
|
|
29
|
+
const map = new Y.Map()
|
|
30
|
+
applyJsonObjectToYMap(map, v)
|
|
31
|
+
return map
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
throw new Error(`unsupported value type: ${v}`)
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export function applyJsonArrayYArray(dest: Y.Array<unknown>, source: JsonArray) {
|
|
38
|
+
dest.push(source.map(convertJsonToYjsData))
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
export function applyJsonObjectToYMap(dest: Y.Map<unknown>, source: JsonObject) {
|
|
42
|
+
Object.entries(source).forEach(([k, v]) => {
|
|
43
|
+
dest.set(k, convertJsonToYjsData(v))
|
|
44
|
+
})
|
|
45
|
+
}
|
|
@@ -1,85 +1,85 @@
|
|
|
1
|
-
import { Patch } from "mobx-keystone"
|
|
2
|
-
import * as Y from "yjs"
|
|
3
|
-
import {
|
|
4
|
-
import { failure } from "../utils/error"
|
|
5
|
-
|
|
6
|
-
export function convertYjsEventToPatches(event: Y.YEvent<any>): Patch[] {
|
|
7
|
-
const patches: Patch[] = []
|
|
8
|
-
|
|
9
|
-
if (event instanceof Y.YMapEvent) {
|
|
10
|
-
const source = event.target as Y.Map<any>
|
|
11
|
-
|
|
12
|
-
event.changes.keys.forEach((change, key) => {
|
|
13
|
-
const path = [...event.path, key]
|
|
14
|
-
|
|
15
|
-
switch (change.action) {
|
|
16
|
-
case "add":
|
|
17
|
-
patches.push({
|
|
18
|
-
op: "add",
|
|
19
|
-
path,
|
|
20
|
-
value: toPlainValue(source.get(key)),
|
|
21
|
-
})
|
|
22
|
-
break
|
|
23
|
-
|
|
24
|
-
case "update":
|
|
25
|
-
patches.push({
|
|
26
|
-
op: "replace",
|
|
27
|
-
path,
|
|
28
|
-
value: toPlainValue(source.get(key)),
|
|
29
|
-
})
|
|
30
|
-
break
|
|
31
|
-
|
|
32
|
-
case "delete":
|
|
33
|
-
patches.push({
|
|
34
|
-
op: "remove",
|
|
35
|
-
path,
|
|
36
|
-
})
|
|
37
|
-
break
|
|
38
|
-
|
|
39
|
-
default:
|
|
40
|
-
throw failure(`unsupported Yjs map event action: ${change.action}`)
|
|
41
|
-
}
|
|
42
|
-
})
|
|
43
|
-
} else if (event instanceof Y.YArrayEvent) {
|
|
44
|
-
let retain = 0
|
|
45
|
-
event.changes.delta.forEach((change) => {
|
|
46
|
-
if (change.retain) {
|
|
47
|
-
retain += change.retain
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
if (change.delete) {
|
|
51
|
-
// remove X items at retain position
|
|
52
|
-
const path = [...event.path, retain]
|
|
53
|
-
for (let i = 0; i < change.delete; i++) {
|
|
54
|
-
patches.push({
|
|
55
|
-
op: "remove",
|
|
56
|
-
path,
|
|
57
|
-
})
|
|
58
|
-
}
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
if (change.insert) {
|
|
62
|
-
const newValues = Array.isArray(change.insert) ? change.insert : [change.insert]
|
|
63
|
-
newValues.forEach((v) => {
|
|
64
|
-
const path = [...event.path, retain]
|
|
65
|
-
patches.push({
|
|
66
|
-
op: "add",
|
|
67
|
-
path,
|
|
68
|
-
value: toPlainValue(v),
|
|
69
|
-
})
|
|
70
|
-
retain++
|
|
71
|
-
})
|
|
72
|
-
}
|
|
73
|
-
})
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
return patches
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
function toPlainValue(v: Y.Map<any> | Y.Array<any> |
|
|
80
|
-
if (v instanceof Y.Map || v instanceof Y.Array) {
|
|
81
|
-
return v.toJSON() as
|
|
82
|
-
} else {
|
|
83
|
-
return v
|
|
84
|
-
}
|
|
85
|
-
}
|
|
1
|
+
import { Patch } from "mobx-keystone"
|
|
2
|
+
import * as Y from "yjs"
|
|
3
|
+
import { JsonArray, JsonObject, JsonValue } from "../jsonTypes"
|
|
4
|
+
import { failure } from "../utils/error"
|
|
5
|
+
|
|
6
|
+
export function convertYjsEventToPatches(event: Y.YEvent<any>): Patch[] {
|
|
7
|
+
const patches: Patch[] = []
|
|
8
|
+
|
|
9
|
+
if (event instanceof Y.YMapEvent) {
|
|
10
|
+
const source = event.target as Y.Map<any>
|
|
11
|
+
|
|
12
|
+
event.changes.keys.forEach((change, key) => {
|
|
13
|
+
const path = [...event.path, key]
|
|
14
|
+
|
|
15
|
+
switch (change.action) {
|
|
16
|
+
case "add":
|
|
17
|
+
patches.push({
|
|
18
|
+
op: "add",
|
|
19
|
+
path,
|
|
20
|
+
value: toPlainValue(source.get(key)),
|
|
21
|
+
})
|
|
22
|
+
break
|
|
23
|
+
|
|
24
|
+
case "update":
|
|
25
|
+
patches.push({
|
|
26
|
+
op: "replace",
|
|
27
|
+
path,
|
|
28
|
+
value: toPlainValue(source.get(key)),
|
|
29
|
+
})
|
|
30
|
+
break
|
|
31
|
+
|
|
32
|
+
case "delete":
|
|
33
|
+
patches.push({
|
|
34
|
+
op: "remove",
|
|
35
|
+
path,
|
|
36
|
+
})
|
|
37
|
+
break
|
|
38
|
+
|
|
39
|
+
default:
|
|
40
|
+
throw failure(`unsupported Yjs map event action: ${change.action}`)
|
|
41
|
+
}
|
|
42
|
+
})
|
|
43
|
+
} else if (event instanceof Y.YArrayEvent) {
|
|
44
|
+
let retain = 0
|
|
45
|
+
event.changes.delta.forEach((change) => {
|
|
46
|
+
if (change.retain) {
|
|
47
|
+
retain += change.retain
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
if (change.delete) {
|
|
51
|
+
// remove X items at retain position
|
|
52
|
+
const path = [...event.path, retain]
|
|
53
|
+
for (let i = 0; i < change.delete; i++) {
|
|
54
|
+
patches.push({
|
|
55
|
+
op: "remove",
|
|
56
|
+
path,
|
|
57
|
+
})
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
if (change.insert) {
|
|
62
|
+
const newValues = Array.isArray(change.insert) ? change.insert : [change.insert]
|
|
63
|
+
newValues.forEach((v) => {
|
|
64
|
+
const path = [...event.path, retain]
|
|
65
|
+
patches.push({
|
|
66
|
+
op: "add",
|
|
67
|
+
path,
|
|
68
|
+
value: toPlainValue(v),
|
|
69
|
+
})
|
|
70
|
+
retain++
|
|
71
|
+
})
|
|
72
|
+
}
|
|
73
|
+
})
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
return patches
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
function toPlainValue(v: Y.Map<any> | Y.Array<any> | JsonValue) {
|
|
80
|
+
if (v instanceof Y.Map || v instanceof Y.Array) {
|
|
81
|
+
return v.toJSON() as JsonObject | JsonArray
|
|
82
|
+
} else {
|
|
83
|
+
return v
|
|
84
|
+
}
|
|
85
|
+
}
|
package/src/index.ts
CHANGED
|
@@ -1,2 +1,7 @@
|
|
|
1
|
-
export { MobxKeystoneYjsError } from "./utils/error"
|
|
2
|
-
export { bindYjsToMobxKeystone } from "./binding/bindYjsToMobxKeystone"
|
|
1
|
+
export { MobxKeystoneYjsError } from "./utils/error"
|
|
2
|
+
export { bindYjsToMobxKeystone } from "./binding/bindYjsToMobxKeystone"
|
|
3
|
+
export {
|
|
4
|
+
convertJsonToYjsData,
|
|
5
|
+
applyJsonArrayYArray,
|
|
6
|
+
applyJsonObjectToYMap,
|
|
7
|
+
} from "./binding/convertJsonToYjsData"
|
package/src/jsonTypes.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
export type
|
|
2
|
-
export type
|
|
3
|
-
export type
|
|
4
|
-
export interface
|
|
1
|
+
export type JsonPrimitive = string | number | boolean | null
|
|
2
|
+
export type JsonValue = JsonPrimitive | JsonObject | JsonArray
|
|
3
|
+
export type JsonObject = { [key: string]: JsonValue }
|
|
4
|
+
export interface JsonArray extends Array<JsonValue> {}
|
|
@@ -1,41 +0,0 @@
|
|
|
1
|
-
import * as Y from "yjs"
|
|
2
|
-
import { JSONValue, JSONArray, JSONObject, JSONPrimitive } from "../jsonTypes"
|
|
3
|
-
|
|
4
|
-
function isJSONPrimitive(v: JSONValue): v is JSONPrimitive {
|
|
5
|
-
const t = typeof v
|
|
6
|
-
return t === "string" || t === "number" || t === "boolean" || v === null
|
|
7
|
-
}
|
|
8
|
-
|
|
9
|
-
function isJSONArray(v: JSONValue): v is JSONArray {
|
|
10
|
-
return Array.isArray(v)
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
function isJSONObject(v: JSONValue): v is JSONObject {
|
|
14
|
-
return !isJSONArray(v) && typeof v === "object"
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
export function toYDataType(v: JSONValue) {
|
|
18
|
-
if (isJSONPrimitive(v)) {
|
|
19
|
-
return v
|
|
20
|
-
} else if (isJSONArray(v)) {
|
|
21
|
-
const arr = new Y.Array()
|
|
22
|
-
applyJsonArray(arr, v)
|
|
23
|
-
return arr
|
|
24
|
-
} else if (isJSONObject(v)) {
|
|
25
|
-
const map = new Y.Map()
|
|
26
|
-
applyJsonObject(map, v)
|
|
27
|
-
return map
|
|
28
|
-
} else {
|
|
29
|
-
return undefined
|
|
30
|
-
}
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
function applyJsonArray(dest: Y.Array<unknown>, source: JSONArray) {
|
|
34
|
-
dest.push(source.map(toYDataType))
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
function applyJsonObject(dest: Y.Map<unknown>, source: JSONObject) {
|
|
38
|
-
Object.entries(source).forEach(([k, v]) => {
|
|
39
|
-
dest.set(k, toYDataType(v))
|
|
40
|
-
})
|
|
41
|
-
}
|