CUB-41: Add low stock indicators to filament UI

This commit is contained in:
cubecraft-agents[bot]
2026-04-26 14:27:08 +00:00
parent 230c3b295d
commit 3d67610575
34 changed files with 10758 additions and 0 deletions

View File

@@ -0,0 +1,63 @@
<!-- Dashboard Summary Bar — Fleet-wide health at a glance -->
<section class="dashboard-summary" role="status" aria-label="Dashboard summary">
<!-- System Health Indicator -->
<div class="summary-item health-indicator"
[class.healthy]="health().status === 'healthy'"
[class.degraded]="isDegraded()"
[class.down]="isDown()"
[matTooltip]="statusLabel()"
matTooltipPosition="below">
<span class="connection-dot" [class.connected]="health().connected"></span>
<span class="health-label">{{ statusLabel() }}</span>
</div>
<!-- Total Active Agents -->
<div class="summary-item" matTooltip="Total active agents" matTooltipPosition="below">
<mat-icon aria-hidden="true">smart_toy</mat-icon>
<span class="metric-value">{{ summary().active }} / {{ summary().total }}</span>
<span class="metric-label">Active</span>
</div>
<!-- Status Breakdown -->
<div class="summary-item status-breakdown">
<mat-chip-set aria-label="Agent status breakdown">
<mat-chip
class="status-chip chip-active"
[class.has-count]="summary().active > 0"
matTooltip="Active agents">
<mat-icon matChipStart>check_circle</mat-icon>
<span class="chip-count">{{ summary().active }}</span>
<span class="chip-label">Active</span>
</mat-chip>
<mat-chip
class="status-chip chip-idle"
[class.has-count]="summary().idle > 0"
matTooltip="Idle agents">
<mat-icon matChipStart>pause_circle</mat-icon>
<span class="chip-count">{{ summary().idle }}</span>
<span class="chip-label">Idle</span>
</mat-chip>
<mat-chip
class="status-chip chip-thinking"
[class.has-count]="summary().thinking > 0"
matTooltip="Thinking agents">
<mat-icon matChipStart>psychology</mat-icon>
<span class="chip-count">{{ summary().thinking }}</span>
<span class="chip-label">Thinking</span>
</mat-chip>
<mat-chip
class="status-chip chip-error"
[class.has-count]="hasErrors()"
matTooltip="Agents in error">
<mat-icon matChipStart>error</mat-icon>
<span class="chip-count">{{ summary().error }}</span>
<span class="chip-label">Error</span>
</mat-chip>
</mat-chip-set>
</div>
</section>

View File

