tiptap/packages/core/src/helpers/getChangedRanges.ts
Dominik f387ad3dd4
feat(pm): new prosemirror package for dependency resolving
* chore:(core): migrate to tsup

* chore: migrate blockquote and bold to tsup

* chore: migrated bubble-menu and bullet-list to tsup

* chore: migrated more packages to tsup

* chore: migrate code and character extensions to tsup

* chore: update package.json to simplify build for all packages

* chore: move all packages to tsup as a build process

* chore: change ci build task

* feat(pm): add prosemirror meta package

* rfix: resolve issues with build paths & export mappings

* docs: update documentation to include notes for @tiptap/pm

* chore(pm): update tsconfig

* chore(packages): update packages

* fix(pm): add package export infos & fix dependencies

* chore(general): start moving to pm package as deps

* chore: move to tiptap pm package internally

* fix(demos): fix demos working with new pm package

* fix(tables): fix tables package

* fix(tables): fix tables package

* chore(demos): pinned typescript version

* chore: remove unnecessary tsconfig

* chore: fix netlify build

* fix(demos): fix package resolving for pm packages

* fix(tests): fix package resolving for pm packages

* fix(tests): fix package resolving for pm packages

* chore(tests): fix tests not running correctly after pm package

* chore(pm): add files to files array

* chore: update build workflow

* chore(tests): increase timeout time back to 12s

* chore(docs): update docs

* chore(docs): update installation guides & pm information to docs

* chore(docs): add link to prosemirror docs

* fix(vue-3): add missing build step

* chore(docs): comment out cdn link

* chore(docs): remove semicolons from docs

* chore(docs): remove unnecessary installation note

* chore(docs): remove unnecessary installation note
2023-02-02 17:37:33 +01:00

84 lines
2.2 KiB
TypeScript

import { Step, Transform } from '@tiptap/pm/transform'
import { Range } from '../types'
import { removeDuplicates } from '../utilities/removeDuplicates'
export type ChangedRange = {
oldRange: Range,
newRange: Range,
}
/**
* Removes duplicated ranges and ranges that are
* fully captured by other ranges.
*/
function simplifyChangedRanges(changes: ChangedRange[]): ChangedRange[] {
const uniqueChanges = removeDuplicates(changes)
return uniqueChanges.length === 1
? uniqueChanges
: uniqueChanges.filter((change, index) => {
const rest = uniqueChanges.filter((_, i) => i !== index)
return !rest.some(otherChange => {
return change.oldRange.from >= otherChange.oldRange.from
&& change.oldRange.to <= otherChange.oldRange.to
&& change.newRange.from >= otherChange.newRange.from
&& change.newRange.to <= otherChange.newRange.to
})
})
}
/**
* Returns a list of changed ranges
* based on the first and last state of all steps.
*/
export function getChangedRanges(transform: Transform): ChangedRange[] {
const { mapping, steps } = transform
const changes: ChangedRange[] = []
mapping.maps.forEach((stepMap, index) => {
const ranges: Range[] = []
// This accounts for step changes where no range was actually altered
// e.g. when setting a mark, node attribute, etc.
// @ts-ignore
if (!stepMap.ranges.length) {
const { from, to } = steps[index] as Step & {
from?: number,
to?: number,
}
if (from === undefined || to === undefined) {
return
}
ranges.push({ from, to })
} else {
stepMap.forEach((from, to) => {
ranges.push({ from, to })
})
}
ranges.forEach(({ from, to }) => {
const newStart = mapping.slice(index).map(from, -1)
const newEnd = mapping.slice(index).map(to)
const oldStart = mapping.invert().map(newStart, -1)
const oldEnd = mapping.invert().map(newEnd)
changes.push({
oldRange: {
from: oldStart,
to: oldEnd,
},
newRange: {
from: newStart,
to: newEnd,
},
})
})
})
return simplifyChangedRanges(changes)
}