@atlaskit/editor-plugin-show-diff
Version:
ShowDiff plugin for @atlaskit/editor-core
59 lines (56 loc) • 2.51 kB
JavaScript
import { AnalyticsStep } from '@atlaskit/adf-schema/steps';
import { areNodesEqualIgnoreAttrs } from '@atlaskit/editor-common/utils/document';
import { ReplaceStep } from '@atlaskit/editor-prosemirror/transform';
/**
* Attempts to merge two consecutive ReplaceStep operations.
* This merges steps where:
* - Both steps replace content at the same starting position
* - The second step's range encompasses the first step's range
* - The second step has content to insert
* Returns null if steps cannot be merged.
*/
function mergeReplaceSteps(step1, step2) {
if (!(step1 instanceof ReplaceStep) || !(step2 instanceof ReplaceStep)) {
return null;
}
// Confirm the step overlaps exactly with the previous
const step2Length = step2.to - step2.from;
if (step1.from === step2.from && step2.to >= step1.to && step2.slice.size >= step2Length && step2Length === step1.slice.size && step2.slice.size >= step1.slice.size && step1.slice.openEnd === step2.slice.openEnd && step1.slice.openStart === step2.slice.openStart) {
return new ReplaceStep(step1.from, step1.to, step2.slice);
}
return null;
}
// Simplifies the steps to improve performance and reduce fragmentation in diffs
export function simplifySteps(steps, originalDoc) {
const stepsToFilter = removeUnusedSteps(steps, originalDoc);
return stepsToFilter
// Remove steps that don't affect document structure or content
.filter(step => !(step instanceof AnalyticsStep))
// Merge consecutive steps where possible
.reduce((acc, step) => {
var _lastStep$merge, _lastStep$merge2;
const lastStep = acc[acc.length - 1];
const merged = (_lastStep$merge = lastStep === null || lastStep === void 0 ? void 0 : (_lastStep$merge2 = lastStep.merge) === null || _lastStep$merge2 === void 0 ? void 0 : _lastStep$merge2.call(lastStep, step)) !== null && _lastStep$merge !== void 0 ? _lastStep$merge : mergeReplaceSteps(lastStep, step);
if (merged) {
acc[acc.length - 1] = merged;
} else {
acc.push(step);
}
return acc;
}, []);
}
/**
* Does a first pass to remove steps that don't impact the document
*/
function removeUnusedSteps(steps, originalDoc) {
const finalSteps = [];
let firstPassDoc = originalDoc;
for (const step of steps) {
const result = step.apply(firstPassDoc);
if (result.failed === null && result.doc && !areNodesEqualIgnoreAttrs(firstPassDoc, result.doc, ['localId'])) {
finalSteps.push(step);
firstPassDoc = result.doc;
}
}
return finalSteps;
}