learning_ai_common_plat/packages/ui
saravanakumardb1 dd90f709e1 fix(gitea): set ROOT_URL=host.docker.internal, NO_PROXY for host (F17)
Resolves F17 in docker-build-optimization-roadmap.

Root cause:
  Gitea's app.ini ROOT_URL was http://localhost:3300/. Gitea bakes
  ROOT_URL into the dist.tarball field of every published package's
  metadata. Inside a Docker container, 'localhost' is the container
  itself, not the host \u2014 so any 'pnpm install' that needed to fetch
  a tarball would ECONNREFUSED, even though the registry metadata
  itself was reachable via host.docker.internal.

Server-side fix (not in git, requires manual replication on each dev
machine; documented in roadmap \u00a73 A-pre-6):
  - Edit /opt/homebrew/var/gitea/custom/conf/app.ini:
    ROOT_URL = http://host.docker.internal:3300/
  - brew services restart gitea
  - sudo sh -c 'echo "127.0.0.1 host.docker.internal" >> /etc/hosts'

Repo-side fix (this commit):
  - switch-network.sh: add host.docker.internal to NO_PROXY +
    NPM_CONFIG_NOPROXY when NETWORK=corp. Required so host-side curl/
    pnpm/npm bypass the corporate proxy (cso.proxy.att.com) when
    resolving host.docker.internal. Without this, host installs fail
    with the corp proxy's 'Unknown Host' 504 page.

Republished all 64 @bytelyst/* packages so tarball URLs reflect the
new ROOT_URL:
  - .publish-manifest.json: 64 entries with new content hashes
  - packages/*/package.json: 64 patch-version bumps
    (auto-bumped by publish-outdated-packages.sh because previous
    versions already existed in registry)

Verification:
  curl http://localhost:3300/.../@bytelyst%2Ferrors | jq .dist.tarball
  → http://host.docker.internal:3300/.../errors-0.1.11.tgz  (was localhost:3300)
  workspace:* refs across all 64 packages: 0

Unblocks: A0-V on every pilot. Verified PASSING on learning_ai_clock:
  backend cold build: 59.2 s
  web cold build:     3:13 (193 s)
  Both via Gitea registry, no docker-prep.sh tarballs needed.
2026-05-27 01:51:43 -07:00
..
.storybook feat(ui): add Storybook for @bytelyst/ui component library 2026-03-28 00:59:25 -07:00
src feat(ui): add <CardButton> primitive (UI audit Pattern A fix) 2026-05-10 09:31:29 +00:00
eslint.config.js chore(audit): unblock workspace lint pipeline + 13 mechanical fixes 2026-05-04 14:21:34 -07:00
package.json fix(gitea): set ROOT_URL=host.docker.internal, NO_PROXY for host (F17) 2026-05-27 01:51:43 -07:00
README.md feat(ui): add Sidebar, StatCard, LoadingSpinner components + README docs — Level 3 component coverage 2026-03-29 00:19:45 -07:00
tsconfig.json chore(platform): align docker and package outputs 2026-03-29 23:41:08 +00:00

@bytelyst/ui

Shared component library for the ByteLyst ecosystem. Built with Radix UI primitives, Lucide icons, and CSS custom properties from @bytelyst/design-tokens.

Install

pnpm add @bytelyst/ui

Peer dependencies: react, react-dom.

Components (15)

Button

5 variants (primary, secondary, ghost, destructive, outline), 3 sizes, loading state with spinner.

import { Button } from '@bytelyst/ui';

<Button variant="primary" size="md" loading={saving}>Save</Button>
<Button variant="destructive" onClick={onDelete}>Delete</Button>
<Button variant="ghost" size="sm">Cancel</Button>

Input

Text input with error/success states. Supports aria-invalid automatically when error is truthy.

import { Input } from '@bytelyst/ui';

<Input placeholder="Email" type="email" />
<Input error="Required field" />

Textarea

Auto-resize textarea with error states.

import { Textarea } from '@bytelyst/ui';

<Textarea placeholder="Description" rows={4} />;

Select

Styled <select> with keyboard navigation.

import { Select } from '@bytelyst/ui';

