Files
bknd/app/src/ui/routes/auth/auth.index.tsx

93 lines
3.5 KiB
TypeScript

import { useApiQuery } from "ui/client";
import { useBknd } from "ui/client/bknd";
import { useBkndAuth } from "ui/client/schema/auth/use-bknd-auth";
import { ButtonLink, type ButtonLinkProps } from "ui/components/buttons/Button";
import { Alert } from "ui/components/display/Alert";
import * as AppShell from "ui/layouts/AppShell/AppShell";
import { routes } from "ui/lib/routes";
export function AuthIndex() {
const { app } = useBknd();
const {
config: { roles, strategies, entity_name, enabled }
} = useBkndAuth();
const users_entity = entity_name;
const $q = useApiQuery((api) => api.data.count(users_entity));
const usersTotal = $q.data?.count ?? 0;
const rolesTotal = Object.keys(roles ?? {}).length ?? 0;
const strategiesTotal = Object.keys(strategies ?? {}).length ?? 0;
const usersLink = app.getAbsolutePath("/data/" + routes.data.entity.list(users_entity));
const rolesLink = routes.auth.roles.list();
const strategiesLink = app.getSettingsPath(["auth", "strategies"]);
return (
<>
<AppShell.SectionHeader>Overview</AppShell.SectionHeader>
<AppShell.Scrollable>
<Alert.Warning
visible={!enabled}
title="Auth not enabled"
message="To use authentication features, please enable it in the settings."
/>
<div className="flex flex-col flex-grow p-3 gap-3">
<div className="grid xs:grid-cols-1 sm:grid-cols-2 xl:grid-cols-4 gap-5">
<KpiCard
title="Users registered"
value={!enabled ? 0 : usersTotal}
actions={[
{
label: "View all",
href: usersLink
},
{ label: "Add new", variant: "default", href: usersLink }
]}
/>
<KpiCard
title="Roles"
value={!enabled ? 0 : rolesTotal}
actions={[
{ label: "View all", href: rolesLink },
{ label: "Add new", variant: "default", href: rolesLink }
]}
/>
<KpiCard
title="Strategies enabled"
value={!enabled ? 0 : strategiesTotal}
actions={[
{ label: "View all", href: strategiesLink },
{ label: "Add new", variant: "default", href: strategiesLink }
]}
/>
</div>
</div>
</AppShell.Scrollable>
</>
);
}
type KpiCardProps = {
title: string;
value: number;
actions: (Omit<ButtonLinkProps, "href"> & { label: string; href?: string })[];
};
const KpiCard: React.FC<KpiCardProps> = ({ title, value, actions }) => (
<div className="flex flex-col border border-muted">
<div className="flex flex-col gap-2 px-5 pt-3.5 pb-4 border-b border-b-muted">
<div>
<span className="opacity-50">{title}</span>
{/*<span>+6.1%</span>*/}
</div>
<div className="text-4xl font-medium">{value}</div>
</div>
<div className="flex flex-row gap-3 p-3 justify-between">
{actions.map((action, i) => (
<ButtonLink key={i} size="small" variant="ghost" href="#" {...action}>
{action.label}
</ButtonLink>
))}
</div>
</div>
);