247 lines
7.9 KiB
TypeScript
247 lines
7.9 KiB
TypeScript
|
import { isMacLike, safeRun } from "../common/util.ts";
|
||
|
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 { Action, AppViewState, initialViewState } from "./types.ts";
|
||
|
import {
|
||
|
BookIcon,
|
||
|
HomeIcon,
|
||
|
preactRender,
|
||
|
TerminalIcon,
|
||
|
useEffect,
|
||
|
useReducer,
|
||
|
} from "./deps.ts";
|
||
|
import type { Editor } from "./editor.ts";
|
||
|
import { Panel } from "./components/panel.tsx";
|
||
|
import { h } from "./deps.ts";
|
||
|
|
||
|
export class MainUI {
|
||
|
viewState: AppViewState = initialViewState;
|
||
|
viewDispatch: (action: Action) => void = () => {};
|
||
|
|
||
|
constructor(private editor: Editor) {
|
||
|
}
|
||
|
|
||
|
ViewComponent() {
|
||
|
const [viewState, dispatch] = useReducer(reducer, initialViewState);
|
||
|
this.viewState = viewState;
|
||
|
this.viewDispatch = dispatch;
|
||
|
|
||
|
const editor = this.editor;
|
||
|
|
||
|
useEffect(() => {
|
||
|
if (viewState.currentPage) {
|
||
|
document.title = viewState.currentPage;
|
||
|
}
|
||
|
}, [viewState.currentPage]);
|
||
|
|
||
|
useEffect(() => {
|
||
|
if (editor.editorView) {
|
||
|
editor.tweakEditorDOM(
|
||
|
editor.editorView.contentDOM,
|
||
|
);
|
||
|
}
|
||
|
}, [viewState.uiOptions.forcedROMode]);
|
||
|
|
||
|
useEffect(() => {
|
||
|
this.editor.rebuildEditorState();
|
||
|
this.editor.dispatchAppEvent("editor:modeswitch");
|
||
|
}, [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}
|
||
|
currentPage={editor.currentPage}
|
||
|
completer={editor.miniEditorComplete.bind(editor)}
|
||
|
vimMode={viewState.uiOptions.vimMode}
|
||
|
darkMode={viewState.uiOptions.darkMode}
|
||
|
onNavigate={(page) => {
|
||
|
dispatch({ type: "stop-navigate" });
|
||
|
setTimeout(() => {
|
||
|
editor.focus();
|
||
|
});
|
||
|
if (page) {
|
||
|
safeRun(async () => {
|
||
|
await editor.navigate(page);
|
||
|
});
|
||
|
}
|
||
|
}}
|
||
|
/>
|
||
|
)}
|
||
|
{viewState.showCommandPalette && (
|
||
|
<CommandPalette
|
||
|
onTrigger={(cmd) => {
|
||
|
dispatch({ type: "hide-palette" });
|
||
|
setTimeout(() => {
|
||
|
editor.focus();
|
||
|
});
|
||
|
if (cmd) {
|
||
|
dispatch({ type: "command-run", command: cmd.command.name });
|
||
|
cmd
|
||
|
.run()
|
||
|
.catch((e: any) => {
|
||
|
console.error("Error running command", e.message);
|
||
|
})
|
||
|
.then(() => {
|
||
|
// Always be focusing the editor after running a command
|
||
|
editor.focus();
|
||
|
});
|
||
|
}
|
||
|
}}
|
||
|
commands={editor.getCommandsByContext(viewState)}
|
||
|
vimMode={viewState.uiOptions.vimMode}
|
||
|
darkMode={viewState.uiOptions.darkMode}
|
||
|
completer={editor.miniEditorComplete.bind(editor)}
|
||
|
recentCommands={viewState.recentCommands}
|
||
|
/>
|
||
|
)}
|
||
|
{viewState.showFilterBox && (
|
||
|
<FilterList
|
||
|
label={viewState.filterBoxLabel}
|
||
|
placeholder={viewState.filterBoxPlaceHolder}
|
||
|
options={viewState.filterBoxOptions}
|
||
|
vimMode={viewState.uiOptions.vimMode}
|
||
|
darkMode={viewState.uiOptions.darkMode}
|
||
|
allowNew={false}
|
||
|
completer={editor.miniEditorComplete.bind(editor)}
|
||
|
helpText={viewState.filterBoxHelpText}
|
||
|
onSelect={viewState.filterBoxOnSelect}
|
||
|
/>
|
||
|
)}
|
||
|
{viewState.showPrompt && (
|
||
|
<Prompt
|
||
|
message={viewState.promptMessage!}
|
||
|
defaultValue={viewState.promptDefaultValue}
|
||
|
vimMode={viewState.uiOptions.vimMode}
|
||
|
darkMode={viewState.uiOptions.darkMode}
|
||
|
completer={editor.miniEditorComplete.bind(editor)}
|
||
|
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}
|
||
|
synced={viewState.synced}
|
||
|
unsavedChanges={viewState.unsavedChanges}
|
||
|
isLoading={viewState.isLoading}
|
||
|
vimMode={viewState.uiOptions.vimMode}
|
||
|
darkMode={viewState.uiOptions.darkMode}
|
||
|
progressPerc={viewState.progressPerc}
|
||
|
completer={editor.miniEditorComplete.bind(editor)}
|
||
|
onRename={async (newName) => {
|
||
|
if (!newName) {
|
||
|
// Always move cursor to the start of the page
|
||
|
editor.editorView?.dispatch({
|
||
|
selection: { anchor: 0 },
|
||
|
});
|
||
|
editor.focus();
|
||
|
return;
|
||
|
}
|
||
|
console.log("Now renaming page to...", newName);
|
||
|
await editor.system.system.loadedPlugs.get("core")!.invoke(
|
||
|
"renamePage",
|
||
|
[{ page: newName }],
|
||
|
);
|
||
|
editor.focus();
|
||
|
}}
|
||
|
actionButtons={[
|
||
|
{
|
||
|
icon: HomeIcon,
|
||
|
description: `Go home (Alt-h)`,
|
||
|
callback: () => {
|
||
|
editor.navigate("");
|
||
|
},
|
||
|
href: "",
|
||
|
},
|
||
|
{
|
||
|
icon: BookIcon,
|
||
|
description: `Open page (${isMacLike() ? "Cmd-k" : "Ctrl-k"})`,
|
||
|
callback: () => {
|
||
|
dispatch({ type: "start-navigate" });
|
||
|
editor.space.updatePageList();
|
||
|
},
|
||
|
},
|
||
|
{
|
||
|
icon: TerminalIcon,
|
||
|
description: `Run command (${isMacLike() ? "Cmd-/" : "Ctrl-/"})`,
|
||
|
callback: () => {
|
||
|
dispatch({
|
||
|
type: "show-palette",
|
||
|
context: editor.getContext(),
|
||
|
});
|
||
|
},
|
||
|
},
|
||
|
]}
|
||
|
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 }}
|
||
|
/>
|
||
|
)}
|
||
|
/>
|
||
|
<div id="sb-main">
|
||
|
{!!viewState.panels.lhs.mode && (
|
||
|
<Panel config={viewState.panels.lhs} editor={editor} />
|
||
|
)}
|
||
|
<div id="sb-editor" />
|
||
|
{!!viewState.panels.rhs.mode && (
|
||
|
<Panel config={viewState.panels.rhs} editor={editor} />
|
||
|
)}
|
||
|
</div>
|
||
|
{!!viewState.panels.modal.mode && (
|
||
|
<div
|
||
|
className="sb-modal"
|
||
|
style={{ inset: `${viewState.panels.modal.mode}px` }}
|
||
|
>
|
||
|
<Panel config={viewState.panels.modal} editor={editor} />
|
||
|
</div>
|
||
|
)}
|
||
|
{!!viewState.panels.bhs.mode && (
|
||
|
<div className="sb-bhs">
|
||
|
<Panel config={viewState.panels.bhs} editor={editor} />
|
||
|
</div>
|
||
|
)}
|
||
|
</>
|
||
|
);
|
||
|
}
|
||
|
|
||
|
render(container: Element) {
|
||
|
// const ViewComponent = this.ui.ViewComponent.bind(this.ui);
|
||
|
preactRender(h(this.ViewComponent.bind(this), {}), container);
|
||
|
}
|
||
|
}
|