UNPKG

json-joy

Version:

Collection of libraries for building collaborative editing apps.

122 lines (121 loc) 3.54 kB
import { AbstractOp } from './AbstractOp'; import { find, isObjectReference, isArrayReference, formatJsonPointer } from '@jsonjoy.com/json-pointer'; import { isTextNode, isElementNode } from '../util'; import { OPCODE } from '../constants'; /** * @category JSON Patch Extended */ export class OpSplit extends AbstractOp { pos; props; constructor(path, pos, props) { super(path); this.pos = pos; this.props = props; } op() { return 'split'; } code() { return OPCODE.split; } apply(doc) { const ref = find(doc, this.path); if (ref.val === undefined) throw new Error('NOT_FOUND'); const tuple = this.split(ref.val); if (isObjectReference(ref)) ref.obj[ref.key] = tuple; else if (isArrayReference(ref)) { ref.obj[ref.key] = tuple[0]; ref.obj.splice(ref.key + 1, 0, tuple[1]); } else doc = tuple; return { doc, old: ref.val }; } split(node) { if (typeof node === 'string') { const { pos, props } = this; const before = node.slice(0, pos); const after = node.slice(pos); if (!props) return [before, after]; const textNodes = [ { ...props, text: before, }, { ...props, text: after, }, ]; return textNodes; } else if (isTextNode(node)) { const { pos, props } = this; const before = node.text.slice(0, pos); const after = node.text.slice(pos); const textNodes = [ { ...node, ...props, text: before, }, { ...node, ...props, text: after, }, ]; return textNodes; } else if (isElementNode(node)) { const { pos, props } = this; const before = node.children.slice(0, pos); const after = node.children.slice(pos); const elementNodes = [ { ...node, ...props, children: before, }, { ...node, ...props, children: after, }, ]; return elementNodes; } else if (typeof node === 'number') { const { pos } = this; return [pos, node - pos]; } else return [node, node]; } toJson(parent) { const op = { op: 'split', path: formatJsonPointer(this.path), pos: this.pos, }; if (this.props) op.props = this.props; return op; } toCompact(parent, verbose) { const opcode = verbose ? 'split' : OPCODE.split; return this.props ? [opcode, this.path, this.pos, this.props] : [opcode, this.path, this.pos]; } encode(encoder, parent) { encoder.encodeArrayHeader(this.props ? 4 : 3); encoder.writer.u8(OPCODE.split); encoder.encodeArray(this.path); encoder.encodeNumber(this.pos); if (this.props) encoder.encodeObject(this.props); } }