silverbullet/web/components/page_navigator.tsx

187 lines
5.9 KiB
TypeScript

import { FilterList } from "./filter.tsx";
import { FilterOption } from "$lib/web.ts";
import { CompletionContext, CompletionResult } from "@codemirror/autocomplete";
import { PageMeta } from "../../plug-api/types.ts";
import { tagRegex as mdTagRegex } from "$common/markdown_parser/parser.ts";
const tagRegex = new RegExp(mdTagRegex.source, "g");
export function PageNavigator({
allPages,
onNavigate,
onModeSwitch,
completer,
vimMode,
mode,
darkMode,
currentPage,
}: {
allPages: PageMeta[];
vimMode: boolean;
darkMode: boolean;
mode: "page" | "meta" | "all";
onNavigate: (page: string | undefined) => void;
onModeSwitch: (mode: "page" | "meta" | "all") => void;
completer: (context: CompletionContext) => Promise<CompletionResult | null>;
currentPage?: string;
}) {
const options: FilterOption[] = [];
for (const pageMeta of allPages) {
// Sanitize the page name
if (!pageMeta.name) {
pageMeta.name = pageMeta.ref;
}
// Order by last modified date in descending order
let orderId = -new Date(pageMeta.lastModified).getTime();
// Unless it was opened in this session
if (pageMeta.lastOpened) {
orderId = -pageMeta.lastOpened;
}
// Or it's the currently open page
if (currentPage && currentPage === pageMeta.name) {
// ... then we put it all the way to the end
orderId = Infinity;
}
if (mode === "page") {
// Special behavior for regular pages
let description: string | undefined;
let aliases: string[] = [];
if (pageMeta.displayName) {
aliases.push(pageMeta.displayName);
}
if (Array.isArray(pageMeta.aliases)) {
aliases = aliases.concat(pageMeta.aliases);
}
if (aliases.length > 0) {
description = "(a.k.a. " + aliases.join(", ") + ") ";
}
if (pageMeta.tags) {
description = (description || "") +
pageMeta.tags.map((tag) => `#${tag}`).join(" ");
}
options.push({
...pageMeta,
name: (pageMeta.pageDecoration?.prefix ?? "") + pageMeta.name,
description,
orderId: orderId,
hint: pageMeta._isBrokenLink ? "Create page" : undefined,
});
} else if (mode === "meta") {
// Special behavior for #template and #meta pages
if (pageMeta._isBrokenLink) {
// Skip over broken links
continue;
}
options.push({
...pageMeta,
// Use the displayName or last bit of the path as the name
name: pageMeta.displayName || pageMeta.name.split("/").pop()!,
// And use the full path as the description
description: pageMeta.name,
hint: pageMeta.tags![0],
orderId: orderId,
});
} else { // all
// In mode "all" just show the full path and all tags
let description: string | undefined;
if (pageMeta.tags) {
description = (description || "") +
pageMeta.tags.map((tag) => `#${tag}`).join(" ");
}
options.push({
...pageMeta,
name: pageMeta.name,
description,
orderId: orderId,
});
}
}
let completePrefix = currentPage + "/";
if (currentPage && currentPage.includes("/")) {
const pieces = currentPage.split("/");
completePrefix = pieces.slice(0, pieces.length - 1).join("/") + "/";
} else if (currentPage && currentPage.includes(" ")) {
completePrefix = currentPage.split(" ")[0] + " ";
}
const pageNoun = mode === "meta" ? mode : "page";
return (
<FilterList
placeholder={mode === "page"
? "Page"
: (mode === "meta"
? "#template or #meta page"
: "Any page, also hidden")}
label="Open"
options={options}
vimMode={vimMode}
darkMode={darkMode}
completer={completer}
phrasePreprocessor={(phrase) => {
phrase = phrase.replaceAll(tagRegex, "").trim();
return phrase;
}}
onKeyPress={(key, text) => {
// Pages cannot start with ^, as documented in Page Name Rules
if (key === "^" && text === "^") {
switch (mode) {
case "page":
onModeSwitch("meta");
break;
case "meta":
onModeSwitch("all");
break;
case "all":
onModeSwitch("page");
break;
}
}
}}
preFilter={(options, phrase) => {
if (mode === "page") {
const allTags = phrase.match(tagRegex);
if (allTags) {
// Search phrase contains hash tags, let's pre-filter the results based on this
const filterTags = allTags.map((t) => t.slice(1));
options = options.filter((pageMeta) => {
if (!pageMeta.tags) {
return false;
}
return filterTags.every((tag) =>
pageMeta.tags.find((itemTag: string) => itemTag.startsWith(tag))
);
});
}
// Remove pages that are tagged as templates or meta
options = options.filter((pageMeta) => {
return !pageMeta.tags?.includes("template") &&
!pageMeta.tags?.includes("meta");
});
} else if (mode === "meta") {
// Filter on pages tagged with "template" or "meta"
options = options.filter((pageMeta) => {
return pageMeta.tags?.includes("template") ||
pageMeta.tags?.includes("meta");
});
}
if (mode !== "all") {
// Filter out hidden pages
options = options.filter((page) =>
!(page.pageDecoration?.hide === true)
);
}
return options;
}}
allowNew={true}
helpText={`Press <code>Enter</code> to open the selected ${pageNoun}, or <code>Shift-Enter</code> to create a new ${pageNoun} with this exact name.`}
newHint={`Create ${pageNoun}`}
completePrefix={completePrefix}
onSelect={(opt) => {
onNavigate(opt?.ref || opt?.name);
}}
/>
);
}