silverbullet/web/components/top_bar.tsx

136 lines
3.9 KiB
TypeScript
Raw Normal View History

2024-01-25 18:42:36 +08:00
import { CompletionContext, CompletionResult, useEffect } from "../deps.ts";
import type { ComponentChildren, FunctionalComponent } from "../deps.ts";
2024-02-09 04:12:23 +08:00
import { Notification } from "../../type/web.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;
2023-08-30 03:17:29 +08:00
class?: string;
callback: () => void;
href?: string;
mobile?: boolean;
};
export function TopBar({
pageName,
unsavedChanges,
2023-08-16 17:40:31 +08:00
syncFailures,
isLoading,
isMobile,
notifications,
onRename,
actionButtons,
darkMode,
vimMode,
2023-06-15 02:58:08 +08:00
progressPerc,
completer,
2022-04-05 00:33:13 +08:00
lhs,
onClick,
2022-04-05 00:33:13 +08:00
rhs,
}: {
pageName?: string;
unsavedChanges: boolean;
2023-08-16 17:40:31 +08:00
syncFailures: number;
isLoading: boolean;
isMobile: boolean;
notifications: Notification[];
darkMode: boolean;
vimMode: boolean;
2023-06-15 02:58:08 +08:00
progressPerc?: number;
onRename: (newName?: string) => Promise<void>;
onClick: () => void;
completer: (context: CompletionContext) => Promise<CompletionResult | null>;
actionButtons: ActionButton[];
lhs?: ComponentChildren;
rhs?: ComponentChildren;
}) {
return (
<div
id="sb-top"
className={syncFailures > 1 ? "sb-sync-error" : undefined}
onClick={onClick}
>
2022-04-05 00:33:13 +08:00
{lhs}
<div className="main">
<div className="inner">
2022-12-16 18:39:46 +08:00
<div className="wrapper">
<span
id="sb-current-page"
className={isLoading
? "sb-loading"
: unsavedChanges
2024-02-27 01:10:31 +08:00
? "sb-unsaved"
: "sb-saved"}
2022-12-16 18:39:46 +08:00
>
<MiniEditor
text={pageName ?? ""}
vimMode={vimMode}
darkMode={darkMode}
onBlur={(newName) => {
if (newName !== pageName) {
return onRename(newName);
} else {
return onRename();
2022-12-16 18:39:46 +08:00
}
}}
completer={completer}
onEnter={(newName) => {
onRename(newName);
2022-12-16 18:39:46 +08:00
}}
/>
</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">
2023-06-15 02:58:08 +08:00
{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);`}
>
2023-06-15 02:58:08 +08:00
{progressPerc}%
</div>
</div>
)}
{actionButtons.map((actionButton) => {
const button = (
<button
onClick={(e) => {
e.preventDefault();
e.stopPropagation();
actionButton.callback();
2023-06-15 02:58:08 +08:00
}}
title={actionButton.description}
2023-08-30 03:17:29 +08:00
className={actionButton.class}
2023-06-15 02:58:08 +08:00
>
<actionButton.icon size={18} />
</button>
);
2023-06-15 02:58:08 +08:00
return actionButton.href !== undefined
? <a href={actionButton.href}>{button}</a>
: button;
})}
2022-04-05 00:33:13 +08:00
</div>
2022-05-07 00:55:04 +08:00
</div>
2022-04-05 00:33:13 +08:00
</div>
</div>
2022-04-05 00:33:13 +08:00
{rhs}
</div>
);
}