silverbullet/web/editor_state.ts

400 lines
13 KiB
TypeScript
Raw Normal View History

2023-07-14 19:58:16 +08:00
import { readonlyMode } from "./cm_plugins/readonly.ts";
import customMarkdownStyle from "./style.ts";
import {
history,
historyKeymap,
indentWithTab,
standardKeymap,
} from "@codemirror/commands";
2023-07-14 19:58:16 +08:00
import {
autocompletion,
closeBrackets,
closeBracketsKeymap,
completionKeymap,
} from "@codemirror/autocomplete";
import {
codeFolding,
indentOnInput,
LanguageDescription,
LanguageSupport,
syntaxHighlighting,
} from "@codemirror/language";
import { EditorSelection, EditorState } from "@codemirror/state";
import {
2023-07-14 19:58:16 +08:00
dropCursor,
EditorView,
highlightSpecialChars,
KeyBinding,
keymap,
layer,
RectangleMarker,
2023-07-14 19:58:16 +08:00
ViewPlugin,
ViewUpdate,
} from "@codemirror/view";
import { vim } from "@replit/codemirror-vim";
import { markdown } from "@codemirror/lang-markdown";
2023-07-14 22:56:20 +08:00
import { Client } from "./client.ts";
2023-07-14 19:58:16 +08:00
import { inlineImagesPlugin } from "./cm_plugins/inline_image.ts";
import { cleanModePlugins } from "./cm_plugins/clean.ts";
import { lineWrapper } from "./cm_plugins/line_wrapper.ts";
import { smartQuoteKeymap } from "./cm_plugins/smart_quotes.ts";
2024-02-29 22:23:05 +08:00
import { ClickEvent } from "../plug-api/types.ts";
2023-07-14 19:58:16 +08:00
import {
attachmentExtension,
pasteLinkExtension,
} from "./cm_plugins/editor_paste.ts";
import { TextChange } from "./change.ts";
import { postScriptPrefacePlugin } from "./cm_plugins/top_bottom_panels.ts";
import { languageFor } from "$common/languages.ts";
import { plugLinter } from "./cm_plugins/lint.ts";
import { Compartment, Extension } from "@codemirror/state";
import { extendedMarkdownLanguage } from "$common/markdown_parser/parser.ts";
import { parseCommand } from "$common/command.ts";
import { safeRun } from "$lib/async.ts";
import { codeCopyPlugin } from "./cm_plugins/code_copy.ts";
2023-07-14 19:58:16 +08:00
export function createEditorState(
client: Client,
2023-07-14 19:58:16 +08:00
pageName: string,
text: string,
readOnly: boolean,
): EditorState {
let touchCount = 0;
// Ugly: keep the keyhandler compartment in the client, to be replaced later once more commands are loaded
client.keyHandlerCompartment = new Compartment();
const keyBindings = client.keyHandlerCompartment.of(
createKeyBindings(client),
);
2023-07-14 19:58:16 +08:00
return EditorState.create({
doc: text,
extensions: [
// Not using CM theming right now, but some extensions depend on the "dark" thing
2023-07-14 20:22:26 +08:00
EditorView.theme({}, {
dark: client.ui.viewState.uiOptions.darkMode,
2023-07-14 20:22:26 +08:00
}),
2023-07-14 19:58:16 +08:00
// Enable vim mode, or not
[
...client.ui.viewState.uiOptions.vimMode ? [vim({ status: true })] : [],
2023-07-14 20:22:26 +08:00
],
[
...readOnly || client.ui.viewState.uiOptions.forcedROMode
2023-07-14 19:58:16 +08:00
? [readonlyMode()]
: [],
],
2023-07-14 19:58:16 +08:00
// The uber markdown mode
markdown({
2024-01-24 20:34:12 +08:00
base: extendedMarkdownLanguage,
codeLanguages: (info) => {
const lang = languageFor(info);
if (lang) {
return LanguageDescription.of({
name: info,
support: new LanguageSupport(lang),
});
}
return null;
},
2023-07-14 19:58:16 +08:00
addKeymap: true,
}),
2024-01-24 20:34:12 +08:00
extendedMarkdownLanguage.data.of({
2023-07-14 19:58:16 +08:00
closeBrackets: { brackets: ["(", "{", "[", "`"] },
}),
2024-01-24 20:34:12 +08:00
syntaxHighlighting(customMarkdownStyle()),
2023-07-14 19:58:16 +08:00
autocompletion({
override: [
client.editorComplete.bind(client),
client.clientSystem.slashCommandHook.slashCommandCompleter.bind(
client.clientSystem.slashCommandHook,
2023-07-14 19:58:16 +08:00
),
],
}),
inlineImagesPlugin(client),
codeCopyPlugin(client),
2023-07-14 19:58:16 +08:00
highlightSpecialChars(),
history(),
dropCursor(),
codeFolding({
placeholderText: "…",
}),
indentOnInput(),
...cleanModePlugins(client),
2023-07-14 19:58:16 +08:00
EditorView.lineWrapping,
plugLinter(client),
// Taken from https://github.com/codemirror/view/blob/main/src/draw-selection.ts
layer({
above: true,
markers(view) {
const safari = /Apple Computer/.test(navigator.vendor);
const ios = safari &&
(/Mobile\/\w+/.test(navigator.userAgent) ||
navigator.maxTouchPoints > 2);
const { state } = view;
const cursors = [];
for (const r of state.selection.ranges) {
const prim = r == state.selection.main;
if (!r.empty || !prim || !ios) {
const className = prim
? "cm-cursor cm-cursor-primary"
: "cm-cursor cm-cursor-secondary";
const cursor = r.empty
? r
: EditorSelection.cursor(r.head, r.head > r.anchor ? -1 : 1);
for (
const piece of RectangleMarker.forRange(view, className, cursor)
) cursors.push(piece);
}
}
return cursors;
},
update(update, dom) {
if (update.transactions.some((tr) => tr.selection)) {
dom.style.animationName = dom.style.animationName == "cm-blink"
? "cm-blink2"
: "cm-blink";
}
return update.docChanged || update.selectionSet;
},
mount(dom) {
dom.style.animationDuration = "1200ms";
},
class: "cm-cursorLayer",
}),
2023-11-25 20:40:56 +08:00
postScriptPrefacePlugin(client),
2023-07-14 19:58:16 +08:00
lineWrapper([
{ selector: "ATXHeading1", class: "sb-line-h1" },
{ selector: "ATXHeading2", class: "sb-line-h2" },
{ selector: "ATXHeading3", class: "sb-line-h3" },
{ selector: "ATXHeading4", class: "sb-line-h4" },
2024-05-14 19:24:33 +08:00
{ selector: "ATXHeading5", class: "sb-line-h5" },
{ selector: "ATXHeading6", class: "sb-line-h6" },
2023-07-14 19:58:16 +08:00
{ selector: "ListItem", class: "sb-line-li", nesting: true },
{ selector: "Blockquote", class: "sb-line-blockquote" },
{ selector: "Task", class: "sb-line-task" },
{ selector: "CodeBlock", class: "sb-line-code" },
{
selector: "FencedCode",
class: "sb-line-fenced-code",
disableSpellCheck: true,
},
{ selector: "Comment", class: "sb-line-comment" },
{ selector: "BulletList", class: "sb-line-ul" },
{ selector: "OrderedList", class: "sb-line-ol" },
{ selector: "TableHeader", class: "sb-line-tbl-header" },
{
selector: "FrontMatter",
class: "sb-frontmatter",
disableSpellCheck: true,
2023-07-14 19:58:16 +08:00
},
]),
keyBindings,
2023-07-14 19:58:16 +08:00
EditorView.domEventHandlers({
// This may result in duplicated touch events on mobile devices
touchmove: () => {
touchCount++;
},
touchend: (event: TouchEvent, view: EditorView) => {
if (touchCount === 0) {
safeRun(async () => {
const touch = event.changedTouches.item(0)!;
if (!event.altKey && event.target instanceof Element) {
// prevent the browser from opening the link twice
const parentA = event.target.closest("a");
if (parentA) {
event.preventDefault();
}
}
2023-11-29 23:50:53 +08:00
const pos = view.posAtCoords({
x: touch.clientX,
y: touch.clientY,
})!;
const potentialClickEvent: ClickEvent = {
2023-07-14 19:58:16 +08:00
page: pageName,
ctrlKey: event.ctrlKey,
metaKey: event.metaKey,
altKey: event.altKey,
2023-11-29 23:50:53 +08:00
pos: pos,
2023-07-14 19:58:16 +08:00
};
2023-11-29 23:50:53 +08:00
const distanceX = touch.clientX - view.coordsAtPos(pos)!.left;
// What we're trying to determine here is if the tap occured anywhere near the looked up position
// this may not be the case with locations that expand signifcantly based on live preview (such as links), we don't want any accidental clicks
// Fixes #585
//
if (distanceX <= view.defaultCharacterWidth) {
await client.dispatchAppEvent(
"page:click",
potentialClickEvent,
);
}
2023-07-14 19:58:16 +08:00
});
}
touchCount = 0;
},
click: (event: MouseEvent, view: EditorView) => {
2024-01-14 01:42:40 +08:00
const pos = view.posAtCoords(event);
if (event.button !== 0) {
return;
}
if (!pos) {
return;
}
2023-07-14 19:58:16 +08:00
safeRun(async () => {
const potentialClickEvent: ClickEvent = {
page: pageName,
ctrlKey: event.ctrlKey,
metaKey: event.metaKey,
altKey: event.altKey,
pos: view.posAtCoords({
x: event.x,
y: event.y,
})!,
};
// Make sure <a> tags are clicked without moving the cursor there
if (!event.altKey && event.target instanceof Element) {
const parentA = event.target.closest("a");
if (parentA) {
event.stopPropagation();
event.preventDefault();
await client.dispatchAppEvent(
2023-07-14 19:58:16 +08:00
"page:click",
potentialClickEvent,
);
return;
}
}
const distanceX = event.x - view.coordsAtPos(pos)!.left;
// What we're trying to determine here is if the click occured anywhere near the looked up position
// this may not be the case with locations that expand signifcantly based on live preview (such as links), we don't want any accidental clicks
// Fixes #357
if (distanceX <= view.defaultCharacterWidth) {
await client.dispatchAppEvent("page:click", potentialClickEvent);
2023-07-14 19:58:16 +08:00
}
});
},
}),
ViewPlugin.fromClass(
class {
update(update: ViewUpdate): void {
if (update.docChanged) {
WIP: Plug API document change event (#488) * add support for basic on doc change event * move change API core into plug-api/lib; add docs * add overlap utility * Maintain modal focus * Federated URL backend handling * Fix small typo in Query.md (#483) * Federation progress * Cleanup and federation prep * Robustness and federation sync * Federation: rewrite page references in federated content * Don't sync service worker and index.json to client on silverbullet.md * Federation listing timeouts * Switching onboarding over to federation links * Reduce amount of sync related log messages a bit * Attribute indexing and code completion * Shift-Enter in the page navigator now takes the input literally * Updated changelog * Completion for handlebar template variables * Make 'pos' a number in tasks * Updated install instructions to include edge builds * WIP: CLI running of plugs * Upgrade deno in Docker to 1.36.0 * Implement CLI store using Deno store * Rerun directives * Fixes #485 * 0.3.8 * 0.3.9 * Changelog * Instantly sync updated pages when ticking off a task in a directive * Sync current open page every 5s * Optimize requests * Make attribute extensible * Debugging sync getting stuck * Misaligning sync cycles (to avoid no-op cycles) * Fixes #500: New apply page template command * Changelog * More sync debugging statements * More sync debugging * Even more debug * Dial down excessive debug logging * Fixes #115: By introducing MQ workers * Use MQ for updating directives in entire space * Work on plug:run * touch up docs * Fix htmlLanguage dependency --------- Co-authored-by: Zef Hemel <zef@zef.me> Co-authored-by: johnl <johnlunney@users.noreply.github.com>
2023-08-16 21:15:19 +08:00
const changes: TextChange[] = [];
update.changes.iterChanges((fromA, toA, fromB, toB, inserted) =>
changes.push({
inserted: inserted.toString(),
oldRange: { from: fromA, to: toA },
newRange: { from: fromB, to: toB },
})
);
client.dispatchAppEvent("editor:pageModified", { changes });
client.ui.viewDispatch({ type: "page-changed" });
client.debouncedUpdateEvent();
client.save().catch((e) => console.error("Error saving", e));
2023-07-14 19:58:16 +08:00
}
}
},
),
pasteLinkExtension,
attachmentExtension(client),
2023-07-14 19:58:16 +08:00
closeBrackets(),
],
});
}
2024-01-24 21:03:14 +08:00
export function createCommandKeyBindings(client: Client): KeyBinding[] {
const commandKeyBindings: KeyBinding[] = [];
// Track which keyboard shortcuts for which commands we've overridden, so we can skip them later
const overriddenCommands = new Set<string>();
// Keyboard shortcuts from SETTINGS take precedense
if (client.settings?.shortcuts) {
for (const shortcut of client.settings.shortcuts) {
// Figure out if we're using the command link syntax here, if so: parse it out
2024-01-25 18:42:36 +08:00
const parsedCommand = parseCommand(shortcut.command);
if (parsedCommand.args.length === 0) {
// If there was no "specialization" of this command (that is, we effectively created a keybinding for an existing command but with arguments), let's add it to the overridden command set:
2024-01-25 18:42:36 +08:00
overriddenCommands.add(parsedCommand.name);
}
commandKeyBindings.push({
key: shortcut.key,
mac: shortcut.mac,
run: (): boolean => {
2024-01-25 18:42:36 +08:00
client.runCommandByName(parsedCommand.name, parsedCommand.args).catch(
(e: any) => {
console.error(e);
client.flashNotification(
`Error running command: ${e.message}`,
"error",
);
},
2024-01-26 02:46:08 +08:00
).then((returnValue: any) => {
// Always be focusing the editor after running a command
2024-01-26 02:46:08 +08:00
if (returnValue !== false) {
client.focus();
}
});
return true;
},
});
}
}
// Then add bindings for plug commands
for (const def of client.clientSystem.commandHook.editorCommands.values()) {
if (def.command.key) {
// If we've already overridden this command, skip it
if (overriddenCommands.has(def.command.name)) {
continue;
}
commandKeyBindings.push({
key: def.command.key,
mac: def.command.mac,
run: (): boolean => {
if (def.command.contexts) {
const context = client.getContext();
if (!context || !def.command.contexts.includes(context)) {
return false;
}
}
Promise.resolve([])
.then(def.run)
.catch((e: any) => {
console.error(e);
client.flashNotification(
`Error running command: ${e.message}`,
"error",
);
2024-01-26 02:46:08 +08:00
}).then((returnValue: any) => {
// Always be focusing the editor after running a command
2024-01-26 02:46:08 +08:00
if (returnValue !== false) {
client.focus();
}
});
2024-01-26 02:46:08 +08:00
return true;
},
});
}
}
2024-01-24 21:03:14 +08:00
return commandKeyBindings;
}
export function createKeyBindings(client: Client): Extension {
return keymap.of([
2024-01-24 21:03:14 +08:00
...createCommandKeyBindings(client),
...smartQuoteKeymap,
...closeBracketsKeymap,
...standardKeymap,
...completionKeymap,
indentWithTab,
]);
}