mobx-keystone-yjs 1.1.0 → 1.2.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 CHANGED
@@ -1,5 +1,9 @@
1
1
  # Change Log
2
2
 
3
+ ## 1.2.0
4
+
5
+ - Added `yjsBindingContext` so bound objects offer a context with the Y.js doc, bound object, etc.
6
+
3
7
  ## 1.1.0
4
8
 
5
9
  - Added the `convertJsonToYjsData`, `applyJsonArrayToYArray` and `applyJsonObjectToYMap` function to help with first migrations from snapshots to Y.js states.
@@ -1,4 +1,4 @@
1
- import { onPatches, onSnapshot, getParentToChildPath, onGlobalPatches, fromSnapshot, applyPatches } from "mobx-keystone";
1
+ import { createContext, onPatches, onSnapshot, getParentToChildPath, onGlobalPatches, fromSnapshot, applyPatches } from "mobx-keystone";
2
2
  import * as Y from "yjs";
3
3
  class MobxKeystoneYjsError extends Error {
4
4
  constructor(msg) {
@@ -184,7 +184,15 @@ function toPlainValue(v) {
184
184
  return v;
185
185
  }
186
186
  }
187
+ const yjsBindingContext = createContext(void 0);
187
188
  function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
189
+ const yjsOrigin = Symbol("bindYjsToMobxKeystoneTransactionOrigin");
190
+ const bindingContext = {
191
+ yjsDoc,
192
+ yjsObject,
193
+ mobxKeystoneType,
194
+ yjsOrigin
195
+ };
188
196
  const yjsJson = yjsObject.toJSON();
189
197
  const initializationGlobalPatches = [];
190
198
  const createBoundObject = () => {
@@ -192,14 +200,15 @@ function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
192
200
  initializationGlobalPatches.push({ target, patches });
193
201
  });
194
202
  try {
195
- return fromSnapshot(mobxKeystoneType, yjsJson);
203
+ const boundObject2 = yjsBindingContext.apply(() => fromSnapshot(mobxKeystoneType, yjsJson), bindingContext);
204
+ yjsBindingContext.set(boundObject2, bindingContext);
205
+ return boundObject2;
196
206
  } finally {
197
207
  disposeOnGlobalPatches();
198
208
  }
199
209
  };
200
210
  const boundObject = createBoundObject();
201
211
  let applyingMobxKeystoneChanges = 0;
