Skip to content
ts
import 'prosekit/basic/style.css'
import 'prosekit/basic/typography.css'

import {
  html,
  LitElement,
} from 'lit'
import {
  customElement,
  property,
  state,
} from 'lit/decorators.js'
import {
  createRef,
  ref,
  type Ref,
} from 'lit/directives/ref.js'
import {
  createEditor,
  type Editor,
  type NodeJSON,
} from 'prosekit/core'

import {
  defineExtension,
  type EditorExtension,
} from './extension'

@customElement('example-lit-dom')
export class MyEditor extends LitElement {
  override createRenderRoot() {
    return this
  }

  @state()
  editor?: Editor<EditorExtension>

  @property({ type: Object, attribute: false })
  defaultContent?: NodeJSON

  private editorRef: Ref<HTMLDivElement> = createRef()

  protected override firstUpdated(): void {
    if (!this.editor) {
      const extension = defineExtension()
      this.editor = createEditor({
        extension,
        defaultContent: this.defaultContent || defaultContent,
      })
    }

    this.editor.mount(this.editorRef.value)
  }

  override render() {
    return html`
      <div class='box-border h-full w-full min-h-36 overflow-y-hidden overflow-x-hidden rounded-md border border-solid border-gray-200 shadow dark:border-zinc-700 flex flex-col bg-white dark:bg-neutral-900'>
        <div class='relative w-full flex-1 box-border overflow-y-scroll'>
          <div class='ProseMirror box-border min-h-full px-[max(4rem,_calc(50%-20rem))] py-8 outline-none outline-0 [&_span[data-mention="user"]]:text-blue-500 [&_span[data-mention="tag"]]:text-violet-500 [&_pre]:text-white [&_pre]:bg-zinc-800' ${ref(this.editorRef)}></div>
        </div>
      </div>
    `
  }
}

const defaultContent: NodeJSON = {
  type: 'doc',
  content: [
    {
      type: 'heading',
      attrs: {
        level: 1,
      },
      content: [
        {
          type: 'text',
          text: 'Image',
        },
      ],
    },
    {
      type: 'image',
      attrs: {
        src: 'https://placehold.co/120x80',
      },
    },
    {
      type: 'heading',
      attrs: {
        level: 1,
      },
      content: [
        {
          type: 'text',
          text: 'Code Block',
        },
      ],
    },
    {
      type: 'codeBlock',
      attrs: {
        language: 'python',
      },
      content: [
        {
          type: 'text',
          text: 'if __name__ == "__main__":\n    print("hello world!")\n\n'.repeat(
            20,
          ),
        },
      ],
    },
  ],
}
ts
import { defineBasicExtension } from 'prosekit/basic'
import { union } from 'prosekit/core'
import {
  defineCodeBlock,
  defineCodeBlockShiki,
} from 'prosekit/extensions/code-block'

import { defineCodeBlockView } from './code-block-view'

export function defineExtension() {
  return union(
    defineBasicExtension(),
    defineCodeBlock(),
    defineCodeBlockShiki(),
    defineCodeBlockView(),
  )
}

export type EditorExtension = ReturnType<typeof defineExtension>
ts
import {
  defineNodeView,
  setNodeAttrs,
} from 'prosekit/core'

import { createElement } from './create-element'
import { createLanguageSelector } from './language-selector'

export function defineCodeBlockView() {
  return defineNodeView({
    name: 'codeBlock',
    constructor: (node, view, getPos) => {
      const language = node.attrs.language as string

      const setLanguage = (language: string) => {
        const pos = getPos()!
        const attrs = { language }
        const command = setNodeAttrs({ type: 'codeBlock', attrs, pos })
        command(view.state, view.dispatch)
      }

      const type = node.type

      const code = createElement('code', {})

      const dom = createElement(
        'div',
        {
          'data-node-view-root': 'true',
        },
        createLanguageSelector({ language, setLanguage }),
        createElement('pre', {}, code),
      )

      return {
        dom: dom,
        contentDOM: code,
        update: (node) => {
          if (node.type !== type) {
            return false
          }
          code.textContent = node.textContent
          return true
        },
        ignoreMutation: () => {
          return true
        },
      }
    },
  })
}
ts
function createElement<K extends keyof HTMLElementTagNameMap>(
  tagName: K,
  attributes: Record<string, string>,
  ...children: (string | HTMLElement)[]
): HTMLElementTagNameMap[K]
function createElement(
  tagName: string,
  attributes: Record<string, string>,
  ...children: (string | HTMLElement)[]
): HTMLElement {
  const element = document.createElement(tagName)
  const { class: className, ...rest } = attributes
  if (className) {
    for (const c of className.split(' ')) {
      element.classList.add(c)
    }
  }
  Object.entries(rest).forEach(([key, value]) => {
    element.setAttribute(key, value)
  })
  children.forEach((child) => {
    if (typeof child === 'string') {
      element.appendChild(document.createTextNode(child))
    } else {
      element.appendChild(child)
    }
  })
  return element
}

export { createElement }
ts
export function getId() {
  return Math.random().toString(36).slice(2)
}
ts
import { shikiBundledLanguagesInfo } from 'prosekit/extensions/code-block'

import { createElement } from './create-element'

export function createLanguageSelector({
  language,
  setLanguage,
}: {
  language?: string
  setLanguage: (language: string) => void
}) {
  const select = createElement(
    'select',
    { class: 'outline-unset focus:outline-unset relative box-border w-auto cursor-pointer select-none appearance-none rounded border-none bg-transparent px-2 py-1 text-xs transition text-white opacity-0 hover:opacity-80 [div[data-node-view-root]:hover_&]:opacity-50 [div[data-node-view-root]:hover_&]:hover:opacity-80' },
    createElement('option', { value: '' }, 'Plain Text'),
    ...shikiBundledLanguagesInfo.map((info) => {
      return createElement('option', { value: info.id }, info.name)
    }),
  )

  select.value = language || ''
  select.addEventListener('change', (event) => {
    setLanguage((event.target as HTMLSelectElement).value)
  })

  return createElement(
    'div',
    { class: 'relative mx-2 top-3 h-0 select-none overflow-visible text-xs', contenteditable: 'false' },
    select,
  )
}