mirror of
https://github.com/ueberdosis/tiptap.git
synced 2024-12-17 12:37:49 +08:00
71 lines
1.5 KiB
TypeScript
71 lines
1.5 KiB
TypeScript
import { Editor as CoreEditor, EditorOptions } from '@tiptap/core'
|
|
import {
|
|
markRaw,
|
|
Ref,
|
|
customRef,
|
|
ComponentInternalInstance,
|
|
ComponentPublicInstance,
|
|
reactive,
|
|
} from 'vue'
|
|
import { EditorState } from 'prosemirror-state'
|
|
import { VueRenderer } from './VueRenderer'
|
|
|
|
function useDebouncedRef<T>(value: T) {
|
|
return customRef<T>((track, trigger) => {
|
|
return {
|
|
get() {
|
|
track()
|
|
return value
|
|
},
|
|
set(newValue) {
|
|
// update state
|
|
value = newValue
|
|
|
|
// update view as soon as possible
|
|
requestAnimationFrame(() => {
|
|
requestAnimationFrame(() => {
|
|
trigger()
|
|
})
|
|
})
|
|
},
|
|
}
|
|
})
|
|
}
|
|
|
|
export type ContentComponent = ComponentInternalInstance & {
|
|
ctx: ComponentPublicInstance,
|
|
}
|
|
|
|
declare module '@tiptap/core' {
|
|
interface Editor {
|
|
contentComponent: ContentComponent | null,
|
|
vueRenderers: Map<string, VueRenderer>,
|
|
}
|
|
}
|
|
|
|
export class Editor extends CoreEditor {
|
|
private reactiveState: Ref<EditorState>
|
|
|
|
public vueRenderers = reactive<Map<string, VueRenderer>>(new Map())
|
|
|
|
public contentComponent: ContentComponent | null = null
|
|
|
|
constructor(options: Partial<EditorOptions> = {}) {
|
|
super(options)
|
|
|
|
this.reactiveState = useDebouncedRef(this.view.state)
|
|
|
|
this.on('transaction', () => {
|
|
this.reactiveState.value = this.view.state
|
|
})
|
|
|
|
return markRaw(this)
|
|
}
|
|
|
|
get state() {
|
|
return this.reactiveState
|
|
? this.reactiveState.value
|
|
: this.view.state
|
|
}
|
|
}
|