2022-10-10 20:50:21 +08:00
|
|
|
import { FilterList } from "./filter.tsx";
|
2023-10-03 20:16:33 +08:00
|
|
|
import { FilterOption } from "../types.ts";
|
2022-12-21 21:55:24 +08:00
|
|
|
import { CompletionContext, CompletionResult } from "../deps.ts";
|
2023-10-03 20:16:33 +08:00
|
|
|
import { PageMeta } from "$sb/types.ts";
|
2023-11-12 17:43:08 +08:00
|
|
|
import { isFederationPath } from "$sb/lib/resolve.ts";
|
2022-03-20 16:56:28 +08:00
|
|
|
|
|
|
|
export function PageNavigator({
|
|
|
|
allPages,
|
|
|
|
onNavigate,
|
2022-12-21 21:55:24 +08:00
|
|
|
completer,
|
|
|
|
vimMode,
|
|
|
|
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;
|
2022-03-20 16:56:28 +08:00
|
|
|
onNavigate: (page: string | undefined) => 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) {
|
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;
|
|
|
|
}
|
2023-11-12 17:43:08 +08:00
|
|
|
// And deprioritize federated pages too
|
|
|
|
if (isFederationPath(pageMeta.name)) {
|
|
|
|
orderId = Math.round(orderId / 10); // Just 10x lower the timestamp to push them down, should work
|
|
|
|
}
|
2022-03-20 16:56:28 +08:00
|
|
|
options.push({
|
|
|
|
...pageMeta,
|
|
|
|
orderId: orderId,
|
|
|
|
});
|
|
|
|
}
|
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
|
|
|
}
|
2022-03-20 16:56:28 +08:00
|
|
|
return (
|
|
|
|
<FilterList
|
|
|
|
placeholder="Page"
|
|
|
|
label="Open"
|
|
|
|
options={options}
|
2022-12-21 21:55:24 +08:00
|
|
|
vimMode={vimMode}
|
|
|
|
darkMode={darkMode}
|
|
|
|
completer={completer}
|
2022-03-20 16:56:28 +08:00
|
|
|
allowNew={true}
|
2023-11-15 21:56:34 +08:00
|
|
|
helpText="Press <code>Enter</code> to open the selected page, or <code>Shift-Enter</code> to create a new page with this exact name."
|
2022-03-20 16:56:28 +08:00
|
|
|
newHint="Create page"
|
2022-04-01 21:02:35 +08:00
|
|
|
completePrefix={completePrefix}
|
2022-03-20 16:56:28 +08:00
|
|
|
onSelect={(opt) => {
|
|
|
|
onNavigate(opt?.name);
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|