@atlaskit/editor-plugin-list
Version:
List plugin for @atlaskit/editor-core
55 lines (52 loc) • 2.47 kB
JavaScript
import { JOIN_SCENARIOS_WHEN_TYPING_TO_INSERT_LIST } from '@atlaskit/editor-common/analytics';
import { createRule } from '@atlaskit/editor-common/utils';
import { canJoin, findWrapping } from '@atlaskit/editor-prosemirror/transform';
import { findParentNodeOfTypeClosestToPos } from '@atlaskit/editor-prosemirror/utils';
// eslint-disable-next-line @typescript-eslint/no-explicit-any
export const createWrappingJoinRule = ({
match,
nodeType,
getAttrs,
joinPredicate
}) => {
const handler = (state, match, start, end) => {
const attrs = (getAttrs instanceof Function ? getAttrs(match) : getAttrs) || {};
const tr = state.tr;
const fixedStart = Math.max(start, 1);
tr.delete(fixedStart, end);
const $start = tr.doc.resolve(fixedStart);
const range = $start.blockRange();
const wrapping = range && findWrapping(range, nodeType, attrs);
if (!wrapping || !range) {
return null;
}
const parentNodePosMapped = tr.mapping.map(range.start);
const parentNode = tr.doc.nodeAt(parentNodePosMapped);
const lastWrap = wrapping[wrapping.length - 1];
if (parentNode && lastWrap) {
const allowedMarks = lastWrap.type.allowedMarks(parentNode.marks) || [];
tr.setNodeMarkup(parentNodePosMapped, parentNode.type, parentNode.attrs, allowedMarks);
}
tr.wrap(range, wrapping);
// if an orderedList node would be inserted by the input rule match, and
// that orderedList node is being added directly after another orderedList
const $end = tr.doc.resolve(tr.mapping.map(end));
const nodeWithPos = findParentNodeOfTypeClosestToPos($end, nodeType);
if (nodeType === state.schema.nodes.orderedList) {
// otherwise join the lists
if (nodeWithPos) {
const nodeEnd = nodeWithPos.pos + nodeWithPos.node.nodeSize;
const after = tr.doc.resolve(nodeEnd).nodeAfter;
if (after && after.type === nodeType && canJoin(tr.doc, nodeEnd) && (!joinPredicate || joinPredicate(match, after, JOIN_SCENARIOS_WHEN_TYPING_TO_INSERT_LIST.JOINED_TO_LIST_BELOW))) {
tr.join(nodeEnd);
}
}
}
const before = tr.doc.resolve(fixedStart - 1).nodeBefore;
if (before && before.type === nodeType && canJoin(tr.doc, fixedStart - 1) && (!joinPredicate || joinPredicate(match, before, JOIN_SCENARIOS_WHEN_TYPING_TO_INSERT_LIST.JOINED_TO_LIST_ABOVE))) {
tr.join(fixedStart - 1);
}
return tr;
};
return createRule(match, handler);
};