85 lines
2.3 KiB
TypeScript
85 lines
2.3 KiB
TypeScript
import { KeyBinding } from "@codemirror/view";
|
||
import { syntaxTree } from "@codemirror/language";
|
||
import { EditorSelection } from "@codemirror/state";
|
||
|
||
const straightQuoteContexts = [
|
||
"CommentBlock",
|
||
"CodeBlock",
|
||
"FencedCode",
|
||
"InlineCode",
|
||
"FrontMatterCode",
|
||
"Attribute",
|
||
"CommandLink",
|
||
"TemplateDirective",
|
||
];
|
||
|
||
// TODO: Add support for selection (put quotes around or create blockquote block?)
|
||
function keyBindingForQuote(
|
||
originalQuote: string,
|
||
left: string,
|
||
right: string,
|
||
): KeyBinding {
|
||
return {
|
||
any: (target, event): boolean => {
|
||
// Moving this check here rather than using the regular "key" property because
|
||
// for some reason the "ä" key is not recognized as a quote key by CodeMirror.
|
||
if (event.key !== originalQuote) {
|
||
return false;
|
||
}
|
||
const cursorPos = target.state.selection.main.from;
|
||
const chBefore = target.state.sliceDoc(cursorPos - 1, cursorPos);
|
||
|
||
// Figure out the context, if in some sort of code/comment fragment don't be smart
|
||
let node = syntaxTree(target.state).resolveInner(cursorPos);
|
||
while (node) {
|
||
if (straightQuoteContexts.includes(node.type.name)) {
|
||
return false;
|
||
}
|
||
if (node.parent) {
|
||
node = node.parent;
|
||
} else {
|
||
break;
|
||
}
|
||
}
|
||
|
||
// Ok, still here, let's use a smart quote
|
||
const changes = target.state.changeByRange((range) => {
|
||
if (!range.empty) {
|
||
return {
|
||
changes: [
|
||
{ insert: left, from: range.from },
|
||
{ insert: right, from: range.to },
|
||
],
|
||
range: EditorSelection.range(
|
||
range.anchor + left.length,
|
||
range.head + left.length,
|
||
),
|
||
};
|
||
} else {
|
||
const quote = (/\W/.exec(chBefore) && !/[!\?,\.\-=“]/.exec(chBefore))
|
||
? left
|
||
: right;
|
||
|
||
return {
|
||
changes: {
|
||
insert: quote,
|
||
from: cursorPos,
|
||
},
|
||
range: EditorSelection.cursor(
|
||
range.anchor + quote.length,
|
||
),
|
||
};
|
||
}
|
||
});
|
||
target.dispatch(changes);
|
||
|
||
return true;
|
||
},
|
||
};
|
||
}
|
||
|
||
export const smartQuoteKeymap: KeyBinding[] = [
|
||
keyBindingForQuote('"', "“", "”"),
|
||
keyBindingForQuote("'", "‘", "’"),
|
||
];
|