@@ -0,0 +1,174 @@
/**
* Dashboard Summary Component Styles
* Touch-optimized for kiosk (Raspberry Pi 5) and mobile PWA
* Uses Angular Material utility classes where possible
*/
// Touch-optimized sizing
$touch-target-min: 48px;
$kiosk-font-primary: 20px;
$mobile-font-primary: 16px;
$spacing-unit: 8px;
// Status colors — high contrast for workshop/bright environments
$color-active: #4ade70; // Green — printing/active
$color-idle: #94a3b8; // Gray — idle/offline
$color-thinking: #60a5fa; // Blue — thinking/processing
$color-error: #f87171; // Red — error/failed
$color-connected: #4ade70; // Green — SignalR connected
$color-disconnected: #f87171; // Red — disconnected
.dashboard-summary {
display: flex;
align-items: center;
gap: $spacing-unit * 2;
padding: $spacing-unit * 2;
overflow-x: auto;
-webkit-overflow-scrolling: touch;
// Responsive: on mobile, allow horizontal scroll
@media (max-width: 480px) {
padding: $spacing-unit;
gap: $spacing-unit;
}
}
.summary-item {
display: flex;
align-items: center;
gap: $spacing-unit;
min-height: $touch-target-min;
white-space: nowrap;
.metric-value {
font-size: $kiosk-font-primary;
font-weight: 600;
line-height: 1.2;
@media (max-width: 480px) {
font-size: $mobile-font-primary;
}
}
.metric-label {
font-size: 12px;
color: rgba(255, 255, 255, 0.7);
text-transform: uppercase;
letter-spacing: 0.05em;
@media (max-width: 480px) {
font-size: 10px;
}
}
}
// Health indicator
.health-indicator {
padding: $spacing-unit $spacing-unit * 2;
border-radius: 24px;
transition: background-color 0.3s ease;
&.healthy {
background-color: rgba($color-active, 0.15);
}
&.degraded {
background-color: rgba($color-thinking, 0.15);
}
&.down {
background-color: rgba($color-error, 0.15);
}
.connection-dot {
width: 12px;
height: 12px;
border-radius: 50%;
display: inline-block;
transition: background-color 0.3s ease;
&.connected {
background-color: $color-connected;
box-shadow: 0 0 6px $color-connected;
}
&:not(.connected) {
background-color: $color-disconnected;
box-shadow: 0 0 6px $color-disconnected;
}
}
.health-label {
font-size: 14px;
font-weight: 500;
@media (max-width: 480px) {
font-size: 12px;
}
}
}
// Status breakdown chips
.status-breakdown {
flex-shrink: 0;
}
.status-chip {
min-height: $touch-target-min !important;
font-size: 14px !important;
@media (max-width: 480px) {
min-height: 40px !important;
font-size: 12px !important;
padding: 0 8px !important;
}
.chip-count {
font-weight: 700;
margin: 0 4px;
}
.chip-label {
font-size: 12px;
opacity: 0.8;
}
mat-icon {
font-size: 18px !important;
width: 18px !important;
height: 18px !important;
}
}
// Status chip color variants
.chip-active {
--mdc-chip-outline-color: #{$color-active};
&.has-count {
background-color: rgba($color-active, 0.15) !important;
}
}
.chip-idle {
--mdc-chip-outline-color: #{$color-idle};
&.has-count {
background-color: rgba($color-idle, 0.15) !important;
}
}
.chip-thinking {
--mdc-chip-outline-color: #{$color-thinking};
&.has-count {
background-color: rgba($color-thinking, 0.15) !important;
}
}
.chip-error {
--mdc-chip-outline-color: #{$color-error};
&.has-count {
background-color: rgba($color-error, 0.2) !important;
}
}

View File

@@ -0,0 +1,103 @@
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { DashboardSummaryComponent } from './dashboard-summary.component';
import { AgentSummary, SystemHealth } from '../../models/agent.model';
describe('DashboardSummaryComponent', () => {
let component: DashboardSummaryComponent;
let fixture: ComponentFixture<DashboardSummaryComponent>;
const mockSummary: AgentSummary = {
total: 7,
active: 4,
idle: 1,
thinking: 1,
error: 1,
};
const mockHealthy: SystemHealth = {
connected: true,
status: 'healthy',
};
beforeEach(async () => {
await TestBed.configureTestingModule({
imports: [DashboardSummaryComponent],
}).compileComponents();
fixture = TestBed.createComponent(DashboardSummaryComponent);
component = fixture.componentInstance;
});
it('should create', () => {
expect(component).toBeTruthy();
});
it('should default to zeroed summary', () => {
const summary = component.summary();
expect(summary.total).toBe(0);
expect(summary.active).toBe(0);
expect(summary.idle).toBe(0);
expect(summary.thinking).toBe(0);
expect(summary.error).toBe(0);
});
it('should default to disconnected/down health', () => {
const health = component.health();
expect(health.connected).toBe(false);
expect(health.status).toBe('down');
});
it('should update summary data', () => {
component.updateSummary(mockSummary);
expect(component.summary()).toEqual(mockSummary);
});
it('should update health data', () => {
component.updateHealth(mockHealthy);
expect(component.health()).toEqual(mockHealthy);
});
it('should compute hasErrors correctly', () => {
expect(component.hasErrors()).toBe(false);
component.updateSummary({ ...mockSummary, error: 2 });
expect(component.hasErrors()).toBe(true);
});
it('should compute connectionColor correctly', () => {
expect(component.connectionColor()).toBe('disconnected');
component.updateHealth({ connected: true, status: 'healthy' });
expect(component.connectionColor()).toBe('connected');
});
it('should compute statusLabel for each state', () => {
component.updateHealth({ connected: true, status: 'healthy' });
expect(component.statusLabel()).toBe('All Systems Go');
component.updateHealth({ connected: true, status: 'degraded' });
expect(component.statusLabel()).toBe('Degraded');
component.updateHealth({ connected: false, status: 'down' });
expect(component.statusLabel()).toBe('Offline');
});
it('should render summary values in template', () => {
component.updateSummary(mockSummary);
component.updateHealth(mockHealthy);
fixture.detectChanges();
const compiled = fixture.nativeElement as HTMLElement;
expect(compiled.textContent).toContain('4 / 7');
expect(compiled.textContent).toContain('Active');
expect(compiled.textContent).toContain('All Systems Go');
});
it('should render status breakdown chips', () => {
component.updateSummary(mockSummary);
fixture.detectChanges();
const compiled = fixture.nativeElement as HTMLElement;
expect(compiled.textContent).toContain('4'); // active count
expect(compiled.textContent).toContain('1'); // idle count (multiple)
expect(compiled.textContent).toContain('Error');
});
});

