You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
96 lines
3.3 KiB
96 lines
3.3 KiB
3 years ago
|
import {Mark, MarkType, Slice, Fragment, NodeType} from "prosemirror-model"
|
||
|
|
||
|
import {Step} from "./step"
|
||
|
import {Transform} from "./transform"
|
||
|
import {AddMarkStep, RemoveMarkStep} from "./mark_step"
|
||
|
import {ReplaceStep} from "./replace_step"
|
||
|
|
||
|
export function addMark(tr: Transform, from: number, to: number, mark: Mark) {
|
||
|
let removed: Step[] = [], added: Step[] = []
|
||
|
let removing: RemoveMarkStep | undefined, adding: AddMarkStep | undefined
|
||
|
tr.doc.nodesBetween(from, to, (node, pos, parent) => {
|
||
|
if (!node.isInline) return
|
||
|
let marks = node.marks
|
||
|
if (!mark.isInSet(marks) && parent!.type.allowsMarkType(mark.type)) {
|
||
|
let start = Math.max(pos, from), end = Math.min(pos + node.nodeSize, to)
|
||
|
let newSet = mark.addToSet(marks)
|
||
|
|
||
|
for (let i = 0; i < marks.length; i++) {
|
||
|
if (!marks[i].isInSet(newSet)) {
|
||
|
if (removing && removing.to == start && removing.mark.eq(marks[i]))
|
||
|
(removing as any).to = end
|
||
|
else
|
||
|
removed.push(removing = new RemoveMarkStep(start, end, marks[i]))
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (adding && adding.to == start)
|
||
|
(adding as any).to = end
|
||
|
else
|
||
|
added.push(adding = new AddMarkStep(start, end, mark))
|
||
|
}
|
||
|
})
|
||
|
|
||
|
removed.forEach(s => tr.step(s))
|
||
|
added.forEach(s => tr.step(s))
|
||
|
}
|
||
|
|
||
|
export function removeMark(tr: Transform, from: number, to: number, mark?: Mark | MarkType | null) {
|
||
|
let matched: {style: Mark, from: number, to: number, step: number}[] = [], step = 0
|
||
|
tr.doc.nodesBetween(from, to, (node, pos) => {
|
||
|
if (!node.isInline) return
|
||
|
step++
|
||
|
let toRemove = null
|
||
|
if (mark instanceof MarkType) {
|
||
|
let set = node.marks, found
|
||
|
while (found = mark.isInSet(set)) {
|
||
|
;(toRemove || (toRemove = [])).push(found)
|
||
|
set = found.removeFromSet(set)
|
||
|
}
|
||
|
} else if (mark) {
|
||
|
if (mark.isInSet(node.marks)) toRemove = [mark]
|
||
|
} else {
|
||
|
toRemove = node.marks
|
||
|
}
|
||
|
if (toRemove && toRemove.length) {
|
||
|
let end = Math.min(pos + node.nodeSize, to)
|
||
|
for (let i = 0; i < toRemove.length; i++) {
|
||
|
let style = toRemove[i], found
|
||
|
for (let j = 0; j < matched.length; j++) {
|
||
|
let m = matched[j]
|
||
|
if (m.step == step - 1 && style.eq(matched[j].style)) found = m
|
||
|
}
|
||
|
if (found) {
|
||
|
found.to = end
|
||
|
found.step = step
|
||
|
} else {
|
||
|
matched.push({style, from: Math.max(pos, from), to: end, step})
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
matched.forEach(m => tr.step(new RemoveMarkStep(m.from, m.to, m.style)))
|
||
|
}
|
||
|
|
||
|
export function clearIncompatible(tr: Transform, pos: number, parentType: NodeType, match = parentType.contentMatch) {
|
||
|
let node = tr.doc.nodeAt(pos)!
|
||
|
let delSteps: Step[] = [], cur = pos + 1
|
||
|
for (let i = 0; i < node.childCount; i++) {
|
||
|
let child = node.child(i), end = cur + child.nodeSize
|
||
|
let allowed = match.matchType(child.type)
|
||
|
if (!allowed) {
|
||
|
delSteps.push(new ReplaceStep(cur, end, Slice.empty))
|
||
|
} else {
|
||
|
match = allowed
|
||
|
for (let j = 0; j < child.marks.length; j++) if (!parentType.allowsMarkType(child.marks[j].type))
|
||
|
tr.step(new RemoveMarkStep(cur, end, child.marks[j]))
|
||
|
}
|
||
|
cur = end
|
||
|
}
|
||
|
if (!match.validEnd) {
|
||
|
let fill = match.fillBefore(Fragment.empty, true)
|
||
|
tr.replace(cur, cur, new Slice(fill!, 0, 0))
|
||
|
}
|
||
|
for (let i = delSteps.length - 1; i >= 0; i--) tr.step(delSteps[i])
|
||
|
}
|