Hybrid traditional + AI-first webapp scaffold. Sibling to crema-app-template, adds the AI assistant surface, command bus, scripts dialog, and virtual cursor. What's pre-wired: - 6 routes: Overview, Resources, Activity, Assistant, Library, Settings - Collapsible rail + appbar + avatar dropdown shell (template code, not a lib) - Mobile sheet at <md - /assistant: streaming chat via @crema/llm-ui, mock fallback, model selector, token meter, retry probe, stop-while-streaming, persistent UI Control toggle - /settings: editable LM Studio endpoint + context window + response cap, with test-connection button - Markdown rendering for assistant replies; ```action``` blocks rendered as a small "Ran N actions" pill - ⌘⇧P script runner dialog + Play icon in the appbar - Two demo scripts in public/scripts/ - mightypix theme as default, scoped via <AppShell theme="mightypix"> Libs wired in tsconfig + app.css: - @crema/action-bus (the bus, parser, runner, cursor, provider, ws, llm-bridge) - @crema/llm-ui, @crema/chat-ui, @crema/aifirst-ui, @crema/notification-ui - lib-theme-mightypix Docs: - README.md — pitch + quick start + structure - docs/AI_FIRST.md — full system tour (data-action contract, bus, DSL, scripts, cursor, LLM integration) - app/components/layout/THEME_CONTRACT.md — every CSS variable a theme must declare - CLAUDE.md — orientation for an LLM working in the repo Genericized from comfy-cloud (the original prototype): - Brand defaults to "App" / Sparkles icon (override via app/lib/identity.ts) - User defaults to a stub (swap useUser() for real auth) - localStorage namespace is "crema.*" (was "comfy.*") Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
91 lines
2.5 KiB
TypeScript
91 lines
2.5 KiB
TypeScript
// Persisted LLM settings — base URL, context budget, response cap.
|
|
// Reactive across tabs (storage event) and within the same tab (custom event).
|
|
|
|
import { useEffect, useSyncExternalStore } from "react"
|
|
|
|
export type LLMSettings = {
|
|
baseURL: string
|
|
contextTokens: number
|
|
responseBudget: number
|
|
}
|
|
|
|
export const DEFAULT_SETTINGS: LLMSettings = {
|
|
baseURL: "http://localhost:1234/v1",
|
|
contextTokens: 9000,
|
|
responseBudget: 512,
|
|
}
|
|
|
|
const STORAGE_KEY = "crema.llm.settings"
|
|
const CHANGE_EVENT = "comfy:llm-settings-change"
|
|
|
|
function readFromStorage(): LLMSettings {
|
|
if (typeof window === "undefined") return DEFAULT_SETTINGS
|
|
try {
|
|
const raw = localStorage.getItem(STORAGE_KEY)
|
|
if (!raw) return DEFAULT_SETTINGS
|
|
const parsed = JSON.parse(raw) as Partial<LLMSettings>
|
|
return {
|
|
baseURL: typeof parsed.baseURL === "string" ? parsed.baseURL : DEFAULT_SETTINGS.baseURL,
|
|
contextTokens:
|
|
Number.isFinite(parsed.contextTokens) && (parsed.contextTokens as number) > 0
|
|
? (parsed.contextTokens as number)
|
|
: DEFAULT_SETTINGS.contextTokens,
|
|
responseBudget:
|
|
Number.isFinite(parsed.responseBudget) && (parsed.responseBudget as number) > 0
|
|
? (parsed.responseBudget as number)
|
|
: DEFAULT_SETTINGS.responseBudget,
|
|
}
|
|
} catch {
|
|
return DEFAULT_SETTINGS
|
|
}
|
|
}
|
|
|
|
export function loadLLMSettings(): LLMSettings {
|
|
return readFromStorage()
|
|
}
|
|
|
|
export function saveLLMSettings(next: LLMSettings) {
|
|
if (typeof window === "undefined") return
|
|
localStorage.setItem(STORAGE_KEY, JSON.stringify(next))
|
|
window.dispatchEvent(new CustomEvent(CHANGE_EVENT))
|
|
}
|
|
|
|
export function resetLLMSettings() {
|
|
saveLLMSettings(DEFAULT_SETTINGS)
|
|
}
|
|
|
|
let cached: LLMSettings | null = null
|
|
|
|
function subscribe(cb: () => void): () => void {
|
|
const onChange = () => {
|
|
cached = null
|
|
cb()
|
|
}
|
|
window.addEventListener(CHANGE_EVENT, onChange)
|
|
window.addEventListener("storage", (e) => {
|
|
if (e.key === STORAGE_KEY) onChange()
|
|
})
|
|
return () => {
|
|
window.removeEventListener(CHANGE_EVENT, onChange)
|
|
}
|
|
}
|
|
|
|
function getSnapshot(): LLMSettings {
|
|
if (!cached) cached = readFromStorage()
|
|
return cached
|
|
}
|
|
|
|
function getServerSnapshot(): LLMSettings {
|
|
return DEFAULT_SETTINGS
|
|
}
|
|
|
|
export function useLLMSettings(): LLMSettings {
|
|
// useSyncExternalStore avoids hydration flicker and stays reactive.
|
|
const value = useSyncExternalStore(subscribe, getSnapshot, getServerSnapshot)
|
|
// Re-read after mount to pick up localStorage on first client render.
|
|
useEffect(() => {
|
|
cached = null
|
|
}, [])
|
|
return value
|
|
}
|