View File

@@ -0,0 +1,80 @@
import { ChangeDetectionStrategy, Component, Input, OnDestroy, signal, computed } from '@angular/core';
import { CommonModule } from '@angular/common';
import { MatButtonModule } from '@angular/material/button';
import { MatIconModule } from '@angular/material/icon';
import { MatChipsModule } from '@angular/material/chips';
import { MatTooltipModule } from '@angular/material/tooltip';
import { AgentSummary, SystemHealth } from '../../models/agent.model';
@Component({
selector: 'app-dashboard-summary',
standalone: true,
imports: [
CommonModule,
MatButtonModule,
MatIconModule,
MatChipsModule,
MatTooltipModule,
],
templateUrl: './dashboard-summary.component.html',
styleUrls: ['./dashboard-summary.component.scss'],
changeDetection: ChangeDetectionStrategy.OnPush,
})
export class DashboardSummaryComponent implements OnDestroy {
/** Agent summary data — reactive signal, updatable via updateSummary() */
readonly summary = signal<AgentSummary>({
total: 0,
active: 0,
idle: 0,
thinking: 0,
error: 0,
});
/** System health data — reactive signal, updatable via updateHealth() */
readonly health = signal<SystemHealth>({
connected: false,
status: 'down',
});
/** Computed signal: whether there are errors to highlight */
readonly hasErrors = computed(() => this.summary().error > 0);
/** Computed signal: whether system is degraded */
readonly isDegraded = computed(() => this.health().status === 'degraded');
/** Computed signal: whether system is down */
readonly isDown = computed(() => this.health().status === 'down');
/** Computed signal: connection indicator color */
readonly connectionColor = computed(() =>
this.health().connected ? 'connected' : 'disconnected'
);
/** Computed signal: overall status label */
readonly statusLabel = computed(() => {
const h = this.health();
if (h.status === 'healthy') return 'All Systems Go';
if (h.status === 'degraded') return 'Degraded';
return 'Offline';
});
/**
* Update the agent summary. Called by the parent or a service
* when new data arrives (e.g., via SignalR).
*/
updateSummary(data: AgentSummary): void {
this.summary.set(data);
}
/**
* Update the system health. Called by the parent or a service
* when the connection state changes.
*/
updateHealth(data: SystemHealth): void {
this.health.set(data);
}
ngOnDestroy(): void {
// Cleanup handled by signals — no manual subscription teardown needed
}
}