tiptap/packages/core/src/commands/insertHTML.ts

37 lines
1.3 KiB
TypeScript
Raw Normal View History

2019-12-15 06:54:20 +08:00
import { DOMParser } from 'prosemirror-model'
2020-09-21 05:19:27 +08:00
import { Selection } from 'prosemirror-state'
import { Command } from '../Editor'
2019-12-17 06:51:18 +08:00
import elementFromString from '../utils/elementFromString'
2020-09-21 05:19:27 +08:00
import {ReplaceStep, ReplaceAroundStep} from "prosemirror-transform"
2019-12-15 06:54:20 +08:00
2020-09-21 05:19:27 +08:00
type InsertHTMLCommand = (value: string) => Command
2020-04-22 05:22:27 +08:00
2019-12-15 06:54:20 +08:00
declare module '../Editor' {
interface Editor {
insertHTML: InsertHTMLCommand,
2019-12-15 06:54:20 +08:00
}
}
2020-09-21 05:19:27 +08:00
// TODO: move to utils
// https://github.com/ProseMirror/prosemirror-state/blob/master/src/selection.js#L466
function selectionToInsertionEnd(tr, startLen, bias) {
let last = tr.steps.length - 1
if (last < startLen) return
let step = tr.steps[last]
if (!(step instanceof ReplaceStep || step instanceof ReplaceAroundStep)) return
let map = tr.mapping.maps[last], end
map.forEach((_from, _to, _newFrom, newTo) => { if (end == null) end = newTo })
tr.setSelection(Selection.near(tr.doc.resolve(end), bias))
}
export const insertHTML: InsertHTMLCommand = value => ({ tr, editor }) => {
const { state } = editor
const { selection } = tr
2019-12-17 06:51:18 +08:00
const element = elementFromString(value)
2019-12-15 06:54:20 +08:00
const slice = DOMParser.fromSchema(state.schema).parseSlice(element)
2020-09-21 05:19:27 +08:00
tr.insert(selection.anchor, slice.content)
selectionToInsertionEnd(tr, tr.steps.length - 1, -1)
return true
2019-12-15 06:54:20 +08:00
}