2022-02-24 16:24:49 +00:00
|
|
|
import {
|
|
|
|
autocompletion,
|
2022-02-25 14:34:00 +00:00
|
|
|
Completion,
|
2022-02-24 16:24:49 +00:00
|
|
|
CompletionContext,
|
|
|
|
completionKeymap,
|
|
|
|
CompletionResult,
|
|
|
|
} from "@codemirror/autocomplete";
|
2022-02-21 08:32:36 +00:00
|
|
|
import { closeBrackets, closeBracketsKeymap } from "@codemirror/closebrackets";
|
|
|
|
import { indentWithTab, standardKeymap } from "@codemirror/commands";
|
|
|
|
import { history, historyKeymap } from "@codemirror/history";
|
2022-02-24 16:24:49 +00:00
|
|
|
import { indentOnInput, syntaxTree } from "@codemirror/language";
|
2022-02-21 08:32:36 +00:00
|
|
|
import { bracketMatching } from "@codemirror/matchbrackets";
|
|
|
|
import { searchKeymap } from "@codemirror/search";
|
2022-02-21 10:27:30 +00:00
|
|
|
import { EditorState, StateField, Transaction } from "@codemirror/state";
|
2022-02-21 08:32:36 +00:00
|
|
|
import {
|
|
|
|
drawSelection,
|
|
|
|
dropCursor,
|
|
|
|
EditorView,
|
|
|
|
highlightSpecialChars,
|
2022-02-25 14:34:00 +00:00
|
|
|
KeyBinding,
|
2022-02-21 08:32:36 +00:00
|
|
|
keymap,
|
|
|
|
} from "@codemirror/view";
|
2022-02-24 16:24:49 +00:00
|
|
|
import React, { useEffect, useReducer } from "react";
|
2022-02-21 10:27:30 +00:00
|
|
|
import ReactDOM from "react-dom";
|
2022-02-24 16:24:49 +00:00
|
|
|
import coreManifest from "../../plugins/dist/core.plugin.json";
|
|
|
|
import { buildContext } from "./buildContext";
|
2022-02-21 08:32:36 +00:00
|
|
|
import * as commands from "./commands";
|
2022-02-25 10:27:58 +00:00
|
|
|
import { CommandPalette } from "./components/command_palette";
|
2022-02-24 16:24:49 +00:00
|
|
|
import { NavigationBar } from "./components/navigation_bar";
|
2022-02-25 10:27:58 +00:00
|
|
|
import { NuggetNavigator } from "./components/nugget_navigator";
|
2022-02-24 16:24:49 +00:00
|
|
|
import { StatusBar } from "./components/status_bar";
|
2022-02-25 14:34:00 +00:00
|
|
|
import { FileSystem } from "./fs";
|
2022-02-21 08:32:36 +00:00
|
|
|
import { lineWrapper } from "./lineWrapper";
|
2022-02-21 10:27:30 +00:00
|
|
|
import { markdown } from "./markdown";
|
2022-02-21 08:32:36 +00:00
|
|
|
import customMarkDown from "./parser";
|
2022-02-24 16:24:49 +00:00
|
|
|
import { BrowserSystem } from "./plugins/browser_system";
|
2022-02-25 14:34:00 +00:00
|
|
|
import { Manifest, slashCommandRegexp } from "./plugins/types";
|
2022-02-22 13:18:37 +00:00
|
|
|
import reducer from "./reducer";
|
2022-02-21 08:32:36 +00:00
|
|
|
import customMarkdownStyle from "./style";
|
2022-02-24 16:24:49 +00:00
|
|
|
import dbSyscalls from "./syscalls/db.localstorage";
|
2022-02-26 11:59:16 +00:00
|
|
|
import { Plugin } from "./plugins/runtime";
|
2022-02-24 16:24:49 +00:00
|
|
|
import editorSyscalls from "./syscalls/editor.browser";
|
|
|
|
import {
|
|
|
|
Action,
|
|
|
|
AppCommand,
|
2022-02-26 11:59:16 +00:00
|
|
|
AppEvent,
|
2022-02-24 16:24:49 +00:00
|
|
|
AppViewState,
|
|
|
|
CommandContext,
|
|
|
|
initialViewState,
|
2022-02-25 14:34:00 +00:00
|
|
|
NuggetMeta,
|
2022-02-24 16:24:49 +00:00
|
|
|
} from "./types";
|
|
|
|
import { safeRun } from "./util";
|
2022-02-23 13:09:26 +00:00
|
|
|
|
2022-02-25 10:27:58 +00:00
|
|
|
class NuggetState {
|
2022-02-23 13:09:26 +00:00
|
|
|
editorState: EditorState;
|
|
|
|
scrollTop: number;
|
2022-02-25 14:34:00 +00:00
|
|
|
meta: NuggetMeta;
|
2022-02-23 13:09:26 +00:00
|
|
|
|
2022-02-25 14:34:00 +00:00
|
|
|
constructor(editorState: EditorState, scrollTop: number, meta: NuggetMeta) {
|
2022-02-23 13:09:26 +00:00
|
|
|
this.editorState = editorState;
|
|
|
|
this.scrollTop = scrollTop;
|
2022-02-25 14:34:00 +00:00
|
|
|
this.meta = meta;
|
2022-02-23 13:09:26 +00:00
|
|
|
}
|
|
|
|
}
|
2022-02-22 16:36:24 +00:00
|
|
|
|
2022-02-25 14:34:00 +00:00
|
|
|
const watchInterval = 5000;
|
|
|
|
|
2022-02-24 16:24:49 +00:00
|
|
|
export class Editor {
|
2022-02-23 13:09:26 +00:00
|
|
|
editorView?: EditorView;
|
|
|
|
viewState: AppViewState;
|
|
|
|
viewDispatch: React.Dispatch<Action>;
|
|
|
|
$hashChange?: () => void;
|
2022-02-25 10:27:58 +00:00
|
|
|
openNuggets: Map<string, NuggetState>;
|
2022-02-23 13:09:26 +00:00
|
|
|
fs: FileSystem;
|
2022-02-24 16:24:49 +00:00
|
|
|
editorCommands: Map<string, AppCommand>;
|
2022-02-26 11:59:16 +00:00
|
|
|
plugins: Plugin[];
|
2022-02-23 13:09:26 +00:00
|
|
|
|
|
|
|
constructor(fs: FileSystem, parent: Element) {
|
2022-02-24 16:24:49 +00:00
|
|
|
this.editorCommands = new Map();
|
2022-02-25 10:27:58 +00:00
|
|
|
this.openNuggets = new Map();
|
2022-02-26 11:59:16 +00:00
|
|
|
this.plugins = [];
|
2022-02-23 13:09:26 +00:00
|
|
|
this.fs = fs;
|
|
|
|
this.viewState = initialViewState;
|
|
|
|
this.viewDispatch = () => {};
|
|
|
|
this.render(parent);
|
|
|
|
this.editorView = new EditorView({
|
|
|
|
state: this.createEditorState(""),
|
|
|
|
parent: document.getElementById("editor")!,
|
2022-02-21 08:32:36 +00:00
|
|
|
});
|
2022-02-23 13:09:26 +00:00
|
|
|
this.addListeners();
|
2022-02-26 11:59:16 +00:00
|
|
|
// this.watch();
|
2022-02-24 16:24:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async init() {
|
2022-02-25 10:27:58 +00:00
|
|
|
await this.loadNuggetList();
|
2022-02-24 16:24:49 +00:00
|
|
|
await this.loadPlugins();
|
2022-02-23 13:09:26 +00:00
|
|
|
this.$hashChange!();
|
2022-02-24 16:24:49 +00:00
|
|
|
this.focus();
|
2022-02-26 11:59:16 +00:00
|
|
|
await this.dispatchAppEvent("ready");
|
2022-02-24 16:24:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async loadPlugins() {
|
|
|
|
const system = new BrowserSystem("plugin");
|
|
|
|
system.registerSyscalls(dbSyscalls, editorSyscalls(this));
|
|
|
|
|
|
|
|
await system.bootServiceWorker();
|
|
|
|
console.log("Now loading core plugin");
|
2022-02-26 11:59:16 +00:00
|
|
|
let mainPlugin = await system.load("core", coreManifest);
|
|
|
|
this.plugins.push(mainPlugin);
|
2022-02-24 16:24:49 +00:00
|
|
|
this.editorCommands = new Map<string, AppCommand>();
|
2022-02-26 11:59:16 +00:00
|
|
|
for (let plugin of this.plugins) {
|
|
|
|
this.buildCommands(plugin);
|
|
|
|
}
|
|
|
|
this.viewDispatch({
|
|
|
|
type: "update-commands",
|
|
|
|
commands: this.editorCommands,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
private buildCommands(plugin: Plugin) {
|
|
|
|
const cmds = plugin.manifest!.commands;
|
2022-02-24 16:24:49 +00:00
|
|
|
for (let name in cmds) {
|
|
|
|
let cmd = cmds[name];
|
|
|
|
this.editorCommands.set(name, {
|
|
|
|
command: cmd,
|
|
|
|
run: async (arg: CommandContext): Promise<any> => {
|
2022-02-26 11:59:16 +00:00
|
|
|
return await plugin.invoke(cmd.invoke, [arg]);
|
2022-02-24 16:24:49 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
2022-02-26 11:59:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Parallelize?
|
|
|
|
async dispatchAppEvent(name: AppEvent, data?: any) {
|
|
|
|
for (let plugin of this.plugins) {
|
|
|
|
await plugin.dispatchEvent(name, data);
|
|
|
|
}
|
2022-02-23 13:09:26 +00:00
|
|
|
}
|
|
|
|
|
2022-02-25 14:34:00 +00:00
|
|
|
get currentNugget(): NuggetMeta | undefined {
|
2022-02-25 10:27:58 +00:00
|
|
|
return this.viewState.currentNugget;
|
2022-02-21 08:32:36 +00:00
|
|
|
}
|
|
|
|
|
2022-02-21 10:27:30 +00:00
|
|
|
createEditorState(text: string): EditorState {
|
2022-02-22 13:18:37 +00:00
|
|
|
const editor = this;
|
2022-02-24 16:24:49 +00:00
|
|
|
let commandKeyBindings: KeyBinding[] = [];
|
|
|
|
for (let def of this.editorCommands.values()) {
|
|
|
|
if (def.command.key) {
|
|
|
|
commandKeyBindings.push({
|
|
|
|
key: def.command.key,
|
|
|
|
mac: def.command.mac,
|
|
|
|
run: (): boolean => {
|
|
|
|
Promise.resolve()
|
|
|
|
.then(async () => {
|
|
|
|
await def.run(buildContext(def, this));
|
|
|
|
})
|
|
|
|
.catch((e) => console.error(e));
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2022-02-21 08:32:36 +00:00
|
|
|
return EditorState.create({
|
|
|
|
doc: text,
|
|
|
|
extensions: [
|
|
|
|
highlightSpecialChars(),
|
|
|
|
history(),
|
|
|
|
drawSelection(),
|
|
|
|
dropCursor(),
|
|
|
|
indentOnInput(),
|
|
|
|
customMarkdownStyle,
|
|
|
|
bracketMatching(),
|
|
|
|
closeBrackets(),
|
2022-02-22 16:36:24 +00:00
|
|
|
autocompletion({
|
2022-02-25 14:34:00 +00:00
|
|
|
override: [
|
|
|
|
this.nuggetCompleter.bind(this),
|
|
|
|
this.commandCompleter.bind(this),
|
|
|
|
],
|
2022-02-22 16:36:24 +00:00
|
|
|
}),
|
2022-02-21 08:32:36 +00:00
|
|
|
EditorView.lineWrapping,
|
|
|
|
lineWrapper([
|
|
|
|
{ selector: "ATXHeading1", class: "line-h1" },
|
|
|
|
{ selector: "ATXHeading2", class: "line-h2" },
|
|
|
|
{ selector: "ListItem", class: "line-li" },
|
|
|
|
{ selector: "Blockquote", class: "line-blockquote" },
|
|
|
|
{ selector: "CodeBlock", class: "line-code" },
|
|
|
|
{ selector: "FencedCode", class: "line-fenced-code" },
|
|
|
|
]),
|
|
|
|
keymap.of([
|
|
|
|
...closeBracketsKeymap,
|
|
|
|
...standardKeymap,
|
|
|
|
...searchKeymap,
|
|
|
|
...historyKeymap,
|
|
|
|
...completionKeymap,
|
|
|
|
indentWithTab,
|
2022-02-24 16:24:49 +00:00
|
|
|
...commandKeyBindings,
|
2022-02-21 08:32:36 +00:00
|
|
|
{
|
|
|
|
key: "Ctrl-b",
|
|
|
|
mac: "Cmd-b",
|
|
|
|
run: commands.insertMarker("**"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: "Ctrl-i",
|
|
|
|
mac: "Cmd-i",
|
|
|
|
run: commands.insertMarker("_"),
|
|
|
|
},
|
|
|
|
{
|
2022-02-25 14:34:00 +00:00
|
|
|
key: "Ctrl-e",
|
|
|
|
mac: "Cmd-e",
|
|
|
|
run: (): boolean => {
|
|
|
|
window.open(location.href, "_blank")!.focus();
|
2022-02-21 08:32:36 +00:00
|
|
|
return true;
|
|
|
|
},
|
|
|
|
},
|
2022-02-21 10:27:30 +00:00
|
|
|
{
|
|
|
|
key: "Ctrl-p",
|
|
|
|
mac: "Cmd-p",
|
|
|
|
run: (target): boolean => {
|
2022-02-23 13:09:26 +00:00
|
|
|
this.viewDispatch({ type: "start-navigate" });
|
2022-02-21 10:27:30 +00:00
|
|
|
return true;
|
|
|
|
},
|
|
|
|
},
|
2022-02-22 13:18:37 +00:00
|
|
|
{
|
|
|
|
key: "Ctrl-.",
|
|
|
|
mac: "Cmd-.",
|
|
|
|
run: (target): boolean => {
|
2022-02-25 14:34:00 +00:00
|
|
|
this.viewDispatch({
|
|
|
|
type: "show-palette",
|
|
|
|
});
|
2022-02-22 13:18:37 +00:00
|
|
|
return true;
|
|
|
|
},
|
|
|
|
},
|
2022-02-21 08:32:36 +00:00
|
|
|
]),
|
|
|
|
EditorView.domEventHandlers({
|
2022-02-21 10:27:30 +00:00
|
|
|
click: this.click.bind(this),
|
2022-02-21 08:32:36 +00:00
|
|
|
}),
|
|
|
|
markdown({
|
|
|
|
base: customMarkDown,
|
|
|
|
}),
|
|
|
|
StateField.define({
|
|
|
|
create: () => null,
|
2022-02-21 10:27:30 +00:00
|
|
|
update: this.update.bind(this),
|
2022-02-21 08:32:36 +00:00
|
|
|
}),
|
|
|
|
],
|
|
|
|
});
|
|
|
|
}
|
2022-02-21 10:27:30 +00:00
|
|
|
|
2022-02-25 10:27:58 +00:00
|
|
|
nuggetCompleter(ctx: CompletionContext): CompletionResult | null {
|
2022-02-25 14:34:00 +00:00
|
|
|
let prefix = ctx.matchBefore(/\[\[[\w\s]*/);
|
2022-02-22 16:36:24 +00:00
|
|
|
if (!prefix) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
from: prefix.from + 2,
|
2022-02-25 10:27:58 +00:00
|
|
|
options: this.viewState.allNuggets.map((nuggetMeta) => ({
|
|
|
|
label: nuggetMeta.name,
|
|
|
|
type: "nugget",
|
2022-02-22 16:36:24 +00:00
|
|
|
})),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-02-25 14:34:00 +00:00
|
|
|
commandCompleter(ctx: CompletionContext): CompletionResult | null {
|
|
|
|
let prefix = ctx.matchBefore(slashCommandRegexp);
|
|
|
|
if (!prefix) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
let options: Completion[] = [];
|
|
|
|
for (let [name, def] of this.viewState.commands) {
|
|
|
|
if (!def.command.slashCommand) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
options.push({
|
|
|
|
label: def.command.slashCommand,
|
|
|
|
detail: name,
|
|
|
|
apply: () => {
|
|
|
|
this.editorView?.dispatch({
|
|
|
|
changes: {
|
|
|
|
from: prefix!.from,
|
|
|
|
to: ctx.pos,
|
|
|
|
insert: "",
|
|
|
|
},
|
|
|
|
});
|
|
|
|
safeRun(async () => {
|
|
|
|
def.run(buildContext(def, this));
|
|
|
|
});
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
from: prefix.from + 1,
|
|
|
|
options: options,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-02-21 10:27:30 +00:00
|
|
|
update(value: null, transaction: Transaction): null {
|
|
|
|
if (transaction.docChanged) {
|
2022-02-23 13:09:26 +00:00
|
|
|
this.viewDispatch({
|
2022-02-25 10:27:58 +00:00
|
|
|
type: "nugget-updated",
|
2022-02-21 10:27:30 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
click(event: MouseEvent, view: EditorView) {
|
|
|
|
if (event.metaKey || event.ctrlKey) {
|
2022-02-22 13:18:37 +00:00
|
|
|
let coords = view.posAtCoords(event)!;
|
|
|
|
let node = syntaxTree(view.state).resolveInner(coords);
|
|
|
|
if (node && node.name === "WikiLinkPage") {
|
2022-02-25 10:27:58 +00:00
|
|
|
let nuggetName = view.state.sliceDoc(node.from, node.to);
|
|
|
|
this.navigate(nuggetName);
|
2022-02-22 13:18:37 +00:00
|
|
|
}
|
2022-02-22 16:36:24 +00:00
|
|
|
if (node && node.name === "TaskMarker") {
|
|
|
|
let checkBoxText = view.state.sliceDoc(node.from, node.to);
|
|
|
|
if (checkBoxText === "[x]" || checkBoxText === "[X]") {
|
|
|
|
view.dispatch({
|
|
|
|
changes: { from: node.from, to: node.to, insert: "[ ]" },
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
view.dispatch({
|
|
|
|
changes: { from: node.from, to: node.to, insert: "[x]" },
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2022-02-21 10:27:30 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async save() {
|
2022-02-23 13:09:26 +00:00
|
|
|
const editorState = this.editorView!.state;
|
|
|
|
|
2022-02-25 10:27:58 +00:00
|
|
|
if (!this.currentNugget) {
|
2022-02-23 13:09:26 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Write to file system
|
2022-02-25 14:34:00 +00:00
|
|
|
let nuggetMeta = await this.fs.writeNugget(
|
|
|
|
this.currentNugget.name,
|
2022-02-23 13:09:26 +00:00
|
|
|
editorState.sliceDoc()
|
2022-02-22 13:18:37 +00:00
|
|
|
);
|
2022-02-23 13:09:26 +00:00
|
|
|
|
2022-02-25 10:27:58 +00:00
|
|
|
// Update in open nugget cache
|
|
|
|
this.openNuggets.set(
|
2022-02-25 14:34:00 +00:00
|
|
|
this.currentNugget.name,
|
|
|
|
new NuggetState(
|
|
|
|
editorState,
|
|
|
|
this.editorView!.scrollDOM.scrollTop,
|
|
|
|
nuggetMeta
|
|
|
|
)
|
2022-02-23 13:09:26 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
// Dispatch update to view
|
2022-02-25 14:34:00 +00:00
|
|
|
this.viewDispatch({ type: "nugget-saved", meta: nuggetMeta });
|
2022-02-23 13:09:26 +00:00
|
|
|
|
2022-02-25 10:27:58 +00:00
|
|
|
// If a new nugget was created, let's refresh the nugget list
|
2022-02-25 14:34:00 +00:00
|
|
|
if (nuggetMeta.created) {
|
2022-02-25 10:27:58 +00:00
|
|
|
await this.loadNuggetList();
|
2022-02-22 13:18:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-25 10:27:58 +00:00
|
|
|
async loadNuggetList() {
|
|
|
|
let nuggetsMeta = await this.fs.listNuggets();
|
2022-02-23 13:09:26 +00:00
|
|
|
this.viewDispatch({
|
2022-02-25 10:27:58 +00:00
|
|
|
type: "nuggets-listed",
|
|
|
|
nuggets: nuggetsMeta,
|
2022-02-22 13:18:37 +00:00
|
|
|
});
|
2022-02-21 10:27:30 +00:00
|
|
|
}
|
|
|
|
|
2022-02-25 14:34:00 +00:00
|
|
|
watch() {
|
|
|
|
setInterval(() => {
|
|
|
|
safeRun(async () => {
|
|
|
|
if (!this.currentNugget) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const currentNuggetName = this.currentNugget.name;
|
|
|
|
let newNuggetMeta = await this.fs.getMeta(currentNuggetName);
|
|
|
|
if (
|
|
|
|
this.currentNugget.lastModified.getTime() <
|
|
|
|
newNuggetMeta.lastModified.getTime()
|
|
|
|
) {
|
|
|
|
console.log("File changed on disk, reloading");
|
|
|
|
let nuggetData = await this.fs.readNugget(currentNuggetName);
|
|
|
|
this.openNuggets.set(
|
|
|
|
newNuggetMeta.name,
|
|
|
|
new NuggetState(
|
|
|
|
this.createEditorState(nuggetData.text),
|
|
|
|
0,
|
|
|
|
newNuggetMeta
|
|
|
|
)
|
|
|
|
);
|
|
|
|
await this.loadNugget(currentNuggetName);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}, watchInterval);
|
|
|
|
}
|
|
|
|
|
2022-02-21 10:27:30 +00:00
|
|
|
focus() {
|
2022-02-23 13:09:26 +00:00
|
|
|
this.editorView!.focus();
|
2022-02-21 10:27:30 +00:00
|
|
|
}
|
2022-02-21 12:25:41 +00:00
|
|
|
|
2022-02-23 13:09:26 +00:00
|
|
|
async navigate(name: string) {
|
2022-02-21 12:25:41 +00:00
|
|
|
location.hash = encodeURIComponent(name);
|
|
|
|
}
|
2022-02-21 08:32:36 +00:00
|
|
|
|
2022-02-23 13:09:26 +00:00
|
|
|
hashChange() {
|
|
|
|
Promise.resolve()
|
|
|
|
.then(async () => {
|
|
|
|
await this.save();
|
2022-02-25 10:27:58 +00:00
|
|
|
const nuggetName = decodeURIComponent(location.hash.substring(1));
|
|
|
|
console.log("Now navigating to", nuggetName);
|
2022-02-23 13:09:26 +00:00
|
|
|
|
|
|
|
if (!this.editorView) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-02-25 14:34:00 +00:00
|
|
|
await this.loadNugget(nuggetName);
|
2022-02-23 13:09:26 +00:00
|
|
|
})
|
|
|
|
.catch((e) => {
|
|
|
|
console.error(e);
|
|
|
|
});
|
|
|
|
}
|
2022-02-21 08:32:36 +00:00
|
|
|
|
2022-02-25 14:34:00 +00:00
|
|
|
async loadNugget(nuggetName: string) {
|
|
|
|
let nuggetState = this.openNuggets.get(nuggetName);
|
|
|
|
if (!nuggetState) {
|
|
|
|
let nuggetData = await this.fs.readNugget(nuggetName);
|
|
|
|
nuggetState = new NuggetState(
|
|
|
|
this.createEditorState(nuggetData.text),
|
|
|
|
0,
|
|
|
|
nuggetData.meta
|
|
|
|
);
|
|
|
|
this.openNuggets.set(nuggetName, nuggetState!);
|
|
|
|
}
|
|
|
|
this.editorView!.setState(nuggetState!.editorState);
|
|
|
|
this.editorView!.scrollDOM.scrollTop = nuggetState!.scrollTop;
|
|
|
|
|
|
|
|
this.viewDispatch({
|
|
|
|
type: "nugget-loaded",
|
|
|
|
meta: nuggetState.meta,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-02-23 13:09:26 +00:00
|
|
|
addListeners() {
|
|
|
|
this.$hashChange = this.hashChange.bind(this);
|
|
|
|
window.addEventListener("hashchange", this.$hashChange);
|
2022-02-22 13:18:37 +00:00
|
|
|
}
|
2022-02-21 10:27:30 +00:00
|
|
|
|
2022-02-23 13:09:26 +00:00
|
|
|
dispose() {
|
|
|
|
if (this.$hashChange) {
|
|
|
|
window.removeEventListener("hashchange", this.$hashChange);
|
2022-02-21 12:25:41 +00:00
|
|
|
}
|
2022-02-23 13:09:26 +00:00
|
|
|
}
|
2022-02-21 10:27:30 +00:00
|
|
|
|
2022-02-23 13:09:26 +00:00
|
|
|
ViewComponent(): React.ReactElement {
|
|
|
|
const [viewState, dispatch] = useReducer(reducer, initialViewState);
|
|
|
|
this.viewState = viewState;
|
|
|
|
this.viewDispatch = dispatch;
|
2022-02-21 08:32:36 +00:00
|
|
|
|
2022-02-23 13:09:26 +00:00
|
|
|
useEffect(() => {
|
|
|
|
if (!location.hash) {
|
|
|
|
this.navigate("start");
|
2022-02-22 13:18:37 +00:00
|
|
|
}
|
2022-02-23 13:09:26 +00:00
|
|
|
}, []);
|
|
|
|
|
|
|
|
// Auto save
|
|
|
|
useEffect(() => {
|
|
|
|
const id = setTimeout(() => {
|
|
|
|
if (!viewState.isSaved) {
|
|
|
|
this.save();
|
|
|
|
}
|
|
|
|
}, 2000);
|
|
|
|
return () => {
|
|
|
|
clearTimeout(id);
|
|
|
|
};
|
|
|
|
}, [viewState.isSaved]);
|
|
|
|
|
|
|
|
let editor = this;
|
|
|
|
|
2022-02-25 10:27:58 +00:00
|
|
|
useEffect(() => {
|
|
|
|
if (viewState.currentNugget) {
|
2022-02-25 14:34:00 +00:00
|
|
|
document.title = viewState.currentNugget.name;
|
2022-02-25 10:27:58 +00:00
|
|
|
}
|
|
|
|
}, [viewState.currentNugget]);
|
2022-02-23 13:09:26 +00:00
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
2022-02-25 10:27:58 +00:00
|
|
|
{viewState.showNuggetNavigator && (
|
|
|
|
<NuggetNavigator
|
|
|
|
allNuggets={viewState.allNuggets}
|
|
|
|
onNavigate={(nugget) => {
|
2022-02-23 13:09:26 +00:00
|
|
|
dispatch({ type: "stop-navigate" });
|
|
|
|
editor!.focus();
|
2022-02-25 10:27:58 +00:00
|
|
|
if (nugget) {
|
2022-02-23 13:09:26 +00:00
|
|
|
editor
|
|
|
|
?.save()
|
|
|
|
.then(() => {
|
2022-02-25 10:27:58 +00:00
|
|
|
editor!.navigate(nugget);
|
2022-02-23 13:09:26 +00:00
|
|
|
})
|
|
|
|
.catch((e) => {
|
2022-02-25 10:27:58 +00:00
|
|
|
alert("Could not save nugget, not switching");
|
2022-02-23 13:09:26 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
{viewState.showCommandPalette && (
|
|
|
|
<CommandPalette
|
|
|
|
onTrigger={(cmd) => {
|
|
|
|
dispatch({ type: "hide-palette" });
|
|
|
|
editor!.focus();
|
|
|
|
if (cmd) {
|
2022-02-24 16:24:49 +00:00
|
|
|
safeRun(async () => {
|
|
|
|
let result = await cmd.run(buildContext(cmd, editor));
|
|
|
|
console.log("Result of command", result);
|
|
|
|
});
|
2022-02-23 13:09:26 +00:00
|
|
|
}
|
|
|
|
}}
|
2022-02-24 16:24:49 +00:00
|
|
|
commands={viewState.commands}
|
2022-02-23 13:09:26 +00:00
|
|
|
/>
|
|
|
|
)}
|
|
|
|
<NavigationBar
|
2022-02-25 10:27:58 +00:00
|
|
|
currentNugget={viewState.currentNugget}
|
2022-02-23 13:09:26 +00:00
|
|
|
onClick={() => {
|
|
|
|
dispatch({ type: "start-navigate" });
|
2022-02-22 13:18:37 +00:00
|
|
|
}}
|
|
|
|
/>
|
2022-02-23 13:09:26 +00:00
|
|
|
<div id="editor"></div>
|
|
|
|
<StatusBar isSaved={viewState.isSaved} editorView={this.editorView} />
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
render(container: ReactDOM.Container) {
|
|
|
|
const ViewComponent = this.ViewComponent.bind(this);
|
|
|
|
ReactDOM.render(<ViewComponent />, container);
|
|
|
|
}
|
2022-02-21 08:32:36 +00:00
|
|
|
}
|