a2dbf7b3db
* Prep for in-process plug loading (e.g. for CF workers, Deno Deploy) * Prototype of fixed in-process loading plugs * Fix: buttons not to scroll with content * Better positioning of modal especially on mobile * Move query caching outside query * Fix annoying mouse behavior when filter box appears * Page navigator search tweaks
167 lines
4.9 KiB
TypeScript
167 lines
4.9 KiB
TypeScript
import {
|
|
CompletionContext,
|
|
CompletionResult,
|
|
useEffect,
|
|
useRef,
|
|
} from "../deps.ts";
|
|
import type { ComponentChildren, FunctionalComponent } from "../deps.ts";
|
|
import { Notification } from "../types.ts";
|
|
import { FeatherProps } from "https://esm.sh/v99/preact-feather@4.2.1/dist/types";
|
|
import { MiniEditor } from "./mini_editor.tsx";
|
|
|
|
export type ActionButton = {
|
|
icon: FunctionalComponent<FeatherProps>;
|
|
description: string;
|
|
class?: string;
|
|
callback: () => void;
|
|
href?: string;
|
|
};
|
|
|
|
export function TopBar({
|
|
pageName,
|
|
unsavedChanges,
|
|
syncFailures,
|
|
isLoading,
|
|
notifications,
|
|
onRename,
|
|
actionButtons,
|
|
darkMode,
|
|
vimMode,
|
|
progressPerc,
|
|
completer,
|
|
lhs,
|
|
onClick,
|
|
rhs,
|
|
}: {
|
|
pageName?: string;
|
|
unsavedChanges: boolean;
|
|
syncFailures: number;
|
|
isLoading: boolean;
|
|
notifications: Notification[];
|
|
darkMode: boolean;
|
|
vimMode: boolean;
|
|
progressPerc?: number;
|
|
onRename: (newName?: string) => Promise<void>;
|
|
onClick: () => void;
|
|
completer: (context: CompletionContext) => Promise<CompletionResult | null>;
|
|
actionButtons: ActionButton[];
|
|
lhs?: ComponentChildren;
|
|
rhs?: ComponentChildren;
|
|
}) {
|
|
// Another one of my less proud moments:
|
|
// Somehow I cannot seem to proerply limit the width of the page name, so I'm doing
|
|
// it this way. If you have a better way to do this, please let me know!
|
|
useEffect(() => {
|
|
function resizeHandler() {
|
|
const editorWidth = parseInt(
|
|
getComputedStyle(document.getElementById("sb-root")!).getPropertyValue(
|
|
"--editor-width",
|
|
),
|
|
);
|
|
const currentPageElement = document.getElementById("sb-current-page");
|
|
if (currentPageElement) {
|
|
// Temporarily make it very narrow to give the parent space
|
|
currentPageElement.style.width = "10px";
|
|
const innerDiv = currentPageElement.parentElement!.parentElement!;
|
|
|
|
// Then calculate a new width
|
|
currentPageElement.style.width = `${
|
|
Math.min(editorWidth - 170, innerDiv.clientWidth - 170)
|
|
}px`;
|
|
}
|
|
}
|
|
globalThis.addEventListener("resize", resizeHandler);
|
|
|
|
// Stop listening on unmount
|
|
return () => {
|
|
globalThis.removeEventListener("resize", resizeHandler);
|
|
};
|
|
}, []);
|
|
|
|
return (
|
|
<div
|
|
id="sb-top"
|
|
className={syncFailures > 1 ? "sb-sync-error" : undefined}
|
|
onClick={onClick}
|
|
>
|
|
{lhs}
|
|
<div className="main">
|
|
<div className="inner">
|
|
<div className="wrapper">
|
|
<span
|
|
id="sb-current-page"
|
|
className={isLoading
|
|
? "sb-loading"
|
|
: unsavedChanges
|
|
? "sb-unsaved"
|
|
: "sb-saved"}
|
|
>
|
|
<MiniEditor
|
|
text={pageName ?? ""}
|
|
vimMode={vimMode}
|
|
darkMode={darkMode}
|
|
onBlur={(newName) => {
|
|
if (newName !== pageName) {
|
|
return onRename(newName);
|
|
} else {
|
|
return onRename();
|
|
}
|
|
}}
|
|
completer={completer}
|
|
onEnter={(newName) => {
|
|
onRename(newName);
|
|
}}
|
|
/>
|
|
</span>
|
|
{notifications.length > 0 && (
|
|
<div className="sb-notifications">
|
|
{notifications.map((notification) => (
|
|
<div
|
|
key={notification.id}
|
|
className={`sb-notification-${notification.type}`}
|
|
>
|
|
{notification.message}
|
|
</div>
|
|
))}
|
|
</div>
|
|
)}
|
|
<div className="sb-actions">
|
|
{progressPerc !== undefined &&
|
|
(
|
|
<div className="progress-wrapper" title={`${progressPerc}%`}>
|
|
<div
|
|
className="progress-bar"
|
|
style={`background: radial-gradient(closest-side, white 79%, transparent 80% 100%), conic-gradient(#282828 ${progressPerc}%, #adadad 0);`}
|
|
>
|
|
{progressPerc}%
|
|
</div>
|
|
</div>
|
|
)}
|
|
{actionButtons.map((actionButton) => {
|
|
const button = (
|
|
<button
|
|
onClick={(e) => {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
actionButton.callback();
|
|
}}
|
|
title={actionButton.description}
|
|
className={actionButton.class}
|
|
>
|
|
<actionButton.icon size={18} />
|
|
</button>
|
|
);
|
|
|
|
return actionButton.href !== undefined
|
|
? <a href={actionButton.href}>{button}</a>
|
|
: button;
|
|
})}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{rhs}
|
|
</div>
|
|
);
|
|
}
|