<Select
  options={[
    { value: 'a', label: 'Option A' },
    { value: 'b', label: 'Option B' },
  ]}
  value={selected}
  onChange={setSelected}
/>;

Modal / Dialog

Radix @radix-ui/react-dialog with focus trap, Esc close, and aria-modal.

import { Modal } from '@bytelyst/ui';

<Modal open={isOpen} onOpenChange={setOpen} title="Edit item">
  <p>Modal content here</p>
</Modal>;

ConfirmDialog

Radix AlertDialog for destructive/warning confirmations. Supports async onConfirm.

import { ConfirmDialog } from '@bytelyst/ui';

<ConfirmDialog
  open={showConfirm}
  onOpenChange={setShowConfirm}
  title="Delete item?"
  description="This action cannot be undone."
  onConfirm={handleDelete}
  variant="destructive"
/>;

Toast

4 types (success, error, warning, info), auto-dismiss, role="alert". Use ToastProvider in your root layout and toast() anywhere.

import { ToastProvider, toast } from '@bytelyst/ui';

// In providers.tsx
<ToastProvider>{children}</ToastProvider>;

// Anywhere in app
toast('Saved successfully', 'success');
toast('Something went wrong', 'error');

Card

Container with default, elevated, and interactive variants. Includes CardHeader, CardTitle, CardDescription sub-components.

import { Card, CardHeader, CardTitle, CardDescription } from '@bytelyst/ui';

<Card>
  <CardHeader>
    <CardTitle>Dashboard</CardTitle>
    <CardDescription>Overview of your data</CardDescription>
  </CardHeader>
  {children}
</Card>;

Badge

Status/count/risk-level badges, color-coded by semantic tokens.

import { Badge } from '@bytelyst/ui';

<Badge variant="success">Active</Badge>
<Badge variant="danger">Critical</Badge>
<Badge variant="warning" count={5} />

EmptyState

Placeholder for empty lists with icon, description, and optional CTA.

import { EmptyState } from '@bytelyst/ui';

<EmptyState
  icon={<FileText size={48} />}
  title="No items yet"
  description="Create your first item to get started."
  action={<Button onClick={onCreate}>Create</Button>}
/>;

Label

Styled form label.

import { Label } from '@bytelyst/ui';

<Label htmlFor="email">Email address</Label>;

Separator

Horizontal/vertical divider.

import { Separator } from '@bytelyst/ui';

<Separator />
<Separator orientation="vertical" />

Sidebar

Responsive collapsible sidebar with mobile toggle, overlay, and navigation items.

import { Sidebar, SidebarItem } from '@bytelyst/ui';

<Sidebar
  collapsed={collapsed}
  onToggle={() => setCollapsed(!collapsed)}
  header={<span>MyApp</span>}
  footer="v1.0.0"
>
  <SidebarItem href="/dashboard" label="Dashboard" icon={<Home size={18} />} active />
  <SidebarItem href="/settings" label="Settings" icon={<Settings size={18} />} />
</Sidebar>;

StatCard

Dashboard metric card with trend indicator.

import { StatCard } from '@bytelyst/ui';

<StatCard label="Total Users" value="1,234" trend="up" trendValue="+12%" />
<StatCard label="Errors" value={42} trend="down" trendValue="-8%" icon={<AlertCircle />} />

LoadingSpinner

Accessible loading indicator with aria-busy and prefers-reduced-motion support.

import { LoadingSpinner } from '@bytelyst/ui';

<LoadingSpinner size="md" label="Loading data…" />
<LoadingSpinner size="sm" label="" />  {/* spinner only, no text */}

Design Token Integration

All components use CSS custom properties with --bl- prefix and sensible fallbacks. Override them per-product:

:root {
  --bl-accent: var(--jj-accent); /* JarvisJr */
  --bl-surface-card: var(--fm-surface); /* FlowMonk */
}

Accessibility

Every component includes:

  • focus-visible ring styles
  • ARIA attributes (aria-label, aria-modal, aria-busy, aria-invalid, aria-current)
  • Keyboard navigation (Esc to close, Tab focus trap in modals)
  • prefers-reduced-motion respected via Tailwind's animate-spin

Storybook

pnpm --filter @bytelyst/ui storybook

Storybook is configured with @storybook/addon-a11y for automated accessibility checks.