2022-10-10 20:50:21 +08:00
|
|
|
import { FilterList } from "./filter.tsx";
|
2024-07-14 17:29:43 +08:00
|
|
|
import { FilterOption } from "$lib/web.ts";
|
2024-03-16 22:29:24 +08:00
|
|
|
import { CompletionContext, CompletionResult } from "@codemirror/autocomplete";
|
2024-02-29 22:23:05 +08:00
|
|
|
import { PageMeta } from "../../plug-api/types.ts";
|
2024-02-27 01:03:55 +08:00
|
|
|
import { tagRegex as mdTagRegex } from "$common/markdown_parser/parser.ts";
|
2022-03-20 16:56:28 +08:00
|
|
|
|
2024-02-29 22:23:05 +08:00
|
|
|
const tagRegex = new RegExp(mdTagRegex.source, "g");
|
2023-12-22 02:49:25 +08:00
|
|
|
|
2022-03-20 16:56:28 +08:00
|
|
|
export function PageNavigator({
|
|
|
|
allPages,
|
|
|
|
onNavigate,
|
2024-07-17 23:03:25 +08:00
|
|
|
onModeSwitch,
|
2022-12-21 21:55:24 +08:00
|
|
|
completer,
|
|
|
|
vimMode,
|
2024-01-21 02:16:07 +08:00
|
|
|
mode,
|
2022-12-21 21:55:24 +08:00
|
|
|
darkMode,
|
2022-03-20 16:56:28 +08:00
|
|
|
currentPage,
|
|
|
|
}: {
|
2023-05-24 02:53:53 +08:00
|
|
|
allPages: PageMeta[];
|
2022-12-21 21:55:24 +08:00
|
|
|
vimMode: boolean;
|
|
|
|
darkMode: boolean;
|
2024-07-19 23:06:40 +08:00
|
|
|
mode: "page" | "meta" | "all";
|
2022-03-20 16:56:28 +08:00
|
|
|
onNavigate: (page: string | undefined) => void;
|
2024-07-19 23:06:40 +08:00
|
|
|
onModeSwitch: (mode: "page" | "meta" | "all") => void;
|
2022-12-21 21:55:24 +08:00
|
|
|
completer: (context: CompletionContext) => Promise<CompletionResult | null>;
|
2022-03-20 16:56:28 +08:00
|
|
|
currentPage?: string;
|
|
|
|
}) {
|
2022-10-10 20:50:21 +08:00
|
|
|
const options: FilterOption[] = [];
|
|
|
|
for (const pageMeta of allPages) {
|
2023-12-22 18:27:07 +08:00
|
|
|
// Sanitize the page name
|
|
|
|
if (!pageMeta.name) {
|
|
|
|
pageMeta.name = pageMeta.ref;
|
|
|
|
}
|
2022-03-20 16:56:28 +08:00
|
|
|
// Order by last modified date in descending order
|
2023-11-12 17:43:08 +08:00
|
|
|
let orderId = -new Date(pageMeta.lastModified).getTime();
|
2022-04-10 17:04:07 +08:00
|
|
|
// Unless it was opened in this session
|
2022-03-20 16:56:28 +08:00
|
|
|
if (pageMeta.lastOpened) {
|
|
|
|
orderId = -pageMeta.lastOpened;
|
|
|
|
}
|
2022-08-01 21:06:32 +08:00
|
|
|
// Or it's the currently open page
|
|
|
|
if (currentPage && currentPage === pageMeta.name) {
|
|
|
|
// ... then we put it all the way to the end
|
|
|
|
orderId = Infinity;
|
|
|
|
}
|
2024-01-24 21:52:07 +08:00
|
|
|
|
|
|
|
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,
|
2024-07-14 17:29:43 +08:00
|
|
|
name: (pageMeta.pageDecoration?.prefix ?? "") + pageMeta.name,
|
2024-01-24 21:52:07 +08:00
|
|
|
description,
|
|
|
|
orderId: orderId,
|
2024-07-25 21:18:58 +08:00
|
|
|
hint: pageMeta._isBrokenLink ? "Create page" : undefined,
|
2024-01-24 21:52:07 +08:00
|
|
|
});
|
2024-07-19 23:06:40 +08:00
|
|
|
} else if (mode === "meta") {
|
2024-07-07 18:12:05 +08:00
|
|
|
// Special behavior for #template and #meta pages
|
2024-07-27 16:55:45 +08:00
|
|
|
if (pageMeta._isBrokenLink) {
|
|
|
|
// Skip over broken links
|
|
|
|
continue;
|
|
|
|
}
|
2024-01-24 21:52:07 +08:00
|
|
|
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,
|
2024-07-07 18:12:05 +08:00
|
|
|
hint: pageMeta.tags![0],
|
2024-01-24 21:52:07 +08:00
|
|
|
orderId: orderId,
|
|
|
|
});
|
2024-07-27 17:01:28 +08:00
|
|
|
} else { // all
|
2024-07-19 23:06:40 +08:00
|
|
|
// 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,
|
|
|
|
});
|
2023-12-22 01:37:50 +08:00
|
|
|
}
|
2022-03-20 16:56:28 +08:00
|
|
|
}
|
2023-11-10 17:57:57 +08:00
|
|
|
let completePrefix = currentPage + "/";
|
2022-04-01 21:02:35 +08:00
|
|
|
if (currentPage && currentPage.includes("/")) {
|
|
|
|
const pieces = currentPage.split("/");
|
|
|
|
completePrefix = pieces.slice(0, pieces.length - 1).join("/") + "/";
|
2022-07-04 17:38:16 +08:00
|
|
|
} else if (currentPage && currentPage.includes(" ")) {
|
|
|
|
completePrefix = currentPage.split(" ")[0] + " ";
|
2022-04-01 21:02:35 +08:00
|
|
|
}
|
2024-07-25 21:18:58 +08:00
|
|
|
|
2024-07-19 23:06:40 +08:00
|
|
|
const pageNoun = mode === "meta" ? mode : "page";
|
2022-03-20 16:56:28 +08:00
|
|
|
return (
|
|
|
|
<FilterList
|
2024-07-25 21:18:58 +08:00
|
|
|
placeholder={mode === "page"
|
|
|
|
? "Page"
|
|
|
|
: (mode === "meta"
|
|
|
|
? "#template or #meta page"
|
|
|
|
: "Any page, also hidden")}
|
2022-03-20 16:56:28 +08:00
|
|
|
label="Open"
|
|
|
|
options={options}
|
2022-12-21 21:55:24 +08:00
|
|
|
vimMode={vimMode}
|
|
|
|
darkMode={darkMode}
|
|
|
|
completer={completer}
|
2023-12-22 02:49:25 +08:00
|
|
|
phrasePreprocessor={(phrase) => {
|
|
|
|
phrase = phrase.replaceAll(tagRegex, "").trim();
|
|
|
|
return phrase;
|
|
|
|
}}
|
2024-07-17 23:03:25 +08:00
|
|
|
onKeyPress={(key, text) => {
|
2024-07-19 23:06:40 +08:00
|
|
|
// Pages cannot start with ^, as documented in Page Name Rules
|
|
|
|
if (key === "^" && text === "^") {
|
2024-07-25 21:18:58 +08:00
|
|
|
switch (mode) {
|
2024-07-19 23:06:40 +08:00
|
|
|
case "page":
|
|
|
|
onModeSwitch("meta");
|
|
|
|
break;
|
|
|
|
case "meta":
|
|
|
|
onModeSwitch("all");
|
|
|
|
break;
|
|
|
|
case "all":
|
|
|
|
onModeSwitch("page");
|
|
|
|
break;
|
|
|
|
}
|
2024-07-17 23:03:25 +08:00
|
|
|
}
|
|
|
|
}}
|
2023-12-22 02:49:25 +08:00
|
|
|
preFilter={(options, phrase) => {
|
2024-01-21 02:16:07 +08:00
|
|
|
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))
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
2024-07-07 18:12:05 +08:00
|
|
|
// Remove pages that are tagged as templates or meta
|
2023-12-22 02:49:25 +08:00
|
|
|
options = options.filter((pageMeta) => {
|
2024-07-07 18:12:05 +08:00
|
|
|
return !pageMeta.tags?.includes("template") &&
|
|
|
|
!pageMeta.tags?.includes("meta");
|
2023-12-22 02:49:25 +08:00
|
|
|
});
|
2024-07-19 23:06:40 +08:00
|
|
|
} else if (mode === "meta") {
|
2024-07-07 18:12:05 +08:00
|
|
|
// Filter on pages tagged with "template" or "meta"
|
2024-01-21 02:16:07 +08:00
|
|
|
options = options.filter((pageMeta) => {
|
2024-07-07 18:12:05 +08:00
|
|
|
return pageMeta.tags?.includes("template") ||
|
|
|
|
pageMeta.tags?.includes("meta");
|
2024-01-21 02:16:07 +08:00
|
|
|
});
|
2023-12-22 02:49:25 +08:00
|
|
|
}
|
2024-07-19 23:06:40 +08:00
|
|
|
|
|
|
|
if (mode !== "all") {
|
|
|
|
// Filter out hidden pages
|
2024-07-25 21:18:58 +08:00
|
|
|
options = options.filter((page) =>
|
|
|
|
!(page.pageDecoration?.hide === true)
|
|
|
|
);
|
2024-07-19 23:06:40 +08:00
|
|
|
}
|
|
|
|
return options;
|
2023-12-22 02:49:25 +08:00
|
|
|
}}
|
2022-03-20 16:56:28 +08:00
|
|
|
allowNew={true}
|
2024-07-19 23:06:40 +08:00
|
|
|
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}`}
|
2022-04-01 21:02:35 +08:00
|
|
|
completePrefix={completePrefix}
|
2022-03-20 16:56:28 +08:00
|
|
|
onSelect={(opt) => {
|
2024-01-24 21:56:02 +08:00
|
|
|
onNavigate(opt?.ref || opt?.name);
|
2022-03-20 16:56:28 +08:00
|
|
|
}}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|