@atlaskit/editor-common
Version:
A package that contains common classes and components for editor and renderer
57 lines (55 loc) • 1.96 kB
JavaScript
import { canJoin } from '@atlaskit/editor-prosemirror/transform';
/**
* Checks whether two adjacent nodes can be joined. If so, the document
* will be updated to join those nodes. If not, the original transaction
* remains untouched.
*
* Nodes are considered joinable if the `isJoinable` predicate returns true or,
* if an array of strings was passed, if their node type name is in that array.
*
* Adapted from https://github.com/ProseMirror/prosemirror-commands/blob/master/src/commands.js#L597-L610
*/
export function autoJoinTr(tr, isJoinable) {
if (Array.isArray(isJoinable)) {
let types = isJoinable;
isJoinable = node => types.indexOf(node.type.name) > -1;
}
const ranges = [];
for (let i = 0; i < tr.mapping.maps.length; i++) {
const map = tr.mapping.maps[i];
for (let j = 0; j < ranges.length; j++) {
ranges[j] = map.map(ranges[j]);
}
map.forEach((_s, _e, from, to) => ranges.push(from, to));
}
// Figure out which joinable points exist inside those ranges,
// by checking all node boundaries in their parent nodes.
const joinable = [];
for (let i = 0; i < ranges.length; i += 2) {
const from = ranges[i];
const to = ranges[i + 1];
const $from = tr.doc.resolve(from);
const depth = $from.sharedDepth(to);
const parent = $from.node(depth);
for (let index = $from.indexAfter(depth), pos = $from.after(depth + 1); pos <= to; ++index) {
const after = parent.maybeChild(index);
if (!after) {
break;
}
if (index && joinable.indexOf(pos) === -1) {
const before = parent.child(index - 1);
if (before.type === after.type && isJoinable(before, after)) {
joinable.push(pos);
}
}
pos += after.nodeSize;
}
}
// Join the joinable points
joinable.sort((a, b) => a - b);
for (let i = joinable.length - 1; i >= 0; i--) {
if (canJoin(tr.doc, joinable[i])) {
tr.join(joinable[i]);
}
}
}