2024-02-25 04:26:14 +09:00
|
|
|
import type { ExtendedRegExpMatchArray } from '@tiptap/core'
|
2022-12-28 03:38:57 +09:00
|
|
|
import {
|
|
|
|
Node,
|
|
|
|
mergeAttributes,
|
|
|
|
nodeInputRule,
|
2023-01-05 17:50:11 +09:00
|
|
|
nodePasteRule,
|
2022-12-28 03:38:57 +09:00
|
|
|
} from '@tiptap/core'
|
2023-01-02 13:53:53 +09:00
|
|
|
import { emojiRegEx, getEmojiAttributes } from '~/config/emojis'
|
2022-12-28 03:38:57 +09:00
|
|
|
|
2024-02-25 04:26:14 +09:00
|
|
|
function wrapHandler<T extends (...args: any[]) => any>(handler: T): T {
|
|
|
|
return <T>((...args: any[]) => {
|
|
|
|
try {
|
|
|
|
return handler(...args)
|
|
|
|
}
|
|
|
|
catch (e) {
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-03-31 04:01:24 +09:00
|
|
|
function createEmojiRule<NR extends typeof nodeInputRule | typeof nodePasteRule>(nodeRule: NR,
|
|
|
|
type: Parameters<NR>[0]['type']): ReturnType<NR>[] {
|
2023-01-05 17:50:11 +09:00
|
|
|
const rule = nodeRule({
|
|
|
|
find: emojiRegEx as RegExp,
|
|
|
|
type,
|
2024-02-25 04:26:14 +09:00
|
|
|
getAttributes: (match: ExtendedRegExpMatchArray) => {
|
2023-01-05 17:50:11 +09:00
|
|
|
const [native] = match
|
|
|
|
return getEmojiAttributes(native)
|
|
|
|
},
|
|
|
|
}) as ReturnType<NR>
|
|
|
|
|
|
|
|
// Error catch for unsupported emoji
|
2024-02-25 04:26:14 +09:00
|
|
|
rule.handler = wrapHandler(rule.handler.bind(rule))
|
2023-01-05 17:50:11 +09:00
|
|
|
|
2024-02-25 04:26:14 +09:00
|
|
|
return [rule]
|
2023-01-05 17:50:11 +09:00
|
|
|
}
|
|
|
|
|
2023-01-16 20:40:47 +09:00
|
|
|
export const TiptapPluginEmoji = Node.create({
|
2022-12-28 04:13:50 +09:00
|
|
|
name: 'em-emoji',
|
2022-12-28 03:38:57 +09:00
|
|
|
|
2022-12-28 04:13:50 +09:00
|
|
|
inline: () => true,
|
|
|
|
group: () => 'inline',
|
|
|
|
draggable: false,
|
2022-12-28 03:38:57 +09:00
|
|
|
|
2022-12-28 04:13:50 +09:00
|
|
|
parseHTML() {
|
|
|
|
return [
|
|
|
|
{
|
2023-01-02 13:53:53 +09:00
|
|
|
tag: 'img.iconify-emoji',
|
2022-12-28 04:13:50 +09:00
|
|
|
},
|
|
|
|
]
|
2022-12-28 03:38:57 +09:00
|
|
|
},
|
|
|
|
|
|
|
|
addAttributes() {
|
|
|
|
return {
|
2023-01-02 13:53:53 +09:00
|
|
|
alt: {
|
2022-12-28 03:38:57 +09:00
|
|
|
default: null,
|
|
|
|
},
|
2023-01-02 13:53:53 +09:00
|
|
|
src: {
|
|
|
|
default: null,
|
|
|
|
},
|
|
|
|
class: {
|
2022-12-28 05:42:58 +09:00
|
|
|
default: null,
|
|
|
|
},
|
2022-12-28 03:38:57 +09:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2022-12-28 04:13:50 +09:00
|
|
|
renderHTML(args) {
|
2023-01-02 13:53:53 +09:00
|
|
|
return ['img', mergeAttributes(this.options.HTMLAttributes, args.HTMLAttributes)]
|
2022-12-28 03:38:57 +09:00
|
|
|
},
|
|
|
|
|
|
|
|
addCommands() {
|
|
|
|
return {
|
2023-01-02 13:53:53 +09:00
|
|
|
insertEmoji: code => ({ commands }) => {
|
2022-12-28 03:38:57 +09:00
|
|
|
return commands.insertContent({
|
|
|
|
type: this.name,
|
2023-01-02 13:53:53 +09:00
|
|
|
attrs: getEmojiAttributes(code),
|
2022-12-28 03:38:57 +09:00
|
|
|
})
|
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
addInputRules() {
|
2023-01-05 17:50:11 +09:00
|
|
|
return createEmojiRule(nodeInputRule, this.type)
|
|
|
|
},
|
|
|
|
|
|
|
|
addPasteRules() {
|
|
|
|
return createEmojiRule(nodePasteRule, this.type)
|
2022-12-28 03:38:57 +09:00
|
|
|
},
|
|
|
|
})
|