// ───────────────────────────────────────────────────────────────────────────── // Component: Proposals list (.prl-* prefix) // File: src/scss/components/_proposals.scss // // All visual rules — colors, typography, borders, backgrounds. // Page layout lives in pages/_proposals.scss. // ───────────────────────────────────────────────────────────────────────────── $proposal-open: #3fb950; $proposal-merged: #a371f7; $proposal-closed: #f85149; // ── Hero ────────────────────────────────────────────────────────────────────── .prl-hero { position: relative; overflow: hidden; background: linear-gradient(to bottom, color-mix(in srgb, var(--bg-canvas) 60%, transparent) 0%, var(--bg-surface) 100%); border-bottom: 1px solid var(--border-default); padding: var(--space-10) var(--space-8) var(--space-8); } .prl-hero-glow { position: absolute; inset: 0; pointer-events: none; opacity: 0.06; &--open { background: radial-gradient(ellipse 55% 60% at 15% 0%, #{$proposal-open} 0%, transparent 70%); } &--merged { background: radial-gradient(ellipse 55% 60% at 15% 0%, #{$proposal-merged} 0%, transparent 70%); } &--closed { background: radial-gradient(ellipse 55% 60% at 15% 0%, #{$proposal-closed} 0%, transparent 70%); } &--all { background: radial-gradient(ellipse 55% 60% at 15% 0%, var(--color-accent) 0%, transparent 70%); } } .prl-hero-eyebrow { display: inline-flex; align-items: center; gap: 6px; font-size: 11px; font-weight: var(--weight-medium); color: var(--text-muted); text-transform: uppercase; letter-spacing: 0.07em; margin-bottom: var(--space-1); } .prl-hero-title { font-size: 28px; font-weight: var(--weight-bold); color: var(--text-primary); line-height: 1.15; margin: 0; letter-spacing: -0.02em; } .prl-hero-sub { font-size: 13px; color: var(--text-muted); line-height: 1.55; margin: 0; max-width: 440px; } .prl-new-btn { display: inline-flex; align-items: center; gap: 7px; padding: 8px 16px; font-size: 13px; font-weight: var(--weight-semibold); color: var(--text-primary); background: transparent; border: 1px solid var(--border-default); border-radius: var(--radius-md); text-decoration: none; transition: border-color var(--transition-fast), background var(--transition-fast), color var(--transition-fast); &:hover { text-decoration: none; border-color: var(--color-accent); color: var(--color-accent); background: color-mix(in srgb, var(--color-accent) 7%, transparent); } } // ── Filter / sort bar ───────────────────────────────────────────────────────── .prl-filterbar { display: flex; align-items: stretch; border-bottom: 1px solid var(--border-subtle); background: var(--bg-surface); padding: 0 var(--space-6); gap: 0; min-height: 44px; } .prl-tabs { display: flex; flex: 1; gap: 0; overflow-x: auto; scrollbar-width: none; &::-webkit-scrollbar { display: none; } } .prl-tab { display: inline-flex; align-items: center; gap: 6px; padding: 0 14px; font-size: 12px; font-weight: var(--weight-medium); color: var(--text-muted); text-decoration: none; border-bottom: 2px solid transparent; white-space: nowrap; transition: color var(--transition-fast), border-color var(--transition-fast); min-height: 44px; &:hover { color: var(--text-secondary); text-decoration: none; } &--active { color: var(--text-primary); border-bottom-color: var(--color-accent); font-weight: var(--weight-semibold); } } .prl-tab-pip { width: 7px; height: 7px; border-radius: 50%; flex-shrink: 0; &--open { background: $proposal-open; } &--merged { background: $proposal-merged; } &--closed { background: $proposal-closed; } } .prl-tab-ct { background: var(--bg-overlay); border: 1px solid var(--border-subtle); border-radius: var(--radius-full); font-size: 10px; font-weight: var(--weight-semibold); font-family: var(--font-mono); padding: 1px 5px; min-width: 16px; text-align: center; color: var(--text-muted); } .prl-controls-divider { width: 1px; height: 16px; background: var(--border-subtle); flex-shrink: 0; } .prl-pill-group { display: flex; align-items: center; gap: 1px; } .prl-pill { font-size: 11px; font-weight: var(--weight-medium); color: var(--text-muted); text-decoration: none; padding: 3px 9px; border-radius: var(--radius-full); transition: background var(--transition-fast), color var(--transition-fast); white-space: nowrap; &:hover { color: var(--text-secondary); text-decoration: none; background: var(--bg-overlay); } &--active { color: var(--text-primary); font-weight: var(--weight-semibold); background: var(--bg-overlay); border: 1px solid var(--border-subtle); } } // ── Proposal row ────────────────────────────────────────────────────────────── .prl-row { display: flex; align-items: flex-start; gap: var(--space-4); padding: 20px var(--space-6); border-bottom: 1px solid var(--border-subtle); text-decoration: none; cursor: pointer; transition: background var(--transition-fast); animation: prl-row-in 180ms ease both; &:last-child { border-bottom: none; } &:nth-child(even) { background: color-mix(in srgb, var(--bg-overlay) 80%, transparent); } &:hover { background: var(--bg-overlay); text-decoration: none; } } @keyframes prl-row-in { from { opacity: 0; transform: translateY(4px); } to { opacity: 1; transform: translateY(0); } } @for $i from 1 through 8 { .prl-row:nth-child(#{$i}) { animation-delay: #{($i - 1) * 28}ms; } } .prl-row-num { font-size: 11px; font-family: var(--font-mono); color: var(--text-muted); opacity: 0.4; flex-shrink: 0; width: 36px; text-align: right; } .prl-row-title { font-size: 14px; font-weight: var(--weight-semibold); color: var(--text-primary); line-height: 1.4; display: block; margin: 6px 0 4px; } // ── Key-value chip — universal badge component ──────────────────────────────── // Each chip has a dim key zone and a highlighted value zone. // Outer element carries the color variant class. .prl-kv { display: inline-flex; align-items: stretch; font-size: 10px; font-family: var(--font-mono); border-radius: var(--radius-sm); border: 1px solid var(--border-subtle); overflow: hidden; flex-shrink: 0; letter-spacing: 0.03em; } .prl-kv-k { display: inline-flex; align-items: center; padding: 2px 5px; color: var(--text-muted); font-weight: 400; background: color-mix(in srgb, var(--text-primary) 5%, transparent); border-right: 1px solid var(--border-subtle); gap: 3px; } .prl-kv-v { display: inline-flex; align-items: center; gap: 4px; padding: 2px 7px; font-weight: var(--weight-semibold); color: var(--text-secondary); } // The risk value zone contains an inline progress bar .prl-kv-risk-bar { display: inline-block; height: 5px; min-width: 3px; max-width: 36px; border-radius: 2px; opacity: 0.7; } // ── Type variants ───────────────────────────────────────────────────────────── .prl-kv--type { border-color: color-mix(in srgb, var(--color-accent) 20%, transparent); .prl-kv-v { color: var(--color-accent); } } .prl-kv--type-identity-transition { border-color: color-mix(in srgb, #a78bfa 25%, transparent); .prl-kv-v { color: #a78bfa; } } .prl-kv--type-canonical-release { border-color: color-mix(in srgb, #34d399 25%, transparent); .prl-kv-v { color: #34d399; } } .prl-kv--type-stem-integration { border-color: color-mix(in srgb, #60a5fa 25%, transparent); .prl-kv-v { color: #60a5fa; } } .prl-kv--type-midi-evolution { border-color: color-mix(in srgb, #f472b6 25%, transparent); .prl-kv-v { color: #f472b6; } } .prl-kv--type-payment-settlement { border-color: color-mix(in srgb, #fbbf24 25%, transparent); .prl-kv-v { color: #fbbf24; } } .prl-kv--type-agent-delegation { border-color: color-mix(in srgb, #2dd4bf 25%, transparent); .prl-kv-v { color: #2dd4bf; } } // ── Risk variants ───────────────────────────────────────────────────────────── .prl-kv--risk-low { border-color: color-mix(in srgb, var(--color-success) 25%, transparent); .prl-kv-v { color: var(--color-success); } .prl-kv-risk-bar { background: var(--color-success); } } .prl-kv--risk-medium { border-color: color-mix(in srgb, var(--color-warning) 30%, transparent); .prl-kv-v { color: var(--color-warning); } .prl-kv-risk-bar { background: var(--color-warning); } } .prl-kv--risk-high { border-color: color-mix(in srgb, var(--color-orange, #f97316) 30%, transparent); .prl-kv-v { color: #f97316; } .prl-kv-risk-bar { background: #f97316; } } .prl-kv--risk-critical { border-color: color-mix(in srgb, var(--color-danger) 35%, transparent); .prl-kv-k { background: color-mix(in srgb, var(--color-danger) 8%, transparent); } .prl-kv-v { color: $proposal-closed; } .prl-kv-risk-bar { background: var(--color-danger); } } // ── Strategy variant ────────────────────────────────────────────────────────── .prl-kv--strategy { border-color: color-mix(in srgb, var(--color-accent) 18%, transparent); .prl-kv-v { color: var(--text-secondary); } } // ── Blocked variant ─────────────────────────────────────────────────────────── .prl-kv--blocked { border-color: color-mix(in srgb, var(--color-danger) 30%, transparent); .prl-kv-k { color: var(--color-danger); background: color-mix(in srgb, var(--color-danger) 8%, transparent); } .prl-kv-v { color: $proposal-closed; } } .prl-kv--blocks { border-color: color-mix(in srgb, var(--color-warning) 30%, transparent); .prl-kv-k { color: var(--color-warning); background: color-mix(in srgb, var(--color-warning) 8%, transparent); } .prl-kv-v { color: var(--color-warning); } } // ── Health variants ─────────────────────────────────────────────────────────── .prl-kv--breakage { border-color: color-mix(in srgb, var(--color-danger) 25%, transparent); .prl-kv-k { color: $proposal-closed; background: color-mix(in srgb, var(--color-danger) 8%, transparent); } .prl-kv-v { color: $proposal-closed; } } .prl-kv--gap { border-color: color-mix(in srgb, var(--color-warning) 25%, transparent); .prl-kv-k { color: var(--color-warning); background: color-mix(in srgb, var(--color-warning) 8%, transparent); } .prl-kv-v { color: var(--color-warning); } } // ── Semver variants ─────────────────────────────────────────────────────────── .prl-kv--semver-major { border-color: color-mix(in srgb, var(--color-danger) 30%, transparent); .prl-kv-v { color: $proposal-closed; font-weight: var(--font-weight-black); } } .prl-kv--semver-minor { border-color: color-mix(in srgb, var(--color-success) 25%, transparent); .prl-kv-v { color: $proposal-open; } } .prl-kv--semver-patch { border-color: var(--border-subtle); .prl-kv-v { color: var(--text-muted); } } // ── Status variants ─────────────────────────────────────────────────────────── .prl-kv--ready { border-color: color-mix(in srgb, var(--color-success) 25%, transparent); .prl-kv-k { color: var(--color-success); background: color-mix(in srgb, var(--color-success) 8%, transparent); } .prl-kv-v { color: var(--color-success); } } .prl-kv--settling { border-color: color-mix(in srgb, var(--color-warning) 25%, transparent); .prl-kv-k { color: var(--color-warning); background: color-mix(in srgb, var(--color-warning) 8%, transparent); } .prl-kv-v { color: var(--color-warning); } } .prl-kv--draft { border-color: var(--border-subtle); .prl-kv-k { color: var(--text-muted); } .prl-kv-v { color: var(--text-muted); } } // Commit-type badge .prl-badge { display: inline-block; font-size: 10px; font-weight: var(--weight-bold); font-family: var(--font-mono); padding: 1px 5px; border-radius: var(--radius-sm); text-transform: lowercase; letter-spacing: 0.02em; background: color-mix(in srgb, var(--color-accent) 10%, transparent); color: var(--color-accent); border: 1px solid color-mix(in srgb, var(--color-accent) 22%, transparent); flex-shrink: 0; &[data-type="feat"], &[data-type="feature"] { background: color-mix(in srgb, var(--color-success) 10%, transparent); color: $proposal-open; border-color: color-mix(in srgb, var(--color-success) 22%, transparent); } &[data-type="fix"], &[data-type="bugfix"], &[data-type="hotfix"] { background: color-mix(in srgb, var(--color-danger) 10%, transparent); color: $proposal-closed; border-color: color-mix(in srgb, var(--color-danger) 22%, transparent); } &[data-type="refactor"] { background: color-mix(in srgb, var(--color-purple) 10%, transparent); color: $proposal-merged; border-color: color-mix(in srgb, var(--color-purple) 22%, transparent); } &[data-type="docs"] { background: color-mix(in srgb, var(--color-warning) 10%, transparent); color: var(--color-warning); border-color: color-mix(in srgb, var(--color-warning) 22%, transparent); } &[data-type="test"] { background: color-mix(in srgb, var(--color-teal) 10%, transparent); color: var(--color-teal); border-color: rgba(45, 212, 191, 0.22); } &[data-type="chore"], &[data-type="style"] { background: color-mix(in srgb, var(--text-secondary) 10%, transparent); color: var(--text-secondary); border-color: color-mix(in srgb, var(--text-secondary) 22%, transparent); } &[data-type="redesign"] { background: rgba(245, 158, 11, 0.1); color: #f59e0b; border-color: rgba(245, 158, 11, 0.22); } &[data-type="breaking"], &[data-type="major"] { background: color-mix(in srgb, var(--color-danger) 15%, transparent); color: $proposal-closed; border-color: color-mix(in srgb, var(--color-danger) 30%, transparent); font-weight: var(--font-weight-black); } } // Semver impact pill .prl-impact { display: inline-block; font-size: 9px; font-weight: var(--font-weight-black); font-family: var(--font-mono); padding: 1px 4px; border-radius: 2px; text-transform: uppercase; letter-spacing: 0.06em; flex-shrink: 0; &--major { background: color-mix(in srgb, var(--color-danger) 15%, transparent); color: $proposal-closed; } &--minor { background: color-mix(in srgb, var(--color-success) 12%, transparent); color: $proposal-open; } &--patch { background: color-mix(in srgb, var(--text-secondary) 12%, transparent); color: var(--text-secondary); } } .prl-badge--breakage { background: color-mix(in srgb, var(--color-danger) 10%, transparent); color: $proposal-closed; border-color: color-mix(in srgb, var(--color-danger) 22%, transparent); display: inline-flex; align-items: center; gap: 3px; flex-shrink: 0; } .prl-badge--gap { background: color-mix(in srgb, var(--color-warning) 10%, transparent); color: var(--color-warning); border-color: color-mix(in srgb, var(--color-warning) 22%, transparent); display: inline-flex; align-items: center; gap: 3px; flex-shrink: 0; } .prl-branch { display: inline-flex; align-items: center; gap: 3px; font-size: 11px; font-family: var(--font-mono); color: var(--text-muted); background: var(--bg-overlay); border: 1px solid var(--border-subtle); border-radius: var(--radius-sm); padding: 1px 5px; max-width: 200px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; &--from { color: var(--color-accent); border-color: color-mix(in srgb, var(--color-accent) 22%, transparent); background: color-mix(in srgb, var(--color-accent) 7%, transparent); } } .prl-arrow { flex-shrink: 0; color: var(--text-muted); opacity: 0.5; } .prl-meta-dot { opacity: 0.3; flex-shrink: 0; } .prl-actor { display: inline-flex; align-items: center; gap: 4px; color: var(--text-secondary); font-weight: var(--weight-medium); flex-shrink: 0; } .prl-sigil { width: 20px; height: 20px; border-radius: 4px; flex-shrink: 0; display: block; } .prl-date { color: var(--text-muted); flex-shrink: 0; } .prl-snippet { font-size: 11px; color: var(--text-muted); overflow: hidden; white-space: nowrap; text-overflow: ellipsis; opacity: 0.75; min-width: 0; max-width: 320px; } .prl-merged-label { display: inline-flex; align-items: center; gap: 4px; font-size: 11px; color: $proposal-merged; opacity: 0.8; } .prl-closed-label { font-size: 11px; color: $proposal-closed; opacity: 0.7; } .prl-sha { font-size: 10px; font-family: var(--font-mono); color: var(--text-muted); opacity: 0.6; text-decoration: none; transition: opacity var(--transition-fast); &:hover { opacity: 1; text-decoration: underline; } } .prl-chevron { flex-shrink: 0; color: var(--text-muted); opacity: 0; transition: opacity var(--transition-fast); } .prl-row:hover .prl-chevron { opacity: 0.5; } // ── Empty state ─────────────────────────────────────────────────────────────── .prl-empty-icon { opacity: 0.18; margin-bottom: var(--space-2); } .prl-empty-title { font-size: 18px; font-weight: var(--weight-semibold); color: var(--text-secondary); } .prl-empty-desc { font-size: 13px; color: var(--text-muted); line-height: 1.6; max-width: 380px; } .prl-empty-cli { display: inline-flex; align-items: center; gap: 8px; margin-top: var(--space-2); padding: var(--space-2) var(--space-4); border-radius: var(--radius-md); background: var(--bg-overlay); border: 1px solid var(--border-subtle); font-size: 12px; } .prl-empty-cli-prompt { font-family: var(--font-mono); color: var(--text-muted); opacity: 0.5; } .prl-empty-cli-cmd { font-family: var(--font-mono); color: var(--text-primary); font-weight: var(--weight-medium); } .prl-empty-cli-hint { color: var(--text-muted); border-left: 1px solid var(--border-subtle); padding-left: 8px; } // ── Proposal type badge (ptype) ─────────────────────────────────────────────── .prl-badge--ptype { font-size: 9px; font-weight: var(--weight-bold); font-family: var(--font-mono); text-transform: uppercase; letter-spacing: 0.07em; padding: 2px 6px; border-radius: var(--radius-sm); flex-shrink: 0; background: color-mix(in srgb, var(--color-accent) 12%, transparent); color: var(--color-accent); border: 1px solid color-mix(in srgb, var(--color-accent) 25%, transparent); } // ── Inline risk display (line 1 of row) ─────────────────────────────────────── .prl-risk-inline { display: inline-flex; align-items: center; gap: 5px; flex-shrink: 0; } .prl-risk-inline-bar { display: inline-block; height: 6px; min-width: 4px; max-width: 48px; border-radius: 2px; } .prl-risk-inline-label { font-size: 9px; font-weight: var(--weight-bold); font-family: var(--font-mono); text-transform: uppercase; letter-spacing: 0.06em; } .prl-risk-inline--low { .prl-risk-inline-bar { background: var(--color-success); } .prl-risk-inline-label { color: var(--color-success); } } .prl-risk-inline--medium { .prl-risk-inline-bar { background: var(--color-warning); } .prl-risk-inline-label { color: var(--color-warning); } } .prl-risk-inline--high { .prl-risk-inline-bar { background: var(--color-orange); } .prl-risk-inline-label { color: var(--color-orange); } } .prl-risk-inline--critical { .prl-risk-inline-bar { background: var(--color-danger-critical); } .prl-risk-inline-label { color: var(--color-danger-critical); } } // ── Status indicator badges (blocked / settling / ready) ────────────────────── .prl-dep-lock { display: inline-flex; align-items: center; gap: 4px; font-size: 10px; font-family: var(--font-mono); font-weight: var(--weight-semibold); color: var(--color-danger); background: color-mix(in srgb, var(--color-danger) 8%, transparent); border: 1px solid color-mix(in srgb, var(--color-danger) 22%, transparent); border-radius: var(--radius-sm); padding: 1px 6px; flex-shrink: 0; } .prl-settling-badge { display: inline-flex; align-items: center; gap: 4px; font-size: 10px; font-family: var(--font-mono); font-weight: var(--weight-semibold); color: var(--color-warning); background: color-mix(in srgb, var(--color-warning) 8%, transparent); border: 1px solid color-mix(in srgb, var(--color-warning) 22%, transparent); border-radius: var(--radius-sm); padding: 1px 6px; flex-shrink: 0; } .prl-merge-ready-badge { display: inline-flex; align-items: center; gap: 4px; font-size: 10px; font-family: var(--font-mono); font-weight: var(--weight-semibold); color: var(--color-success); background: color-mix(in srgb, var(--color-success) 8%, transparent); border: 1px solid color-mix(in srgb, var(--color-success) 22%, transparent); border-radius: var(--radius-sm); padding: 1px 6px; flex-shrink: 0; } // ── Approval pips ───────────────────────────────────────────────────────────── .prl-approval-dots { display: inline-flex; align-items: center; gap: 3px; flex-shrink: 0; } .prl-approval-pip { width: 7px; height: 7px; border-radius: 50%; border: 1.5px solid var(--text-muted); opacity: 0.4; flex-shrink: 0; &--filled { background: var(--color-success); border-color: var(--color-success); opacity: 1; } } .prl-approval-label { font-size: 10px; font-family: var(--font-mono); color: var(--text-muted); margin-left: 2px; } // ── Conflict count ──────────────────────────────────────────────────────────── .prl-conflict-count { display: inline-flex; align-items: center; gap: 3px; font-size: 10px; font-family: var(--font-mono); color: var(--color-warning); flex-shrink: 0; &--clean { color: var(--color-success); } } // ── Row footer (author · date · snippet) ────────────────────────────────────── .prl-row-footer { display: flex; align-items: center; flex-wrap: wrap; gap: 5px; font-size: 11px; color: var(--text-muted); min-width: 0; } .prl-author-badge { display: inline-flex; align-items: center; &--agent { color: var(--color-accent); opacity: 0.8; } } // ── Risk badge ──────────────────────────────────────────────────────────────── .prl-risk-badge { display: inline-flex; align-items: center; font-size: 0.62rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.07em; padding: 2px 7px; border-radius: 9999px; border: 1px solid; font-family: var(--font-mono); white-space: nowrap; &--low { color: var(--color-success); border-color: color-mix(in srgb, var(--color-success) 35%, transparent); background: color-mix(in srgb, var(--color-success) 8%, transparent); } &--medium { color: var(--color-warning); border-color: color-mix(in srgb, var(--color-warning) 35%, transparent); background: color-mix(in srgb, var(--color-warning) 8%, transparent); } &--high { color: var(--color-orange); border-color: color-mix(in srgb, var(--color-orange) 35%, transparent); background: color-mix(in srgb, var(--color-orange) 8%, transparent); } &--critical { color: var(--color-danger-critical); border-color: color-mix(in srgb, var(--color-danger-critical) 35%, transparent); background: color-mix(in srgb, var(--color-danger-critical) 8%, transparent); } }