silverbullet/web/editor_ui.tsx

374 lines
13 KiB
TypeScript
Raw Permalink Normal View History

2023-07-14 20:22:26 +08:00
import { Confirm, Prompt } from "./components/basic_modals.tsx";
import { CommandPalette } from "./components/command_palette.tsx";
import { FilterList } from "./components/filter.tsx";
import { PageNavigator } from "./components/page_navigator.tsx";
import { TopBar } from "./components/top_bar.tsx";
import reducer from "./reducer.ts";
import { type Action, type AppViewState, initialViewState } from "./type.ts";
import * as featherIcons from "preact-feather";
import * as mdi from "./filtered_material_icons.ts";
import { h, render as preactRender } from "preact";
import { useEffect, useReducer } from "preact/hooks";
import { closeSearchPanel } from "@codemirror/search";
import { runScopeHandlers } from "@codemirror/view";
2023-07-14 22:56:20 +08:00
import type { Client } from "./client.ts";
2023-07-14 20:22:26 +08:00
import { Panel } from "./components/panel.tsx";
import { safeRun, sleep } from "../lib/async.ts";
import { parseCommand } from "$common/command.ts";
2024-08-15 22:39:06 +08:00
import { defaultActionButtons } from "@silverbulletmd/silverbullet/type/config";
2023-07-14 20:22:26 +08:00
export class MainUI {
viewState: AppViewState = initialViewState;
viewDispatch: (action: Action) => void = () => {};
constructor(private client: Client) {
2023-07-14 22:48:35 +08:00
// Make keyboard shortcuts work even when the editor is in read only mode or not focused
globalThis.addEventListener("keydown", (ev) => {
if (!client.editorView.hasFocus) {
2024-01-26 02:46:08 +08:00
const target = ev.target as HTMLElement;
if (target.className === "cm-textfield" && ev.key === "Escape") {
// Search panel is open, let's close it
console.log("Closing search panel");
closeSearchPanel(client.editorView);
return;
} else if (
target.className === "cm-textfield" || target.closest(".cm-content")
) {
2023-07-14 22:48:35 +08:00
// In some cm element, let's back out
return;
}
2024-01-24 21:03:14 +08:00
console.log("Delegated keydown", ev, "to editor");
if (runScopeHandlers(client.editorView, ev, "editor")) {
2023-07-14 22:48:35 +08:00
ev.preventDefault();
}
}
});
globalThis.addEventListener("touchstart", (ev) => {
// Launch the page picker on a two-finger tap
if (ev.touches.length === 2) {
ev.stopPropagation();
ev.preventDefault();
client.startPageNavigate("page");
2023-07-14 22:48:35 +08:00
}
// Launch the command palette using a three-finger tap
if (ev.touches.length === 3) {
ev.stopPropagation();
ev.preventDefault();
this.viewDispatch({
type: "show-palette",
context: client.getContext(),
2023-07-14 22:48:35 +08:00
});
}
});
globalThis.addEventListener("mouseup", (_) => {
setTimeout(() => {
client.editorView.dispatch({});
});
});
2023-07-14 20:22:26 +08:00
}
ViewComponent() {
const [viewState, dispatch] = useReducer(reducer, initialViewState);
this.viewState = viewState;
this.viewDispatch = dispatch;
2023-12-22 01:37:50 +08:00
const client = this.client;
2023-07-14 20:22:26 +08:00
useEffect(() => {
if (viewState.currentPage) {
2024-07-13 21:00:04 +08:00
document.title =
2024-07-14 17:29:43 +08:00
(viewState.currentPageMeta?.pageDecoration?.prefix ?? "") +
2024-07-13 21:00:04 +08:00
viewState.currentPage;
2023-07-14 20:22:26 +08:00
}
2024-07-13 21:00:04 +08:00
}, [viewState.currentPage, viewState.currentPageMeta]);
2023-07-14 20:22:26 +08:00
useEffect(() => {
2023-12-22 01:37:50 +08:00
client.tweakEditorDOM(
client.editorView.contentDOM,
2023-07-27 17:41:44 +08:00
);
2023-07-14 20:22:26 +08:00
}, [viewState.uiOptions.forcedROMode]);
useEffect(() => {
this.client.rebuildEditorState();
this.client.dispatchAppEvent("editor:modeswitch");
2023-07-14 20:22:26 +08:00
}, [viewState.uiOptions.vimMode]);
useEffect(() => {
document.documentElement.dataset.theme = viewState.uiOptions.darkMode
? "dark"
: "light";
}, [viewState.uiOptions.darkMode]);
useEffect(() => {
// Need to dispatch a resize event so that the top_bar can pick it up
globalThis.dispatchEvent(new Event("resize"));
}, [viewState.panels]);
return (
<>
{viewState.showPageNavigator && (
<PageNavigator
allPages={viewState.allPages}
2023-12-22 01:37:50 +08:00
currentPage={client.currentPage}
mode={viewState.pageNavigatorMode}
2023-12-22 01:37:50 +08:00
completer={client.miniEditorComplete.bind(client)}
2023-07-14 20:22:26 +08:00
vimMode={viewState.uiOptions.vimMode}
darkMode={viewState.uiOptions.darkMode}
onModeSwitch={(mode) => {
dispatch({ type: "stop-navigate" });
setTimeout(() => {
dispatch({ type: "start-navigate", mode });
});
}}
2023-07-14 20:22:26 +08:00
onNavigate={(page) => {
dispatch({ type: "stop-navigate" });
setTimeout(() => {
2023-12-22 01:37:50 +08:00
client.focus();
2023-07-14 20:22:26 +08:00
});
if (page) {
safeRun(async () => {
await client.navigate({ page });
2023-07-14 20:22:26 +08:00
});
}
}}
/>
)}
{viewState.showCommandPalette && (
<CommandPalette
onTrigger={(cmd) => {
dispatch({ type: "hide-palette" });
if (cmd) {
dispatch({ type: "command-run", command: cmd.command.name });
cmd
.run()
.catch((e: any) => {
console.error("Error running command", e.message);
})
2024-01-26 02:46:08 +08:00
.then((returnValue: any) => {
2023-07-14 20:22:26 +08:00
// Always be focusing the editor after running a command
2024-01-26 02:46:08 +08:00
if (returnValue !== false) {
client.focus();
}
2023-07-14 20:22:26 +08:00
});
}
}}
2023-12-22 01:37:50 +08:00
commands={client.getCommandsByContext(viewState)}
2023-07-14 20:22:26 +08:00
vimMode={viewState.uiOptions.vimMode}
darkMode={viewState.uiOptions.darkMode}
2023-12-22 01:37:50 +08:00
completer={client.miniEditorComplete.bind(client)}
2023-07-14 20:22:26 +08:00
recentCommands={viewState.recentCommands}
config={this.client.config}
2023-07-14 20:22:26 +08:00
/>
)}
{viewState.showFilterBox && (
<FilterList
label={viewState.filterBoxLabel}
placeholder={viewState.filterBoxPlaceHolder}
options={viewState.filterBoxOptions}
vimMode={viewState.uiOptions.vimMode}
darkMode={viewState.uiOptions.darkMode}
allowNew={false}
2023-12-22 01:37:50 +08:00
completer={client.miniEditorComplete.bind(client)}
2023-07-14 20:22:26 +08:00
helpText={viewState.filterBoxHelpText}
onSelect={viewState.filterBoxOnSelect}
/>
)}
{viewState.showPrompt && (
<Prompt
message={viewState.promptMessage!}
defaultValue={viewState.promptDefaultValue}
vimMode={viewState.uiOptions.vimMode}
darkMode={viewState.uiOptions.darkMode}
2023-12-22 01:37:50 +08:00
completer={client.miniEditorComplete.bind(client)}
2023-07-14 20:22:26 +08:00
callback={(value) => {
dispatch({ type: "hide-prompt" });
viewState.promptCallback!(value);
}}
/>
)}
{viewState.showConfirm && (
<Confirm
message={viewState.confirmMessage!}
callback={(value) => {
dispatch({ type: "hide-confirm" });
viewState.confirmCallback!(value);
}}
/>
)}
<TopBar
pageName={viewState.currentPage}
notifications={viewState.notifications}
2023-08-16 17:40:31 +08:00
syncFailures={viewState.syncFailures}
2023-07-14 20:22:26 +08:00
unsavedChanges={viewState.unsavedChanges}
isLoading={viewState.isLoading}
vimMode={viewState.uiOptions.vimMode}
darkMode={viewState.uiOptions.darkMode}
progressPerc={viewState.progressPerc}
2023-12-22 01:37:50 +08:00
completer={client.miniEditorComplete.bind(client)}
onClick={() => {
2023-12-22 01:37:50 +08:00
client.editorView.scrollDOM.scrollTop = 0;
}}
2023-07-14 20:22:26 +08:00
onRename={async (newName) => {
if (!newName) {
// Always move cursor to the start of the page
2023-12-22 01:37:50 +08:00
client.editorView.dispatch({
2023-07-14 20:22:26 +08:00
selection: { anchor: 0 },
});
2023-12-22 01:37:50 +08:00
client.focus();
2023-07-14 20:22:26 +08:00
return;
}
console.log("Now renaming page to...", newName);
await client.clientSystem.system.invokeFunction(
"index.renamePageCommand",
2023-07-14 20:22:26 +08:00
[{ page: newName }],
);
2023-12-22 01:37:50 +08:00
client.focus();
2023-07-14 20:22:26 +08:00
}}
actionButtons={[
// Sync button
2024-10-10 18:52:28 +08:00
...(!globalThis.silverBulletConfig.syncOnly &&
!viewState.config.hideSyncButton)
2023-08-30 23:25:54 +08:00
// If we support syncOnly, don't show this toggle button
2023-08-30 03:17:29 +08:00
? [{
2024-01-25 18:42:36 +08:00
icon: featherIcons.RefreshCw,
description: this.client.syncMode
? "Currently in Sync mode, click to switch to Online mode"
: "Currently in Online mode, click to switch to Sync mode",
class: this.client.syncMode ? "sb-enabled" : undefined,
2023-08-30 03:17:29 +08:00
callback: () => {
(async () => {
const newValue = !this.client.syncMode;
2023-08-30 03:17:29 +08:00
if (newValue) {
2023-09-05 02:02:35 +08:00
localStorage.setItem("syncMode", "true");
this.client.flashNotification(
2023-09-05 02:02:35 +08:00
"Now switching to sync mode, one moment please...",
);
await sleep(1000);
location.reload();
2023-08-30 03:17:29 +08:00
} else {
2023-09-05 02:02:35 +08:00
localStorage.removeItem("syncMode");
this.client.flashNotification(
2023-09-07 18:52:40 +08:00
"Now switching to online mode, one moment please...",
2023-09-05 02:02:35 +08:00
);
await sleep(1000);
location.reload();
2023-08-30 03:17:29 +08:00
}
})().catch(console.error);
},
}]
: [],
// Edit (reader/writer) button ONLY on mobile
...(viewState.isMobile && !viewState.config.hideEditButton)
? [{
icon: featherIcons.Edit3,
description: viewState.uiOptions.forcedROMode
? "Currently in reader mode, click to switch to writer mode"
: "Currently in writer mode, click to switch to reader mode",
class: !viewState.uiOptions.forcedROMode
? "sb-enabled"
: undefined,
callback: () => {
dispatch({
type: "set-ui-option",
key: "forcedROMode",
value: !viewState.uiOptions.forcedROMode,
});
// After a tick (to have the dispatch update the state), rebuild the editor
setTimeout(() => {
client.rebuildEditorState();
});
},
}]
: [],
// Custom action buttons
...(viewState.config.actionButtons.length > 0
? viewState.config.actionButtons
: defaultActionButtons)
.filter((button) =>
(typeof button.mobile === "undefined") ||
(button.mobile === viewState.isMobile)
)
.map((button) => {
const parsedCommand = parseCommand(button.command);
const mdiIcon = (mdi as any)[kebabToCamel(button.icon)];
let featherIcon =
(featherIcons as any)[kebabToCamel(button.icon)];
if (!featherIcon) {
featherIcon = featherIcons.HelpCircle;
}
return {
icon: mdiIcon ? mdiIcon : featherIcon,
description: button.description || "",
callback: () => {
client.runCommandByName(
parsedCommand.name,
parsedCommand.args,
);
},
href: "",
};
}),
2023-07-14 20:22:26 +08:00
]}
rhs={!!viewState.panels.rhs.mode && (
<div
className="panel"
style={{ flex: viewState.panels.rhs.mode }}
/>
)}
lhs={!!viewState.panels.lhs.mode && (
<div
className="panel"
style={{ flex: viewState.panels.lhs.mode }}
/>
)}
2024-07-14 17:29:43 +08:00
pageNamePrefix={viewState.currentPageMeta?.pageDecoration
?.prefix ??
2024-07-13 21:00:04 +08:00
""}
cssClass={viewState.currentPageMeta?.pageDecoration?.cssClasses
? viewState.currentPageMeta?.pageDecoration?.cssClasses
.join(" ").replaceAll(/[^a-zA-Z0-9-_ ]/g, "")
: ""}
2023-07-14 20:22:26 +08:00
/>
<div id="sb-main">
{!!viewState.panels.lhs.mode && (
2023-12-22 01:37:50 +08:00
<Panel config={viewState.panels.lhs} editor={client} />
2023-07-14 20:22:26 +08:00
)}
<div id="sb-editor" />
{!!viewState.panels.rhs.mode && (
2023-12-22 01:37:50 +08:00
<Panel config={viewState.panels.rhs} editor={client} />
2023-07-14 20:22:26 +08:00
)}
</div>
{!!viewState.panels.modal.mode && (
<div
className="sb-modal"
style={{ inset: `${viewState.panels.modal.mode}px` }}
>
2023-12-22 01:37:50 +08:00
<Panel config={viewState.panels.modal} editor={client} />
2023-07-14 20:22:26 +08:00
</div>
)}
{!!viewState.panels.bhs.mode && (
<div className="sb-bhs">
2023-12-22 01:37:50 +08:00
<Panel config={viewState.panels.bhs} editor={client} />
2023-07-14 20:22:26 +08:00
</div>
)}
</>
);
}
render(container: Element) {
// const ViewComponent = this.ui.ViewComponent.bind(this.ui);
2024-01-28 22:08:35 +08:00
container.innerHTML = "";
2023-07-14 20:22:26 +08:00
preactRender(h(this.ViewComponent.bind(this), {}), container);
}
}
function kebabToCamel(str: string) {
return str.replace(/-([a-z])/g, (g) => g[1].toUpperCase()).replace(
/^./,
(g) => g.toUpperCase(),
);
}