2022-03-31 20:28:07 +08:00
|
|
|
import { Prec } from "@codemirror/state";
|
|
|
|
import { KeyBinding, keymap } from "@codemirror/view";
|
2022-06-14 00:31:36 +08:00
|
|
|
import {
|
|
|
|
Language,
|
|
|
|
LanguageSupport,
|
|
|
|
LanguageDescription,
|
|
|
|
} from "@codemirror/language";
|
2022-03-31 20:28:07 +08:00
|
|
|
import { MarkdownExtension, MarkdownParser, parseCode } from "@lezer/markdown";
|
|
|
|
import { html } from "@codemirror/lang-html";
|
2022-06-14 00:31:36 +08:00
|
|
|
import {
|
|
|
|
commonmarkLanguage,
|
|
|
|
markdownLanguage,
|
|
|
|
mkLang,
|
|
|
|
getCodeParser,
|
|
|
|
} from "./markdown";
|
|
|
|
import { insertNewlineContinueMarkup, deleteMarkupBackward } from "./commands";
|
2022-03-31 20:28:07 +08:00
|
|
|
export {
|
|
|
|
commonmarkLanguage,
|
|
|
|
markdownLanguage,
|
|
|
|
insertNewlineContinueMarkup,
|
|
|
|
deleteMarkupBackward,
|
|
|
|
};
|
2022-03-20 16:56:28 +08:00
|
|
|
|
|
|
|
/// A small keymap with Markdown-specific bindings. Binds Enter to
|
|
|
|
/// [`insertNewlineContinueMarkup`](#lang-markdown.insertNewlineContinueMarkup)
|
|
|
|
/// and Backspace to
|
|
|
|
/// [`deleteMarkupBackward`](#lang-markdown.deleteMarkupBackward).
|
|
|
|
export const markdownKeymap: readonly KeyBinding[] = [
|
2022-03-31 20:28:07 +08:00
|
|
|
{ key: "Enter", run: insertNewlineContinueMarkup },
|
|
|
|
{ key: "Backspace", run: deleteMarkupBackward },
|
|
|
|
];
|
2022-03-20 16:56:28 +08:00
|
|
|
|
2022-03-31 20:28:07 +08:00
|
|
|
const htmlNoMatch = html({ matchClosingTags: false });
|
2022-03-20 16:56:28 +08:00
|
|
|
|
|
|
|
/// Markdown language support.
|
2022-03-31 20:28:07 +08:00
|
|
|
export function markdown(
|
2022-03-31 23:25:34 +08:00
|
|
|
config: {
|
|
|
|
/// When given, this language will be used by default to parse code
|
|
|
|
/// blocks.
|
|
|
|
defaultCodeLanguage?: Language | LanguageSupport;
|
2022-06-14 00:31:36 +08:00
|
|
|
/// A source of language support for highlighting fenced code
|
|
|
|
/// blocks. When it is an array, the parser will use
|
|
|
|
/// [`LanguageDescription.matchLanguageName`](#language.LanguageDescription^matchLanguageName)
|
|
|
|
/// with the fenced code info to find a matching language. When it
|
|
|
|
/// is a function, will be called with the info string and may
|
|
|
|
/// return a language or `LanguageDescription` object.
|
|
|
|
codeLanguages?:
|
|
|
|
| readonly LanguageDescription[]
|
|
|
|
| ((info: string) => Language | LanguageDescription | null);
|
2022-03-31 23:25:34 +08:00
|
|
|
/// Set this to false to disable installation of the Markdown
|
|
|
|
/// [keymap](#lang-markdown.markdownKeymap).
|
|
|
|
addKeymap?: boolean;
|
|
|
|
/// Markdown parser
|
|
|
|
/// [extensions](https://github.com/lezer-parser/markdown#user-content-markdownextension)
|
|
|
|
/// to add to the parser.
|
|
|
|
extensions?: MarkdownExtension;
|
|
|
|
/// The base language to use. Defaults to
|
|
|
|
/// [`commonmarkLanguage`](#lang-markdown.commonmarkLanguage).
|
|
|
|
base?: Language;
|
|
|
|
} = {}
|
2022-03-31 20:28:07 +08:00
|
|
|
) {
|
|
|
|
let {
|
|
|
|
codeLanguages,
|
|
|
|
defaultCodeLanguage,
|
|
|
|
addKeymap = true,
|
2022-03-31 23:25:34 +08:00
|
|
|
base: { parser } = commonmarkLanguage,
|
2022-03-31 20:28:07 +08:00
|
|
|
} = config;
|
|
|
|
if (!(parser instanceof MarkdownParser))
|
|
|
|
throw new RangeError(
|
2022-03-31 23:25:34 +08:00
|
|
|
"Base parser provided to `markdown` should be a Markdown parser"
|
2022-03-31 20:28:07 +08:00
|
|
|
);
|
|
|
|
let extensions = config.extensions ? [config.extensions] : [];
|
|
|
|
let support = [htmlNoMatch.support],
|
2022-03-31 23:25:34 +08:00
|
|
|
defaultCode;
|
2022-03-20 16:56:28 +08:00
|
|
|
if (defaultCodeLanguage instanceof LanguageSupport) {
|
2022-03-31 20:28:07 +08:00
|
|
|
support.push(defaultCodeLanguage.support);
|
|
|
|
defaultCode = defaultCodeLanguage.language;
|
2022-03-20 16:56:28 +08:00
|
|
|
} else if (defaultCodeLanguage) {
|
2022-03-31 20:28:07 +08:00
|
|
|
defaultCode = defaultCodeLanguage;
|
2022-03-20 16:56:28 +08:00
|
|
|
}
|
2022-03-31 20:28:07 +08:00
|
|
|
let codeParser =
|
2022-03-31 23:25:34 +08:00
|
|
|
codeLanguages || defaultCode
|
2022-06-14 00:31:36 +08:00
|
|
|
? getCodeParser(codeLanguages, defaultCode)
|
2022-03-31 23:25:34 +08:00
|
|
|
: undefined;
|
2022-03-31 20:28:07 +08:00
|
|
|
extensions.push(
|
2022-03-31 23:25:34 +08:00
|
|
|
parseCode({ codeParser, htmlParser: htmlNoMatch.language.parser })
|
2022-03-31 20:28:07 +08:00
|
|
|
);
|
|
|
|
if (addKeymap) support.push(Prec.high(keymap.of(markdownKeymap)));
|
|
|
|
return new LanguageSupport(mkLang(parser.configure(extensions)), support);
|
2022-03-20 16:56:28 +08:00
|
|
|
}
|