202
- const yjsOrigin = Symbol("bindYjsToMobxKeystoneTransactionOrigin");
203
212
  const observeDeepCb = (events) => {
204
213
  const patches = [];
205
214
  events.forEach((event) => {
@@ -271,6 +280,7 @@ export {
271
280
  applyJsonArrayYArray,
272
281
  applyJsonObjectToYMap,
273
282
  bindYjsToMobxKeystone,
274
- convertJsonToYjsData
283
+ convertJsonToYjsData,
284
+ yjsBindingContext
275
285
  };
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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"}
286
+ //# 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/yjsBindingContext.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 { AnyType, createContext } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\n\nexport interface YjsBindingContext {\n  yjsDoc: Y.Doc\n  yjsObject: Y.Map<unknown> | Y.Array<unknown>\n  mobxKeystoneType: AnyType\n  yjsOrigin: symbol\n}\n\nexport const yjsBindingContext = createContext<YjsBindingContext | undefined>(undefined)\n","import {\n  AnyDataModel,\n  AnyModel,\n  AnyStandardType,\n  ModelClass,\n  Patch,\n  TypeToData,\n  applyPatches,\n  fromSnapshot,\n  getParentToChildPath,\n  onGlobalPatches,\n  onPatches,\n  onSnapshot,\n} from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { applyMobxKeystonePatchToYjsObject } from \"./applyMobxKeystonePatchToYjsObject\"\nimport { convertYjsEventToPatches } from \"./convertYjsEventToPatches\"\nimport { YjsBindingContext, yjsBindingContext } from \"./yjsBindingContext\"\n\nexport function bindYjsToMobxKeystone<\n  TType extends AnyStandardType | ModelClass<AnyModel> | ModelClass<AnyDataModel>,\n>({\n  yjsDoc,\n  yjsObject,\n  mobxKeystoneType,\n}: {\n  yjsDoc: Y.Doc\n  yjsObject: Y.Map<unknown> | Y.Array<unknown>\n  mobxKeystoneType: TType\n}): {\n  boundObject: TypeToData<TType>\n  dispose(): void\n  yjsOrigin: symbol\n} {\n  const yjsOrigin = Symbol(\"bindYjsToMobxKeystoneTransactionOrigin\")\n\n  const bindingContext: YjsBindingContext = {\n    yjsDoc,\n    yjsObject,\n    mobxKeystoneType,\n    yjsOrigin,\n  }\n\n  const yjsJson = yjsObject.toJSON()\n\n  const initializationGlobalPatches: { target: object; patches: Patch[] }[] = []\n\n  const createBoundObject = () => {\n    const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {\n      initializationGlobalPatches.push({ target, patches })\n    })\n\n    try {\n      const boundObject = yjsBindingContext.apply(\n        () => fromSnapshot(mobxKeystoneType, yjsJson as any),\n        bindingContext\n      )\n      yjsBindingContext.set(boundObject, bindingContext)\n      return boundObject\n    } finally {\n      disposeOnGlobalPatches()\n    }\n  }\n\n  const boundObject = createBoundObject()\n\n  let applyingMobxKeystoneChanges = 0\n\n  // bind any changes from yjs to mobx-keystone\n  const observeDeepCb = (events: Y.YEvent<any>[]) => {\n    const patches: Patch[] = []\n    events.forEach((event) => {\n      if (event.transaction.origin !== yjsOrigin) {\n        patches.push(...convertYjsEventToPatches(event))\n      }\n    })\n\n    if (patches.length > 0) {\n      applyingMobxKeystoneChanges++\n      try {\n        applyPatches(boundObject, patches)\n      } finally {\n        applyingMobxKeystoneChanges--\n      }\n    }\n  }\n\n  yjsObject.observeDeep(observeDeepCb)\n\n  // bind any changes from mobx-keystone to yjs\n  let pendingPatches: Patch[] = []\n  const disposeOnPatches = onPatches(boundObject, (patches) => {\n    if (applyingMobxKeystoneChanges > 0) {\n      return\n    }\n\n    pendingPatches.push(...patches)\n  })\n\n  // this is only used so we can transact all patches to the snapshot boundary\n  const disposeOnSnapshot = onSnapshot(boundObject, () => {\n    if (pendingPatches.length === 0) {\n      return\n    }\n\n    const patches = pendingPatches\n    pendingPatches = []\n\n    yjsDoc.transact(() => {\n      patches.forEach((patch) => {\n        applyMobxKeystonePatchToYjsObject(patch, yjsObject)\n      })\n    }, yjsOrigin)\n  })\n\n  // sync initial patches, that might include setting defaults, IDs, etc\n  yjsDoc.transact(() => {\n    // we need to skip initializations until we hit the initialization of the bound object\n    // this is because default objects might be created and initialized before the main object\n    // but we just need to catch when those are actually assigned to the bound object\n    let boundObjectFound = false\n\n    initializationGlobalPatches.forEach(({ target, patches }) => {\n      if (!boundObjectFound) {\n        if (target !== boundObject) {\n          return // skip\n        }\n        boundObjectFound = true\n      }\n\n      const parentToChildPath = getParentToChildPath(boundObject, target)\n      // this is undefined only if target is not a child of boundModel\n      if (parentToChildPath !== undefined) {\n        patches.forEach((patch) => {\n          applyMobxKeystonePatchToYjsObject(\n            {\n              ...patch,\n              path: [...parentToChildPath, ...patch.path],\n            },\n            yjsObject\n          )\n        })\n      }\n    })\n  }, yjsOrigin)\n\n  return {\n    boundObject,\n    dispose: () => {\n      disposeOnPatches()\n      disposeOnSnapshot()\n      yjsObject.unobserveDeep(observeDeepCb)\n    },\n    yjsOrigin,\n  }\n}\n"],"names":["boundObject"],"mappings":"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"}
@@ -1,4 +1,4 @@
1
- import { onPatches, onSnapshot, getParentToChildPath, onGlobalPatches, fromSnapshot, applyPatches } from "mobx-keystone";
1
+ import { createContext, onPatches, onSnapshot, getParentToChildPath, onGlobalPatches, fromSnapshot, applyPatches } from "mobx-keystone";
2
2
  import * as Y from "yjs";
3
3
  class MobxKeystoneYjsError extends Error {
4
4
  constructor(msg) {
@@ -184,7 +184,15 @@ function toPlainValue(v) {
184
184
  return v;
185
185
  }
186
186
  }
187
+ const yjsBindingContext = createContext(void 0);
187
188
  function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
189
+ const yjsOrigin = Symbol("bindYjsToMobxKeystoneTransactionOrigin");
190
+ const bindingContext = {
191
+ yjsDoc,
192
+ yjsObject,
193
+ mobxKeystoneType,
194
+ yjsOrigin
195
+ };
188
196
  const yjsJson = yjsObject.toJSON();
189
197
  const initializationGlobalPatches = [];
190
198
  const createBoundObject = () => {
@@ -192,14 +200,15 @@ function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
192
200
  initializationGlobalPatches.push({ target, patches });
193
201
  });
194
202
  try {
195
- return fromSnapshot(mobxKeystoneType, yjsJson);
203
+ const boundObject2 = yjsBindingContext.apply(() => fromSnapshot(mobxKeystoneType, yjsJson), bindingContext);
204
+ yjsBindingContext.set(boundObject2, bindingContext);
205
+ return boundObject2;
196
206
  } finally {
197
207
  disposeOnGlobalPatches();
198
208
  }
199
209
  };
200
210
  const boundObject = createBoundObject();
201
211
  let applyingMobxKeystoneChanges = 0;
202
- const yjsOrigin = Symbol("bindYjsToMobxKeystoneTransactionOrigin");
203
212
  const observeDeepCb = (events) => {
204
213
  const patches = [];
205
214
  events.forEach((event) => {
@@ -271,6 +280,7 @@ export {
271
280
  applyJsonArrayYArray,
272
281
  applyJsonObjectToYMap,
273
282
  bindYjsToMobxKeystone,
274
- convertJsonToYjsData
283
+ convertJsonToYjsData,
284
+ yjsBindingContext
275
285
  };
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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"}
286
+ //# 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/yjsBindingContext.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 { AnyType, createContext } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\n\nexport interface YjsBindingContext {\n  yjsDoc: Y.Doc\n  yjsObject: Y.Map<unknown> | Y.Array<unknown>\n  mobxKeystoneType: AnyType\n  yjsOrigin: symbol\n}\n\nexport const yjsBindingContext = createContext<YjsBindingContext | undefined>(undefined)\n","import {\n  AnyDataModel,\n  AnyModel,\n  AnyStandardType,\n  ModelClass,\n  Patch,\n  TypeToData,\n  applyPatches,\n  fromSnapshot,\n  getParentToChildPath,\n  onGlobalPatches,\n  onPatches,\n  onSnapshot,\n} from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { applyMobxKeystonePatchToYjsObject } from \"./applyMobxKeystonePatchToYjsObject\"\nimport { convertYjsEventToPatches } from \"./convertYjsEventToPatches\"\nimport { YjsBindingContext, yjsBindingContext } from \"./yjsBindingContext\"\n\nexport function bindYjsToMobxKeystone<\n  TType extends AnyStandardType | ModelClass<AnyModel> | ModelClass<AnyDataModel>,\n>({\n  yjsDoc,\n  yjsObject,\n  mobxKeystoneType,\n}: {\n  yjsDoc: Y.Doc\n  yjsObject: Y.Map<unknown> | Y.Array<unknown>\n  mobxKeystoneType: TType\n}): {\n  boundObject: TypeToData<TType>\n  dispose(): void\n  yjsOrigin: symbol\n} {\n  const yjsOrigin = Symbol(\"bindYjsToMobxKeystoneTransactionOrigin\")\n\n  const bindingContext: YjsBindingContext = {\n    yjsDoc,\n    yjsObject,\n    mobxKeystoneType,\n    yjsOrigin,\n  }\n\n  const yjsJson = yjsObject.toJSON()\n\n  const initializationGlobalPatches: { target: object; patches: Patch[] }[] = []\n\n  const createBoundObject = () => {\n    const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {\n      initializationGlobalPatches.push({ target, patches })\n    })\n\n    try {\n      const boundObject = yjsBindingContext.apply(\n        () => fromSnapshot(mobxKeystoneType, yjsJson as any),\n        bindingContext\n      )\n      yjsBindingContext.set(boundObject, bindingContext)\n      return boundObject\n    } finally {\n      disposeOnGlobalPatches()\n    }\n  }\n\n  const boundObject = createBoundObject()\n\n  let applyingMobxKeystoneChanges = 0\n\n  // bind any changes from yjs to mobx-keystone\n  const observeDeepCb = (events: Y.YEvent<any>[]) => {\n    const patches: Patch[] = []\n    events.forEach((event) => {\n      if (event.transaction.origin !== yjsOrigin) {\n        patches.push(...convertYjsEventToPatches(event))\n      }\n    })\n\n    if (patches.length > 0) {\n      applyingMobxKeystoneChanges++\n      try {\n        applyPatches(boundObject, patches)\n      } finally {\n        applyingMobxKeystoneChanges--\n      }\n    }\n  }\n\n  yjsObject.observeDeep(observeDeepCb)\n\n  // bind any changes from mobx-keystone to yjs\n  let pendingPatches: Patch[] = []\n  const disposeOnPatches = onPatches(boundObject, (patches) => {\n    if (applyingMobxKeystoneChanges > 0) {\n      return\n    }\n\n    pendingPatches.push(...patches)\n  })\n\n  // this is only used so we can transact all patches to the snapshot boundary\n  const disposeOnSnapshot = onSnapshot(boundObject, () => {\n    if (pendingPatches.length === 0) {\n      return\n    }\n\n    const patches = pendingPatches\n    pendingPatches = []\n\n    yjsDoc.transact(() => {\n      patches.forEach((patch) => {\n        applyMobxKeystonePatchToYjsObject(patch, yjsObject)\n      })\n    }, yjsOrigin)\n  })\n\n  // sync initial patches, that might include setting defaults, IDs, etc\n  yjsDoc.transact(() => {\n    // we need to skip initializations until we hit the initialization of the bound object\n    // this is because default objects might be created and initialized before the main object\n    // but we just need to catch when those are actually assigned to the bound object\n    let boundObjectFound = false\n\n    initializationGlobalPatches.forEach(({ target, patches }) => {\n      if (!boundObjectFound) {\n        if (target !== boundObject) {\n          return // skip\n        }\n        boundObjectFound = true\n      }\n\n      const parentToChildPath = getParentToChildPath(boundObject, target)\n      // this is undefined only if target is not a child of boundModel\n      if (parentToChildPath !== undefined) {\n        patches.forEach((patch) => {\n          applyMobxKeystonePatchToYjsObject(\n            {\n              ...patch,\n              path: [...parentToChildPath, ...patch.path],\n            },\n            yjsObject\n          )\n        })\n      }\n    })\n  }, yjsOrigin)\n\n  return {\n    boundObject,\n    dispose: () => {\n      disposeOnPatches()\n      disposeOnSnapshot()\n      yjsObject.unobserveDeep(observeDeepCb)\n    },\n    yjsOrigin,\n  }\n}\n"],"names":["boundObject"],"mappings":"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"}
@@ -203,7 +203,15 @@
203
203
  return v;
204
204
  }
205
205
  }
206
+ const yjsBindingContext = mobxKeystone.createContext(void 0);
206
207
  function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
208
+ const yjsOrigin = Symbol("bindYjsToMobxKeystoneTransactionOrigin");
209
+ const bindingContext = {
210
+ yjsDoc,
211
+ yjsObject,
212
+ mobxKeystoneType,
213
+ yjsOrigin
214
+ };
207
215
  const yjsJson = yjsObject.toJSON();
208
216
  const initializationGlobalPatches = [];
209
217
  const createBoundObject = () => {
@@ -211,14 +219,15 @@
211
219
  initializationGlobalPatches.push({ target, patches });
212
220
  });
213
221
  try {
214
- return mobxKeystone.fromSnapshot(mobxKeystoneType, yjsJson);
222
+ const boundObject2 = yjsBindingContext.apply(() => mobxKeystone.fromSnapshot(mobxKeystoneType, yjsJson), bindingContext);
223
+ yjsBindingContext.set(boundObject2, bindingContext);
224
+ return boundObject2;
215
225
  } finally {
216
226
  disposeOnGlobalPatches();
217
227
  }
218
228
  };
219
229
  const boundObject = createBoundObject();
220
230
  let applyingMobxKeystoneChanges = 0;
221
- const yjsOrigin = Symbol("bindYjsToMobxKeystoneTransactionOrigin");
222
231
  const observeDeepCb = (events) => {
223
232
  const patches = [];
224
233
  events.forEach((event) => {
@@ -290,6 +299,7 @@
290
299
  exports2.applyJsonObjectToYMap = applyJsonObjectToYMap;
291
300
  exports2.bindYjsToMobxKeystone = bindYjsToMobxKeystone;
292
301
  exports2.convertJsonToYjsData = convertJsonToYjsData;
302
+ exports2.yjsBindingContext = yjsBindingContext;
293
303
  Object.defineProperty(exports2, Symbol.toStringTag, { value: "Module" });
294
304
  });
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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"}
305
+ //# 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/yjsBindingContext.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 { AnyType, createContext } from \"mobx-keystone\"\nimport * as Y from \"yjs\"\n\nexport interface YjsBindingContext {\n  yjsDoc: Y.Doc\n  yjsObject: Y.Map<unknown> | Y.Array<unknown>\n  mobxKeystoneType: AnyType\n  yjsOrigin: symbol\n}\n\nexport const yjsBindingContext = createContext<YjsBindingContext | undefined>(undefined)\n","import {\n  AnyDataModel,\n  AnyModel,\n  AnyStandardType,\n  ModelClass,\n  Patch,\n  TypeToData,\n  applyPatches,\n  fromSnapshot,\n  getParentToChildPath,\n  onGlobalPatches,\n  onPatches,\n  onSnapshot,\n} from \"mobx-keystone\"\nimport * as Y from \"yjs\"\nimport { applyMobxKeystonePatchToYjsObject } from \"./applyMobxKeystonePatchToYjsObject\"\nimport { convertYjsEventToPatches } from \"./convertYjsEventToPatches\"\nimport { YjsBindingContext, yjsBindingContext } from \"./yjsBindingContext\"\n\nexport function bindYjsToMobxKeystone<\n  TType extends AnyStandardType | ModelClass<AnyModel> | ModelClass<AnyDataModel>,\n>({\n  yjsDoc,\n  yjsObject,\n  mobxKeystoneType,\n}: {\n  yjsDoc: Y.Doc\n  yjsObject: Y.Map<unknown> | Y.Array<unknown>\n  mobxKeystoneType: TType\n}): {\n  boundObject: TypeToData<TType>\n  dispose(): void\n  yjsOrigin: symbol\n} {\n  const yjsOrigin = Symbol(\"bindYjsToMobxKeystoneTransactionOrigin\")\n\n  const bindingContext: YjsBindingContext = {\n    yjsDoc,\n    yjsObject,\n    mobxKeystoneType,\n    yjsOrigin,\n  }\n\n  const yjsJson = yjsObject.toJSON()\n\n  const initializationGlobalPatches: { target: object; patches: Patch[] }[] = []\n\n  const createBoundObject = () => {\n    const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {\n      initializationGlobalPatches.push({ target, patches })\n    })\n\n    try {\n      const boundObject = yjsBindingContext.apply(\n        () => fromSnapshot(mobxKeystoneType, yjsJson as any),\n        bindingContext\n      )\n      yjsBindingContext.set(boundObject, bindingContext)\n      return boundObject\n    } finally {\n      disposeOnGlobalPatches()\n    }\n  }\n\n  const boundObject = createBoundObject()\n\n  let applyingMobxKeystoneChanges = 0\n\n  // bind any changes from yjs to mobx-keystone\n  const observeDeepCb = (events: Y.YEvent<any>[]) => {\n    const patches: Patch[] = []\n    events.forEach((event) => {\n      if (event.transaction.origin !== yjsOrigin) {\n        patches.push(...convertYjsEventToPatches(event))\n      }\n    })\n\n    if (patches.length > 0) {\n      applyingMobxKeystoneChanges++\n      try {\n        applyPatches(boundObject, patches)\n      } finally {\n        applyingMobxKeystoneChanges--\n      }\n    }\n  }\n\n  yjsObject.observeDeep(observeDeepCb)\n\n  // bind any changes from mobx-keystone to yjs\n  let pendingPatches: Patch[] = []\n  const disposeOnPatches = onPatches(boundObject, (patches) => {\n    if (applyingMobxKeystoneChanges > 0) {\n      return\n    }\n\n    pendingPatches.push(...patches)\n  })\n\n  // this is only used so we can transact all patches to the snapshot boundary\n  const disposeOnSnapshot = onSnapshot(boundObject, () => {\n    if (pendingPatches.length === 0) {\n      return\n    }\n\n    const patches = pendingPatches\n    pendingPatches = []\n\n    yjsDoc.transact(() => {\n      patches.forEach((patch) => {\n        applyMobxKeystonePatchToYjsObject(patch, yjsObject)\n      })\n    }, yjsOrigin)\n  })\n\n  // sync initial patches, that might include setting defaults, IDs, etc\n  yjsDoc.transact(() => {\n    // we need to skip initializations until we hit the initialization of the bound object\n    // this is because default objects might be created and initialized before the main object\n    // but we just need to catch when those are actually assigned to the bound object\n    let boundObjectFound = false\n\n    initializationGlobalPatches.forEach(({ target, patches }) => {\n      if (!boundObjectFound) {\n        if (target !== boundObject) {\n          return // skip\n        }\n        boundObjectFound = true\n      }\n\n      const parentToChildPath = getParentToChildPath(boundObject, target)\n      // this is undefined only if target is not a child of boundModel\n      if (parentToChildPath !== undefined) {\n        patches.forEach((patch) => {\n          applyMobxKeystonePatchToYjsObject(\n            {\n              ...patch,\n              path: [...parentToChildPath, ...patch.path],\n            },\n            yjsObject\n          )\n        })\n      }\n    })\n  }, yjsOrigin)\n\n  return {\n    boundObject,\n    dispose: () => {\n      disposeOnPatches()\n      disposeOnSnapshot()\n      yjsObject.unobserveDeep(observeDeepCb)\n    },\n    yjsOrigin,\n  }\n}\n"],"names":["Y","createContext","onGlobalPatches","boundObject","fromSnapshot","applyPatches","onPatches","onSnapshot","getParentToChildPath"],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ import { AnyType } from "mobx-keystone";
2
+ import * as Y from "yjs";
3
+ export interface YjsBindingContext {
4
+ yjsDoc: Y.Doc;
5
+ yjsObject: Y.Map<unknown> | Y.Array<unknown>;
6
+ mobxKeystoneType: AnyType;
7
+ yjsOrigin: symbol;
8
+ }
9
+ export declare const yjsBindingContext: import("mobx-keystone").Context<YjsBindingContext | undefined>;
@@ -1,3 +1,5 @@
1
- export { MobxKeystoneYjsError } from "./utils/error";
2
1
  export { bindYjsToMobxKeystone } from "./binding/bindYjsToMobxKeystone";
3
- export { convertJsonToYjsData, applyJsonArrayYArray, applyJsonObjectToYMap, } from "./binding/convertJsonToYjsData";
2
+ export { applyJsonArrayYArray, applyJsonObjectToYMap, convertJsonToYjsData, } from "./binding/convertJsonToYjsData";
3
+ export { yjsBindingContext } from "./binding/yjsBindingContext";
4
+ export type { YjsBindingContext } from "./binding/yjsBindingContext";
5
+ export { MobxKeystoneYjsError } from "./utils/error";
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "mobx-keystone-yjs",
3
- "version": "1.1.0",
3
+ "version": "1.2.0",
4
4
  "description": "Yjs bindings for mobx-keystone",
5
5
  "keywords": [
6
6
  "mobx",
@@ -1,142 +1,156 @@
1
- import {
2
- AnyDataModel,
3
- AnyModel,
4
- AnyStandardType,
5
- ModelClass,
6
- Patch,
7
- TypeToData,
8
- applyPatches,
9
- fromSnapshot,
10
- getParentToChildPath,
11
- onGlobalPatches,
12
- onPatches,
13
- onSnapshot,
14
- } from "mobx-keystone"
15
- import * as Y from "yjs"
16
- import { applyMobxKeystonePatchToYjsObject } from "./applyMobxKeystonePatchToYjsObject"
17
- import { convertYjsEventToPatches } from "./convertYjsEventToPatches"
18
-
19
- export function bindYjsToMobxKeystone<
20
- TType extends AnyStandardType | ModelClass<AnyModel> | ModelClass<AnyDataModel>,
21
- >({
22
- yjsDoc,
23
- yjsObject,
24
- mobxKeystoneType,
25
- }: {
26
- yjsDoc: Y.Doc
27
- yjsObject: Y.Map<unknown> | Y.Array<unknown>
28
- mobxKeystoneType: TType
29
- }): {
30
- boundObject: TypeToData<TType>
31
- dispose(): void
32
- yjsOrigin: symbol
33
- } {
34
- const yjsJson = yjsObject.toJSON()
35
-
36
- const initializationGlobalPatches: { target: object; patches: Patch[] }[] = []
37
-
38
- const createBoundObject = () => {
39
- const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {
40
- initializationGlobalPatches.push({ target, patches })
41
- })
42
-
43
- try {
44
- return fromSnapshot(mobxKeystoneType, yjsJson as any)
45
- } finally {
46
- disposeOnGlobalPatches()
47
- }
48
- }
49
-
50
- const boundObject = createBoundObject()
51
-
52
- let applyingMobxKeystoneChanges = 0
53
- const yjsOrigin = Symbol("bindYjsToMobxKeystoneTransactionOrigin")
54
-
55
- // bind any changes from yjs to mobx-keystone
56
- const observeDeepCb = (events: Y.YEvent<any>[]) => {
57
- const patches: Patch[] = []
58
- events.forEach((event) => {
59
- if (event.transaction.origin !== yjsOrigin) {
60
- patches.push(...convertYjsEventToPatches(event))
61
- }
62
- })
63
-
64
- if (patches.length > 0) {
65
- applyingMobxKeystoneChanges++
66
- try {
67
- applyPatches(boundObject, patches)
68
- } finally {
69
- applyingMobxKeystoneChanges--
70
- }
71
- }
72
- }
73
-
74
- yjsObject.observeDeep(observeDeepCb)
75
-
76
- // bind any changes from mobx-keystone to yjs
77
- let pendingPatches: Patch[] = []
78
- const disposeOnPatches = onPatches(boundObject, (patches) => {
79
- if (applyingMobxKeystoneChanges > 0) {
80
- return
81
- }
82
-
83
- pendingPatches.push(...patches)
84
- })
85
-
86
- // this is only used so we can transact all patches to the snapshot boundary
87
- const disposeOnSnapshot = onSnapshot(boundObject, () => {
88
- if (pendingPatches.length === 0) {
89
- return
90
- }
91
-
92
- const patches = pendingPatches
93
- pendingPatches = []
94
-
95
- yjsDoc.transact(() => {
96
- patches.forEach((patch) => {
97
- applyMobxKeystonePatchToYjsObject(patch, yjsObject)
98
- })
99
- }, yjsOrigin)
100
- })
101
-
102
- // sync initial patches, that might include setting defaults, IDs, etc
103
- yjsDoc.transact(() => {
104
- // we need to skip initializations until we hit the initialization of the bound object
105
- // this is because default objects might be created and initialized before the main object
106
- // but we just need to catch when those are actually assigned to the bound object
107
- let boundObjectFound = false
108
-
109
- initializationGlobalPatches.forEach(({ target, patches }) => {
110
- if (!boundObjectFound) {
111
- if (target !== boundObject) {
112
- return // skip
113
- }
114
- boundObjectFound = true
115
- }
116
-
117
- const parentToChildPath = getParentToChildPath(boundObject, target)
118
- // this is undefined only if target is not a child of boundModel
119
- if (parentToChildPath !== undefined) {
120
- patches.forEach((patch) => {
121
- applyMobxKeystonePatchToYjsObject(
122
- {
123
- ...patch,
124
- path: [...parentToChildPath, ...patch.path],
125
- },
126
- yjsObject
127
- )
128
- })
129
- }
130
- })
131
- }, yjsOrigin)
132
-
133
- return {
134
- boundObject,
135
- dispose: () => {
136
- disposeOnPatches()
137
- disposeOnSnapshot()
138
- yjsObject.unobserveDeep(observeDeepCb)
139
- },
140
- yjsOrigin,
141
- }
142
- }
1
+ import {
2
+ AnyDataModel,
3
+ AnyModel,
4
+ AnyStandardType,
5
+ ModelClass,
6
+ Patch,
7
+ TypeToData,
8
+ applyPatches,
9
+ fromSnapshot,
10
+ getParentToChildPath,
11
+ onGlobalPatches,
12
+ onPatches,
13
+ onSnapshot,
14
+ } from "mobx-keystone"
15
+ import * as Y from "yjs"
16
+ import { applyMobxKeystonePatchToYjsObject } from "./applyMobxKeystonePatchToYjsObject"
17
+ import { convertYjsEventToPatches } from "./convertYjsEventToPatches"
18
+ import { YjsBindingContext, yjsBindingContext } from "./yjsBindingContext"
19
+
20
+ export function bindYjsToMobxKeystone<
21
+ TType extends AnyStandardType | ModelClass<AnyModel> | ModelClass<AnyDataModel>,
22
+ >({
23
+ yjsDoc,
24
+ yjsObject,
25
+ mobxKeystoneType,
26
+ }: {
27
+ yjsDoc: Y.Doc
28
+ yjsObject: Y.Map<unknown> | Y.Array<unknown>
29
+ mobxKeystoneType: TType
30
+ }): {
31
+ boundObject: TypeToData<TType>
32
+ dispose(): void
33
+ yjsOrigin: symbol
34
+ } {
35
+ const yjsOrigin = Symbol("bindYjsToMobxKeystoneTransactionOrigin")
36
+
37
+ const bindingContext: YjsBindingContext = {
38
+ yjsDoc,
39
+ yjsObject,
40
+ mobxKeystoneType,
41
+ yjsOrigin,
42
+ }
43
+
44
+ const yjsJson = yjsObject.toJSON()
45
+
46
+ const initializationGlobalPatches: { target: object; patches: Patch[] }[] = []
47
+
48
+ const createBoundObject = () => {
49
+ const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {
50
+ initializationGlobalPatches.push({ target, patches })
51
+ })
52
+
53
+ try {
54
+ const boundObject = yjsBindingContext.apply(
55
+ () => fromSnapshot(mobxKeystoneType, yjsJson as any),
56
+ bindingContext
57
+ )
58
+ yjsBindingContext.set(boundObject, bindingContext)
59
+ return boundObject
60
+ } finally {
61
+ disposeOnGlobalPatches()
62
+ }
63
+ }
64
+
65
+ const boundObject = createBoundObject()
66
+
67
+ let applyingMobxKeystoneChanges = 0
68
+
69
+ // bind any changes from yjs to mobx-keystone
70
+ const observeDeepCb = (events: Y.YEvent<any>[]) => {
71
+ const patches: Patch[] = []
72
+ events.forEach((event) => {
73
+ if (event.transaction.origin !== yjsOrigin) {
74
+ patches.push(...convertYjsEventToPatches(event))
75
+ }
76
+ })
77
+
78
+ if (patches.length > 0) {
79
+ applyingMobxKeystoneChanges++
80
+ try {
81
+ applyPatches(boundObject, patches)
82
+ } finally {
83
+ applyingMobxKeystoneChanges--
84
+ }
85
+ }
86
+ }
87
+
88
+ yjsObject.observeDeep(observeDeepCb)
89
+
90
+ // bind any changes from mobx-keystone to yjs
91
+ let pendingPatches: Patch[] = []
92
+ const disposeOnPatches = onPatches(boundObject, (patches) => {
93
+ if (applyingMobxKeystoneChanges > 0) {
94
+ return
95
+ }
96
+
97
+ pendingPatches.push(...patches)
98
+ })
99
+
100
+ // this is only used so we can transact all patches to the snapshot boundary
101
+ const disposeOnSnapshot = onSnapshot(boundObject, () => {
102
+ if (pendingPatches.length === 0) {
103
+ return
104
+ }
105
+
106
+ const patches = pendingPatches
107
+ pendingPatches = []
108
+
109
+ yjsDoc.transact(() => {
110
+ patches.forEach((patch) => {
111
+ applyMobxKeystonePatchToYjsObject(patch, yjsObject)
112
+ })
113
+ }, yjsOrigin)
114
+ })
115
+
116
+ // sync initial patches, that might include setting defaults, IDs, etc
117
+ yjsDoc.transact(() => {
118
+ // we need to skip initializations until we hit the initialization of the bound object
119
+ // this is because default objects might be created and initialized before the main object
120
+ // but we just need to catch when those are actually assigned to the bound object
121
+ let boundObjectFound = false
122
+
123
+ initializationGlobalPatches.forEach(({ target, patches }) => {
124
+ if (!boundObjectFound) {
125
+ if (target !== boundObject) {
126
+ return // skip
127
+ }
128
+ boundObjectFound = true
129
+ }
130
+
131
+ const parentToChildPath = getParentToChildPath(boundObject, target)
132
+ // this is undefined only if target is not a child of boundModel
133
+ if (parentToChildPath !== undefined) {
134
+ patches.forEach((patch) => {
135
+ applyMobxKeystonePatchToYjsObject(
136
+ {
137
+ ...patch,
138
+ path: [...parentToChildPath, ...patch.path],
139
+ },
140
+ yjsObject
141
+ )
142
+ })
143
+ }
144
+ })
145
+ }, yjsOrigin)
146
+
147
+ return {
148
+ boundObject,
149
+ dispose: () => {
150
+ disposeOnPatches()
151
+ disposeOnSnapshot()
152
+ yjsObject.unobserveDeep(observeDeepCb)
153
+ },
154
+ yjsOrigin,
155
+ }
156
+ }
@@ -0,0 +1,11 @@
1
+ import { AnyType, createContext } from "mobx-keystone"
2
+ import * as Y from "yjs"
3
+
4
+ export interface YjsBindingContext {
5
+ yjsDoc: Y.Doc
6
+ yjsObject: Y.Map<unknown> | Y.Array<unknown>
7
+ mobxKeystoneType: AnyType
8
+ yjsOrigin: symbol
9
+ }
10
+
11
+ export const yjsBindingContext = createContext<YjsBindingContext | undefined>(undefined)
package/src/index.ts CHANGED
@@ -1,7 +1,9 @@
1
- export { MobxKeystoneYjsError } from "./utils/error"
2
1
  export { bindYjsToMobxKeystone } from "./binding/bindYjsToMobxKeystone"
3
2
  export {
4
- convertJsonToYjsData,
5
3
  applyJsonArrayYArray,
6
4
  applyJsonObjectToYMap,
5
+ convertJsonToYjsData,
7
6
  } from "./binding/convertJsonToYjsData"
7
+ export { yjsBindingContext } from "./binding/yjsBindingContext"
8
+ export type { YjsBindingContext } from "./binding/yjsBindingContext"
9
+ export { MobxKeystoneYjsError } from "./utils/error"