- 客單價趨勢 (AOV Trend)
+ 平均單價趨勢
diff --git a/web/static/css/page-growth-bem.css b/web/static/css/page-growth-bem.css
index 8f43b71..74fb739 100644
--- a/web/static/css/page-growth-bem.css
+++ b/web/static/css/page-growth-bem.css
@@ -9,72 +9,82 @@
display: flex;
align-items: center;
justify-content: space-between;
- gap: var(--momo-space-3, 16px);
- padding: var(--momo-space-3, 16px) var(--momo-space-4, 20px);
- margin-top: var(--momo-space-3, 16px);
- background: var(--momo-surface-1);
+ gap: var(--momo-space-4, 16px);
+ padding: var(--momo-space-4, 16px) var(--momo-space-5, 24px);
+ margin-top: var(--momo-space-3, 12px);
+ background: var(--momo-bg-surface);
border: 1px solid var(--momo-border-subtle);
- border-radius: var(--momo-radius-md, 8px);
- box-shadow: var(--momo-shadow-soft);
+ border-radius: var(--momo-radius-md, 4px);
+ box-shadow: var(--momo-shadow-md);
}
.growth-analysis-page .ga-page-head__title {
display: flex;
align-items: center;
- gap: var(--momo-space-2, 12px);
+ gap: var(--momo-space-2, 8px);
+ min-width: 0;
}
.growth-analysis-page .ga-page-head__icon {
- font-size: 1.25rem;
- color: var(--momo-warm-olive, #6f7a4a);
+ color: var(--momo-page-accent);
+ font-size: var(--momo-text-title);
}
.growth-analysis-page .ga-page-head__h1 {
- font-size: var(--momo-text-h4, 1.25rem);
- font-weight: 800;
- color: var(--momo-text-strong);
margin: 0;
+ color: var(--momo-page-ink, var(--momo-text-primary));
+ font-family: var(--momo-font-display);
+ font-size: var(--momo-text-headline);
+ font-weight: var(--momo-font-weight-black);
+ line-height: var(--momo-line-height-tight);
+ letter-spacing: 0;
}
.growth-analysis-page .ga-page-head__meta {
- font-size: var(--momo-text-sm, 0.85rem);
- color: var(--momo-text-tertiary);
+ flex: 0 0 auto;
+ color: var(--momo-text-secondary);
+ font-size: var(--momo-text-meta);
+ line-height: var(--momo-line-height-base);
+ text-align: right;
}
.growth-analysis-page .ga-page-head__meta strong {
- color: var(--momo-text-strong);
- font-weight: 700;
+ color: var(--momo-text-primary);
+ font-family: var(--momo-font-mono);
+ font-weight: var(--momo-font-weight-bold);
+ font-feature-settings: "tnum";
}
.growth-analysis-page .ga-empty-state {
display: grid;
gap: var(--momo-space-2, 8px);
- padding: var(--momo-space-8, 40px) var(--momo-space-5, 24px);
- margin-bottom: var(--momo-space-4, 20px);
- color: var(--momo-ink-secondary);
- background: var(--momo-surface-0);
+ padding: var(--momo-space-7, 48px) var(--momo-space-5, 24px);
+ margin-bottom: var(--momo-space-4, 16px);
+ color: var(--momo-text-tertiary);
+ background: var(--momo-bg-surface);
border: 1px solid var(--momo-border-light);
- border-radius: var(--momo-radius-md, 8px);
+ border-radius: var(--momo-radius-md, 4px);
text-align: center;
}
.growth-analysis-page .ga-empty-state i {
- color: var(--momo-warm-honey-deep);
- font-size: var(--momo-text-2xl, 1.5rem);
+ color: var(--momo-page-accent-dark);
+ font-size: var(--momo-text-headline);
}
.growth-analysis-page .ga-empty-state h2 {
margin: 0;
- color: var(--momo-ink-primary);
- font-size: var(--momo-text-lg, 1.125rem);
- font-weight: var(--momo-font-bold, 700);
+ color: var(--momo-text-primary);
+ font-size: var(--momo-text-title);
+ font-weight: var(--momo-font-weight-bold);
+ line-height: var(--momo-line-height-tight);
}
.growth-analysis-page .ga-empty-state p {
margin: 0;
- color: var(--momo-ink-tertiary);
+ color: var(--momo-text-secondary);
}
/* ── KPI Row ────────────────────────────────────────── */
.growth-analysis-page .ga-kpi-row {
display: grid;
grid-template-columns: repeat(3, minmax(0, 1fr));
- gap: var(--momo-space-3, 16px);
+ gap: var(--momo-space-3, 12px);
}
@media (max-width: 992px) {
.growth-analysis-page .ga-kpi-row { grid-template-columns: 1fr; }
@@ -83,45 +93,46 @@
.growth-analysis-page .ga-kpi {
position: relative;
overflow: hidden;
- padding: var(--momo-space-4, 24px);
- border-radius: var(--momo-radius-md, 8px);
- color: var(--momo-text-strong);
- background: var(--momo-surface-raised);
+ min-height: 132px;
+ padding: var(--momo-space-5, 24px);
+ color: var(--momo-text-primary);
+ background: var(--momo-bg-elevated);
border: 1px solid var(--momo-border-strong);
border-left: 4px solid var(--ga-kpi-accent, var(--momo-page-accent));
- box-shadow: var(--momo-shadow-soft);
- min-height: 132px;
+ border-radius: var(--momo-radius-md, 4px);
+ box-shadow: var(--momo-shadow-md);
}
.growth-analysis-page .ga-kpi--revenue {
--ga-kpi-accent: var(--momo-page-accent);
}
.growth-analysis-page .ga-kpi--aov {
- --ga-kpi-accent: var(--momo-tag-honey);
+ --ga-kpi-accent: var(--momo-warning-text);
}
.growth-analysis-page .ga-kpi--orders {
- --ga-kpi-accent: var(--momo-tag-olive);
+ --ga-kpi-accent: var(--momo-success-text);
}
.growth-analysis-page .ga-kpi__label {
position: relative;
z-index: 1;
- font-size: var(--momo-text-sm, 0.85rem);
- font-weight: 800;
- text-transform: uppercase;
- letter-spacing: 0.04em;
- color: var(--momo-text-muted);
margin-bottom: var(--momo-space-2, 8px);
+ color: var(--momo-text-secondary);
+ font-size: var(--momo-text-body-sm);
+ font-weight: var(--momo-font-weight-bold);
+ letter-spacing: 0;
+ line-height: var(--momo-line-height-tight);
}
.growth-analysis-page .ga-kpi__value {
position: relative;
z-index: 1;
- color: var(--momo-text-strong);
- font-family: var(--momo-font-mono, ui-monospace, monospace);
- font-size: 2rem;
- font-weight: 800;
- letter-spacing: 0;
margin-bottom: var(--momo-space-1, 4px);
- line-height: 1.08;
+ color: var(--momo-text-primary);
+ font-family: var(--momo-font-mono);
+ font-feature-settings: "tnum";
+ font-size: var(--momo-text-display);
+ font-weight: var(--momo-font-weight-black);
+ letter-spacing: 0;
+ line-height: var(--momo-line-height-tight);
}
.growth-analysis-page .ga-kpi__delta {
position: relative;
@@ -129,39 +140,43 @@
display: flex;
flex-wrap: wrap;
align-items: center;
- gap: var(--momo-space-1, 6px);
+ gap: var(--momo-space-1, 4px);
margin-top: var(--momo-space-2, 8px);
}
.growth-analysis-page .ga-kpi__chip {
display: inline-flex;
align-items: center;
- padding: 2px 10px;
- background: color-mix(in srgb, var(--ga-kpi-accent) 12%, var(--momo-surface));
- color: var(--ga-kpi-accent);
- border: 1px solid color-mix(in srgb, var(--ga-kpi-accent) 28%, var(--momo-border-subtle));
- font-size: 0.72rem;
- font-weight: 700;
- border-radius: 6px;
+ padding: var(--momo-space-1, 4px) var(--momo-space-2, 8px);
+ color: var(--momo-tag-honey-text);
+ background: var(--momo-tag-honey-bg);
+ border: 1px solid var(--momo-tag-honey-border);
+ border-radius: var(--momo-radius-sm, 2px);
+ font-size: var(--momo-text-label);
+ font-weight: var(--momo-font-weight-bold);
+ line-height: 1;
+ letter-spacing: 0;
}
.growth-analysis-page .ga-kpi__yoy {
- font-weight: 700;
- font-size: 0.95rem;
+ font-family: var(--momo-font-mono);
+ font-feature-settings: "tnum";
+ font-size: var(--momo-text-body-sm);
+ font-weight: var(--momo-font-weight-bold);
}
-.growth-analysis-page .ga-kpi__yoy.is-up { color: var(--momo-tag-rust); }
-.growth-analysis-page .ga-kpi__yoy.is-down { color: var(--momo-tag-olive); }
+.growth-analysis-page .ga-kpi__yoy.is-up { color: var(--momo-danger-text); }
+.growth-analysis-page .ga-kpi__yoy.is-down { color: var(--momo-success-text); }
.growth-analysis-page .ga-kpi__hint {
position: relative;
z-index: 1;
- font-size: var(--momo-text-xs, 0.78rem);
- color: var(--momo-text-muted);
- margin-top: 2px;
+ margin-top: var(--momo-space-1, 4px);
+ color: var(--momo-text-secondary);
+ font-size: var(--momo-text-meta);
}
.growth-analysis-page .ga-kpi__icon-bg {
position: absolute;
- right: -15px;
- bottom: -15px;
- font-size: 6rem;
+ right: calc(var(--momo-space-4, 16px) * -1);
+ bottom: calc(var(--momo-space-4, 16px) * -1);
color: var(--ga-kpi-accent);
+ font-size: 88px;
opacity: 0.1;
transform: rotate(-15deg);
pointer-events: none;
@@ -170,7 +185,7 @@
/* ── Chart Row 變體 ─────────────────────────────────── */
.growth-analysis-page .ga-chart-row {
display: grid;
- gap: var(--momo-space-3, 16px);
+ gap: var(--momo-space-3, 12px);
}
.growth-analysis-page .ga-chart-row--8-4 { grid-template-columns: 2fr 1fr; }
.growth-analysis-page .ga-chart-row--6-6 { grid-template-columns: 1fr 1fr; }
@@ -183,29 +198,32 @@
.growth-analysis-page .ga-chart-card {
display: flex;
flex-direction: column;
- background: var(--momo-surface-1);
+ background: var(--momo-bg-surface);
border: 1px solid var(--momo-border-strong);
- border-radius: var(--momo-radius-md, 8px);
- box-shadow: var(--momo-shadow-soft);
+ border-radius: var(--momo-radius-md, 4px);
+ box-shadow: var(--momo-shadow-md);
overflow: hidden;
}
.growth-analysis-page .ga-chart-card__head {
- padding: 1rem 1.25rem;
- background: var(--momo-surface-2);
+ padding: var(--momo-space-4, 16px) var(--momo-space-5, 24px);
+ background: var(--momo-bg-elevated);
border-bottom: 1px solid var(--momo-border-subtle);
}
.growth-analysis-page .ga-chart-card__title {
display: inline-flex;
align-items: center;
- gap: var(--momo-space-1, 8px);
- font-weight: 800;
- color: var(--momo-text-strong);
+ gap: var(--momo-space-2, 8px);
+ color: var(--momo-text-primary);
+ font-size: var(--momo-text-title);
+ font-weight: var(--momo-font-weight-bold);
+ line-height: var(--momo-line-height-tight);
+ letter-spacing: 0;
}
.growth-analysis-page .ga-chart-card__title i {
- color: var(--momo-warm-olive, #6f7a4a);
+ color: var(--momo-page-accent);
}
.growth-analysis-page .ga-chart-card__body {
- padding: var(--momo-space-3, 16px);
+ padding: var(--momo-space-4, 16px);
height: var(--ga-chart-h, 320px);
}
.growth-analysis-page .ga-chart-card__body canvas {
@@ -217,19 +235,29 @@
.growth-analysis-page .ga-page-head {
align-items: flex-start;
flex-direction: column;
- padding: 14px 16px;
+ padding: var(--momo-space-3, 12px) var(--momo-space-4, 16px);
+ }
+
+ .growth-analysis-page .ga-page-head__h1 {
+ font-size: var(--momo-text-title);
+ }
+
+ .growth-analysis-page .ga-page-head__meta {
+ width: 100%;
+ text-align: left;
}
.growth-analysis-page .ga-kpi {
min-height: 112px;
- padding: 16px;
+ padding: var(--momo-space-4, 16px);
}
.growth-analysis-page .ga-kpi__value {
- font-size: 1.45rem;
+ font-size: var(--momo-text-headline);
}
.growth-analysis-page .ga-chart-card__body {
height: 260px !important;
+ padding: var(--momo-space-3, 12px);
}
}
diff --git a/web/static/css/page-growth.css b/web/static/css/page-growth.css
index 3e66f2b..389d7af 100644
--- a/web/static/css/page-growth.css
+++ b/web/static/css/page-growth.css
@@ -7,23 +7,24 @@
.growth-analysis-page {
display: flex;
flex-direction: column;
- gap: var(--momo-space-3, 18px);
+ gap: var(--momo-space-4, 16px);
}
.growth-analysis-page .card {
border: 1px solid var(--momo-border-strong);
- border-radius: var(--momo-radius-md, 8px);
- box-shadow: var(--momo-shadow-soft);
- margin-bottom: 1.5rem;
- background: var(--momo-surface-1, rgba(255, 253, 248, 0.94));
+ border-radius: var(--momo-radius-md, 4px);
+ box-shadow: var(--momo-shadow-md);
+ margin-bottom: var(--momo-space-5, 24px);
+ background: var(--momo-bg-surface);
}
.growth-analysis-page .card-header {
- background: var(--momo-surface-2, rgba(250, 247, 240, 0.9));
+ background: var(--momo-bg-elevated);
border-bottom: 1px solid var(--momo-border-subtle);
- font-weight: 800;
- color: var(--momo-text-strong);
- padding: 1rem 1.25rem;
+ font-size: var(--momo-text-title);
+ font-weight: var(--momo-font-weight-bold);
+ color: var(--momo-text-primary);
+ padding: var(--momo-space-4, 16px) var(--momo-space-5, 24px);
}
/* ── KPI 卡片 ────────────────────────────────────────── */
@@ -34,25 +35,26 @@
}
.growth-analysis-page .kpi-card .icon-bg {
position: absolute;
- right: -15px;
- bottom: -15px;
- font-size: 6rem;
+ right: calc(var(--momo-space-4, 16px) * -1);
+ bottom: calc(var(--momo-space-4, 16px) * -1);
+ font-size: 88px;
opacity: 0.15;
transform: rotate(-15deg);
pointer-events: none;
}
.growth-analysis-page .kpi-value {
- font-size: 2rem;
- font-weight: 800;
+ font-family: var(--momo-font-mono);
+ font-feature-settings: "tnum";
+ font-size: var(--momo-text-display);
+ font-weight: var(--momo-font-weight-black);
letter-spacing: 0;
- margin-bottom: 0.2rem;
+ margin-bottom: var(--momo-space-1, 4px);
}
.growth-analysis-page .kpi-label {
- font-size: 0.85rem;
- font-weight: 600;
- text-transform: uppercase;
- letter-spacing: 0.5px;
- opacity: 0.9;
+ font-size: var(--momo-text-body-sm);
+ font-weight: var(--momo-font-weight-semibold);
+ letter-spacing: 0;
+ color: var(--momo-text-secondary);
}
/* ── Bootstrap 覆寫:用 page palette 漸層取代原色 ─── */
@@ -69,8 +71,8 @@
/* ── 趨勢色 ─────────────────────────────────────────── */
.growth-analysis-page .text-success,
.growth-analysis-page .trend-up {
- color: var(--momo-warm-honey, #c89043) !important;
+ color: var(--momo-warning-text) !important;
}
.growth-analysis-page .trend-down {
- color: var(--momo-warm-rust, #b5342f) !important;
+ color: var(--momo-danger-text) !important;
}
diff --git a/web/static/js/page-growth.js b/web/static/js/page-growth.js
index e456844..2dc48f1 100644
--- a/web/static/js/page-growth.js
+++ b/web/static/js/page-growth.js
@@ -7,22 +7,22 @@
'use strict';
const data = JSON.parse(document.getElementById('chart-data').textContent);
+ const rootStyle = getComputedStyle(document.documentElement);
+ const token = (name, fallback) => rootStyle.getPropertyValue(name).trim() || fallback;
// 與 design system page-group=analytics 對齊的暖色 palette
const chartPalette = {
- caramel: 'rgba(201, 100, 66, 1)',
- caramelSoft: 'rgba(201, 100, 66, 0.58)',
- honey: 'rgba(184, 132, 22, 1)',
- honeySoft: 'rgba(184, 132, 22, 0.58)',
- rust: 'rgba(181, 52, 47, 1)',
- rustSoft: 'rgba(181, 52, 47, 0.48)',
- mahogany: 'rgba(143, 69, 48, 1)',
- mahoganySoft: 'rgba(143, 69, 48, 0.12)'
+ caramel: token('--momo-page-chart-2', '#c96442'),
+ caramelSoft: token('--momo-warm-caramel-soft', 'rgba(201, 100, 66, 0.58)'),
+ honey: token('--momo-page-accent', '#c89043'),
+ honeySoft: token('--momo-page-accent-soft', 'rgba(200, 144, 67, 0.14)'),
+ rust: token('--momo-danger-text', '#7a3210'),
+ rustSoft: token('--momo-danger-bg', '#efd3c4')
};
- Chart.defaults.color = '#6f665a';
- Chart.defaults.borderColor = 'rgba(126, 111, 92, 0.18)';
- Chart.defaults.font.family = "'Noto Sans TC', 'Inter', system-ui, sans-serif";
+ Chart.defaults.color = token('--momo-text-secondary', '#6b6155');
+ Chart.defaults.borderColor = token('--momo-border-light', 'rgba(42, 37, 32, 0.10)');
+ Chart.defaults.font.family = token('--momo-font-family', "'Inter', system-ui, sans-serif");
// 1) Revenue + YoY
new Chart(document.getElementById('revenueChart'), {
@@ -70,10 +70,10 @@
data: {
labels: data.labels,
datasets: [{
- label: '平均客單價 ($)',
+ label: '平均單價 ($)',
data: data.aov,
borderColor: chartPalette.caramel,
- backgroundColor: 'rgba(201, 100, 66, 0.12)',
+ backgroundColor: chartPalette.caramelSoft,
fill: true, tension: 0.4
}]
},
@@ -92,7 +92,7 @@
label: '毛利率 (%)',
data: data.margin_rate,
borderColor: chartPalette.honey,
- backgroundColor: 'rgba(184, 132, 22, 0.12)',
+ backgroundColor: chartPalette.honeySoft,
fill: true, tension: 0.4
}]
},