Components
List
A vertical stack of focusable rows (watchOS list spirit). Keep it short — a glanceable HUD caps at 3–5 rows. Compose List with ListRow (leading glyph, label, trailing value).
Installation
npx @glasskit-ui/cli add listInstall the SDK (it provides GlassViewport, useDpad and the stylesheet), then copy these files into your project:
npm install @glasskit-ui/react// components/lib/utils.tsexport type ClassValue = string | number | null | undefined | false;/** * Join truthy class names. Dependency-free on purpose: the lens components * style via bespoke semantic classes (no conflicting Tailwind utilities to * de-dupe), so this needs no clsx/tailwind-merge and resolves from anywhere * the registry is vendored. */export function cn(...inputs: ClassValue[]): string { return inputs.filter(Boolean).join(" ");}/** * Accessible name from a free-form `label` prop: the label itself when it's a * plain string, otherwise undefined (a ReactNode can't become an aria-label). */export function stringLabel(label: unknown): string | undefined { return typeof label === "string" ? label : undefined;}// components/glasskit/list.tsx"use client";import { useEffect, useRef, type ReactNode } from "react";import { cn } from "../lib/utils";/** * <List> — a vertical stack of focusable rows (watchOS list spirit) that fills * the stage and scrolls (D-pad scrollIntoView). A position indicator on the * inline-end edge tracks scroll — sized to the content and hidden when the list * fits. Compose with <ListRow>. */export function List({ children, className,}: { children: ReactNode; className?: string;}) { const scrollRef = useRef<HTMLDivElement>(null); const railRef = useRef<HTMLSpanElement>(null); const thumbRef = useRef<HTMLSpanElement>(null); useEffect(() => { const sc = scrollRef.current; const rail = railRef.current; const thumb = thumbRef.current; if (!sc || !rail || !thumb) return; const update = () => { const { scrollTop, scrollHeight, clientHeight } = sc; const overflow = scrollHeight - clientHeight; if (overflow <= 1) { rail.dataset.show = "false"; return; } rail.dataset.show = "true"; const trackH = rail.clientHeight; const thumbH = Math.max(trackH * (clientHeight / scrollHeight), 26); const pos = (scrollTop / overflow) * (trackH - thumbH); thumb.style.height = `${thumbH}px`; thumb.style.transform = `translateY(${pos}px)`; }; update(); sc.addEventListener("scroll", update, { passive: true }); const ro = typeof ResizeObserver !== "undefined" ? new ResizeObserver(update) : null; ro?.observe(sc); return () => { sc.removeEventListener("scroll", update); ro?.disconnect(); }; }, []); return ( <div className={cn("gk-list", className)}> <div className="gk-list__scroll" ref={scrollRef}> {children} </div> <span className="gk-list__rail" ref={railRef} data-show="false" aria-hidden="true" > <span className="gk-list__bar" ref={thumbRef} /> </span> </div> );}/** * <ListRow> — a D-pad-focusable row: leading glyph, label, trailing value. * Renders a real <button> with the `focusable` class so `useDpad()` walks * it. Logical layout (leading = inline-start) for RTL safety. */export function ListRow({ children, leading, trailing, onClick, disabled, className,}: { /** The row label. */ children: ReactNode; /** Optional inline-start glyph — typically a <GlowIcon>. */ leading?: ReactNode; /** Optional inline-end value/affordance. */ trailing?: ReactNode; onClick?: () => void; disabled?: boolean; className?: string;}) { return ( <button type="button" disabled={disabled} onClick={onClick} className={cn("focusable gk-list-row t-body", className)} > {leading} <span className="gk-list-row__label">{children}</span> {trailing != null ? ( <span className="gk-list-row__trailing t-caption">{trailing}</span> ) : null} </button> );}Usage
<List> <ListRow leading={<GlowIcon size="sm"><NavIcon /></GlowIcon>}> Navigate </ListRow> <ListRow trailing="2">Messages</ListRow></List>Props
Prop
Type
Confirm
A decision screen: a prompt plus a two-button action bar. Drop it into a Screen stage; useDpad seeds focus on the primary action.
QuickReplyChips
Tappable canned replies (the comms job — there is no keyboard on the lens, text is voice). Each chip is D-pad-focusable. Keep the set short and glanceable.