diff --git a/README.md b/README.md index f69fe1f..de9179f 100644 --- a/README.md +++ b/README.md @@ -46,4 +46,6 @@ Editor: ## License +TrackChange based on: [TrackChangeExtension](https://github.com/chenyuncai/tiptap-track-change-extension) + This code is distributed under the MIT license. See `LICENSE` in this directory. diff --git a/components/editor/diff/diff.js b/components/editor/diff/diff.js deleted file mode 100644 index e3da66a..0000000 --- a/components/editor/diff/diff.js +++ /dev/null @@ -1,2 +0,0 @@ -// diff format - diff --git a/components/editor/diff/track-change-extension.js b/components/editor/diff/track-change-extension.js new file mode 100644 index 0000000..5fbea5f --- /dev/null +++ b/components/editor/diff/track-change-extension.js @@ -0,0 +1,540 @@ +/** + * License: MIT + * based on: https://github.com/chenyuncai/tiptap-track-change-extension + */ +import { ReplaceStep, Step } from '@tiptap/pm/transform' +import { TextSelection, Plugin, PluginKey } from '@tiptap/pm/state' +import { Slice, Fragment } from '@tiptap/pm/model' +import { Extension, Mark, getMarkRange, getMarksBetween, isMarkActive, mergeAttributes } from '@tiptap/core' +// import type { CommandProps, Editor, MarkRange } from '@tiptap/core' +// import type { Transaction } from '@tiptap/pm/state' + +const LOG_ENABLED = true + +export const MARK_DELETION = 'deletion' +export const MARK_INSERTION = 'insertion' +export const EXTENSION_NAME = 'trackchange' + +// Track Change Operations +export const TRACK_COMMAND_ACCEPT = 'accept' +export const TRACK_COMMAND_ACCEPT_ALL = 'accept-all' +export const TRACK_COMMAND_REJECT = 'reject' +export const TRACK_COMMAND_REJECT_ALL = 'reject-all' + +// export type TRACK_COMMAND_TYPE = 'accept' | 'accept-all' | 'reject' | 'reject-all' + +// insert mark +export const InsertionMark = Mark.create({ + name: MARK_INSERTION, + addAttributes () { + return { + 'data-op-user-id': { + type: 'string', + default: () => '', + }, + 'data-op-user-nickname': { + type: 'string', + default: () => '', + }, + 'data-op-date': { + type: 'string', + default: () => '', + } + } + }, + parseHTML () { + return [ + { tag: 'insert' } + ] + }, + renderHTML ({ HTMLAttributes }) { + return ['insert', mergeAttributes(this.options.HTMLAttributes, HTMLAttributes), 0] + } +}) + +// delete mark +export const DeletionMark = Mark.create({ + name: MARK_DELETION, + addAttributes () { + return { + 'data-op-user-id': { + type: 'string', + default: () => '', + }, + 'data-op-user-nickname': { + type: 'string', + default: () => '', + }, + 'data-op-date': { + type: 'string', + default: () => '', + } + } + }, + parseHTML () { + return [ + { tag: 'delete' } + ] + }, + renderHTML ({ HTMLAttributes }) { + return ['delete', mergeAttributes(this.options.HTMLAttributes, HTMLAttributes), 0] + } +}) + +// save the ime-mode status, when input chinese char, the extension needs to deal the change with a special strategy +// TODO: Is it necessary to save these two variable into a tiptap instance when someone open two editor +const IME_STATUS_NORMAL = 0 +const IME_STATUS_START = 1 +const IME_STATUS_CONTINUE = 2 +const IME_STATUS_FINISHED = 3 +// type IME_STATUS_TYPE = 0 | 1 | 2 | 3 +let composingStatus = 0 +let isStartChineseInput = false + +// get self extension instance by name +const getSelfExt = (editor) => editor.extensionManager.extensions.find(item => item.type === 'extension' && item.name === EXTENSION_NAME) + +// get the current minute time, avoid two char with different time splitted with too many marks +const getMinuteTime = () => Math.round(new Date().getTime() / 1000 / 60) * 1000 * 60 + +/** + * accept or reject tracked changes for all content or just the selection + * @param opType operation to apply + * @param param a command props, so we can get the editor, tr prop + * @returns null + */ +const changeTrack = (opType, param) => { + /** + * get the range to deal, use selection default + */ + const from = param.editor.state.selection.from + const to = param.editor.state.selection.to + /** + * find all the mark ranges to deal and remove mark or remove content according by opType + * if got accept all or reject all, just set 'from' to 0 and 'to' to content size + * if got just a part range, + */ + let markRanges = [] + /** + * deal a part and no selection contents, need to recognize the left mark near by cursor + */ + if ((opType === TRACK_COMMAND_ACCEPT || opType === TRACK_COMMAND_REJECT) && from === to) { + // detect left mark + const isInsertBeforeCursor = isMarkActive(param.editor.state, MARK_INSERTION) + const isDeleteBeforeCursor = isMarkActive(param.editor.state, MARK_DELETION) + let leftRange + if (isInsertBeforeCursor) { + leftRange = getMarkRange(param.editor.state.selection.$from, param.editor.state.doc.type.schema.marks.insertion) + } else if (isDeleteBeforeCursor) { + leftRange = getMarkRange(param.editor.state.selection.$from, param.editor.state.doc.type.schema.marks.deletion) + } + if (leftRange) { + markRanges = getMarksBetween(leftRange.from, leftRange.to, param.editor.state.doc) + } + } else if (opType === TRACK_COMMAND_ACCEPT_ALL || opType === TRACK_COMMAND_REJECT_ALL) { + // all editor content + markRanges = getMarksBetween(0, param.editor.state.doc.content.size, param.editor.state.doc) + // change the opType to normal + opType = opType === TRACK_COMMAND_ACCEPT_ALL ? TRACK_COMMAND_ACCEPT : TRACK_COMMAND_REJECT + } else { + // just the selection + markRanges = getMarksBetween(from, to, param.editor.state.doc) + } + // just deal the track change nodes + markRanges = markRanges.filter(markRange => markRange.mark.type.name === MARK_DELETION || markRange.mark.type.name === MARK_INSERTION) + if (!markRanges.length) { return false } + + const currentTr = param.tr + /** + * mark type and opType compose: + * 1. accept with insert mark: remove insert mark + * 2. accept with delete mark: remove content + * 3. reject with insert mark: remove content + * 4. reject with delete mark: remove delete mark + * so + * 1 and 4 need to remove mark + * 2 and 3 need to remove content + */ + // record offset when delete some content to find the correct pos for next range + let offset = 0 + const removeInsertMark = param.editor.state.doc.type.schema.marks.insertion.create() + const removeDeleteMark = param.editor.state.doc.type.schema.marks.deletion.create() + markRanges.forEach((markRange) => { + const isAcceptInsert = opType === TRACK_COMMAND_ACCEPT && markRange.mark.type.name === MARK_INSERTION + const isRejectDelete = opType === TRACK_COMMAND_REJECT && markRange.mark.type.name === MARK_DELETION + if (isAcceptInsert || isRejectDelete) { + // 1 and 4: remove mark + currentTr.removeMark(markRange.from - offset, markRange.to - offset, removeInsertMark.type) + currentTr.removeMark(markRange.from - offset, markRange.to - offset, removeDeleteMark.type) + } else { + // 2 and 3 remove content + currentTr.deleteRange(markRange.from - offset, markRange.to - offset) + // change the offset + offset += (markRange.to - markRange.from) + } + }) + if (currentTr.steps.length) { + // set a custom meta to tail Our TrackChangeExtension to ignore this change + // TODO: is there any official field to do this? + currentTr.setMeta('trackManualChanged', true) + // apply to current editor state and get a new state + const newState = param.editor.state.apply(currentTr) + // update the new state to editor to render new content + param.editor.view.updateState(newState) + } + return false +} + +// @ts-ignore +/** + * TODO: some problems to fix or feature to implement + * 1. when delete content includes two and more paragraphs, cannot mark the new paragraph as insert mark, because the mark is inline, can we add global attrs? + * 2. when delete content includes two and more paragraphs, connot ignore the insert mark inside the content. Currently, the insert mark is marked as deleted. But it need to be delete directly. + * 3. select two chars and inout a chinese char, the new char was input with wrong position. (fixed by stop input action) + * 4. how to toggle to "hide" mode and can record the change ranges too, just look likes the office word + */ +export const TrackChangeExtension = Extension.create({ + name: EXTENSION_NAME, + onCreate () { + if (this.options.onStatusChange) { + this.options.onStatusChange(this.options.enabled) + } + }, + addExtensions () { + return [InsertionMark, DeletionMark] + }, + addCommands: () => { + return { + setTrackChangeStatus: (enabled) => (param) => { + const thisExtension = getSelfExt(param.editor) + thisExtension.options.enabled = enabled + if (thisExtension.options.onStatusChange) { + thisExtension.options.onStatusChange(thisExtension.options.enabled) + } + return false + }, + toggleTrackChangeStatus: () => (param) => { + const thisExtension = getSelfExt(param.editor) + thisExtension.options.enabled = !thisExtension.options.enabled + if (thisExtension.options.onStatusChange) { + thisExtension.options.onStatusChange(thisExtension.options.enabled) + } + return false + }, + getTrackChangeStatus: () => (param) => { + const thisExtension = getSelfExt(param.editor) + return thisExtension.options.enabled + }, + acceptChange: () => (param) => { + changeTrack('accept', param) + return false + }, + acceptAllChanges: () => (param) => { + changeTrack('accept-all', param) + return false + }, + rejectChange: () => (param) => { + changeTrack('reject', param) + return false + + }, + rejectAllChanges: () => (param) => { + changeTrack('reject-all', param) + return false + }, + updateOpUserOption: (opUserId, opUserNickname) => (param) => { + const thisExtension = getSelfExt(param.editor) + thisExtension.options.dataOpUserId = opUserId + thisExtension.options.dataOpUserNickname = opUserNickname + return false + } + } + }, + // @ts-ignore + onSelectionUpdate (p) { + // log the status for debug + LOG_ENABLED && console.log('selection and input status', p.transaction.selection.from, p.transaction.selection.to, p.editor.view.composing) + }, + // @ts-ignore + addProseMirrorPlugins () { + return [ + new Plugin({ + key: new PluginKey('composing-check'), + props: { + handleDOMEvents: { + compositionstart: (_event) => { + LOG_ENABLED && console.log('start chinese input') + // start and update will fire same time + isStartChineseInput = true + }, + compositionupdate: (_event) => { + LOG_ENABLED && console.log('chinese input continue') + composingStatus = IME_STATUS_CONTINUE + } + } + } + }) + ] + }, + // @ts-ignore + onTransaction: ({ editor, transaction }) => { + // chinese input status check + const isChineseStart = isStartChineseInput && composingStatus === IME_STATUS_CONTINUE + const isChineseInputting = !isStartChineseInput && composingStatus === IME_STATUS_CONTINUE + const isNormalInput = composingStatus === IME_STATUS_NORMAL // not ime mode + composingStatus = IME_STATUS_NORMAL // reset for next change + isStartChineseInput = false + /** + * normal ime mode start, we can ignore the input and deal change when last confirm + * but what if it has a selection before? the selected content will loss for real. so we have to deal with it + */ + // no change, ignore + if (!transaction.docChanged) { return } + // check if it is change by accept or reject + if (transaction.getMeta('trackManualChanged')) { return } + // if this is a redo or undo, ignore it. + if (transaction.getMeta('history$')) { return } + // check if it is synced by another client or the server, need to ignore too + const syncMeta = transaction.getMeta('y-sync$') + if (syncMeta && syncMeta.isChangeOrigin) { + LOG_ENABLED && console.log('sync from origin', syncMeta) + return + } + // has no real step + if (!transaction.steps.length) { + LOG_ENABLED && console.log('none content change') + return + } + // check if this tr was applied to editor: vue2 and vue3 have different result + const isThisTrApplied = transaction.before !== editor.state.tr.doc + const thisExtension = getSelfExt(editor) + const trackChangeEnabled = thisExtension.options.enabled + LOG_ENABLED && console.warn('内容变化,执行跟踪修订相关逻辑', transaction.steps.length, transaction) + /** + * Two main process + * + * 1. When Status disabled + * a. the new insert content cannot be wrapped with insert or delete mark (avoid the original auto style by browser) + * b. consider the ime mode + * c. make the cursor pos right + * d. can "remove the auto-used style" and "user edit" op be a single operation to history? + * + * 2. When Status Enabled(with more complex process) + * a. mark the new content with insert + * b. remove auto-used delete mark for new content + * c. readd the deleted normal content and mark as delete + * d. readd the deleted "delete char" + * e. if "insert char" be deleted, just let it go + * f. correct the final cursor position, too many cases to consider + * h. careful with ime mode + * i. ignore history updates + * + * Test Case: + * 1. enabled: input char + * 1. enabled: delete normal char + * 1. enabled: delete the 'delete' char + * 1. enabled: input after the 'delete' char + * 1. enabled: delete the 'insert' char + * 1. enabled: input after the 'insert' char + * 1. enabled: make a selection and delete then + * 1. enabled: select normal char and input new text + * 1. enabled: select 'insert' chars and input new text + * 1. enabled: ime input after normal char + * 1. enabled: ime input after insert char + * 1. enabled: ime input after delete char + * 1. enabled: ime input when select normal chars + * 1. enabled: ime input when select insert chars + * 1. enabled: ime input when select delete chars + * 1. enabled: ime input when select both insert and delete chars + * + * + * 2. disabled: input normal char + * 2. disabled: input after insert char + * 2. disabled: input after delete char + */ + // copy the step, avoid the change the origin step info + const allSteps = transaction.steps.map(step => Step.fromJSON(editor.state.doc.type.schema, step.toJSON())) + LOG_ENABLED && console.log('allSteps', allSteps) + // latest cursor pos + // TODO: recognize selection direction and backspace/delete to get calculate new cursor pos + const currentNewPos = transaction.selection.from + LOG_ENABLED && console.log('currentNewPos', currentNewPos) + + // cursor offset, try to reset the pos in the end process + let posOffset = 0 + let hasAddAndDelete = false + allSteps.forEach((step, _index, _arr) => { + if (step instanceof ReplaceStep) { + // loss chars + let delCount = 0 + if (step.from !== step.to) { + /** + * means some content will be removed, but in this extension, we will reAdd it back + * note: if the old content is already 'insert', just let it go. + */ + const slice = transaction.docs[_index].slice(step.from, step.to) + slice.content.forEach(node => { + const isInsertNode = node.marks.find(m => m.type.name === MARK_INSERTION) + if (!isInsertNode) { + // sum the none-insert mark size + // TODO: nodeSize or size filed? what if the node is a image or some other custom node + delCount += node.nodeSize + } + }) + } + posOffset += delCount + // added chars in this step + const newCount = step.slice ? step.slice.size : 0 + if (newCount && delCount) { + // if only add or del, don't need to change selection, it's already right + hasAddAndDelete = true + } + } + }) + if (isNormalInput) { + // none ime mode + if (!hasAddAndDelete) { + // just delete or add, no need to correct cursor + posOffset = 0 + } + } else if (isChineseStart) { + // first chinese char + if (hasAddAndDelete) { + /** + * input chinese char when a selection content exist + * need the readd the selected chars, so we need update the cursor + * but we cannot change the cursor in "inputting time", ,,,,need some more test + */ + // posOffset -= 1 + } else { + // just new content, ime software will make it right + posOffset = 0 + } + } else if (isChineseInputting) { + // chinese input, at least two chars, no need to update cursor + posOffset = 0 + } + // TODO: if there has multiple replace steps,we can ignore the cursor + LOG_ENABLED && console.table({ + hasAddAndDelete, + isNormalInput, + isChineseStart, + isChineseInputting, + posOffset + }) + + // get the correct tr to manipulate + // tr had been applied in vue2, and not in vue3 + // so we get a new tr in vue2 by "editor.state.tr" + let newChangeTr = transaction + if (isThisTrApplied) { + newChangeTr = editor.state.tr + } + + let reAddOffset = 0 + allSteps.forEach((step, index) => { + if (step instanceof ReplaceStep) { + const invertedStep = step.invert(transaction.docs[index]) + if (step.slice.size) { + const insertionMark = editor.state.doc.type.schema.marks.insertion.create({ + 'data-op-user-id': thisExtension.options.dataOpUserId, + 'data-op-user-nickname': thisExtension.options.dataOpUserNickname, + 'data-op-date': getMinuteTime() + }) + const deletionMark = editor.state.doc.type.schema.marks.deletion.create() + const from = step.from + reAddOffset + const to = step.from + reAddOffset + step.slice.size + if (trackChangeEnabled) { + // add insert mark to new content + newChangeTr.addMark(from, to, insertionMark) + } else { + // if disabled this extension, remove auto-used track mark for new content + newChangeTr.removeMark(from, to, insertionMark.type) + } + // remove auto-used delete mark for new content anyway + newChangeTr.removeMark(from, to, deletionMark.type) + } + if (step.from !== step.to && trackChangeEnabled) { + LOG_ENABLED && console.log('find content to readd', step) + // get the reverted step, so we can know what content is deleted in this step and readd then + // make sure use related doc in transaction. every step has the new doc with same order + // TODO: is there difference between vue2 and vue3 + const skipSteps = [] + // collect the content we need to ignore readd, because some content is insert mark before, there data need to be allowed delete + LOG_ENABLED && console.log('invertedStep', invertedStep) + const reAddStep = new ReplaceStep( + invertedStep.from + reAddOffset, + invertedStep.from + reAddOffset, + invertedStep.slice, + // @ts-ignore: what is internal means + invertedStep.structure + ) + // make a empty step to replace the original "INSERT" mark, because these content don't need to readd + // the slice content maybe a TextNode or other node with any child content + // so we need to travel all the content to find the "INSERT" mark + let addedEmptyOffset = 0 // when empty step is added, record the offset to correct the next empty step, because the content will b shorter than the current when create next empty step + const travelContent = (content, parentOffset) => { + content.forEach((node, offset) => { + const start = parentOffset + offset + const end = start + node.nodeSize + if (node.content && node.content.size) { + // this node has child content, need to travel + travelContent(node.content, start) + } else { + // this node is a text node, or a node without child content + if (node.marks.find(m => m.type.name === MARK_INSERTION)) { + // construct a empty step, apply this after readd action applied + skipSteps.push(new ReplaceStep(start - addedEmptyOffset, end - addedEmptyOffset, Slice.empty)) + addedEmptyOffset += node.nodeSize + reAddOffset -= node.nodeSize + } + } + }) + } + travelContent(invertedStep.slice.content, invertedStep.from) + reAddOffset += invertedStep.slice.size + // apply readd step action + newChangeTr.step(reAddStep) + const { from } = reAddStep + const to = from + reAddStep.slice.size + // add delete mark for readd content + newChangeTr.addMark(from, to, newChangeTr.doc.type.schema.marks.deletion.create({ + 'data-op-user-id': thisExtension.options.dataOpUserId, + 'data-op-user-nickname': thisExtension.options.dataOpUserNickname, + 'data-op-date': getMinuteTime() + })) + skipSteps.forEach((step) => { + // delete the content if it is already with insert mark + newChangeTr.step(step) + }) + } + const newState = editor.state.apply(newChangeTr) + editor.view.updateState(newState) + // TODO: if there has cursor change action, can there change be merged to one tr with one updateState? + } + }) + + // calculate new cursor postion + const finalNewPos = trackChangeEnabled ? (currentNewPos + posOffset) : currentNewPos + if (trackChangeEnabled) { + const trWithChange = editor.view.state.tr + trWithChange.setSelection(TextSelection.create(editor.view.state.doc, finalNewPos)) + const newStateWithNewSelection = editor.view.state.apply(trWithChange) + LOG_ENABLED && console.log('update cursor', finalNewPos) + editor.view.updateState(newStateWithNewSelection) + } + if (isChineseStart && hasAddAndDelete && trackChangeEnabled) { + // if the first chinese input and have some content selected + // just make it stop by blur action + // delete the selection and use just need paste the new content again + editor.commands.deleteSelection() + editor.commands.blur() + setTimeout(() => { + // focus again + editor.commands.focus() + }, 100) + } + } +}) + +export default TrackChangeExtension diff --git a/components/editor/live-editor.jsx b/components/editor/live-editor.jsx index bde3caf..993fbf0 100644 --- a/components/editor/live-editor.jsx +++ b/components/editor/live-editor.jsx @@ -7,19 +7,19 @@ import React from 'react' import { MenuBar } from './menu-bar' import MarkdownIt from 'markdown-it' -import { DeletionMark, InsertionMark, MenuBubble } from './intelli/menu/menu-bubble' +import { MenuBubble } from './intelli/menu/menu-bubble' import { createSlashExtension } from './intelli/slash-extension' import { CommandFunctions } from './action/command-functions' import { createAiBlock } from './intelli/ai-block-extension' +import TrackChangeExtension from './diff/track-change-extension' const md = new MarkdownIt() const extensions = [ CommandFunctions, - // addExtensions () { - // return [InsertionMark, DeletionMark] - // }, - InsertionMark, DeletionMark, + TrackChangeExtension.configure({ + enabled: false, + }), StarterKit.configure({ bulletList: { keepMarks: true,