Work
This commit is contained in:
parent
6158713dd1
commit
aa7929ea29
@ -2,3 +2,5 @@ Home page
|
|||||||
|
|
||||||
[[Great Parenting]]
|
[[Great Parenting]]
|
||||||
[[1:1s]]
|
[[1:1s]]
|
||||||
|
|
||||||
|
Sup
|
@ -18,7 +18,7 @@ import ReactDOM from "react-dom";
|
|||||||
import * as commands from "./commands";
|
import * as commands from "./commands";
|
||||||
import { CommandPalette } from "./components/commandpalette";
|
import { CommandPalette } from "./components/commandpalette";
|
||||||
import { NoteNavigator } from "./components/notenavigator";
|
import { NoteNavigator } from "./components/notenavigator";
|
||||||
import { HttpFileSystem } from "./fs";
|
import { FileSystem, HttpFileSystem } from "./fs";
|
||||||
import { lineWrapper } from "./lineWrapper";
|
import { lineWrapper } from "./lineWrapper";
|
||||||
import { markdown } from "./markdown";
|
import { markdown } from "./markdown";
|
||||||
import customMarkDown from "./parser";
|
import customMarkDown from "./parser";
|
||||||
@ -30,10 +30,7 @@ import { syntaxTree } from "@codemirror/language";
|
|||||||
import * as util from "./util";
|
import * as util from "./util";
|
||||||
import { NoteMeta } from "./types";
|
import { NoteMeta } from "./types";
|
||||||
|
|
||||||
const fs = new HttpFileSystem("http://localhost:2222/fs");
|
|
||||||
|
|
||||||
const initialViewState: AppViewState = {
|
const initialViewState: AppViewState = {
|
||||||
currentNote: "",
|
|
||||||
isSaved: false,
|
isSaved: false,
|
||||||
showNoteNavigator: false,
|
showNoteNavigator: false,
|
||||||
showCommandPalette: false,
|
showCommandPalette: false,
|
||||||
@ -41,26 +38,44 @@ const initialViewState: AppViewState = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
import { CompletionContext, CompletionResult } from "@codemirror/autocomplete";
|
import { CompletionContext, CompletionResult } from "@codemirror/autocomplete";
|
||||||
|
import { NavigationBar } from "./components/navigation_bar";
|
||||||
|
import { StatusBar } from "./components/status_bar";
|
||||||
|
|
||||||
|
class NoteState {
|
||||||
|
editorState: EditorState;
|
||||||
|
scrollTop: number;
|
||||||
|
|
||||||
|
constructor(editorState: EditorState, scrollTop: number) {
|
||||||
|
this.editorState = editorState;
|
||||||
|
this.scrollTop = scrollTop;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class Editor {
|
class Editor {
|
||||||
view: EditorView;
|
editorView?: EditorView;
|
||||||
currentNote: string;
|
viewState: AppViewState;
|
||||||
dispatch: React.Dispatch<Action>;
|
viewDispatch: React.Dispatch<Action>;
|
||||||
allNotes: NoteMeta[];
|
$hashChange?: () => void;
|
||||||
|
openNotes: Map<string, NoteState>;
|
||||||
|
fs: FileSystem;
|
||||||
|
|
||||||
constructor(
|
constructor(fs: FileSystem, parent: Element) {
|
||||||
parent: Element,
|
this.fs = fs;
|
||||||
currentNote: string,
|
this.viewState = initialViewState;
|
||||||
text: string,
|
this.viewDispatch = () => {};
|
||||||
dispatch: React.Dispatch<Action>
|
this.render(parent);
|
||||||
) {
|
this.editorView = new EditorView({
|
||||||
this.view = new EditorView({
|
state: this.createEditorState(""),
|
||||||
state: this.createEditorState(text),
|
parent: document.getElementById("editor")!,
|
||||||
parent: parent,
|
|
||||||
});
|
});
|
||||||
this.currentNote = currentNote;
|
this.addListeners();
|
||||||
this.dispatch = dispatch;
|
this.loadNoteList();
|
||||||
this.allNotes = [];
|
this.openNotes = new Map();
|
||||||
|
this.$hashChange!();
|
||||||
|
}
|
||||||
|
|
||||||
|
get currentNote(): string | undefined {
|
||||||
|
return this.viewState.currentNote;
|
||||||
}
|
}
|
||||||
|
|
||||||
createEditorState(text: string): EditorState {
|
createEditorState(text: string): EditorState {
|
||||||
@ -141,7 +156,7 @@ class Editor {
|
|||||||
key: "Ctrl-p",
|
key: "Ctrl-p",
|
||||||
mac: "Cmd-p",
|
mac: "Cmd-p",
|
||||||
run: (target): boolean => {
|
run: (target): boolean => {
|
||||||
this.dispatch({ type: "start-navigate" });
|
this.viewDispatch({ type: "start-navigate" });
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@ -149,7 +164,7 @@ class Editor {
|
|||||||
key: "Ctrl-.",
|
key: "Ctrl-.",
|
||||||
mac: "Cmd-.",
|
mac: "Cmd-.",
|
||||||
run: (target): boolean => {
|
run: (target): boolean => {
|
||||||
this.dispatch({ type: "show-palette" });
|
this.viewDispatch({ type: "show-palette" });
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@ -177,7 +192,7 @@ class Editor {
|
|||||||
// TODO: put something in the cm-completionIcon-note style
|
// TODO: put something in the cm-completionIcon-note style
|
||||||
return {
|
return {
|
||||||
from: prefix.from + 2,
|
from: prefix.from + 2,
|
||||||
options: this.allNotes.map((noteMeta) => ({
|
options: this.viewState.allNotes.map((noteMeta) => ({
|
||||||
label: noteMeta.name,
|
label: noteMeta.name,
|
||||||
type: "note",
|
type: "note",
|
||||||
})),
|
})),
|
||||||
@ -186,7 +201,7 @@ class Editor {
|
|||||||
|
|
||||||
update(value: null, transaction: Transaction): null {
|
update(value: null, transaction: Transaction): null {
|
||||||
if (transaction.docChanged) {
|
if (transaction.docChanged) {
|
||||||
this.dispatch({
|
this.viewDispatch({
|
||||||
type: "note-updated",
|
type: "note-updated",
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -194,11 +209,6 @@ class Editor {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
load(name: string, text: string) {
|
|
||||||
this.currentNote = name;
|
|
||||||
this.view.setState(this.createEditorState(text));
|
|
||||||
}
|
|
||||||
|
|
||||||
click(event: MouseEvent, view: EditorView) {
|
click(event: MouseEvent, view: EditorView) {
|
||||||
if (event.metaKey || event.ctrlKey) {
|
if (event.metaKey || event.ctrlKey) {
|
||||||
let coords = view.posAtCoords(event)!;
|
let coords = view.posAtCoords(event)!;
|
||||||
@ -224,11 +234,26 @@ class Editor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async save() {
|
async save() {
|
||||||
const created = await fs.writeNote(
|
const editorState = this.editorView!.state;
|
||||||
|
|
||||||
|
if (!this.currentNote) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Write to file system
|
||||||
|
const created = await this.fs.writeNote(
|
||||||
this.currentNote,
|
this.currentNote,
|
||||||
this.view.state.sliceDoc()
|
editorState.sliceDoc()
|
||||||
);
|
);
|
||||||
this.dispatch({ type: "note-saved" });
|
|
||||||
|
// Update in open note cache
|
||||||
|
this.openNotes.set(
|
||||||
|
this.currentNote,
|
||||||
|
new NoteState(editorState, this.editorView!.scrollDOM.scrollTop)
|
||||||
|
);
|
||||||
|
|
||||||
|
// Dispatch update to view
|
||||||
|
this.viewDispatch({ type: "note-saved" });
|
||||||
|
|
||||||
// If a new note was created, let's refresh the note list
|
// If a new note was created, let's refresh the note list
|
||||||
if (created) {
|
if (created) {
|
||||||
await this.loadNoteList();
|
await this.loadNoteList();
|
||||||
@ -236,115 +261,94 @@ class Editor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async loadNoteList() {
|
async loadNoteList() {
|
||||||
let notesMeta = await fs.listNotes();
|
let notesMeta = await this.fs.listNotes();
|
||||||
this.allNotes = notesMeta;
|
this.viewDispatch({
|
||||||
this.dispatch({
|
|
||||||
type: "notes-listed",
|
type: "notes-listed",
|
||||||
notes: notesMeta,
|
notes: notesMeta,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
focus() {
|
focus() {
|
||||||
this.view.focus();
|
this.editorView!.focus();
|
||||||
}
|
}
|
||||||
|
|
||||||
navigate(name: string) {
|
async navigate(name: string) {
|
||||||
location.hash = encodeURIComponent(name);
|
location.hash = encodeURIComponent(name);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
let editor: Editor | null;
|
hashChange() {
|
||||||
|
Promise.resolve()
|
||||||
function NavigationBar({
|
.then(async () => {
|
||||||
currentNote,
|
await this.save();
|
||||||
onClick,
|
|
||||||
}: {
|
|
||||||
currentNote: string;
|
|
||||||
onClick: () => void;
|
|
||||||
}) {
|
|
||||||
return (
|
|
||||||
<div id="top">
|
|
||||||
<div className="current-note" onClick={onClick}>
|
|
||||||
» {currentNote}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function StatusBar({ isSaved }: { isSaved: boolean }) {
|
|
||||||
let wordCount = 0,
|
|
||||||
readingTime = 0;
|
|
||||||
if (editor) {
|
|
||||||
let text = editor.view.state.sliceDoc();
|
|
||||||
wordCount = util.countWords(text);
|
|
||||||
readingTime = util.readingTime(wordCount);
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
<div id="bottom">
|
|
||||||
{wordCount} words | {readingTime} min | {isSaved ? "Saved" : "Edited"}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function AppView() {
|
|
||||||
const editorRef = useRef<HTMLDivElement>(null);
|
|
||||||
const [appState, dispatch] = useReducer(reducer, initialViewState);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
editor = new Editor(editorRef.current!, "", "", dispatch);
|
|
||||||
editor.focus();
|
|
||||||
// @ts-ignore
|
|
||||||
window.editor = editor;
|
|
||||||
if (!location.hash) {
|
|
||||||
editor.navigate("start");
|
|
||||||
}
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
editor?.loadNoteList();
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
// Auto save
|
|
||||||
useEffect(() => {
|
|
||||||
const id = setTimeout(() => {
|
|
||||||
if (!appState.isSaved) {
|
|
||||||
editor?.save();
|
|
||||||
}
|
|
||||||
}, 2000);
|
|
||||||
return () => {
|
|
||||||
clearTimeout(id);
|
|
||||||
};
|
|
||||||
}, [appState.isSaved]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
function hashChange() {
|
|
||||||
const noteName = decodeURIComponent(location.hash.substring(1));
|
const noteName = decodeURIComponent(location.hash.substring(1));
|
||||||
console.log("Now navigating to", noteName);
|
console.log("Now navigating to", noteName);
|
||||||
|
|
||||||
fs.readNote(noteName)
|
if (!this.editorView) {
|
||||||
.then((text) => {
|
return;
|
||||||
editor!.load(noteName, text);
|
}
|
||||||
dispatch({
|
|
||||||
|
let noteState = this.openNotes.get(noteName);
|
||||||
|
if (!noteState) {
|
||||||
|
let text = await this.fs.readNote(noteName);
|
||||||
|
noteState = new NoteState(this.createEditorState(text), 0);
|
||||||
|
}
|
||||||
|
this.openNotes.set(noteName, noteState!);
|
||||||
|
this.editorView!.setState(noteState!.editorState);
|
||||||
|
this.editorView.scrollDOM.scrollTop = noteState!.scrollTop;
|
||||||
|
|
||||||
|
this.viewDispatch({
|
||||||
type: "note-loaded",
|
type: "note-loaded",
|
||||||
name: noteName,
|
name: noteName,
|
||||||
});
|
});
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
console.error("Error loading note", e);
|
console.error(e);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
hashChange();
|
|
||||||
window.addEventListener("hashchange", hashChange);
|
addListeners() {
|
||||||
return () => {
|
this.$hashChange = this.hashChange.bind(this);
|
||||||
window.removeEventListener("hashchange", hashChange);
|
window.addEventListener("hashchange", this.$hashChange);
|
||||||
};
|
}
|
||||||
|
|
||||||
|
dispose() {
|
||||||
|
if (this.$hashChange) {
|
||||||
|
window.removeEventListener("hashchange", this.$hashChange);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ViewComponent(): React.ReactElement {
|
||||||
|
const [viewState, dispatch] = useReducer(reducer, initialViewState);
|
||||||
|
this.viewState = viewState;
|
||||||
|
this.viewDispatch = dispatch;
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!location.hash) {
|
||||||
|
this.navigate("start");
|
||||||
|
}
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
// Auto save
|
||||||
|
useEffect(() => {
|
||||||
|
const id = setTimeout(() => {
|
||||||
|
if (!viewState.isSaved) {
|
||||||
|
this.save();
|
||||||
|
}
|
||||||
|
}, 2000);
|
||||||
|
return () => {
|
||||||
|
clearTimeout(id);
|
||||||
|
};
|
||||||
|
}, [viewState.isSaved]);
|
||||||
|
|
||||||
|
let editor = this;
|
||||||
|
|
||||||
|
useEffect(() => {}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{appState.showNoteNavigator && (
|
{viewState.showNoteNavigator && (
|
||||||
<NoteNavigator
|
<NoteNavigator
|
||||||
allNotes={appState.allNotes}
|
allNotes={viewState.allNotes}
|
||||||
onNavigate={(note) => {
|
onNavigate={(note) => {
|
||||||
dispatch({ type: "stop-navigate" });
|
dispatch({ type: "stop-navigate" });
|
||||||
editor!.focus();
|
editor!.focus();
|
||||||
@ -361,7 +365,7 @@ function AppView() {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{appState.showCommandPalette && (
|
{viewState.showCommandPalette && (
|
||||||
<CommandPalette
|
<CommandPalette
|
||||||
onTrigger={(cmd) => {
|
onTrigger={(cmd) => {
|
||||||
dispatch({ type: "hide-palette" });
|
dispatch({ type: "hide-palette" });
|
||||||
@ -374,15 +378,29 @@ function AppView() {
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
<NavigationBar
|
<NavigationBar
|
||||||
currentNote={appState.currentNote}
|
currentNote={viewState.currentNote}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
dispatch({ type: "start-navigate" });
|
dispatch({ type: "start-navigate" });
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<div id="editor" ref={editorRef}></div>
|
<div id="editor"></div>
|
||||||
<StatusBar isSaved={appState.isSaved} />
|
<StatusBar isSaved={viewState.isSaved} editorView={this.editorView} />
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
ReactDOM.render(<AppView />, document.getElementById("root"));
|
render(container: ReactDOM.Container) {
|
||||||
|
const ViewComponent = this.ViewComponent.bind(this);
|
||||||
|
ReactDOM.render(<ViewComponent />, container);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let ed = new Editor(
|
||||||
|
new HttpFileSystem("http://localhost:2222/fs"),
|
||||||
|
document.getElementById("root")!
|
||||||
|
);
|
||||||
|
|
||||||
|
ed.focus();
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
window.editor = ed;
|
||||||
|
15
webapp/src/components/navigation_bar.tsx
Normal file
15
webapp/src/components/navigation_bar.tsx
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
export function NavigationBar({
|
||||||
|
currentNote,
|
||||||
|
onClick,
|
||||||
|
}: {
|
||||||
|
currentNote?: string;
|
||||||
|
onClick: () => void;
|
||||||
|
}) {
|
||||||
|
return (
|
||||||
|
<div id="top">
|
||||||
|
<div className="current-note" onClick={onClick}>
|
||||||
|
» {currentNote}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
23
webapp/src/components/status_bar.tsx
Normal file
23
webapp/src/components/status_bar.tsx
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
import { EditorView } from "@codemirror/view";
|
||||||
|
import * as util from "../util";
|
||||||
|
|
||||||
|
export function StatusBar({
|
||||||
|
isSaved,
|
||||||
|
editorView,
|
||||||
|
}: {
|
||||||
|
isSaved: boolean;
|
||||||
|
editorView?: EditorView;
|
||||||
|
}) {
|
||||||
|
let wordCount = 0,
|
||||||
|
readingTime = 0;
|
||||||
|
if (editorView) {
|
||||||
|
let text = editorView.state.sliceDoc();
|
||||||
|
wordCount = util.countWords(text);
|
||||||
|
readingTime = util.readingTime(wordCount);
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<div id="bottom">
|
||||||
|
{wordCount} words | {readingTime} min | {isSaved ? "Saved" : "Edited"}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
2
webapp/src/index.ts
Normal file
2
webapp/src/index.ts
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
|
||||||
|
|
@ -1,4 +1,3 @@
|
|||||||
|
|
||||||
export type NoteMeta = {
|
export type NoteMeta = {
|
||||||
name: string;
|
name: string;
|
||||||
};
|
};
|
||||||
@ -6,10 +5,10 @@ export type NoteMeta = {
|
|||||||
export type AppCommand = {
|
export type AppCommand = {
|
||||||
name: string;
|
name: string;
|
||||||
run: () => void;
|
run: () => void;
|
||||||
}
|
};
|
||||||
|
|
||||||
export type AppViewState = {
|
export type AppViewState = {
|
||||||
currentNote: string;
|
currentNote?: string;
|
||||||
isSaved: boolean;
|
isSaved: boolean;
|
||||||
showNoteNavigator: boolean;
|
showNoteNavigator: boolean;
|
||||||
showCommandPalette: boolean;
|
showCommandPalette: boolean;
|
||||||
@ -24,6 +23,4 @@ export type Action =
|
|||||||
| { type: "start-navigate" }
|
| { type: "start-navigate" }
|
||||||
| { type: "stop-navigate" }
|
| { type: "stop-navigate" }
|
||||||
| { type: "show-palette" }
|
| { type: "show-palette" }
|
||||||
| { type: "hide-palette" }
|
| { type: "hide-palette" };
|
||||||
;
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user