Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: trigger inputrule on enter for code block #1940

Closed
wants to merge 1 commit into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 4 additions & 1 deletion packages/extension-code-block/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,10 @@
"@tiptap/core": "^2.0.0-beta.1"
},
"dependencies": {
"prosemirror-inputrules": "^1.1.3"
"@types/prosemirror-view": "^1.19.1",
"@types/prosemirror-state": "^1.2.7",
"prosemirror-inputrules": "^1.1.3",
"prosemirror-state": "^1.3.4"
},
"repository": {
"type": "git",
Expand Down
17 changes: 15 additions & 2 deletions packages/extension-code-block/src/code-block.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { Node } from '@tiptap/core'
import { textblockTypeInputRule } from 'prosemirror-inputrules'
import enterRules from './plugin'

export interface CodeBlockOptions {
languageClassPrefix: string,
Expand All @@ -21,8 +22,8 @@ declare module '@tiptap/core' {
}
}

export const backtickInputRegex = /^```(?<language>[a-z]*)? $/
export const tildeInputRegex = /^~~~(?<language>[a-z]*)? $/
export const backtickInputRegex = /^```(?<language>[a-z]*)?\s$/
export const tildeInputRegex = /^~~~(?<language>[a-z]*)?\s$/

export const CodeBlock = Node.create<CodeBlockOptions>({
name: 'codeBlock',
Expand Down Expand Up @@ -125,4 +126,16 @@ export const CodeBlock = Node.create<CodeBlockOptions>({
textblockTypeInputRule(tildeInputRegex, this.type, ({ groups }: any) => groups),
]
},

addProseMirrorPlugins() {
return [
...((this.parent && this.parent()) || []),
enterRules({
rules: [
textblockTypeInputRule(backtickInputRegex, this.type, ({ groups }: any) => groups),
textblockTypeInputRule(tildeInputRegex, this.type, ({ groups }: any) => groups),
],
}),
]
},
})
45 changes: 45 additions & 0 deletions packages/extension-code-block/src/plugin.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
// Modified from
// https://github.com/ProseMirror/prosemirror-inputrules/blob/master/src/inputrules.js
// license: MIT

import { InputRule } from 'prosemirror-inputrules'
import { Plugin as ProseMirrorPlugin } from 'prosemirror-state'
import type { Plugin, TextSelection } from 'prosemirror-state'
import type { EditorView } from 'prosemirror-view'

const MAX_MATCH = 500

function run(view: EditorView, from: number, to: number, text: string, rules: any[], plugin: Plugin) {
if (view.composing) return false
const state = view.state; const
$from = state.doc.resolve(from)
if ($from.parent.type.spec.code) return false
const textBefore = $from.parent.textBetween(Math.max(0, $from.parentOffset - MAX_MATCH), $from.parentOffset,
undefined, '\ufffc') + text
for (let i = 0; i < rules.length; i += 1) {
const match = rules[i].match.exec(textBefore)
const tr = match && rules[i].handler(state, match, from - (match[0].length - text.length), to)
if (!tr) continue
view.dispatch(tr.setMeta(plugin, {
transform: tr, from, to, text,
}))
return true
}
return false
}

export default function (options: any = {}) {
const rules: InputRule[] = options.rules || []
const plugin: Plugin = new ProseMirrorPlugin({
props: {
Copy link
Contributor

@BrianHung BrianHung Sep 26, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is missing the compositionend event in the original inputrule, and well as the isInputRules field which allows undoInputRule to work; as well as the original plugin state. Why not just copy the original plugin and add the handleKeyDown field?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ProseMirror/prosemirror-inputrules#6

Because marijnh said {enter} is not a kind of inputrule, so I made a standalone plugin

Copy link
Contributor Author

@fengzilong fengzilong Sep 26, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The origin inputrule plugin is not removed, IMO there is no necessary to add compositionend here maybe.
I'm not familiar with ProseMirror, I'm not sure whether the origin plugin state is needed here 😥

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure whether the origin plugin state is needed here

The plugin state is kept there so that the undoInputRule function (which is binded to backspace) can work using the setMeta and getMeta fields.

https://prosemirror.net/docs/ref/#state.Transaction.setMeta
https://prosemirror.net/docs/ref/#state.Transaction.getMeta

Because marijnh said {enter} is not a kind of inputrule, so I made a standalone plugin

I think you could argue enter is a type of inputrule, but where the trigger isn't an explicit key being entered, but an implicit key being pressed; plus, you're still using the InputRule abstraction, just handling / running it differently.

handleKeyDown(view, event) {
if (event.key !== 'Enter') return false
const { $cursor } = view.state.selection as TextSelection
if ($cursor) return run(view, $cursor.pos, $cursor.pos, '\n', rules, plugin)
return false
},
},
})

return plugin
}