feat(network): add asset style lab mockups

This commit is contained in:
chihlasm
2026-04-14 02:10:48 +00:00
parent 91cc9a4170
commit c37e216e0b
6 changed files with 396 additions and 48 deletions

View File

@@ -1,10 +1,12 @@
import { memo, useState, useRef, useEffect } from 'react'
import { Position, NodeResizer, useReactFlow, type NodeProps } from '@xyflow/react'
import { HardDrive, Network, ShieldCheck } from 'lucide-react'
import { BaseNode, BaseNodeHeader, BaseNodeContent } from '../ui/base-node'
import { BaseHandle } from '../ui/base-handle'
import { NodeStatusIndicator, type NodeStatus } from '../ui/node-status-indicator'
import { NodeTooltip, NodeTooltipTrigger, NodeTooltipContent } from '../ui/node-tooltip'
import { getDeviceRenderConfig } from './deviceRegistry'
import { getDeviceRenderConfig, CATEGORY_LABELS } from './deviceRegistry'
import { cn } from '@/lib/utils'
import type { DeviceProperties } from '@/types'
export interface DeviceNodeData {
@@ -29,9 +31,48 @@ const NODE_DEFAULT = 120 // default square side in px
const NODE_MIN = 80 // minimum square side in px
const NODE_MAX = 280 // maximum square side in px
function DeviceNodeChrome({
category,
accentClass,
glyph,
}: {
category: string
accentClass: string
glyph: string
}) {
const MetaIcon = category === 'network'
? Network
: category === 'security'
? ShieldCheck
: HardDrive
return (
<>
<div className="pointer-events-none absolute inset-x-0 top-0 flex items-center justify-between px-2 py-1">
<span className={cn('rounded-full border px-1.5 py-0.5 text-[8px] font-semibold tracking-[0.18em]', accentClass)}>
{glyph}
</span>
<span className="text-[8px] font-medium uppercase tracking-[0.16em] text-muted-foreground/80">
{CATEGORY_LABELS[category] ?? category}
</span>
</div>
<div className="pointer-events-none absolute inset-x-3 top-6 h-px bg-gradient-to-r from-transparent via-border-default to-transparent" />
<div className="pointer-events-none absolute bottom-2 left-2 flex items-center gap-1 rounded-full border border-default bg-page/80 px-1.5 py-0.5 text-[8px] uppercase tracking-[0.14em] text-muted-foreground">
<MetaIcon size={9} />
<span>Asset</span>
</div>
<div className="pointer-events-none absolute bottom-2 right-2 flex gap-1">
<span className="h-1.5 w-1.5 rounded-full bg-border-default" />
<span className="h-1.5 w-1.5 rounded-full bg-border-default/80" />
<span className="h-1.5 w-1.5 rounded-full bg-border-default/60" />
</div>
</>
)
}
function DeviceNodeComponent({ id, data, selected, width, height }: NodeProps) {
const nodeData = data as unknown as DeviceNodeData
const { icon: Icon, color } = getDeviceRenderConfig(nodeData.deviceType, nodeData.category)
const { icon: Icon, color, accentClass, surfaceClass, category, glyph } = getDeviceRenderConfig(nodeData.deviceType, nodeData.category)
const status = (nodeData.properties?.status || 'unknown') as NodeStatus
const ip = nodeData.properties?.ip
const props = nodeData.properties || {}
@@ -46,6 +87,8 @@ function DeviceNodeComponent({ id, data, selected, width, height }: NodeProps) {
const labelPx = Math.max(9, Math.min(20, Math.round(scale * 11)))
// IP font: 9px at default, clamped to [8, 16]
const ipPx = Math.max(8, Math.min(16, Math.round(scale * 9)))
const typePx = Math.max(8, Math.min(13, Math.round(scale * 8.5)))
const iconPlateSize = Math.round(Math.max(36, Math.min(92, scale * 56)))
const [editing, setEditing] = useState(false)
const [labelValue, setLabelValue] = useState(nodeData.label ?? '')
@@ -81,9 +124,23 @@ function DeviceNodeComponent({ id, data, selected, width, height }: NodeProps) {
<NodeStatusIndicator status={status}>
<NodeTooltip>
<NodeTooltipTrigger>
<BaseNode className="w-full h-full group flex flex-col items-center justify-center">
<BaseNodeHeader className="flex-col gap-1 items-center py-2 px-2">
<Icon size={iconPx} style={{ color }} />
<BaseNode className="group h-full w-full bg-gradient-to-b from-card via-card to-page/80">
<div className={cn('pointer-events-none absolute inset-x-0 top-0 h-14 bg-gradient-to-b', surfaceClass)} />
<DeviceNodeChrome category={category} accentClass={accentClass} glyph={glyph} />
<BaseNodeHeader className="flex h-full flex-col items-center justify-center gap-2 px-2 pt-8 pb-3">
<div
className={cn(
'relative flex items-center justify-center rounded-2xl border shadow-inner',
accentClass,
)}
style={{ width: iconPlateSize, height: iconPlateSize }}
>
<div className="absolute inset-[5px] rounded-[14px] border border-white/8 bg-page/65" />
<div className="absolute inset-x-2 top-2 h-px bg-gradient-to-r from-transparent via-white/20 to-transparent" />
<div className="relative z-10">
<Icon size={iconPx} style={{ color }} />
</div>
</div>
{editing ? (
<input
ref={inputRef}
@@ -109,7 +166,7 @@ function DeviceNodeComponent({ id, data, selected, width, height }: NodeProps) {
) : (
<span
style={{ fontSize: labelPx }}
className="font-medium text-primary text-center leading-tight line-clamp-2 cursor-default"
className="max-w-[88%] cursor-default text-center font-medium leading-tight text-primary line-clamp-2"
onDoubleClick={e => {
e.stopPropagation()
setEditing(true)
@@ -118,10 +175,21 @@ function DeviceNodeComponent({ id, data, selected, width, height }: NodeProps) {
{labelValue}
</span>
)}
<span
style={{ fontSize: typePx }}
className="rounded-full border border-default bg-page/70 px-2 py-0.5 text-[9px] uppercase tracking-[0.18em] text-muted-foreground"
>
{nodeData.deviceType.replace(/-/g, ' ')}
</span>
</BaseNodeHeader>
{ip && (
<BaseNodeContent className="items-center pt-0 pb-1">
<span className="font-mono text-muted-foreground" style={{ fontSize: ipPx }}>{ip}</span>
<BaseNodeContent className="items-center pt-0 pb-3">
<span
className="rounded-full border border-default bg-page/80 px-2 py-0.5 font-mono text-muted-foreground"
style={{ fontSize: ipPx }}
>
{ip}
</span>
</BaseNodeContent>
)}
<BaseHandle type="target" position={Position.Top} />

View File

@@ -9,6 +9,10 @@ import {
export interface DeviceRenderConfig {
icon: LucideIcon
color: string
accentClass: string
surfaceClass: string
category: string
glyph: string
}
// Category-semantic color palette — each color carries meaning:
@@ -27,62 +31,109 @@ export const STORAGE_COLOR = '#a78bfa'
export const CLOUD_COLOR = '#67e8f9'
export const INFRA_COLOR = '#94a3b8'
const CATEGORY_STYLES: Record<string, Pick<DeviceRenderConfig, 'accentClass' | 'surfaceClass'>> = {
network: {
accentClass: 'border-sky-400/40 bg-sky-400/12 text-sky-300',
surfaceClass: 'from-sky-400/12 via-sky-400/4 to-transparent',
},
security: {
accentClass: 'border-rose-400/40 bg-rose-400/12 text-rose-300',
surfaceClass: 'from-rose-400/12 via-rose-400/4 to-transparent',
},
compute: {
accentClass: 'border-emerald-400/40 bg-emerald-400/12 text-emerald-300',
surfaceClass: 'from-emerald-400/12 via-emerald-400/4 to-transparent',
},
storage: {
accentClass: 'border-violet-400/40 bg-violet-400/12 text-violet-300',
surfaceClass: 'from-violet-400/12 via-violet-400/4 to-transparent',
},
cloud: {
accentClass: 'border-cyan-400/40 bg-cyan-400/12 text-cyan-300',
surfaceClass: 'from-cyan-400/12 via-cyan-400/4 to-transparent',
},
endpoint: {
accentClass: 'border-amber-400/40 bg-amber-400/12 text-amber-300',
surfaceClass: 'from-amber-400/12 via-amber-400/4 to-transparent',
},
infrastructure: {
accentClass: 'border-slate-400/40 bg-slate-300/10 text-slate-300',
surfaceClass: 'from-slate-300/10 via-slate-300/4 to-transparent',
},
}
function makeConfig(
icon: LucideIcon,
color: string,
category: string,
glyph: string,
): DeviceRenderConfig {
return {
icon,
color,
category,
glyph,
accentClass: CATEGORY_STYLES[category]?.accentClass ?? CATEGORY_STYLES.infrastructure.accentClass,
surfaceClass: CATEGORY_STYLES[category]?.surfaceClass ?? CATEGORY_STYLES.infrastructure.surfaceClass,
}
}
const SYSTEM_DEVICE_ICONS: Record<string, DeviceRenderConfig> = {
// Network layer
'router': { icon: Router, color: NETWORK_COLOR },
'switch': { icon: Network, color: NETWORK_COLOR },
'access-point': { icon: Wifi, color: NETWORK_COLOR },
'load-balancer': { icon: Gauge, color: NETWORK_COLOR },
'router': makeConfig(Router, NETWORK_COLOR, 'network', 'RTR'),
'switch': makeConfig(Network, NETWORK_COLOR, 'network', 'SW'),
'access-point': makeConfig(Wifi, NETWORK_COLOR, 'network', 'AP'),
'load-balancer': makeConfig(Gauge, NETWORK_COLOR, 'network', 'LB'),
// Security
'firewall': { icon: BrickWallFire, color: SECURITY_COLOR },
'badge-reader': { icon: KeyRound, color: SECURITY_COLOR },
'firewall': makeConfig(BrickWallFire, SECURITY_COLOR, 'security', 'FW'),
'badge-reader': makeConfig(KeyRound, SECURITY_COLOR, 'security', 'BR'),
// Compute
'server': { icon: Server, color: COMPUTE_COLOR },
'vm': { icon: Boxes, color: COMPUTE_COLOR },
'container': { icon: Package, color: COMPUTE_COLOR },
'server': makeConfig(Server, COMPUTE_COLOR, 'compute', 'SRV'),
'vm': makeConfig(Boxes, COMPUTE_COLOR, 'compute', 'VM'),
'container': makeConfig(Package, COMPUTE_COLOR, 'compute', 'CTR'),
// Storage
'nas': { icon: Database, color: STORAGE_COLOR },
'san': { icon: HardDrive, color: STORAGE_COLOR },
'cloud-storage': { icon: CloudCog, color: STORAGE_COLOR },
'nas': makeConfig(Database, STORAGE_COLOR, 'storage', 'NAS'),
'san': makeConfig(HardDrive, STORAGE_COLOR, 'storage', 'SAN'),
'cloud-storage': makeConfig(CloudCog, STORAGE_COLOR, 'storage', 'CS'),
// Cloud / Internet
'cloud': { icon: Cloud, color: CLOUD_COLOR },
'aws': { icon: Cloud, color: CLOUD_COLOR },
'azure': { icon: Cloud, color: CLOUD_COLOR },
'gcp': { icon: Cloud, color: CLOUD_COLOR },
'isp': { icon: Globe, color: CLOUD_COLOR },
'cloud': makeConfig(Cloud, CLOUD_COLOR, 'cloud', 'CLD'),
'aws': makeConfig(Cloud, CLOUD_COLOR, 'cloud', 'AWS'),
'azure': makeConfig(Cloud, CLOUD_COLOR, 'cloud', 'AZ'),
'gcp': makeConfig(Cloud, CLOUD_COLOR, 'cloud', 'GCP'),
'isp': makeConfig(Globe, CLOUD_COLOR, 'cloud', 'WAN'),
// Endpoints
'workstation': { icon: Monitor, color: ENDPOINT_COLOR },
'laptop': { icon: Laptop, color: ENDPOINT_COLOR },
'tablet': { icon: Tablet, color: ENDPOINT_COLOR },
'phone': { icon: Smartphone, color: ENDPOINT_COLOR },
'printer': { icon: Printer, color: ENDPOINT_COLOR },
'workstation': makeConfig(Monitor, ENDPOINT_COLOR, 'endpoint', 'WS'),
'laptop': makeConfig(Laptop, ENDPOINT_COLOR, 'endpoint', 'LTP'),
'tablet': makeConfig(Tablet, ENDPOINT_COLOR, 'endpoint', 'TAB'),
'phone': makeConfig(Smartphone, ENDPOINT_COLOR, 'endpoint', 'PH'),
'printer': makeConfig(Printer, ENDPOINT_COLOR, 'endpoint', 'PRN'),
// Infrastructure / physical
'ups': { icon: BatteryCharging, color: INFRA_COLOR },
'pdu': { icon: PlugZap, color: INFRA_COLOR },
'rack': { icon: RectangleVertical, color: INFRA_COLOR },
'patch-panel': { icon: Cable, color: INFRA_COLOR },
'camera': { icon: Camera, color: INFRA_COLOR },
'nvr': { icon: Video, color: INFRA_COLOR },
'iot': { icon: Radio, color: INFRA_COLOR },
'ups': makeConfig(BatteryCharging, INFRA_COLOR, 'infrastructure', 'UPS'),
'pdu': makeConfig(PlugZap, INFRA_COLOR, 'infrastructure', 'PDU'),
'rack': makeConfig(RectangleVertical, INFRA_COLOR, 'infrastructure', 'RCK'),
'patch-panel': makeConfig(Cable, INFRA_COLOR, 'infrastructure', 'PP'),
'camera': makeConfig(Camera, INFRA_COLOR, 'infrastructure', 'CAM'),
'nvr': makeConfig(Video, INFRA_COLOR, 'infrastructure', 'NVR'),
'iot': makeConfig(Radio, INFRA_COLOR, 'infrastructure', 'IOT'),
}
const CATEGORY_DEFAULTS: Record<string, DeviceRenderConfig> = {
'network': { icon: Router, color: NETWORK_COLOR },
'compute': { icon: Server, color: COMPUTE_COLOR },
'storage': { icon: Database, color: STORAGE_COLOR },
'cloud': { icon: Cloud, color: CLOUD_COLOR },
'endpoint': { icon: Monitor, color: ENDPOINT_COLOR },
'infrastructure': { icon: PlugZap, color: INFRA_COLOR },
'security': { icon: BrickWallFire, color: SECURITY_COLOR },
'network': makeConfig(Router, NETWORK_COLOR, 'network', 'NET'),
'compute': makeConfig(Server, COMPUTE_COLOR, 'compute', 'CPU'),
'storage': makeConfig(Database, STORAGE_COLOR, 'storage', 'DB'),
'cloud': makeConfig(Cloud, CLOUD_COLOR, 'cloud', 'CLD'),
'endpoint': makeConfig(Monitor, ENDPOINT_COLOR, 'endpoint', 'END'),
'infrastructure': makeConfig(PlugZap, INFRA_COLOR, 'infrastructure', 'INF'),
'security': makeConfig(BrickWallFire, SECURITY_COLOR, 'security', 'SEC'),
}
const FALLBACK: DeviceRenderConfig = { icon: Cpu, color: INFRA_COLOR }
const FALLBACK: DeviceRenderConfig = makeConfig(Cpu, INFRA_COLOR, 'infrastructure', 'DEV')
export function getDeviceRenderConfig(slug: string, category?: string): DeviceRenderConfig {
if (SYSTEM_DEVICE_ICONS[slug]) return SYSTEM_DEVICE_ICONS[slug]

View File

@@ -5,8 +5,8 @@ export function BaseNode({ className, ...props }: ComponentProps<'div'>) {
return (
<div
className={cn(
'bg-card text-heading relative rounded-lg border border-default',
'transition-colors hover:border-hover',
'bg-card text-heading relative overflow-hidden rounded-xl border border-default',
'transition-colors hover:border-hover focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent/50',
'in-[.selected]:border-accent',
className,
)}