kua-money-trace/web/dashboard.css

225 lines
16 KiB
CSS

/* money-trace · dashboard layout & charts */
.dash-body { overflow-y: auto; overflow-x: hidden; }
/* tab nav */
.tabs {
display: flex; gap: 2px; padding: 0 22px; border-bottom: 1px solid var(--line);
background: var(--panel); position: sticky; top: 0; z-index: 20;
overflow-x: auto; scrollbar-width: thin;
}
.tabs::-webkit-scrollbar { height: 4px; }
.tabs::-webkit-scrollbar-thumb { background: var(--line-2); border-radius: 2px; }
.tab {
font-family: var(--font-ui); font-size: 13.5px; color: var(--ink-mute);
background: transparent; border: 0; border-bottom: 2px solid transparent;
padding: 15px 18px 13px; cursor: pointer; display: flex; align-items: center; gap: 9px;
transition: .16s; white-space: nowrap;
}
.tab:hover { color: var(--ink); }
.tab.on { color: var(--ink); border-bottom-color: var(--c-income); }
.tab svg { width: 16px; height: 16px; opacity: .85; }
.tab .tnum { font-family: var(--font-num); font-size: 10px; color: var(--ink-dim); }
/* panels */
.panel-wrap { padding: 22px; max-width: 1320px; margin: 0 auto; }
.panel { display: none; animation: fade .22s ease; }
.panel.on { display: block; }
@keyframes fade { from { opacity: 0; transform: translateY(6px); } to { opacity: 1; transform: none; } }
.panel-head { margin-bottom: 18px; }
.panel-head h2 { font-size: 22px; font-weight: 600; margin: 0 0 5px; letter-spacing: -0.01em; }
.panel-head p { font-size: 13.5px; color: var(--ink-mute); margin: 0; max-width: 640px; line-height: 1.5; }
.panel-head p b { color: var(--ink); font-weight: 500; }
/* grid */
.grid { display: grid; gap: 16px; }
.g-2 { grid-template-columns: 1fr 1fr; }
.g-3 { grid-template-columns: repeat(3, 1fr); }
.g-1-2 { grid-template-columns: 320px 1fr; }
.g-2-1 { grid-template-columns: 1fr 340px; }
@media (max-width: 1080px) { .g-2, .g-3, .g-1-2, .g-2-1 { grid-template-columns: 1fr; } }
/* card */
.card { background: var(--panel); border: 1px solid var(--line); border-radius: 14px; padding: 18px; }
.card.tall { min-height: 280px; }
.card h3 { font-size: 12px; letter-spacing: 0.12em; text-transform: uppercase; color: var(--ink-mute); margin: 0 0 16px; font-weight: 600; display: flex; align-items: center; gap: 8px; }
.card h3 .pill { margin-left: auto; font-family: var(--font-num); font-size: 11px; color: var(--ink-dim); text-transform: none; letter-spacing: 0; }
.card .chart-host { width: 100%; }
/* KPI cards */
.kpi { background: var(--panel); border: 1px solid var(--line); border-radius: 14px; padding: 18px 20px; position: relative; overflow: hidden; }
.kpi::before { content: ''; position: absolute; left: 0; top: 0; bottom: 0; width: 3px; background: var(--accent, var(--ink-mute)); }
.kpi .k-lab { font-size: 11px; letter-spacing: 0.1em; text-transform: uppercase; color: var(--ink-mute); margin-bottom: 10px; }
.kpi .k-val { font-size: 30px; font-weight: 500; line-height: 1; font-family: var(--font-num); letter-spacing: -0.01em; }
.kpi .k-val small { font-size: 15px; color: var(--ink-dim); }
.kpi .k-sub { font-size: 11.5px; color: var(--ink-dim); margin-top: 8px; }
.kpi .k-sub b { font-family: var(--font-num); color: var(--ink-mute); font-weight: 500; }
/* hbars */
.hbars { display: flex; flex-direction: column; gap: 11px; }
.hbar-row { display: grid; grid-template-columns: 168px 1fr 96px; align-items: center; gap: 14px; cursor: default; }
.hbar-label { font-size: 13px; color: var(--ink); display: flex; flex-direction: column; gap: 2px; overflow: hidden; }
.hbar-label .hbar-sub { font-size: 10.5px; color: var(--ink-dim); font-family: var(--font-num); }
.hbar-track { height: 22px; background: var(--bg); border-radius: 5px; overflow: hidden; }
.hbar-fill { height: 100%; border-radius: 5px; opacity: 0.9; transition: width .5s cubic-bezier(.3,1,.4,1); }
.hbar-val { text-align: right; font-size: 13px; color: var(--ink); }
.hbar-row:hover .hbar-fill { opacity: 1; }
/* legend chips */
.chips { display: flex; flex-wrap: wrap; gap: 8px 16px; margin-top: 14px; }
.chip { display: flex; align-items: center; gap: 7px; font-size: 12px; color: var(--ink-mute); }
.chip i { width: 10px; height: 10px; border-radius: 3px; }
.chip b { color: var(--ink); font-family: var(--font-num); font-weight: 500; }
/* chart axis text */
.chart-axis { fill: var(--ink-dim); font-size: 10px; font-family: var(--font-num); }
.donut-center-top { fill: var(--ink); font-size: 20px; font-family: var(--font-num); font-weight: 500; }
.donut-center-bot { fill: var(--ink-dim); font-size: 10.5px; letter-spacing: 0.08em; text-transform: uppercase; font-family: var(--font-ui); }
.donut-wrap { display: flex; align-items: center; gap: 22px; }
/* big callout */
.callout { display: flex; align-items: baseline; gap: 14px; padding: 4px 0; }
.callout .big { font-size: 54px; font-weight: 600; font-family: var(--font-num); line-height: 1; letter-spacing: -0.02em; }
.callout .big.mult { color: var(--c-fee); }
.callout .ctxt { font-size: 13px; color: var(--ink-mute); line-height: 1.5; }
.callout .ctxt b { color: var(--ink); }
/* bank table */
.btable { width: 100%; border-collapse: collapse; font-size: 13px; }
.btable th { text-align: right; font-size: 10.5px; letter-spacing: 0.08em; text-transform: uppercase; color: var(--ink-dim); font-weight: 600; padding: 0 14px 12px; border-bottom: 1px solid var(--line); }
.btable th:first-child { text-align: left; }
.btable td { padding: 13px 14px; border-bottom: 1px solid var(--line); text-align: right; font-family: var(--font-num); color: var(--ink); }
.btable td:first-child { text-align: left; font-family: var(--font-ui); }
.btable tr:hover td { background: rgba(255,255,255,0.02); }
.btable .bk-name { display: flex; align-items: center; gap: 10px; }
.btable .bk-dot { width: 9px; height: 9px; border-radius: 3px; flex: none; }
.btable .bk-meta { font-size: 10.5px; color: var(--ink-dim); font-family: var(--font-num); }
.btable .minibar { display: inline-flex; height: 7px; border-radius: 3px; overflow: hidden; width: 120px; background: var(--bg); vertical-align: middle; }
.btable .minibar i { height: 100%; display: block; }
.btable td.muted { color: var(--ink-dim); }
/* insight strip */
.insight { display: flex; gap: 12px; align-items: flex-start; background: var(--panel-2); border: 1px solid var(--line); border-radius: 12px; padding: 14px 16px; font-size: 13px; color: var(--ink-mute); line-height: 1.55; }
.insight .i-ic { width: 26px; height: 26px; border-radius: 7px; display: grid; place-items: center; flex: none; background: var(--bg); color: var(--c-income); }
.insight b { color: var(--ink); font-weight: 500; }
.insight .num { color: var(--ink); }
/* month mini selector for panels */
.mini-months { display: flex; gap: 3px; align-items: center; }
.mini-months .mm { font-family: var(--font-num); font-size: 11px; color: var(--ink-dim); padding: 4px 8px; border-radius: 6px; cursor: pointer; }
.mini-months .mm:hover { color: var(--ink); background: var(--bg); }
.mini-months .mm.on { color: var(--bg); background: var(--c-income); }
/* top merchants list */
.mlist { display: flex; flex-direction: column; }
.mrow { display: flex; align-items: center; gap: 12px; padding: 10px 0; border-bottom: 1px solid var(--line); }
.mrow:last-child { border-bottom: 0; }
.mrow .m-rank { font-family: var(--font-num); font-size: 11px; color: var(--ink-dim); width: 20px; }
.mrow .m-name { flex: 1; font-size: 13px; color: var(--ink); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
.mrow .m-cat { font-size: 10.5px; color: var(--ink-dim); margin-left: 6px; }
.mrow .m-amt { font-family: var(--font-num); font-size: 13px; color: var(--ink); }
.mrow .m-cnt { font-family: var(--font-num); font-size: 10.5px; color: var(--ink-dim); width: 48px; text-align: right; }
/* ---- heatmap ---- */
.heatmap { display: flex; flex-direction: column; gap: 4px; }
.hm-row { display: grid; grid-template-columns: 64px repeat(var(--cols, 31), 1fr); gap: 4px; align-items: center; }
.hm-head { margin-bottom: 2px; }
.hm-rowlab { font-size: 11.5px; color: var(--ink-mute); text-align: right; padding-right: 8px; font-family: var(--font-ui); }
.hm-collab { font-size: 9px; color: var(--ink-dim); text-align: center; font-family: var(--font-num); }
.hm-cell { aspect-ratio: 1; border-radius: 3px; background: color-mix(in oklab, var(--hc) calc(var(--a) * 100%), var(--bg)); border: 1px solid var(--line); transition: transform .1s; min-height: 14px; }
.hm-cell[data-v="0"] { background: var(--bg); }
.hm-cell:hover { transform: scale(1.18); border-color: var(--line-2); cursor: default; }
.hm-scale { display: flex; align-items: center; gap: 8px; font-size: 10.5px; color: var(--ink-dim); margin-top: 12px; justify-content: flex-end; }
.hm-scale .sw { display: flex; gap: 3px; }
.hm-scale .sw i { width: 14px; height: 14px; border-radius: 3px; }
/* ---- account picker chips ---- */
.acct-chips { display: flex; flex-wrap: wrap; gap: 8px; margin-bottom: 16px; }
.acct-chip { display: flex; align-items: center; gap: 8px; padding: 7px 12px; border-radius: 8px; border: 1px solid var(--line); background: var(--panel); cursor: pointer; font-size: 12.5px; color: var(--ink-mute); transition: .14s; }
.acct-chip:hover { border-color: var(--line-2); color: var(--ink); }
.acct-chip.on { color: var(--ink); background: var(--panel-2); }
.acct-chip i { width: 9px; height: 9px; border-radius: 3px; }
.acct-chip .ac-end { font-family: var(--font-num); font-size: 11px; color: var(--ink-dim); }
/* ---- ledger explorer table ---- */
.lx-controls { display: flex; gap: 12px; flex-wrap: wrap; align-items: center; margin-bottom: 16px; }
.lx-search { flex: 1; min-width: 220px; position: relative; }
.lx-search input { width: 100%; background: var(--bg); border: 1px solid var(--line); border-radius: 9px; padding: 11px 14px 11px 38px; color: var(--ink); font-family: var(--font-ui); font-size: 13.5px; }
.lx-search input:focus { outline: none; border-color: var(--line-2); }
.lx-search svg { position: absolute; left: 12px; top: 11px; width: 16px; height: 16px; color: var(--ink-dim); }
.lx-select { background: var(--bg); border: 1px solid var(--line); border-radius: 9px; padding: 11px 13px; color: var(--ink); font-family: var(--font-ui); font-size: 13px; cursor: pointer; }
.lx-count { font-size: 12px; color: var(--ink-dim); font-family: var(--font-num); white-space: nowrap; }
.lx-table-wrap { border: 1px solid var(--line); border-radius: 12px; overflow: hidden; }
.lx-table { width: 100%; border-collapse: collapse; font-size: 12.5px; }
.lx-table thead { position: sticky; top: 0; background: var(--panel-2); z-index: 2; }
.lx-table th { text-align: left; font-size: 10px; letter-spacing: 0.08em; text-transform: uppercase; color: var(--ink-dim); font-weight: 600; padding: 11px 14px; border-bottom: 1px solid var(--line); cursor: pointer; user-select: none; white-space: nowrap; }
.lx-table th:hover { color: var(--ink-mute); }
.lx-table th.r, .lx-table td.r { text-align: right; }
.lx-table td { padding: 10px 14px; border-bottom: 1px solid var(--line); color: var(--ink-mute); }
.lx-table tr:last-child td { border-bottom: 0; }
.lx-table tr:hover td { background: rgba(255,255,255,0.02); }
.lx-table .lx-date { font-family: var(--font-num); font-size: 11.5px; color: var(--ink-dim); white-space: nowrap; }
.lx-table .lx-desc { color: var(--ink); max-width: 320px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
.lx-table .lx-amt { font-family: var(--font-num); white-space: nowrap; }
.lx-table .lx-amt.cr { color: var(--c-income); }
.lx-table .lx-amt.db { color: var(--ink); }
.lx-flow-tag { display: inline-flex; align-items: center; gap: 5px; font-size: 10.5px; padding: 2px 8px; border-radius: 20px; border: 1px solid var(--line-2); color: var(--ink-mute); white-space: nowrap; }
.lx-flow-tag i { width: 6px; height: 6px; border-radius: 50%; }
.lx-pdf { color: var(--ink-dim); font-family: var(--font-num); font-size: 10.5px; }
.lx-more { text-align: center; padding: 14px; color: var(--ink-dim); font-size: 12px; cursor: pointer; }
.lx-more:hover { color: var(--ink); background: rgba(255,255,255,0.02); }
/* ---- coverage grid (data quality) ---- */
.cov-grid { overflow-x: auto; }
.cov-row { display: grid; grid-template-columns: 190px repeat(var(--cols, 15), 1fr); gap: 3px; align-items: center; margin-bottom: 3px; }
.cov-head .cov-cell { background: transparent; }
.cov-rowlab { font-size: 11.5px; color: var(--ink); display: flex; align-items: center; gap: 8px; overflow: hidden; }
.cov-rowlab .cov-dot { width: 8px; height: 8px; border-radius: 3px; flex: none; }
.cov-rowlab .cov-meta { font-size: 9.5px; color: var(--ink-dim); font-family: var(--font-num); }
.cov-collab { font-size: 9px; color: var(--ink-dim); text-align: center; font-family: var(--font-num); writing-mode: vertical-rl; height: 36px; }
.cov-cell { height: 22px; border-radius: 3px; background: var(--bg); border: 1px solid var(--line); position: relative; }
.cov-cell.has { background: color-mix(in oklab, var(--c-income) 30%, var(--bg)); border-color: color-mix(in oklab, var(--c-income) 45%, var(--bg)); }
.cov-cell.has:hover { background: color-mix(in oklab, var(--c-income) 55%, var(--bg)); }
/* ---- mini stat row ---- */
.qstats { display: grid; grid-template-columns: repeat(4, 1fr); gap: 14px; margin-bottom: 16px; }
.qstat { background: var(--panel); border: 1px solid var(--line); border-radius: 12px; padding: 15px 16px; }
.qstat .qv { font-family: var(--font-num); font-size: 24px; color: var(--ink); }
.qstat .ql { font-size: 11px; color: var(--ink-mute); margin-top: 5px; }
.qbar { height: 8px; border-radius: 4px; background: var(--bg); overflow: hidden; margin-top: 10px; display: flex; }
.qbar i { height: 100%; }
@media (max-width: 1080px) { .qstats { grid-template-columns: 1fr 1fr; } }
/* net pill */
.net-pos { color: var(--c-income); }
.net-neg { color: var(--c-spend); }
/* ---- category breakdown accordion ---- */
.bk-section { border-bottom: 1px solid var(--line); }
.bk-section:last-child { border-bottom: 0; }
.bk-header { display: flex; align-items: center; gap: 10px; padding: 11px 14px; cursor: pointer; user-select: none; }
.bk-header:hover { background: rgba(255,255,255,0.025); }
.bk-toggle { font-size: 9px; color: var(--ink-dim); width: 10px; flex-shrink: 0; }
.bk-dot { width: 9px; height: 9px; border-radius: 3px; flex-shrink: 0; }
.bk-name { flex: 1; font-size: 13.5px; color: var(--ink); }
.bk-meta { font-size: 11px; color: var(--ink-dim); font-family: var(--font-num); }
.bk-pct { font-size: 11px; color: var(--ink-dim); font-family: var(--font-num); width: 30px; text-align: right; }
.bk-amt { font-size: 13px; font-family: var(--font-num); color: var(--ink); min-width: 72px; text-align: right; }
.bk-rows { padding: 2px 14px 10px; border-top: 1px solid var(--line); background: rgba(0,0,0,0.18); }
.bk-row { display: grid; grid-template-columns: 1fr 34px 34px 80px; gap: 10px; align-items: baseline; padding: 6px 0; border-bottom: 1px solid rgba(255,255,255,0.04); }
.bk-row:last-child { border-bottom: 0; }
.bk-row-desc { font-size: 12px; color: var(--ink-mute); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; padding-left: 19px; font-family: var(--font-num); }
.bk-row-cnt { font-size: 11px; color: var(--ink-dim); font-family: var(--font-num); text-align: right; }
.bk-row-pct { font-size: 11px; color: var(--ink-dim); font-family: var(--font-num); text-align: right; }
.bk-row-amt { font-size: 12px; color: var(--ink); font-family: var(--font-num); text-align: right; }
.bk-more { font-size: 11px; color: var(--ink-dim); padding: 7px 0 3px 19px; }
/* ---- merchant logos (Brandfetch) ---- */
.mlogo { border-radius: 6px; object-fit: contain; background: rgba(255,255,255,0.06); vertical-align: middle; flex: none; }
.mlogo-fb { display: inline-flex; align-items: center; justify-content: center; border-radius: 6px; font-size: 11px; font-weight: 700; color: #fff; font-family: var(--font-ui); }
.mrow .m-name { display: flex; align-items: center; gap: 9px; }
.bk-row-desc.with-logo { display: flex; align-items: center; gap: 8px; }
.lx-desc .lx-logo-wrap { display: inline-flex; align-items: center; gap: 8px; }
.mlogo-cat { display: inline-flex; align-items: center; justify-content: center; border-radius: 6px; background: rgba(255,255,255,0.06); vertical-align: middle; flex: none; }
.mlogo-cat img { display: block; }