import { createNode, mergeAttributes } from '@tiptap/core' export interface TaskItemOptions { nested: boolean, } const TaskItem = createNode({ name: 'task_item', content: 'paragraph+', // TODO: allow content to be a callback function // content() { // return this.options.nested ? '(paragraph|todo_list)+' : 'paragraph+', // }, defining: true, defaultOptions: { nested: false, }, addAttributes() { return { checked: { default: false, parseHTML: element => ({ checked: element.getAttribute('data-checked') === 'true', }), renderHTML: attributes => ({ 'data-checked': attributes.checked, }), }, } }, parseHTML() { return [ { tag: 'li[data-type="task_item"]', priority: 51, }, ] }, renderHTML({ attributes }) { return ['li', mergeAttributes(attributes, { 'data-type': 'task_item' }), 0] }, addKeyboardShortcuts() { const shortcuts = { Enter: () => this.editor.splitListItem('task_item'), } if (!this.options.nested) { return shortcuts } return { ...shortcuts, Tab: () => this.editor.sinkListItem('task_item'), 'Shift-Tab': () => this.editor.liftListItem('task_item'), } }, addNodeView() { return ({ attributes, getPos, editor }) => { const { view } = editor const listItem = document.createElement('li') const checkbox = document.createElement('input') const content = document.createElement('div') checkbox.type = 'checkbox' checkbox.addEventListener('change', event => { const { checked } = event.target as any if (typeof getPos === 'function') { view.dispatch(view.state.tr.setNodeMarkup(getPos(), undefined, { checked, })) } }) if (attributes['data-checked'] === true) { checkbox.setAttribute('checked', 'checked') } listItem.append(checkbox, content) Object.entries(attributes).forEach(([key, value]) => { listItem.setAttribute(key, value) }) return { dom: listItem, contentDOM: content, } } }, }) export default TaskItem declare module '@tiptap/core/src/Editor' { interface AllExtensions { TaskItem: typeof TaskItem, } }