New component that will replace the task-lane SuggestedFix card. Docks above the chat composer with a 320ms slide-up animation. This commit implements only the Proposed state (Tasks 8 & 9 fill Verifying, Partial, AI-confirming, Nudge, Collapsed). Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
492 lines
15 KiB
CSS
492 lines
15 KiB
CSS
@import 'tailwindcss';
|
|
|
|
@custom-variant dark (&:where(.dark, .dark *));
|
|
|
|
/* React Flow — imported here (not in component JS) so @tailwindcss/vite
|
|
doesn't wrap it in a cascade layer, which would lower its specificity
|
|
below Tailwind's own styles and hide nodes/edges. */
|
|
@import '@xyflow/react/dist/style.css';
|
|
|
|
@theme {
|
|
/* ── Surface colors (Deep Charcoal — sidebar darkest) ─ */
|
|
--color-bg-page: #16181f;
|
|
--color-bg-sidebar: #0e1016;
|
|
--color-bg-card: #1e2028;
|
|
--color-bg-card-hover: #252830;
|
|
--color-bg-input: #252830;
|
|
--color-bg-code: #12141a;
|
|
--color-bg-elevated: #2a2d38;
|
|
--color-bg-raised: #303442;
|
|
|
|
/* ── Text colors ───────────────────────────────── */
|
|
--color-text-heading: #f0f2f5;
|
|
--color-text-primary: #e2e5eb;
|
|
--color-text-secondary: #848b9b;
|
|
--color-text-muted: #4f5666;
|
|
--color-text-rail-label: #e2e5eb;
|
|
|
|
/* ── Border colors ─────────────────────────────── */
|
|
--color-border-default: #2a2e3a;
|
|
--color-border-hover: #3d4252;
|
|
|
|
/* ── Accent (electric blue) ───────────────────────── */
|
|
--color-accent: #60a5fa;
|
|
--color-accent-hover: #3b82f6;
|
|
--color-accent-dim: rgba(96,165,250,0.10);
|
|
--color-accent-text: #93c5fd;
|
|
|
|
/* ── Semantic colors ───────────────────────────── */
|
|
--color-success: #34d399;
|
|
--color-success-dim: rgba(52,211,153,0.10);
|
|
--color-warning: #fbbf24;
|
|
--color-warning-dim: rgba(251,191,36,0.10);
|
|
--color-danger: #f87171;
|
|
--color-danger-dim: rgba(248,113,113,0.10);
|
|
--color-info: #67e8f9;
|
|
--color-info-dim: rgba(103,232,249,0.10);
|
|
|
|
/* ── Tailwind semantic mappings ─────────────────── */
|
|
--color-background: #16181f;
|
|
--color-foreground: #e2e5eb;
|
|
--color-card: #1e2028;
|
|
--color-card-foreground: #e2e5eb;
|
|
--color-popover: #1e2028;
|
|
--color-popover-foreground: #e2e5eb;
|
|
--color-primary: #60a5fa;
|
|
--color-primary-foreground: #ffffff;
|
|
--color-secondary: #2a2d38;
|
|
--color-secondary-foreground: #e2e5eb;
|
|
--color-muted: #2a2d38;
|
|
--color-muted-foreground: #848b9b;
|
|
--color-accent-tw: #2a2d38;
|
|
--color-accent-foreground: #e2e5eb;
|
|
--color-destructive: #f87171;
|
|
--color-destructive-foreground: #ffffff;
|
|
--color-border: #2a2e3a;
|
|
--color-input: #252830;
|
|
--color-ring: #60a5fa;
|
|
|
|
/* ── Radii ─────────────────────────────────────── */
|
|
--radius-sm: 5px;
|
|
--radius-md: 8px;
|
|
--radius-lg: 12px;
|
|
--radius-xl: 16px;
|
|
|
|
/* ── Fonts ─────────────────────────────────────── */
|
|
--font-sans: 'IBM Plex Sans', system-ui, -apple-system, sans-serif;
|
|
--font-heading: 'Bricolage Grotesque', system-ui, sans-serif;
|
|
--font-mono: 'JetBrains Mono', monospace;
|
|
/* font-label removed — migrated to font-mono or font-sans */
|
|
|
|
/* ── Animations ────────────────────────────────── */
|
|
--animate-fade-in: fade-in 200ms ease-out both;
|
|
--animate-fade-in-up: fade-in-up 200ms ease-out both;
|
|
--animate-slide-in-left: slide-in-from-left 200ms ease-out;
|
|
--animate-slide-in-right: slide-in-from-right 200ms ease-out both;
|
|
--animate-slide-in-bottom: slide-in-from-bottom 200ms ease-out both;
|
|
--animate-scale-in: scale-in 150ms ease-out both;
|
|
--animate-fade: fadeIn 300ms ease both;
|
|
--animate-slide-up: slide-up 320ms cubic-bezier(.22,.9,.28,1) both;
|
|
|
|
@keyframes slide-up {
|
|
from { transform: translateY(14px); opacity: 0; }
|
|
to { transform: translateY(0); opacity: 1; }
|
|
}
|
|
@keyframes fade-in {
|
|
from { opacity: 0; } to { opacity: 1; }
|
|
}
|
|
@keyframes fade-in-up {
|
|
from { opacity: 0; transform: translateY(4px); } to { opacity: 1; transform: translateY(0); }
|
|
}
|
|
@keyframes slide-in-from-left {
|
|
from { transform: translateX(-100%); } to { transform: translateX(0); }
|
|
}
|
|
@keyframes slide-in-from-right {
|
|
from { opacity: 0; transform: translateX(16px); } to { opacity: 1; transform: translateX(0); }
|
|
}
|
|
@keyframes slide-in-from-bottom {
|
|
from { opacity: 0; transform: translateY(16px); } to { opacity: 1; transform: translateY(0); }
|
|
}
|
|
@keyframes scale-in {
|
|
from { opacity: 0; transform: scale(0.95); } to { opacity: 1; transform: scale(1); }
|
|
}
|
|
@keyframes fadeIn {
|
|
from { opacity: 0; transform: translateY(6px); } to { opacity: 1; transform: translateY(0); }
|
|
}
|
|
}
|
|
|
|
:root {
|
|
--sidebar-w: 72px;
|
|
--sidebar-w-pinned: 260px;
|
|
--topbar-h: 44px;
|
|
--ease-out-smooth: cubic-bezier(0.4, 0, 0.2, 1);
|
|
}
|
|
|
|
/* ── Base styles ─────────────────────────────────────── */
|
|
@layer base {
|
|
* {
|
|
@apply border-border;
|
|
scrollbar-width: thin;
|
|
scrollbar-color: var(--color-border) transparent;
|
|
}
|
|
*::-webkit-scrollbar {
|
|
width: 6px;
|
|
height: 6px;
|
|
}
|
|
*::-webkit-scrollbar-track {
|
|
background: transparent;
|
|
}
|
|
*::-webkit-scrollbar-thumb {
|
|
background-color: var(--color-border);
|
|
border-radius: 9999px;
|
|
}
|
|
*::-webkit-scrollbar-thumb:hover {
|
|
background-color: var(--color-muted-foreground);
|
|
}
|
|
|
|
body {
|
|
@apply bg-background text-foreground font-sans;
|
|
}
|
|
|
|
button, [role="button"] {
|
|
cursor: pointer;
|
|
}
|
|
|
|
h1, h2, h3, h4, h5, h6 {
|
|
font-family: var(--font-heading);
|
|
font-weight: 700;
|
|
letter-spacing: -0.02em;
|
|
}
|
|
}
|
|
|
|
/* ── Glass/gradient shims removed — Design System v4 migration complete ── */
|
|
|
|
@utility stagger-item {
|
|
opacity: 0;
|
|
animation: fade-in-only 350ms var(--ease-out-smooth) both;
|
|
animation-delay: calc(var(--stagger-index, 0) * 50ms);
|
|
}
|
|
|
|
@utility btn-press {
|
|
@apply active:scale-[0.98] transition-transform;
|
|
}
|
|
|
|
/* ── New component classes (Design System v4) ───── */
|
|
@utility card-flat {
|
|
background: var(--color-bg-card);
|
|
border: 1px solid var(--color-border-default);
|
|
border-radius: 8px;
|
|
}
|
|
|
|
@utility card-interactive {
|
|
background: var(--color-bg-card);
|
|
border: 1px solid var(--color-border-default);
|
|
border-radius: 8px;
|
|
cursor: pointer;
|
|
transition: transform 350ms cubic-bezier(0.34, 1.56, 0.64, 1),
|
|
border-color 200ms ease,
|
|
box-shadow 200ms ease;
|
|
&:hover {
|
|
transform: translateY(-4px);
|
|
border-color: var(--color-border-hover);
|
|
box-shadow: 0 8px 24px rgba(0, 0, 0, 0.25);
|
|
}
|
|
}
|
|
|
|
@utility btn-primary-v4 {
|
|
background: var(--color-accent);
|
|
color: #ffffff;
|
|
font-weight: 550;
|
|
border-radius: 5px;
|
|
padding: 9px 16px;
|
|
font-size: 13px;
|
|
transition: filter 150ms ease, box-shadow 150ms ease, transform 150ms ease;
|
|
&:hover {
|
|
filter: brightness(1.1);
|
|
box-shadow: 0 2px 10px rgba(96, 165, 250, 0.2);
|
|
transform: translateY(-1px);
|
|
}
|
|
&:active {
|
|
box-shadow: 0 0 4px rgba(96, 165, 250, 0.1);
|
|
transform: translateY(0);
|
|
}
|
|
}
|
|
|
|
@utility btn-ghost-v4 {
|
|
background: transparent;
|
|
color: var(--color-text-secondary);
|
|
border: 1px solid var(--color-border-default);
|
|
border-radius: 5px;
|
|
padding: 9px 16px;
|
|
font-size: 13px;
|
|
transition: background 150ms ease, color 150ms ease, border-color 150ms ease, box-shadow 150ms ease, transform 150ms ease;
|
|
&:hover {
|
|
background: var(--color-bg-elevated);
|
|
color: var(--color-text-primary);
|
|
border-color: var(--color-border-hover);
|
|
transform: translateY(-1px);
|
|
}
|
|
&:active {
|
|
transform: translateY(0);
|
|
}
|
|
}
|
|
|
|
/* ── Interactive shadow patterns (dark-mode aware) ── */
|
|
/* On dark backgrounds, use brighter surfaces + faint accent glow
|
|
instead of black shadows (which are invisible on #1a1c23) */
|
|
|
|
/* Tab/toggle group: active item gets elevated surface + faint accent glow */
|
|
@utility tab-active-shadow {
|
|
background: var(--color-bg-elevated);
|
|
box-shadow: 0 1px 4px rgba(96, 165, 250, 0.08);
|
|
}
|
|
|
|
/* Card hover lift — brighter border + subtle accent glow */
|
|
@utility card-lift {
|
|
transition: box-shadow 200ms ease, transform 200ms ease, border-color 200ms ease;
|
|
&:hover {
|
|
border-color: var(--color-border-hover);
|
|
box-shadow: 0 2px 8px rgba(96, 165, 250, 0.06);
|
|
transform: translateY(-2px);
|
|
}
|
|
&:active {
|
|
box-shadow: none;
|
|
transform: translateY(0);
|
|
}
|
|
}
|
|
|
|
@utility rdp-custom {
|
|
@apply text-foreground;
|
|
& .rdp-month { @apply w-full; }
|
|
& .rdp-caption { @apply flex justify-center items-center mb-4; }
|
|
& .rdp-caption_label { @apply text-sm font-medium; }
|
|
& .rdp-nav { @apply flex gap-1; }
|
|
& .rdp-nav_button { @apply h-7 w-7 bg-transparent p-0 opacity-50 hover:opacity-100; }
|
|
& .rdp-table { @apply w-full border-collapse; }
|
|
& .rdp-head_cell { @apply text-muted-foreground font-normal text-xs; }
|
|
& .rdp-cell { @apply text-center text-sm p-0; }
|
|
& .rdp-day { @apply h-9 w-9 p-0 font-normal hover:bg-accent rounded-md transition-colors; }
|
|
& .rdp-day_selected { @apply bg-primary text-primary-foreground hover:bg-primary/90; }
|
|
& .rdp-day_today { @apply bg-accent text-accent-foreground; }
|
|
& .rdp-day_outside { @apply text-muted-foreground opacity-50; }
|
|
& .rdp-day_disabled { @apply text-muted-foreground opacity-50; }
|
|
& .rdp-day_range_middle { @apply bg-accent text-accent-foreground; }
|
|
& .rdp-day_hidden { @apply invisible; }
|
|
}
|
|
|
|
/* ── Layout utilities ────────────────────────────────── */
|
|
@layer utilities {
|
|
.app-shell {
|
|
display: grid;
|
|
grid-template-columns: var(--sidebar-w) 1fr;
|
|
grid-template-rows: auto 1fr;
|
|
height: 100vh;
|
|
overflow: hidden;
|
|
transition: grid-template-columns 200ms ease;
|
|
}
|
|
|
|
.app-shell--pinned {
|
|
--sidebar-w: 260px;
|
|
}
|
|
|
|
/* Legacy collapsed class — kept as alias for pinned inverse */
|
|
.app-shell--collapsed {
|
|
grid-template-columns: 56px 1fr;
|
|
}
|
|
|
|
.topbar {
|
|
grid-column: 1 / -1;
|
|
height: var(--topbar-h);
|
|
}
|
|
|
|
.sidebar {
|
|
min-height: 0;
|
|
height: 100%;
|
|
overflow-y: auto;
|
|
}
|
|
|
|
.main-content {
|
|
min-height: 0;
|
|
min-width: 0;
|
|
/* Each page handles its own scrolling — full-height pages use
|
|
overflow-hidden, scrollable pages use overflow-y-auto */
|
|
}
|
|
|
|
@media (max-width: 767px) {
|
|
.app-shell {
|
|
grid-template-columns: 1fr;
|
|
}
|
|
}
|
|
|
|
/* Staggered fade-in helper — uses opacity-only animation
|
|
so it doesn't block hover transforms via fill-mode */
|
|
.fade-in {
|
|
animation: fade-in-only 0.3s ease both;
|
|
}
|
|
|
|
@keyframes fade-in-only {
|
|
from { opacity: 0; }
|
|
to { opacity: 1; }
|
|
}
|
|
}
|
|
|
|
/* ── Sonner toast overrides ──────────────────────────── */
|
|
[data-sonner-toast] {
|
|
background-color: var(--color-card) !important;
|
|
color: var(--color-card-foreground) !important;
|
|
border: 1px solid var(--color-border) !important;
|
|
box-shadow: 0 10px 15px -3px rgb(0 0 0 / 0.3) !important;
|
|
border-radius: 0.75rem;
|
|
font-family: var(--font-sans);
|
|
}
|
|
|
|
[data-sonner-toast] [data-title] {
|
|
font-family: var(--font-sans);
|
|
font-weight: 600;
|
|
}
|
|
|
|
[data-sonner-toast][data-type='success'] {
|
|
border-color: oklch(0.76 0.15 163 / 0.3) !important;
|
|
}
|
|
[data-sonner-toast][data-type='success'] [data-icon] {
|
|
color: oklch(0.76 0.15 163);
|
|
}
|
|
|
|
[data-sonner-toast][data-type='error'] {
|
|
border-color: oklch(0.7 0.16 22 / 0.3) !important;
|
|
}
|
|
[data-sonner-toast][data-type='error'] [data-icon] {
|
|
color: oklch(0.7 0.16 22);
|
|
}
|
|
|
|
[data-sonner-toast][data-type='info'] {
|
|
border-color: var(--color-border) !important;
|
|
}
|
|
[data-sonner-toast][data-type='info'] [data-icon] {
|
|
color: var(--color-muted-foreground);
|
|
}
|
|
|
|
[data-sonner-toast][data-type='warning'] {
|
|
border-color: oklch(0.82 0.16 85 / 0.3) !important;
|
|
}
|
|
[data-sonner-toast][data-type='warning'] [data-icon] {
|
|
color: oklch(0.82 0.16 85);
|
|
}
|
|
|
|
[data-sonner-toast] [data-close-button] {
|
|
color: var(--color-muted-foreground);
|
|
border-radius: 0.375rem;
|
|
transition: color 150ms, background-color 150ms;
|
|
}
|
|
[data-sonner-toast] [data-close-button]:hover {
|
|
background-color: var(--color-accent);
|
|
color: var(--color-accent-foreground);
|
|
}
|
|
|
|
/* ── React Flow dark theme overrides ─────────────────── */
|
|
/* React Flow v12 uses --xy-* CSS custom properties for theming.
|
|
Override the defaults to match our design system. */
|
|
.react-flow.dark {
|
|
--xy-background-color-default: transparent;
|
|
--xy-edge-stroke-default: var(--color-border);
|
|
--xy-edge-stroke-selected-default: var(--color-primary);
|
|
--xy-edge-label-color-default: var(--color-muted-foreground);
|
|
--xy-edge-label-background-color-default: var(--color-card);
|
|
--xy-node-background-color-default: var(--color-card);
|
|
--xy-node-color-default: var(--color-foreground);
|
|
--xy-node-border-default: 1px solid var(--color-border);
|
|
--xy-handle-background-color-default: var(--color-border);
|
|
--xy-handle-border-color-default: var(--color-card);
|
|
--xy-minimap-background-color-default: var(--color-card);
|
|
--xy-minimap-mask-background-color-default: rgba(28, 31, 42, 0.6);
|
|
--xy-controls-button-background-color-default: var(--color-card);
|
|
--xy-controls-button-background-color-hover-default: var(--color-accent);
|
|
--xy-controls-button-color-default: var(--color-muted-foreground);
|
|
--xy-controls-button-color-hover-default: var(--color-foreground);
|
|
--xy-controls-button-border-color-default: var(--color-border);
|
|
}
|
|
|
|
.react-flow__controls {
|
|
border-radius: 0.75rem !important;
|
|
box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.3) !important;
|
|
overflow: hidden;
|
|
}
|
|
|
|
.react-flow__minimap {
|
|
border-radius: 0.75rem !important;
|
|
}
|
|
|
|
.react-flow__attribution {
|
|
display: none;
|
|
}
|
|
|
|
/* ── Glow edge animations ────────────────────────────── */
|
|
@keyframes glow-flow-downstream {
|
|
from { stroke-dashoffset: 40; }
|
|
to { stroke-dashoffset: 0; }
|
|
}
|
|
|
|
@keyframes glow-flow-upstream {
|
|
from { stroke-dashoffset: 0; }
|
|
to { stroke-dashoffset: 40; }
|
|
}
|
|
|
|
.glow-edge-flow-downstream {
|
|
animation: glow-flow-downstream 1s linear infinite;
|
|
}
|
|
|
|
.glow-edge-flow-upstream {
|
|
animation: glow-flow-upstream 1s linear infinite;
|
|
}
|
|
|
|
/* ── Accessibility: Reduce motion ────────────────────── */
|
|
@media (prefers-reduced-motion: reduce) {
|
|
*,
|
|
*::before,
|
|
*::after {
|
|
animation-duration: 0.01ms !important;
|
|
animation-iteration-count: 1 !important;
|
|
transition-duration: 0.01ms !important;
|
|
scroll-behavior: auto !important;
|
|
}
|
|
}
|
|
|
|
/* ── Print / PDF export ───────────────────────────────────────────────── */
|
|
@media print {
|
|
/* Hide everything that isn't the canvas */
|
|
body > * { display: none !important; }
|
|
|
|
/* Show only the React Flow viewport inside the diagram editor page */
|
|
#root { display: block !important; }
|
|
#root > * { display: none !important; }
|
|
|
|
/* The diagram editor mounts as a child of the app shell — target the canvas wrapper */
|
|
.react-flow__renderer,
|
|
.react-flow__viewport,
|
|
.react-flow {
|
|
display: block !important;
|
|
}
|
|
|
|
/* Make the canvas fill the printed page */
|
|
.react-flow {
|
|
position: fixed !important;
|
|
inset: 0 !important;
|
|
width: 100vw !important;
|
|
height: 100vh !important;
|
|
background: #ffffff !important;
|
|
}
|
|
|
|
/* Force light backgrounds on nodes so they're readable on white paper */
|
|
.react-flow__node {
|
|
print-color-adjust: exact;
|
|
-webkit-print-color-adjust: exact;
|
|
}
|
|
|
|
/* Hide UI chrome */
|
|
.react-flow__controls,
|
|
.react-flow__minimap,
|
|
.react-flow__panel {
|
|
display: none !important;
|
|
}
|
|
}
|