mirror of
https://github.com/hcengineering/platform.git
synced 2025-03-22 23:58:27 +00:00
171 lines
4.7 KiB
TypeScript
171 lines
4.7 KiB
TypeScript
//
|
|
// Copyright © 2023 Hardcore Engineering Inc.
|
|
//
|
|
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License. You may
|
|
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
//
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
//
|
|
|
|
import { generateId, Markup } from '@hcengineering/core'
|
|
import { Extensions, getSchema } from '@tiptap/core'
|
|
import { Node, Schema } from '@tiptap/pm/model'
|
|
import { prosemirrorJSONToYDoc, prosemirrorToYDoc, yDocToProsemirrorJSON } from 'y-prosemirror'
|
|
import {
|
|
Doc as YDoc,
|
|
XmlElement as YXmlElement,
|
|
XmlFragment as YXmlFragment,
|
|
XmlText as YXmlText,
|
|
applyUpdate,
|
|
encodeStateAsUpdate
|
|
} from 'yjs'
|
|
import { defaultExtensions, jsonToMarkup, markupToJSON, markupToPmNode, type MarkupNode } from '@hcengineering/text'
|
|
|
|
const defaultSchema = getSchema(defaultExtensions)
|
|
|
|
/**
|
|
* @public
|
|
*/
|
|
export function markupToYDoc (markup: Markup, field: string, schema?: Schema, extensions?: Extensions): YDoc {
|
|
const node = markupToPmNode(markup, schema, extensions)
|
|
return prosemirrorToYDoc(node, field)
|
|
}
|
|
|
|
/**
|
|
* Convert markup to Y.Doc without using ProseMirror schema
|
|
*
|
|
* @public
|
|
*/
|
|
export function markupToYDocNoSchema (markup: Markup, field: string): YDoc {
|
|
return jsonToYDocNoSchema(markupToJSON(markup), field)
|
|
}
|
|
|
|
/**
|
|
* Convert ProseMirror JSON to Y.Doc without using ProseMirror schema
|
|
*
|
|
* @public
|
|
*/
|
|
export function jsonToYDocNoSchema (json: MarkupNode, field: string): YDoc {
|
|
const ydoc = new YDoc({ guid: generateId() })
|
|
const fragment = ydoc.getXmlFragment(field)
|
|
|
|
const nodes = json.type === 'doc' ? json.content ?? [] : [json]
|
|
nodes.map((p) => nodeToYXmlElement(fragment, p))
|
|
|
|
return ydoc
|
|
}
|
|
|
|
/**
|
|
* Convert ProseMirror JSON Node representation to YXmlElement
|
|
* */
|
|
function nodeToYXmlElement (parent: YXmlFragment, node: MarkupNode): YXmlElement | YXmlText {
|
|
const elem = node.type === 'text' ? new YXmlText() : new YXmlElement(node.type)
|
|
parent.push([elem])
|
|
|
|
if (elem instanceof YXmlElement) {
|
|
if (node.content !== undefined && node.content.length > 0) {
|
|
node.content.map((p) => nodeToYXmlElement(elem, p))
|
|
}
|
|
} else {
|
|
// https://github.com/yjs/y-prosemirror/blob/master/src/plugins/sync-plugin.js#L777
|
|
const attributes: Record<string, any> = {}
|
|
if (node.marks !== undefined) {
|
|
node.marks.forEach((mark) => {
|
|
attributes[mark.type] = mark.attrs ?? {}
|
|
})
|
|
}
|
|
elem.applyDelta([
|
|
{
|
|
insert: node.text ?? '',
|
|
attributes
|
|
}
|
|
])
|
|
}
|
|
|
|
if (node.attrs !== undefined) {
|
|
Object.entries(node.attrs).forEach(([key, value]) => {
|
|
elem.setAttribute(key, value)
|
|
})
|
|
}
|
|
|
|
return elem
|
|
}
|
|
|
|
/**
|
|
* @public
|
|
*/
|
|
export function yDocToMarkup (ydoc: YDoc, field: string): Markup {
|
|
const json = yDocToProsemirrorJSON(ydoc, field)
|
|
return jsonToMarkup(json as MarkupNode)
|
|
}
|
|
|
|
/**
|
|
* Get ProseMirror nodes from Y.Doc content
|
|
*
|
|
* @public
|
|
*/
|
|
export function yDocContentToNodes (content: ArrayBuffer, schema?: Schema, extensions?: Extensions): Node[] {
|
|
schema ??= extensions === undefined ? defaultSchema : getSchema(extensions ?? defaultExtensions)
|
|
|
|
const nodes: Node[] = []
|
|
|
|
try {
|
|
const ydoc = new YDoc({
|
|
gc: false,
|
|
guid: generateId()
|
|
})
|
|
const uint8arr = new Uint8Array(content)
|
|
applyUpdate(ydoc, uint8arr)
|
|
|
|
for (const field of ydoc.share.keys()) {
|
|
try {
|
|
const body = yDocToProsemirrorJSON(ydoc, field)
|
|
nodes.push(schema.nodeFromJSON(body))
|
|
} catch {}
|
|
}
|
|
} catch (err: any) {
|
|
console.error(err)
|
|
}
|
|
|
|
return nodes
|
|
}
|
|
|
|
/**
|
|
* Update Y.Doc content
|
|
*
|
|
* @public
|
|
*/
|
|
export function updateYDocContent (
|
|
content: ArrayBuffer,
|
|
updateFn: (body: Record<string, any>) => Record<string, any>,
|
|
schema?: Schema,
|
|
extensions?: Extensions
|
|
): YDoc | undefined {
|
|
schema ??= extensions === undefined ? defaultSchema : getSchema(extensions ?? defaultExtensions)
|
|
|
|
try {
|
|
const ydoc = new YDoc({ guid: generateId(), gc: false })
|
|
const res = new YDoc({ guid: generateId(), gc: false })
|
|
const uint8arr = new Uint8Array(content)
|
|
applyUpdate(ydoc, uint8arr)
|
|
|
|
for (const field of ydoc.share.keys()) {
|
|
const body = yDocToProsemirrorJSON(ydoc, field)
|
|
const updated = updateFn(body)
|
|
const yDoc = prosemirrorJSONToYDoc(schema, updated, field)
|
|
const update = encodeStateAsUpdate(yDoc)
|
|
applyUpdate(res, update)
|
|
}
|
|
|
|
return res
|
|
} catch (err: any) {
|
|
console.error(err)
|
|
}
|
|
}
|