Files
design-system/src/pages/Inventory/sections/LayoutSection.tsx

143 lines
5.4 KiB
TypeScript
Raw Normal View History

import styles from './LayoutSection.module.css'
import { Sidebar } from '../../../design-system/layout/Sidebar/Sidebar'
import { SidebarTree } from '../../../design-system/layout/Sidebar/SidebarTree'
import type { SidebarTreeNode } from '../../../design-system/layout/Sidebar/SidebarTree'
import { StatusDot } from '../../../design-system/primitives/StatusDot/StatusDot'
import { Box, Settings, FileText, ChevronRight } from 'lucide-react'
import { TopBar } from '../../../design-system/layout/TopBar/TopBar'
// ── DemoCard helper ──────────────────────────────────────────────────────────
interface DemoCardProps {
id: string
title: string
description: string
children: React.ReactNode
}
function DemoCard({ id, title, description, children }: DemoCardProps) {
return (
<div id={id} className={styles.componentCard}>
<h3 className={styles.componentTitle}>{title}</h3>
<p className={styles.componentDesc}>{description}</p>
<div className={styles.demoArea}>{children}</div>
</div>
)
}
// ── Sample tree nodes ────────────────────────────────────────────────────────
const SAMPLE_APP_NODES: SidebarTreeNode[] = [
{
id: 'app1',
label: 'cameleer-prod',
icon: <StatusDot variant="live" />,
badge: '14.3k',
path: '/apps/app1',
starrable: true,
starKey: 'app:app1',
children: [
{ id: 'app1/r1', label: 'order-ingest', icon: <ChevronRight size={12} />, badge: '5,421', path: '/apps/app1/r1' },
{ id: 'app1/r2', label: 'payment-validate', icon: <ChevronRight size={12} />, badge: '3,102', path: '/apps/app1/r2' },
],
},
{
id: 'app2',
label: 'cameleer-staging',
icon: <StatusDot variant="stale" />,
badge: '871',
path: '/apps/app2',
starrable: true,
starKey: 'app:app2',
children: [
{ id: 'app2/r3', label: 'notify-customer', icon: <ChevronRight size={12} />, badge: '2,201', path: '/apps/app2/r3' },
],
},
{
id: 'app3',
label: 'cameleer-dev',
icon: <StatusDot variant="dead" />,
badge: '42',
path: '/apps/app3',
starrable: true,
starKey: 'app:app3',
},
]
// ── LayoutSection ─────────────────────────────────────────────────────────────
export function LayoutSection() {
return (
<section id="layout" className={styles.section}>
<h2 className={styles.sectionTitle}>Layout</h2>
{/* 1. AppShell */}
<DemoCard
id="appshell"
title="AppShell"
description="Full-page shell that composes Sidebar + TopBar + main content area. Cannot be nested — shown as a structural diagram."
>
<div className={styles.shellDiagram}>
<div className={styles.shellDiagramTop}>
TopBar breadcrumb · search · filters · time range · env badge · user avatar
</div>
<div className={styles.shellDiagramBody}>
<div className={styles.shellDiagramSide}>
<span>Sidebar</span>
<span style={{ fontWeight: 400, fontSize: 10, marginTop: 4 }}>Logo</span>
<span style={{ fontWeight: 400, fontSize: 10 }}>Search</span>
<span style={{ fontWeight: 400, fontSize: 10 }}>Navigation</span>
<span style={{ fontWeight: 400, fontSize: 10 }}>Applications tree</span>
<span style={{ fontWeight: 400, fontSize: 10 }}>Agents tree</span>
<span style={{ fontWeight: 400, fontSize: 10 }}>Starred</span>
</div>
<div className={styles.shellDiagramMain}>
&lt;children&gt; page content rendered here
</div>
</div>
</div>
</DemoCard>
{/* 2. Sidebar */}
<DemoCard
id="sidebar"
title="Sidebar"
description="Composable navigation sidebar with sections, tree navigation, and icon-rail collapse mode."
>
<div className={styles.sidebarPreview}>
<Sidebar>
<Sidebar.Header logo={<span style={{ fontSize: 20 }}>&#x1F42A;</span>} title="cameleer" version="v3.2.1" />
<Sidebar.Section label="Applications" icon={<Box size={14} />} open={true} onToggle={() => {}} active={false}>
<SidebarTree nodes={SAMPLE_APP_NODES} isStarred={() => false} onToggleStar={() => {}} />
</Sidebar.Section>
<Sidebar.Footer>
<Sidebar.FooterLink icon={<Settings size={14} />} label="Admin" />
<Sidebar.FooterLink icon={<FileText size={14} />} label="API Docs" />
</Sidebar.Footer>
</Sidebar>
</div>
</DemoCard>
{/* 3. TopBar */}
<DemoCard
id="topbar"
title="TopBar"
description="Top navigation bar with breadcrumb, search trigger, status filters, time range, environment badge, and user avatar."
>
<div className={styles.topbarPreview}>
<TopBar
breadcrumb={[
{ label: 'Dashboard', href: '#' },
{ label: 'Applications', href: '#' },
{ label: 'order-ingest' },
]}
environment="production"
user={{ name: 'Hendrik' }}
/>
</div>
</DemoCard>
</section>
)
}