2022-10-10 20:50:21 +08:00
|
|
|
|
import { KeyBinding } from "./deps.ts";
|
|
|
|
|
import { syntaxTree } from "../common/deps.ts";
|
2022-04-25 00:06:34 +08:00
|
|
|
|
|
2022-10-17 21:48:21 +08:00
|
|
|
|
const straightQuoteContexts = [
|
|
|
|
|
"CommentBlock",
|
|
|
|
|
"FencedCode",
|
|
|
|
|
"InlineCode",
|
|
|
|
|
"FrontMatterCode",
|
|
|
|
|
];
|
2022-03-20 16:56:28 +08:00
|
|
|
|
|
|
|
|
|
// TODO: Add support for selection (put quotes around or create blockquote block?)
|
|
|
|
|
function keyBindingForQuote(
|
|
|
|
|
quote: string,
|
|
|
|
|
left: string,
|
2022-10-10 20:50:21 +08:00
|
|
|
|
right: string,
|
2022-03-20 16:56:28 +08:00
|
|
|
|
): KeyBinding {
|
|
|
|
|
return {
|
|
|
|
|
key: quote,
|
|
|
|
|
run: (target): boolean => {
|
2022-10-17 21:48:21 +08:00
|
|
|
|
const cursorPos = target.state.selection.main.from;
|
|
|
|
|
const chBefore = target.state.sliceDoc(cursorPos - 1, cursorPos);
|
2022-04-25 00:06:34 +08:00
|
|
|
|
|
|
|
|
|
// 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
|
2022-03-20 16:56:28 +08:00
|
|
|
|
let quote = right;
|
|
|
|
|
if (/\W/.exec(chBefore) && !/[!\?,\.\-=“]/.exec(chBefore)) {
|
|
|
|
|
quote = left;
|
|
|
|
|
}
|
|
|
|
|
target.dispatch({
|
|
|
|
|
changes: {
|
|
|
|
|
insert: quote,
|
|
|
|
|
from: cursorPos,
|
|
|
|
|
},
|
|
|
|
|
selection: {
|
|
|
|
|
anchor: cursorPos + 1,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
return true;
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export const smartQuoteKeymap: KeyBinding[] = [
|
|
|
|
|
keyBindingForQuote('"', "“", "”"),
|
|
|
|
|
keyBindingForQuote("'", "‘", "’"),
|
|
|
|
|
];
|