Work on compose
This commit is contained in:
@@ -22,34 +22,24 @@ declare module '@tiptap/core' {
|
||||
}
|
||||
|
||||
export const LinkExtension = Node.create({
|
||||
name: 'link',
|
||||
|
||||
group: 'inline',
|
||||
|
||||
atom: true,
|
||||
|
||||
name: 'link',
|
||||
group: 'inline',
|
||||
inline: true,
|
||||
|
||||
selectable: true,
|
||||
|
||||
draggable: true,
|
||||
|
||||
priority: 1000,
|
||||
|
||||
addAttributes() {
|
||||
return {
|
||||
url: { default: null },
|
||||
}
|
||||
},
|
||||
|
||||
renderHTML(props) {
|
||||
return ['div', { 'data-url': props.node.attrs.url }]
|
||||
},
|
||||
|
||||
renderText(props) {
|
||||
return props.node.attrs.url
|
||||
},
|
||||
|
||||
addStorage() {
|
||||
return {
|
||||
markdown: {
|
||||
@@ -60,7 +50,6 @@ export const LinkExtension = Node.create({
|
||||
},
|
||||
}
|
||||
},
|
||||
|
||||
addCommands() {
|
||||
return {
|
||||
insertLink:
|
||||
@@ -75,7 +64,6 @@ export const LinkExtension = Node.create({
|
||||
},
|
||||
}
|
||||
},
|
||||
|
||||
addPasteRules() {
|
||||
return [
|
||||
nodePasteRule({
|
||||
|
||||
@@ -1,3 +0,0 @@
|
||||
import {createPopoverNode} from '@lib/tiptap/common'
|
||||
|
||||
export const Mention = createPopoverNode('mention', '@')
|
||||
@@ -1,44 +1,26 @@
|
||||
import type {SvelteComponent, ComponentType} from 'svelte'
|
||||
import type {Readable} from 'svelte/store'
|
||||
import tippy, {type Instance} from 'tippy.js'
|
||||
import {mergeAttributes, Node} from '@tiptap/core'
|
||||
import {PluginKey} from '@tiptap/pm/state'
|
||||
import Suggestion from '@tiptap/suggestion'
|
||||
import type {Search} from '@lib/util'
|
||||
|
||||
export type PopoverOptions = {
|
||||
tippyOptions: Record<string, any>
|
||||
getLabel?: (id: string) => string
|
||||
onStart?: (opts: any) => void
|
||||
onUpdate?: (opts: any) => void
|
||||
onKeyDown?: (opts: any) => boolean | undefined
|
||||
onExit?: () => void
|
||||
export type SuggestionsOptions = {
|
||||
char: string,
|
||||
search: Readable<Search<any, any>>
|
||||
select: (value: any, props: any) => void
|
||||
suggestionComponent: ComponentType
|
||||
suggestionsComponent: ComponentType
|
||||
}
|
||||
|
||||
export const createPopoverNode = (name: string, char: string) => {
|
||||
const pluginKey = new PluginKey(name)
|
||||
|
||||
return Node.create<PopoverOptions>({
|
||||
export const createSuggestions = (name: string) =>
|
||||
Node.create<SuggestionsOptions>({
|
||||
name,
|
||||
group: 'inline',
|
||||
inline: true,
|
||||
selectable: false,
|
||||
atom: true,
|
||||
addAttributes: () => ({
|
||||
id: {
|
||||
default: null,
|
||||
parseHTML: el => el.getAttribute('data-id'),
|
||||
renderHTML: ({id}) => id ? {'data-id': id} : {},
|
||||
},
|
||||
}),
|
||||
parseHTML() {
|
||||
return [{tag: `span[data-type="${this.name}"]`}]
|
||||
},
|
||||
renderHTML({node, HTMLAttributes}) {
|
||||
const label = this.options.getLabel?.(node.attrs.id) || node.attrs.id
|
||||
|
||||
return ['span', mergeAttributes({'data-type': this.name}, HTMLAttributes), `${char}${label}`]
|
||||
},
|
||||
renderText({node}) {
|
||||
return `${char}${this.options.getLabel?.(node.attrs.id) || node.attrs.id}`
|
||||
},
|
||||
inline: true,
|
||||
group: 'inline',
|
||||
selectable: false,
|
||||
addKeyboardShortcuts() {
|
||||
return {
|
||||
Backspace: () => this.editor.commands.command(({ tr, state }) => {
|
||||
@@ -66,9 +48,9 @@ export const createPopoverNode = (name: string, char: string) => {
|
||||
addProseMirrorPlugins() {
|
||||
return [
|
||||
Suggestion({
|
||||
char,
|
||||
pluginKey,
|
||||
pluginKey: new PluginKey(name),
|
||||
editor: this.editor,
|
||||
char: this.options.char,
|
||||
command: ({editor, range, props}) => {
|
||||
// increase range.to by one when the next node is of type "text"
|
||||
// and starts with a space character
|
||||
@@ -100,6 +82,14 @@ export const createPopoverNode = (name: string, char: string) => {
|
||||
render: () => {
|
||||
let popover: Instance[]
|
||||
let target: HTMLElement
|
||||
let suggestions: SvelteComponent
|
||||
|
||||
const mapProps = (props: any) => ({
|
||||
term: props.query,
|
||||
search: this.options.search,
|
||||
component: this.options.suggestionComponent,
|
||||
select: (value: string) => this.options.select(value, props),
|
||||
})
|
||||
|
||||
return {
|
||||
onStart: props => {
|
||||
@@ -113,13 +103,12 @@ export const createPopoverNode = (name: string, char: string) => {
|
||||
interactive: true,
|
||||
trigger: "manual",
|
||||
placement: "bottom-start",
|
||||
...this.options.tippyOptions,
|
||||
})
|
||||
|
||||
this.options.onStart?.({target, props})
|
||||
suggestions = new this.options.suggestionsComponent({target, props: mapProps(props)})
|
||||
},
|
||||
onUpdate: props => {
|
||||
this.options.onUpdate?.({props})
|
||||
suggestions.$set(mapProps(props))
|
||||
|
||||
if (props.clientRect) {
|
||||
popover[0].setProps({
|
||||
@@ -134,11 +123,11 @@ export const createPopoverNode = (name: string, char: string) => {
|
||||
return true
|
||||
}
|
||||
|
||||
return Boolean(this.options.onKeyDown?.(props))
|
||||
return Boolean(suggestions.onKeyDown?.(props.event))
|
||||
},
|
||||
onExit: () => {
|
||||
popover[0].destroy()
|
||||
this.options.onExit?.()
|
||||
suggestions.$destroy()
|
||||
},
|
||||
}
|
||||
},
|
||||
@@ -146,4 +135,3 @@ export const createPopoverNode = (name: string, char: string) => {
|
||||
]
|
||||
},
|
||||
})
|
||||
}
|
||||
@@ -1,3 +0,0 @@
|
||||
import {createPopoverNode} from '@lib/tiptap/common'
|
||||
|
||||
export const Topic = createPopoverNode('topic', '#')
|
||||
+19
-2
@@ -1,3 +1,20 @@
|
||||
import type {JSONContent} from '@tiptap/core'
|
||||
|
||||
export {createSuggestions} from '@lib/tiptap/Suggestions'
|
||||
export {LinkExtension} from '@lib/tiptap/LinkExtension'
|
||||
export {Mention} from '@lib//tiptap/Mention'
|
||||
export {Topic} from '@lib//tiptap/Topic'
|
||||
|
||||
export const findNodes = (json: JSONContent, type: string) => {
|
||||
const results: JSONContent[] = []
|
||||
|
||||
for (const node of json.content || []) {
|
||||
if (node.type === type) {
|
||||
results.push(node)
|
||||
}
|
||||
|
||||
for (const result of findNodes(node, type)) {
|
||||
results.push(result)
|
||||
}
|
||||
}
|
||||
|
||||
return results
|
||||
}
|
||||
|
||||
+8
-1
@@ -38,7 +38,14 @@ export type SearchOptions<V, T> = {
|
||||
sortFn?: (items: FuseResult<T>) => any
|
||||
}
|
||||
|
||||
export const createSearch = <V, T>(data: T[], opts: SearchOptions<V, T>) => {
|
||||
export type Search<V, T> = {
|
||||
getValue: (item: T) => V
|
||||
getOption: (value: V) => T | undefined
|
||||
searchOptions: (term: string) => T[]
|
||||
searchValues: (term: string) => V[]
|
||||
}
|
||||
|
||||
export const createSearch = <V, T>(data: T[], opts: SearchOptions<V, T>): Search<V, T> => {
|
||||
const fuse = new Fuse(data, {...opts.fuseOptions, includeScore: true})
|
||||
const map = new Map<V, T>(data.map(item => [opts.getValue(item), item]))
|
||||
|
||||
|
||||
Reference in New Issue
Block a user