Spaces:
Running
Running
| <html lang="pt-BR"> | |
| <head> | |
| <meta charset="UTF-8"> | |
| <meta name="viewport" content="width=device-width, initial-scale=1.0"> | |
| <title>Assistente de Bolso — Dashboard</title> | |
| <link rel="preconnect" href="https://fonts.googleapis.com"> | |
| <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin> | |
| <link href="https://fonts.googleapis.com/css2?family=DM+Serif+Display&family=Inter:wght@400;500;600;700&display=swap" rel="stylesheet"> | |
| <script src="https://cdn.jsdelivr.net/npm/chart.js"></script> | |
| <style> | |
| :root { | |
| --green: #1DE199; | |
| --green-dark: #17B87A; | |
| --green-light: #E8FBF4; | |
| --green-glow: rgba(29, 225, 153, 0.12); | |
| --green-subtle: #F0FDF9; | |
| --white: #FFFFFF; | |
| --black: #0B0B0B; | |
| --gray-50: #FAFAFA; | |
| --gray-100: #F4F4F5; | |
| --gray-200: #E4E4E7; | |
| --gray-300: #D4D4D8; | |
| --gray-400: #A1A1AA; | |
| --gray-500: #71717A; | |
| --gray-600: #52525B; | |
| --gray-700: #3F3F46; | |
| --gray-800: #27272A; | |
| --red: #EF4444; | |
| --red-light: #FEF2F2; | |
| --blue: #3B82F6; | |
| --blue-light: #EFF6FF; | |
| --yellow: #F59E0B; | |
| --yellow-light: #FFFBEB; | |
| --purple: #8B5CF6; | |
| --purple-light: #F5F3FF; | |
| --font-serif: 'DM Serif Display', serif; | |
| --font-sans: 'Inter', -apple-system, BlinkMacSystemFont, sans-serif; | |
| --shadow-xs: 0 1px 2px rgba(0,0,0,0.04); | |
| --shadow-sm: 0 1px 3px rgba(0,0,0,0.06), 0 1px 2px rgba(0,0,0,0.04); | |
| --shadow-md: 0 4px 6px rgba(0,0,0,0.05), 0 2px 4px rgba(0,0,0,0.03); | |
| --shadow-lg: 0 10px 15px rgba(0,0,0,0.08), 0 4px 6px rgba(0,0,0,0.04); | |
| --radius-sm: 6px; | |
| --radius-md: 10px; | |
| --radius-lg: 14px; | |
| --radius-full: 9999px; | |
| --space-1: 4px; | |
| --space-2: 8px; | |
| --space-3: 12px; | |
| --space-4: 16px; | |
| --space-5: 20px; | |
| --space-6: 24px; | |
| --space-8: 32px; | |
| --space-10: 40px; | |
| } | |
| *, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; } | |
| html { scroll-behavior: smooth; } | |
| body { | |
| font-family: var(--font-sans); | |
| background: var(--gray-50); | |
| color: var(--black); | |
| line-height: 1.5; | |
| min-height: 100dvh; | |
| -webkit-font-smoothing: antialiased; | |
| } | |
| /* ===== HEADER ===== */ | |
| .header { | |
| background: var(--white); | |
| border-bottom: 1px solid var(--gray-200); | |
| position: sticky; | |
| top: 0; | |
| z-index: 100; | |
| } | |
| .header-inner { | |
| max-width: 1000px; | |
| margin: 0 auto; | |
| padding: var(--space-3) var(--space-4); | |
| display: flex; | |
| align-items: center; | |
| justify-content: space-between; | |
| } | |
| .logo { | |
| display: flex; | |
| align-items: center; | |
| gap: var(--space-2); | |
| } | |
| .logo-icon { | |
| width: 32px; | |
| height: 32px; | |
| background: linear-gradient(135deg, var(--green), var(--green-dark)); | |
| border-radius: var(--radius-sm); | |
| display: flex; | |
| align-items: center; | |
| justify-content: center; | |
| } | |
| .logo-icon svg { width: 18px; height: 18px; color: var(--white); } | |
| .logo-img { width: 28px; height: 28px; object-fit: contain; } | |
| .logo-text { font-family: var(--font-serif); font-size: 1rem; } | |
| .header-actions { | |
| display: flex; | |
| align-items: center; | |
| gap: var(--space-2); | |
| } | |
| .avatar { | |
| width: 32px; | |
| height: 32px; | |
| border-radius: 50%; | |
| background: var(--gray-200); | |
| display: flex; | |
| align-items: center; | |
| justify-content: center; | |
| font-size: 0.7rem; | |
| font-weight: 600; | |
| color: var(--gray-600); | |
| } | |
| /* ===== TABS NAV ===== */ | |
| .tabs-nav { | |
| background: var(--white); | |
| border-bottom: 1px solid var(--gray-200); | |
| position: sticky; | |
| top: 56px; | |
| z-index: 90; | |
| } | |
| .tabs-nav-inner { | |
| max-width: 1000px; | |
| margin: 0 auto; | |
| display: flex; | |
| padding: 0 var(--space-4); | |
| } | |
| .tab-btn { | |
| font-family: var(--font-sans); | |
| font-size: 0.8125rem; | |
| font-weight: 500; | |
| padding: var(--space-3) var(--space-4); | |
| background: none; | |
| border: none; | |
| color: var(--gray-500); | |
| cursor: pointer; | |
| position: relative; | |
| transition: color 0.15s; | |
| } | |
| .tab-btn:hover { color: var(--black); } | |
| .tab-btn.active { color: var(--black); } | |
| .tab-btn.active::after { | |
| content: ''; | |
| position: absolute; | |
| bottom: 0; | |
| left: var(--space-4); | |
| right: var(--space-4); | |
| height: 2px; | |
| background: var(--green); | |
| border-radius: 2px 2px 0 0; | |
| } | |
| .tab-btn:focus-visible { outline: 2px solid var(--green); outline-offset: -2px; } | |
| /* ===== FILTERS (STRIPE STYLE) ===== */ | |
| .filters-bar { | |
| background: var(--white); | |
| border-bottom: 1px solid var(--gray-200); | |
| } | |
| .filters-inner { | |
| max-width: 1000px; | |
| margin: 0 auto; | |
| padding: var(--space-3) var(--space-4); | |
| display: flex; | |
| flex-wrap: nowrap; | |
| gap: var(--space-2); | |
| align-items: center; | |
| overflow-x: auto; | |
| -webkit-overflow-scrolling: touch; | |
| scrollbar-width: none; | |
| -ms-overflow-style: none; | |
| } | |
| .filters-inner::-webkit-scrollbar { | |
| display: none; | |
| } | |
| .filter-chip { | |
| display: inline-flex; | |
| align-items: center; | |
| gap: var(--space-1); | |
| font-family: var(--font-sans); | |
| font-size: 0.75rem; | |
| font-weight: 500; | |
| padding: 6px 10px; | |
| background: var(--white); | |
| border: 1px solid var(--gray-300); | |
| border-radius: var(--radius-sm); | |
| color: var(--gray-700); | |
| cursor: pointer; | |
| transition: all 0.15s; | |
| white-space: nowrap; | |
| } | |
| .filter-chip:hover { | |
| background: var(--gray-50); | |
| border-color: var(--gray-400); | |
| } | |
| .filter-chip:focus { | |
| outline: none; | |
| border-color: var(--green); | |
| box-shadow: 0 0 0 3px var(--green-glow); | |
| } | |
| .filter-chip svg { | |
| width: 12px; | |
| height: 12px; | |
| color: var(--gray-400); | |
| } | |
| .filter-chip.active { | |
| background: var(--green-light); | |
| border-color: var(--green); | |
| color: var(--green-dark); | |
| } | |
| .filter-chip.active svg { color: var(--green-dark); } | |
| .filter-dropdown { | |
| position: relative; | |
| } | |
| .filter-dropdown-content { | |
| display: none; | |
| position: absolute; | |
| top: calc(100% + 4px); | |
| left: 0; | |
| min-width: 180px; | |
| background: var(--white); | |
| border: 1px solid var(--gray-200); | |
| border-radius: var(--radius-md); | |
| box-shadow: var(--shadow-lg); | |
| z-index: 200; | |
| padding: var(--space-1); | |
| } | |
| .filter-dropdown.open .filter-dropdown-content { display: block; } | |
| .filter-option { | |
| display: block; | |
| width: 100%; | |
| padding: var(--space-2) var(--space-3); | |
| font-family: var(--font-sans); | |
| font-size: 0.8125rem; | |
| text-align: left; | |
| background: none; | |
| border: none; | |
| border-radius: var(--radius-sm); | |
| color: var(--gray-700); | |
| cursor: pointer; | |
| transition: background 0.1s; | |
| } | |
| .filter-option:hover { background: var(--gray-100); } | |
| .filter-option.selected { background: var(--green-light); color: var(--green-dark); font-weight: 500; } | |
| /* ===== MAIN ===== */ | |
| .main { | |
| max-width: 1000px; | |
| margin: 0 auto; | |
| padding: var(--space-4); | |
| } | |
| .tab-content { display: none; } | |
| .tab-content.active { display: block; animation: fadeIn 0.2s ease; } | |
| @keyframes fadeIn { | |
| from { opacity: 0; transform: translateY(6px); } | |
| to { opacity: 1; transform: translateY(0); } | |
| } | |
| /* ===== KPI GRID ===== */ | |
| .kpi-grid { | |
| display: grid; | |
| grid-template-columns: repeat(2, 1fr); | |
| gap: var(--space-3); | |
| margin-bottom: var(--space-4); | |
| } | |
| @media (min-width: 640px) { | |
| .kpi-grid { | |
| grid-template-columns: repeat(4, 1fr); | |
| } | |
| } | |
| .kpi-card { | |
| background: var(--white); | |
| border-radius: var(--radius-md); | |
| padding: var(--space-4); | |
| border: 1px solid var(--gray-200); | |
| position: relative; | |
| } | |
| .kpi-header { | |
| display: flex; | |
| align-items: flex-start; | |
| justify-content: space-between; | |
| margin-bottom: var(--space-2); | |
| } | |
| .kpi-label { | |
| font-size: 0.6875rem; | |
| font-weight: 600; | |
| color: var(--gray-500); | |
| text-transform: uppercase; | |
| letter-spacing: 0.03em; | |
| line-height: 1.3; | |
| } | |
| .info-btn { | |
| width: 16px; | |
| height: 16px; | |
| border-radius: 50%; | |
| background: var(--gray-100); | |
| border: none; | |
| color: var(--gray-400); | |
| font-size: 0.6rem; | |
| font-weight: 700; | |
| cursor: pointer; | |
| display: flex; | |
| align-items: center; | |
| justify-content: center; | |
| flex-shrink: 0; | |
| transition: all 0.15s; | |
| } | |
| .info-btn:hover { background: var(--gray-200); color: var(--gray-600); } | |
| .info-btn:focus-visible { outline: 2px solid var(--green); outline-offset: 2px; } | |
| .kpi-value { | |
| font-family: var(--font-serif); | |
| font-size: 1.375rem; | |
| color: var(--black); | |
| line-height: 1.1; | |
| } | |
| .kpi-value.positive { color: var(--green-dark); } | |
| .kpi-value.negative { color: var(--red); } | |
| .kpi-subtitle { | |
| font-size: 0.6875rem; | |
| color: var(--gray-400); | |
| margin-top: 2px; | |
| } | |
| .connect-link { | |
| display: inline-flex; | |
| align-items: center; | |
| gap: 4px; | |
| margin-top: var(--space-2); | |
| font-size: 0.6875rem; | |
| font-weight: 600; | |
| color: var(--green); | |
| text-decoration: none; | |
| } | |
| .connect-link:hover { text-decoration: underline; } | |
| /* ===== PROJECTION CARD (REDESIGNED - LIGHT VERSION) ===== */ | |
| .projection-card { | |
| background: var(--white); | |
| border-radius: var(--radius-lg); | |
| border: 1px solid var(--gray-200); | |
| margin-bottom: var(--space-4); | |
| overflow: hidden; | |
| } | |
| .projection-header { | |
| display: flex; | |
| align-items: center; | |
| justify-content: space-between; | |
| padding: var(--space-4); | |
| border-bottom: 1px solid var(--gray-100); | |
| } | |
| .projection-title { | |
| display: flex; | |
| align-items: center; | |
| gap: var(--space-3); | |
| } | |
| .projection-icon { | |
| width: 36px; | |
| height: 36px; | |
| border-radius: var(--radius-md); | |
| background: linear-gradient(135deg, var(--green-light), var(--green-subtle)); | |
| display: flex; | |
| align-items: center; | |
| justify-content: center; | |
| color: var(--green-dark); | |
| } | |
| .projection-icon svg { | |
| width: 18px; | |
| height: 18px; | |
| } | |
| .projection-title-text h2 { | |
| font-family: var(--font-serif); | |
| font-size: 1rem; | |
| font-weight: 400; | |
| color: var(--black); | |
| margin-bottom: 0; | |
| line-height: 1.2; | |
| } | |
| .projection-title-text span { | |
| font-size: 0.6875rem; | |
| color: var(--gray-500); | |
| line-height: 1.2; | |
| } | |
| .badge { | |
| font-size: 0.5625rem; | |
| font-weight: 700; | |
| text-transform: uppercase; | |
| letter-spacing: 0.05em; | |
| padding: 4px 10px; | |
| border-radius: var(--radius-full); | |
| background: var(--green-light); | |
| color: var(--green-dark); | |
| border: 1px solid var(--green); | |
| } | |
| .projection-body { | |
| padding: var(--space-4); | |
| } | |
| .projection-metrics { | |
| display: grid; | |
| grid-template-columns: repeat(2, 1fr); | |
| gap: var(--space-3); | |
| margin-bottom: var(--space-4); | |
| } | |
| @media (min-width: 500px) { | |
| .projection-metrics { grid-template-columns: repeat(4, 1fr); } | |
| } | |
| .projection-metric { | |
| background: var(--gray-50); | |
| border-radius: var(--radius-md); | |
| padding: var(--space-3); | |
| text-align: center; | |
| } | |
| .projection-metric-label { | |
| font-size: 0.625rem; | |
| font-weight: 600; | |
| color: var(--gray-500); | |
| text-transform: uppercase; | |
| letter-spacing: 0.03em; | |
| margin-bottom: var(--space-1); | |
| } | |
| .projection-metric-value { | |
| font-family: var(--font-serif); | |
| font-size: 1.125rem; | |
| line-height: 1.2; | |
| color: var(--black); | |
| } | |
| .projection-metric-value.income { color: var(--green-dark); } | |
| .projection-metric-value.expense { color: var(--red); } | |
| .projection-metric-value.result { color: var(--green-dark); } | |
| .projection-metric-value.result.negative { color: var(--red); } | |
| .projection-divider { | |
| display: flex; | |
| align-items: center; | |
| gap: var(--space-3); | |
| margin-bottom: var(--space-3); | |
| } | |
| .projection-divider::before, | |
| .projection-divider::after { | |
| content: ''; | |
| flex: 1; | |
| height: 1px; | |
| background: var(--gray-200); | |
| } | |
| .projection-divider-text { | |
| font-size: 0.625rem; | |
| font-weight: 600; | |
| color: var(--gray-400); | |
| text-transform: uppercase; | |
| letter-spacing: 0.05em; | |
| } | |
| .projection-list { | |
| list-style: none; | |
| display: grid; | |
| gap: var(--space-2); | |
| } | |
| .projection-item { | |
| display: flex; | |
| align-items: center; | |
| justify-content: space-between; | |
| padding: var(--space-3); | |
| background: var(--gray-50); | |
| border-radius: var(--radius-md); | |
| transition: background 0.15s; | |
| } | |
| .projection-item:hover { | |
| background: var(--gray-100); | |
| } | |
| .projection-item-info { | |
| display: flex; | |
| align-items: center; | |
| gap: var(--space-3); | |
| } | |
| .projection-item-icon { | |
| width: 32px; | |
| height: 32px; | |
| border-radius: var(--radius-sm); | |
| display: flex; | |
| align-items: center; | |
| justify-content: center; | |
| font-size: 0.875rem; | |
| } | |
| .projection-item-icon.expense { | |
| background: var(--red-light); | |
| } | |
| .projection-item-icon.income { | |
| background: var(--green-light); | |
| } | |
| .projection-item-text { display: flex; flex-direction: column; } | |
| .projection-item-title { | |
| font-size: 0.8125rem; | |
| font-weight: 500; | |
| color: var(--black); | |
| } | |
| .projection-item-date { | |
| font-size: 0.6875rem; | |
| color: var(--gray-500); | |
| } | |
| .projection-item-value { | |
| font-size: 0.875rem; | |
| font-weight: 600; | |
| text-align: right; | |
| } | |
| .projection-item-value.income { color: var(--green-dark); } | |
| .projection-item-value.expense { color: var(--red); } | |
| .projection-footer { | |
| padding: var(--space-3) var(--space-4); | |
| background: var(--gray-50); | |
| border-top: 1px solid var(--gray-100); | |
| } | |
| .projection-note { | |
| font-size: 0.6875rem; | |
| color: var(--gray-500); | |
| line-height: 1.5; | |
| display: flex; | |
| align-items: flex-start; | |
| gap: var(--space-2); | |
| } | |
| .projection-note svg { | |
| width: 14px; | |
| height: 14px; | |
| color: var(--gray-400); | |
| flex-shrink: 0; | |
| margin-top: 1px; | |
| } | |
| /* ===== CARD ===== */ | |
| .card { | |
| background: var(--white); | |
| border-radius: var(--radius-md); | |
| border: 1px solid var(--gray-200); | |
| margin-bottom: var(--space-4); | |
| overflow: hidden; | |
| } | |
| .card-header { | |
| padding: var(--space-4); | |
| border-bottom: 1px solid var(--gray-100); | |
| } | |
| .card-header h3 { | |
| font-family: var(--font-serif); | |
| font-size: 0.9375rem; | |
| font-weight: 400; | |
| } | |
| .card-body { padding: var(--space-4); } | |
| /* ===== CHARTS ===== */ | |
| .chart-container { | |
| position: relative; | |
| height: 180px; | |
| } | |
| .chart-row { | |
| display: grid; | |
| grid-template-columns: 1fr; | |
| gap: var(--space-4); | |
| } | |
| @media (min-width: 640px) { | |
| .chart-row { grid-template-columns: repeat(2, 1fr); } | |
| } | |
| .mini-chart-container { | |
| position: relative; | |
| height: 140px; | |
| } | |
| /* ===== CATEGORY GRID ===== */ | |
| .category-grid { | |
| display: grid; | |
| grid-template-columns: 1fr; | |
| gap: var(--space-4); | |
| } | |
| @media (min-width: 500px) { | |
| .category-grid { grid-template-columns: 140px 1fr; } | |
| } | |
| .category-chart-wrapper { | |
| display: flex; | |
| justify-content: center; | |
| align-items: center; | |
| } | |
| .category-chart-box { | |
| width: 120px; | |
| height: 120px; | |
| } | |
| .category-table { | |
| width: 100%; | |
| border-collapse: collapse; | |
| font-size: 0.75rem; | |
| } | |
| .category-table th { | |
| text-align: left; | |
| font-size: 0.625rem; | |
| font-weight: 600; | |
| color: var(--gray-500); | |
| text-transform: uppercase; | |
| letter-spacing: 0.03em; | |
| padding: var(--space-2) 0; | |
| border-bottom: 1px solid var(--gray-200); | |
| } | |
| .category-table th:last-child { text-align: right; } | |
| .category-table td { | |
| padding: var(--space-2) 0; | |
| border-bottom: 1px solid var(--gray-100); | |
| } | |
| .category-table td:last-child { text-align: right; font-weight: 600; } | |
| .cat-dot { | |
| display: inline-block; | |
| width: 8px; | |
| height: 8px; | |
| border-radius: 50%; | |
| margin-right: var(--space-2); | |
| } | |
| .cat-percent { color: var(--gray-400); font-size: 0.6875rem; } | |
| /* ===== BREAKDOWN MINI CARDS ===== */ | |
| .breakdown-row { | |
| display: grid; | |
| grid-template-columns: 1fr; | |
| gap: var(--space-3); | |
| margin-bottom: var(--space-4); | |
| } | |
| @media (min-width: 640px) { | |
| .breakdown-row { grid-template-columns: repeat(3, 1fr); } | |
| } | |
| .breakdown-card { | |
| background: var(--white); | |
| border-radius: var(--radius-md); | |
| border: 1px solid var(--gray-200); | |
| padding: var(--space-3); | |
| } | |
| .breakdown-card h4 { | |
| font-size: 0.6875rem; | |
| font-weight: 600; | |
| color: var(--gray-500); | |
| text-transform: uppercase; | |
| letter-spacing: 0.02em; | |
| margin-bottom: var(--space-2); | |
| } | |
| .breakdown-list { list-style: none; } | |
| .breakdown-item { | |
| display: flex; | |
| justify-content: space-between; | |
| align-items: center; | |
| padding: var(--space-1) 0; | |
| font-size: 0.75rem; | |
| } | |
| .breakdown-item-label { | |
| display: flex; | |
| align-items: center; | |
| gap: var(--space-1); | |
| color: var(--gray-700); | |
| } | |
| .breakdown-item-value { font-weight: 600; color: var(--black); } | |
| /* ===== TRANSACTION LIST ===== */ | |
| .search-bar { | |
| display: flex; | |
| flex-wrap: wrap; | |
| gap: var(--space-2); | |
| margin-bottom: var(--space-3); | |
| } | |
| .search-input { | |
| flex: 1; | |
| min-width: 160px; | |
| font-family: var(--font-sans); | |
| font-size: 0.75rem; | |
| padding: var(--space-2) var(--space-3); | |
| padding-left: 32px; | |
| border: 1px solid var(--gray-300); | |
| border-radius: var(--radius-sm); | |
| background: var(--white) url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='14' height='14' viewBox='0 0 24 24' fill='none' stroke='%23A1A1AA' stroke-width='2'%3E%3Ccircle cx='11' cy='11' r='8'/%3E%3Cpath d='m21 21-4.3-4.3'/%3E%3C/svg%3E") no-repeat 10px center; | |
| transition: border-color 0.15s, box-shadow 0.15s; | |
| } | |
| .search-input:focus { | |
| outline: none; | |
| border-color: var(--green); | |
| box-shadow: 0 0 0 3px var(--green-glow); | |
| } | |
| .chip-group { display: flex; gap: var(--space-1); } | |
| .chip { | |
| font-family: var(--font-sans); | |
| font-size: 0.6875rem; | |
| font-weight: 500; | |
| padding: var(--space-1) var(--space-2); | |
| border: 1px solid var(--gray-300); | |
| border-radius: var(--radius-full); | |
| background: var(--white); | |
| color: var(--gray-600); | |
| cursor: pointer; | |
| transition: all 0.15s; | |
| } | |
| .chip:hover { border-color: var(--green); color: var(--green-dark); } | |
| .chip.active { background: var(--green-light); border-color: var(--green); color: var(--green-dark); } | |
| .tx-list { list-style: none; } | |
| .tx-item { | |
| display: flex; | |
| align-items: center; | |
| justify-content: space-between; | |
| padding: var(--space-3) 0; | |
| border-bottom: 1px solid var(--gray-100); | |
| gap: var(--space-3); | |
| } | |
| .tx-item:last-child { border-bottom: none; } | |
| .tx-info { flex: 1; min-width: 0; } | |
| .tx-main { | |
| display: flex; | |
| align-items: center; | |
| gap: var(--space-2); | |
| margin-bottom: 2px; | |
| } | |
| .tx-title { | |
| font-size: 0.8125rem; | |
| font-weight: 500; | |
| color: var(--black); | |
| white-space: nowrap; | |
| overflow: hidden; | |
| text-overflow: ellipsis; | |
| } | |
| .tx-badge { | |
| font-size: 0.5625rem; | |
| font-weight: 700; | |
| text-transform: uppercase; | |
| padding: 2px 5px; | |
| border-radius: 3px; | |
| flex-shrink: 0; | |
| } | |
| .tx-badge.rec { background: var(--yellow-light); color: var(--yellow); } | |
| .tx-badge.parc { background: var(--blue-light); color: var(--blue); } | |
| .tx-badge.pf { background: var(--gray-100); color: var(--gray-600); } | |
| .tx-badge.pj { background: var(--purple-light); color: var(--purple); } | |
| .tx-meta { | |
| font-size: 0.6875rem; | |
| color: var(--gray-500); | |
| display: flex; | |
| gap: var(--space-2); | |
| flex-wrap: wrap; | |
| } | |
| .tx-value { | |
| font-size: 0.8125rem; | |
| font-weight: 600; | |
| text-align: right; | |
| flex-shrink: 0; | |
| } | |
| .tx-value.expense { color: var(--red); } | |
| .tx-value.income { color: var(--green-dark); } | |
| .view-all-btn { | |
| display: flex; | |
| align-items: center; | |
| justify-content: center; | |
| width: 100%; | |
| padding: var(--space-3); | |
| font-family: var(--font-sans); | |
| font-size: 0.75rem; | |
| font-weight: 600; | |
| color: var(--gray-600); | |
| background: var(--gray-50); | |
| border: none; | |
| border-top: 1px solid var(--gray-200); | |
| cursor: pointer; | |
| transition: background 0.15s, color 0.15s; | |
| } | |
| .view-all-btn:hover { background: var(--gray-100); color: var(--black); } | |
| /* ===== MODAL ===== */ | |
| .modal-overlay { | |
| display: none; | |
| position: fixed; | |
| inset: 0; | |
| background: rgba(0,0,0,0.4); | |
| z-index: 1000; | |
| align-items: center; | |
| justify-content: center; | |
| padding: var(--space-4); | |
| } | |
| .modal-overlay.visible { display: flex; animation: fadeIn 0.15s ease; } | |
| .modal { | |
| background: var(--white); | |
| border-radius: var(--radius-lg); | |
| max-width: 500px; | |
| width: 100%; | |
| max-height: 85vh; | |
| overflow-y: auto; | |
| box-shadow: var(--shadow-lg); | |
| } | |
| .modal-header { | |
| display: flex; | |
| align-items: center; | |
| justify-content: space-between; | |
| padding: var(--space-4); | |
| border-bottom: 1px solid var(--gray-200); | |
| } | |
| .modal-header h4 { | |
| font-family: var(--font-serif); | |
| font-size: 1rem; | |
| font-weight: 400; | |
| } | |
| .modal-close { | |
| width: 28px; | |
| height: 28px; | |
| border-radius: var(--radius-sm); | |
| background: none; | |
| border: none; | |
| color: var(--gray-400); | |
| cursor: pointer; | |
| display: flex; | |
| align-items: center; | |
| justify-content: center; | |
| transition: background 0.15s; | |
| } | |
| .modal-close:hover { background: var(--gray-100); color: var(--black); } | |
| .modal-body { padding: var(--space-4); } | |
| .modal-body p { font-size: 0.875rem; color: var(--gray-600); line-height: 1.6; } | |
| .full-table { | |
| width: 100%; | |
| border-collapse: collapse; | |
| font-size: 0.75rem; | |
| } | |
| .full-table th { | |
| text-align: left; | |
| font-size: 0.625rem; | |
| font-weight: 600; | |
| color: var(--gray-500); | |
| text-transform: uppercase; | |
| padding: var(--space-2); | |
| background: var(--gray-50); | |
| border-bottom: 1px solid var(--gray-200); | |
| position: sticky; | |
| top: 0; | |
| } | |
| .full-table td { | |
| padding: var(--space-2); | |
| border-bottom: 1px solid var(--gray-100); | |
| vertical-align: middle; | |
| } | |
| .full-table .val { font-weight: 600; white-space: nowrap; } | |
| .full-table .val.expense { color: var(--red); } | |
| .full-table .val.income { color: var(--green-dark); } | |
| .table-scroll { overflow-x: auto; -webkit-overflow-scrolling: touch; } | |
| </style> | |
| </head> | |
| <body> | |
| <!-- HEADER --> | |
| <header class="header"> | |
| <div class="header-inner"> | |
| <div class="logo"> | |
| <img src="" alt="Assistente de Bolso" class="logo-img"> | |
| <span class="logo-text">Assistente de Bolso</span> | |
| </div> | |
| <div class="header-actions"> | |
| <div class="avatar">DS</div> | |
| </div> | |
| </div> | |
| </header> | |
| <!-- TABS NAV --> | |
| <nav class="tabs-nav" role="tablist"> | |
| <div class="tabs-nav-inner"> | |
| <button class="tab-btn active" role="tab" data-tab="resumo">Resumo</button> | |
| <button class="tab-btn" role="tab" data-tab="despesas">Despesas</button> | |
| <button class="tab-btn" role="tab" data-tab="receitas">Receitas</button> | |
| </div> | |
| </nav> | |
| <!-- FILTERS BAR (STRIPE STYLE) --> | |
| <div class="filters-bar"> | |
| <div class="filters-inner"> | |
| <div class="filter-dropdown" id="filterPeriod"> | |
| <button class="filter-chip" aria-haspopup="true"> | |
| <span>Dezembro 2024</span> | |
| <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="m6 9 6 6 6-6"/></svg> | |
| </button> | |
| <div class="filter-dropdown-content"> | |
| <button class="filter-option selected">Mês atual</button> | |
| <button class="filter-option">Mês anterior</button> | |
| <button class="filter-option">Últimos 3 meses</button> | |
| <button class="filter-option">Últimos 6 meses</button> | |
| <button class="filter-option">Últimos 12 meses</button> | |
| <button class="filter-option">Personalizado</button> | |
| </div> | |
| </div> | |
| <div class="filter-dropdown" id="filterBank"> | |
| <button class="filter-chip" aria-haspopup="true"> | |
| <span>Todos os bancos</span> | |
| <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="m6 9 6 6 6-6"/></svg> | |
| </button> | |
| <div class="filter-dropdown-content"> | |
| <button class="filter-option selected">Todos os bancos</button> | |
| <button class="filter-option">Inter</button> | |
| <button class="filter-option">Nubank</button> | |
| <button class="filter-option">Bradesco</button> | |
| </div> | |
| </div> | |
| <div class="filter-dropdown" id="filterMethod"> | |
| <button class="filter-chip" aria-haspopup="true"> | |
| <span>Todos métodos</span> | |
| <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="m6 9 6 6 6-6"/></svg> | |
| </button> | |
| <div class="filter-dropdown-content"> | |
| <button class="filter-option selected">Todos métodos</button> | |
| <button class="filter-option">Cartão de crédito</button> | |
| <button class="filter-option">PIX</button> | |
| <button class="filter-option">Boleto</button> | |
| <button class="filter-option">Cartão de débito</button> | |
| <button class="filter-option">Não informado</button> | |
| </div> | |
| </div> | |
| <div class="filter-dropdown" id="filterAccount"> | |
| <button class="filter-chip" aria-haspopup="true"> | |
| <span>PF + PJ</span> | |
| <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="m6 9 6 6 6-6"/></svg> | |
| </button> | |
| <div class="filter-dropdown-content"> | |
| <button class="filter-option selected">Ambos</button> | |
| <button class="filter-option">Somente PF</button> | |
| <button class="filter-option">Somente PJ</button> | |
| </div> | |
| </div> | |
| </div> | |
| </div> | |
| <!-- MAIN --> | |
| <main class="main"> | |
| <!-- ===== ABA RESUMO ===== --> | |
| <div class="tab-content active" id="tab-resumo"> | |
| <!-- KPI Grid --> | |
| <div class="kpi-grid"> | |
| <article class="kpi-card"> | |
| <div class="kpi-header"> | |
| <span class="kpi-label">Despesas do mês</span> | |
| <button class="info-btn" data-info="despesas">i</button> | |
| </div> | |
| <div class="kpi-value negative">R$ 7.284,50</div> | |
| <span class="kpi-subtitle">até agora</span> | |
| </article> | |
| <article class="kpi-card"> | |
| <div class="kpi-header"> | |
| <span class="kpi-label">Receitas do mês</span> | |
| <button class="info-btn" data-info="receitas">i</button> | |
| </div> | |
| <div class="kpi-value positive">R$ 11.200,00</div> | |
| <span class="kpi-subtitle">até agora</span> | |
| </article> | |
| <article class="kpi-card"> | |
| <div class="kpi-header"> | |
| <span class="kpi-label">Resultado do mês</span> | |
| <button class="info-btn" data-info="resultado">i</button> | |
| </div> | |
| <div class="kpi-value positive">R$ 3.915,50</div> | |
| <span class="kpi-subtitle">receitas − despesas</span> | |
| </article> | |
| <article class="kpi-card"> | |
| <div class="kpi-header"> | |
| <span class="kpi-label">Saldo atual</span> | |
| <button class="info-btn" data-info="saldo">i</button> | |
| </div> | |
| <div class="kpi-value">R$ 18.742,33</div> | |
| <span class="kpi-subtitle">dinheiro na conta</span> | |
| </article> | |
| </div> | |
| <!-- Projection Card (Redesigned) --> | |
| <section class="projection-card"> | |
| <div class="projection-header"> | |
| <div class="projection-title"> | |
| <div class="projection-icon"> | |
| <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> | |
| <path d="M12 8v4l3 3"/> | |
| <circle cx="12" cy="12" r="10"/> | |
| </svg> | |
| </div> | |
| <div class="projection-title-text"> | |
| <h2>Projeção até o fim do mês</h2> | |
| <span>Baseado em compromissos conhecidos</span> | |
| </div> | |
| </div> | |
| <div style="display: flex; align-items: center; gap: 8px;"> | |
| <span class="badge">Previsão</span> | |
| <button class="info-btn" data-info="projecao">i</button> | |
| </div> | |
| </div> | |
| <div class="projection-body"> | |
| <div class="projection-metrics"> | |
| <div class="projection-metric"> | |
| <div class="projection-metric-label">Entradas</div> | |
| <div class="projection-metric-value income">+R$ 0</div> | |
| </div> | |
| <div class="projection-metric"> | |
| <div class="projection-metric-label">Saídas</div> | |
| <div class="projection-metric-value expense">−R$ 4.120</div> | |
| </div> | |
| <div class="projection-metric"> | |
| <div class="projection-metric-label">Resultado</div> | |
| <div class="projection-metric-value result negative">−R$ 204</div> | |
| </div> | |
| <div class="projection-metric"> | |
| <div class="projection-metric-label">Saldo previsto</div> | |
| <div class="projection-metric-value">R$ 14.622</div> | |
| </div> | |
| </div> | |
| <div class="projection-divider"> | |
| <span class="projection-divider-text">Próximos compromissos</span> | |
| </div> | |
| <ul class="projection-list"> | |
| <li class="projection-item"> | |
| <div class="projection-item-info"> | |
| <div class="projection-item-icon expense">📚</div> | |
| <div class="projection-item-text"> | |
| <span class="projection-item-title">Mensalidade escola</span> | |
| <span class="projection-item-date">Dia 20 • Recorrente</span> | |
| </div> | |
| </div> | |
| <span class="projection-item-value expense">−R$ 1.850</span> | |
| </li> | |
| <li class="projection-item"> | |
| <div class="projection-item-info"> | |
| <div class="projection-item-icon expense">💳</div> | |
| <div class="projection-item-text"> | |
| <span class="projection-item-title">Parcela cartão (Inter)</span> | |
| <span class="projection-item-date">Dia 22 • Fatura</span> | |
| </div> | |
| </div> | |
| <span class="projection-item-value expense">−R$ 1.340</span> | |
| </li> | |
| <li class="projection-item"> | |
| <div class="projection-item-info"> | |
| <div class="projection-item-icon expense">🏥</div> | |
| <div class="projection-item-text"> | |
| <span class="projection-item-title">Plano de saúde</span> | |
| <span class="projection-item-date">Dia 25 • Recorrente</span> | |
| </div> | |
| </div> | |
| <span class="projection-item-value expense">−R$ 680</span> | |
| </li> | |
| <li class="projection-item"> | |
| <div class="projection-item-info"> | |
| <div class="projection-item-icon expense">📶</div> | |
| <div class="projection-item-text"> | |
| <span class="projection-item-title">Internet + Celular</span> | |
| <span class="projection-item-date">Dia 28 • Recorrente</span> | |
| </div> | |
| </div> | |
| <span class="projection-item-value expense">−R$ 250</span> | |
| </li> | |
| </ul> | |
| </div> | |
| <div class="projection-footer"> | |
| <p class="projection-note"> | |
| <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> | |
| <circle cx="12" cy="12" r="10"/> | |
| <path d="M12 16v-4"/> | |
| <path d="M12 8h.01"/> | |
| </svg> | |
| <span>Projeção baseada em despesas recorrentes, parcelas de cartão e compromissos já conhecidos que ainda não foram pagos.</span> | |
| </p> | |
| </div> | |
| </section> | |
| <!-- Charts Row --> | |
| <div class="chart-row"> | |
| <div class="card"> | |
| <div class="card-header"><h3>Despesas vs Receitas</h3></div> | |
| <div class="card-body"> | |
| <div class="mini-chart-container"><canvas id="summaryBarChart"></canvas></div> | |
| </div> | |
| </div> | |
| <div class="card"> | |
| <div class="card-header"><h3>Movimentações no mês</h3></div> | |
| <div class="card-body"> | |
| <div class="mini-chart-container"><canvas id="summaryLineChart"></canvas></div> | |
| </div> | |
| </div> | |
| </div> | |
| </div> | |
| <!-- ===== ABA DESPESAS ===== --> | |
| <div class="tab-content" id="tab-despesas"> | |
| <!-- Category Breakdown --> | |
| <div class="card"> | |
| <div class="card-header"><h3>Despesas por categoria</h3></div> | |
| <div class="card-body"> | |
| <div class="category-grid"> | |
| <div class="category-chart-wrapper"> | |
| <div class="category-chart-box"><canvas id="expCatChart"></canvas></div> | |
| </div> | |
| <div class="table-scroll"> | |
| <table class="category-table"> | |
| <thead><tr><th>Categoria</th><th>Total</th><th>%</th></tr></thead> | |
| <tbody id="expCatTable"></tbody> | |
| </table> | |
| </div> | |
| </div> | |
| </div> | |
| </div> | |
| <!-- Mini Breakdowns --> | |
| <div class="breakdown-row"> | |
| <div class="breakdown-card"> | |
| <h4>Por método de pagamento</h4> | |
| <ul class="breakdown-list" id="expMethodList"></ul> | |
| </div> | |
| <div class="breakdown-card"> | |
| <h4>Por banco</h4> | |
| <ul class="breakdown-list" id="expBankList"></ul> | |
| </div> | |
| <div class="breakdown-card"> | |
| <h4>Por tipo de conta</h4> | |
| <ul class="breakdown-list" id="expAccountList"></ul> | |
| </div> | |
| </div> | |
| <!-- Evolution Chart --> | |
| <div class="card"> | |
| <div class="card-header"><h3>Evolução das despesas</h3></div> | |
| <div class="card-body"> | |
| <div class="chart-container"><canvas id="expEvolutionChart"></canvas></div> | |
| </div> | |
| </div> | |
| <!-- Transaction List --> | |
| <div class="card"> | |
| <div class="card-header"><h3>Últimas despesas</h3></div> | |
| <div class="card-body"> | |
| <div class="search-bar"> | |
| <input type="text" class="search-input" placeholder="Buscar..."> | |
| <div class="chip-group"> | |
| <button class="chip">Recorrentes</button> | |
| <button class="chip">Parceladas</button> | |
| </div> | |
| </div> | |
| <ul class="tx-list" id="expList"></ul> | |
| </div> | |
| <button class="view-all-btn" data-modal="expModal">Ver todas as despesas</button> | |
| </div> | |
| </div> | |
| <!-- ===== ABA RECEITAS ===== --> | |
| <div class="tab-content" id="tab-receitas"> | |
| <!-- Category Breakdown --> | |
| <div class="card"> | |
| <div class="card-header"><h3>Receitas por categoria</h3></div> | |
| <div class="card-body"> | |
| <div class="category-grid"> | |
| <div class="category-chart-wrapper"> | |
| <div class="category-chart-box"><canvas id="incCatChart"></canvas></div> | |
| </div> | |
| <div class="table-scroll"> | |
| <table class="category-table"> | |
| <thead><tr><th>Categoria</th><th>Total</th><th>%</th></tr></thead> | |
| <tbody id="incCatTable"></tbody> | |
| </table> | |
| </div> | |
| </div> | |
| </div> | |
| </div> | |
| <!-- Evolution Chart --> | |
| <div class="card"> | |
| <div class="card-header"><h3>Evolução das receitas</h3></div> | |
| <div class="card-body"> | |
| <div class="chart-container"><canvas id="incEvolutionChart"></canvas></div> | |
| </div> | |
| </div> | |
| <!-- Transaction List --> | |
| <div class="card"> | |
| <div class="card-header"><h3>Últimas receitas</h3></div> | |
| <div class="card-body"> | |
| <div class="search-bar"> | |
| <input type="text" class="search-input" placeholder="Buscar..."> | |
| <div class="chip-group"> | |
| <button class="chip">Recorrentes</button> | |
| </div> | |
| </div> | |
| <ul class="tx-list" id="incList"></ul> | |
| </div> | |
| <button class="view-all-btn" data-modal="incModal">Ver todas as receitas</button> | |
| </div> | |
| </div> | |
| </main> | |
| <!-- INFO MODAL --> | |
| <div class="modal-overlay" id="infoModal"> | |
| <div class="modal"> | |
| <div class="modal-header"> | |
| <h4 id="infoTitle">Info</h4> | |
| <button class="modal-close"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M18 6 6 18"/><path d="m6 6 12 12"/></svg></button> | |
| </div> | |
| <div class="modal-body"><p id="infoText"></p></div> | |
| </div> | |
| </div> | |
| <!-- FULL LIST MODAL - EXPENSES --> | |
| <div class="modal-overlay" id="expModal"> | |
| <div class="modal" style="max-width:700px;"> | |
| <div class="modal-header"> | |
| <h4>Todas as despesas</h4> | |
| <button class="modal-close"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M18 6 6 18"/><path d="m6 6 12 12"/></svg></button> | |
| </div> | |
| <div class="modal-body"> | |
| <div class="table-scroll"> | |
| <table class="full-table"> | |
| <thead><tr><th>Data</th><th>Descrição</th><th>Categoria</th><th>Banco</th><th>Método</th><th>Valor</th></tr></thead> | |
| <tbody id="expFullTable"></tbody> | |
| </table> | |
| </div> | |
| </div> | |
| </div> | |
| </div> | |
| <!-- FULL LIST MODAL - INCOME --> | |
| <div class="modal-overlay" id="incModal"> | |
| <div class="modal" style="max-width:600px;"> | |
| <div class="modal-header"> | |
| <h4>Todas as receitas</h4> | |
| <button class="modal-close"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M18 6 6 18"/><path d="m6 6 12 12"/></svg></button> | |
| </div> | |
| <div class="modal-body"> | |
| <div class="table-scroll"> | |
| <table class="full-table"> | |
| <thead><tr><th>Data</th><th>Descrição</th><th>Categoria</th><th>Valor</th></tr></thead> | |
| <tbody id="incFullTable"></tbody> | |
| </table> | |
| </div> | |
| </div> | |
| </div> | |
| </div> | |
| <script> | |
| // ===== DATA ===== | |
| const infoContent = { | |
| despesas: { title: 'Despesas do mês', text: 'Soma de todos os gastos registrados no mês atual: compras, contas, transferências de saída e qualquer valor que saiu das suas contas.' }, | |
| receitas: { title: 'Receitas do mês', text: 'Soma de todas as entradas registradas no mês atual: salários, vendas, transferências recebidas e qualquer valor que entrou nas suas contas.' }, | |
| resultado: { title: 'Resultado do mês', text: 'Receitas menos despesas. Mostra se você está gastando mais do que ganha (negativo) ou economizando (positivo). Atenção: não é o mesmo que o dinheiro que você tem na conta.' }, | |
| saldo: { title: 'Dinheiro na conta', text: 'Soma dos saldos de todas as suas contas bancárias conectadas. Esse valor pode incluir dinheiro de meses anteriores — é diferente do resultado do mês.' }, | |
| projecao: { title: 'Projeção', text: 'Estimativa de entradas e saídas até o fim do mês, baseada em: despesas recorrentes, parcelas de cartão, e compromissos já conhecidos.' } | |
| }; | |
| const expCategories = { | |
| labels: ['Educação', 'Casa', 'Transporte', 'Saúde', 'Mercado', 'Lazer'], | |
| values: [1850, 1420, 1180, 980, 1120, 734.50], | |
| colors: ['#1DE199', '#17B87A', '#0B0B0B', '#52525B', '#A1A1AA', '#D4D4D8'] | |
| }; | |
| const incCategories = { | |
| labels: ['Salário', 'Freelance', 'Investimentos'], | |
| values: [8500, 2200, 500], | |
| colors: ['#1DE199', '#17B87A', '#0B0B0B'] | |
| }; | |
| const expByMethod = [ | |
| { label: 'Crédito', value: 3840 }, | |
| { label: 'PIX', value: 2120 }, | |
| { label: 'Débito', value: 890 }, | |
| { label: 'Boleto', value: 434.50 } | |
| ]; | |
| const expByBank = [ | |
| { label: 'Inter', value: 3200 }, | |
| { label: 'Nubank', value: 2840 }, | |
| { label: 'Bradesco', value: 1244.50 } | |
| ]; | |
| const expByAccount = [ | |
| { label: 'PF', value: 5420 }, | |
| { label: 'PJ', value: 1864.50 } | |
| ]; | |
| const expenses = [ | |
| { date: '14/12', desc: 'Compras mercado', cat: 'Mercado', bank: 'Nubank', method: 'Crédito', value: 487.32, badge: null, account: 'PF' }, | |
| { date: '12/12', desc: 'Mensalidade escola', cat: 'Educação', bank: 'Inter', method: 'Débito', value: 1850, badge: 'rec', account: 'PF' }, | |
| { date: '10/12', desc: 'Porta banheiro — 2/3', cat: 'Casa', bank: 'Nubank', method: 'Crédito', value: 450, badge: 'parc', account: 'PF' }, | |
| { date: '08/12', desc: 'Combustível', cat: 'Transporte', bank: 'Inter', method: 'PIX', value: 320, badge: null, account: 'PJ' }, | |
| { date: '05/12', desc: 'Netflix + Spotify', cat: 'Lazer', bank: 'Nubank', method: 'Crédito', value: 89.80, badge: 'rec', account: 'PF' }, | |
| { date: '03/12', desc: 'Farmácia', cat: 'Saúde', bank: 'Bradesco', method: 'Débito', value: 156.40, badge: null, account: 'PF' }, | |
| { date: '01/12', desc: 'Contabilidade', cat: 'Casa', bank: 'Inter', method: 'Boleto', value: 434.50, badge: 'rec', account: 'PJ' } | |
| ]; | |
| const incomes = [ | |
| { date: '05/12', desc: 'Salário — Empresa XYZ', cat: 'Salário', value: 8500, badge: 'rec', account: 'PF' }, | |
| { date: '10/12', desc: 'Projeto consultoria', cat: 'Freelance', value: 2200, badge: null, account: 'PJ' }, | |
| { date: '01/12', desc: 'Dividendos FIIs', cat: 'Investimentos', value: 500, badge: 'rec', account: 'PF' } | |
| ]; | |
| const dailyExp = [420, 0, 156.40, 0, 89.80, 0, 0, 320, 0, 450, 0, 1850, 0, 487.32]; | |
| const dailyInc = [500, 0, 0, 0, 8500, 0, 0, 0, 0, 2200, 0, 0, 0, 0]; | |
| const dayLabels = ['01','02','03','04','05','06','07','08','09','10','11','12','13','14']; | |
| // ===== UTILS ===== | |
| const fmt = v => v.toLocaleString('pt-BR', { style: 'currency', currency: 'BRL' }); | |
| // ===== TABS ===== | |
| document.querySelectorAll('.tab-btn').forEach(btn => { | |
| btn.addEventListener('click', () => { | |
| document.querySelectorAll('.tab-btn').forEach(b => b.classList.remove('active')); | |
| btn.classList.add('active'); | |
| document.querySelectorAll('.tab-content').forEach(c => c.classList.remove('active')); | |
| document.getElementById('tab-' + btn.dataset.tab).classList.add('active'); | |
| }); | |
| }); | |
| // ===== FILTER DROPDOWNS ===== | |
| document.querySelectorAll('.filter-dropdown').forEach(dd => { | |
| const trigger = dd.querySelector('.filter-chip'); | |
| trigger.addEventListener('click', e => { | |
| e.stopPropagation(); | |
| document.querySelectorAll('.filter-dropdown').forEach(d => { if (d !== dd) d.classList.remove('open'); }); | |
| dd.classList.toggle('open'); | |
| }); | |
| dd.querySelectorAll('.filter-option').forEach(opt => { | |
| opt.addEventListener('click', () => { | |
| dd.querySelectorAll('.filter-option').forEach(o => o.classList.remove('selected')); | |
| opt.classList.add('selected'); | |
| trigger.querySelector('span').textContent = opt.textContent; | |
| dd.classList.remove('open'); | |
| }); | |
| }); | |
| }); | |
| document.addEventListener('click', () => { | |
| document.querySelectorAll('.filter-dropdown').forEach(d => d.classList.remove('open')); | |
| }); | |
| // ===== INFO MODAL ===== | |
| const infoModal = document.getElementById('infoModal'); | |
| document.querySelectorAll('.info-btn').forEach(btn => { | |
| btn.addEventListener('click', () => { | |
| const key = btn.dataset.info; | |
| const data = infoContent[key] || { title: 'Info', text: '' }; | |
| document.getElementById('infoTitle').textContent = data.title; | |
| document.getElementById('infoText').textContent = data.text; | |
| infoModal.classList.add('visible'); | |
| }); | |
| }); | |
| // ===== FULL LIST MODALS ===== | |
| document.querySelectorAll('.view-all-btn').forEach(btn => { | |
| btn.addEventListener('click', () => { | |
| document.getElementById(btn.dataset.modal).classList.add('visible'); | |
| }); | |
| }); | |
| document.querySelectorAll('.modal-close').forEach(btn => { | |
| btn.addEventListener('click', () => btn.closest('.modal-overlay').classList.remove('visible')); | |
| }); | |
| document.querySelectorAll('.modal-overlay').forEach(ov => { | |
| ov.addEventListener('click', e => { if (e.target === ov) ov.classList.remove('visible'); }); | |
| }); | |
| // ===== CHIP TOGGLE ===== | |
| document.querySelectorAll('.chip').forEach(c => c.addEventListener('click', () => c.classList.toggle('active'))); | |
| // ===== RENDER LISTS ===== | |
| function renderTxList(id, items, type) { | |
| const ul = document.getElementById(id); | |
| ul.innerHTML = items.slice(0, 5).map(i => ` | |
| <li class="tx-item"> | |
| <div class="tx-info"> | |
| <div class="tx-main"> | |
| <span class="tx-title">${i.desc}</span> | |
| ${i.badge === 'rec' ? '<span class="tx-badge rec">Rec</span>' : ''} | |
| ${i.badge === 'parc' ? '<span class="tx-badge parc">Parc</span>' : ''} | |
| </div> | |
| <div class="tx-meta"> | |
| <span>${i.date}</span> | |
| <span>${i.cat}</span> | |
| ${i.bank ? `<span>${i.bank}</span>` : ''} | |
| ${i.method ? `<span>${i.method}</span>` : ''} | |
| <span class="tx-badge ${i.account.toLowerCase()}">${i.account}</span> | |
| </div> | |
| </div> | |
| <span class="tx-value ${type}">${type === 'expense' ? '−' : '+'}${fmt(i.value)}</span> | |
| </li> | |
| `).join(''); | |
| } | |
| renderTxList('expList', expenses, 'expense'); | |
| renderTxList('incList', incomes, 'income'); | |
| // ===== RENDER FULL TABLES ===== | |
| function renderFullTable(id, items, type) { | |
| const tb = document.getElementById(id); | |
| tb.innerHTML = items.map(i => ` | |
| <tr> | |
| <td>${i.date}</td> | |
| <td>${i.desc}${i.badge === 'rec' ? ' <span class="tx-badge rec">Rec</span>' : ''}${i.badge === 'parc' ? ' <span class="tx-badge parc">Parc</span>' : ''}</td> | |
| <td>${i.cat}</td> | |
| ${i.bank !== undefined ? `<td>${i.bank}</td>` : ''} | |
| ${i.method !== undefined ? `<td>${i.method}</td>` : ''} | |
| <td class="val ${type}">${type === 'expense' ? '−' : '+'}${fmt(i.value)}</td> | |
| </tr> | |
| `).join(''); | |
| } | |
| renderFullTable('expFullTable', expenses, 'expense'); | |
| renderFullTable('incFullTable', incomes, 'income'); | |
| // ===== RENDER CATEGORY TABLES ===== | |
| function renderCatTable(id, data) { | |
| const tb = document.getElementById(id); | |
| const total = data.values.reduce((a,b) => a + b, 0); | |
| tb.innerHTML = data.labels.map((l, i) => { | |
| const pct = ((data.values[i] / total) * 100).toFixed(1); | |
| return `<tr><td><span class="cat-dot" style="background:${data.colors[i]}"></span>${l}</td><td>${fmt(data.values[i])}</td><td class="cat-percent">${pct}%</td></tr>`; | |
| }).join(''); | |
| } | |
| renderCatTable('expCatTable', expCategories); | |
| renderCatTable('incCatTable', incCategories); | |
| // ===== RENDER BREAKDOWN LISTS ===== | |
| function renderBreakdown(id, items) { | |
| const ul = document.getElementById(id); | |
| ul.innerHTML = items.map(i => ` | |
| <li class="breakdown-item"> | |
| <span class="breakdown-item-label">${i.label}</span> | |
| <span class="breakdown-item-value">${fmt(i.value)}</span> | |
| </li> | |
| `).join(''); | |
| } | |
| renderBreakdown('expMethodList', expByMethod); | |
| renderBreakdown('expBankList', expByBank); | |
| renderBreakdown('expAccountList', expByAccount); | |
| // ===== CHARTS ===== | |
| Chart.defaults.font.family = 'Inter, sans-serif'; | |
| Chart.defaults.color = '#71717A'; | |
| // Summary Bar | |
| new Chart(document.getElementById('summaryBarChart'), { | |
| type: 'bar', | |
| data: { | |
| labels: ['Dezembro'], | |
| datasets: [ | |
| { label: 'Despesas', data: [7284.50], backgroundColor: '#EF4444', borderRadius: 4, barPercentage: 0.5 }, | |
| { label: 'Receitas', data: [11200], backgroundColor: '#1DE199', borderRadius: 4, barPercentage: 0.5 } | |
| ] | |
| }, | |
| options: { | |
| indexAxis: 'y', | |
| responsive: true, | |
| maintainAspectRatio: false, | |
| scales: { x: { grid: { color: '#E4E4E7' }, ticks: { callback: v => 'R$ ' + (v/1000).toFixed(0) + 'k' } }, y: { display: false } }, | |
| plugins: { legend: { position: 'bottom', labels: { usePointStyle: true, pointStyle: 'circle', padding: 16 } }, tooltip: { backgroundColor: '#0B0B0B', padding: 10, cornerRadius: 6, callbacks: { label: ctx => ctx.dataset.label + ': ' + fmt(ctx.parsed.x) } } } | |
| } | |
| }); | |
| // Summary Line | |
| new Chart(document.getElementById('summaryLineChart'), { | |
| type: 'line', | |
| data: { | |
| labels: dayLabels, | |
| datasets: [ | |
| { label: 'Despesas', data: dailyExp, borderColor: '#EF4444', backgroundColor: 'rgba(239,68,68,0.08)', fill: true, tension: 0.3, pointRadius: 2 }, | |
| { label: 'Receitas', data: dailyInc, borderColor: '#1DE199', backgroundColor: 'rgba(29,225,153,0.08)', fill: true, tension: 0.3, pointRadius: 2 } | |
| ] | |
| }, | |
| options: { | |
| responsive: true, | |
| maintainAspectRatio: false, | |
| interaction: { intersect: false, mode: 'index' }, | |
| scales: { x: { grid: { display: false } }, y: { grid: { color: '#E4E4E7' }, ticks: { callback: v => 'R$ ' + (v/1000).toFixed(0) + 'k' } } }, | |
| plugins: { legend: { position: 'bottom', labels: { usePointStyle: true, pointStyle: 'circle', padding: 16 } }, tooltip: { backgroundColor: '#0B0B0B', padding: 10, cornerRadius: 6, callbacks: { label: ctx => ctx.dataset.label + ': ' + fmt(ctx.parsed.y) } } } | |
| } | |
| }); | |
| // Expense Category Doughnut | |
| new Chart(document.getElementById('expCatChart'), { | |
| type: 'doughnut', | |
| data: { labels: expCategories.labels, datasets: [{ data: expCategories.values, backgroundColor: expCategories.colors, borderWidth: 0 }] }, | |
| options: { responsive: true, maintainAspectRatio: false, cutout: '55%', plugins: { legend: { display: false }, tooltip: { backgroundColor: '#0B0B0B', padding: 8, cornerRadius: 4, callbacks: { label: ctx => fmt(ctx.parsed) } } } } | |
| }); | |
| // Expense Evolution | |
| new Chart(document.getElementById('expEvolutionChart'), { | |
| type: 'bar', | |
| data: { labels: dayLabels, datasets: [{ label: 'Despesas', data: dailyExp, backgroundColor: '#EF4444', borderRadius: 3 }] }, | |
| options: { responsive: true, maintainAspectRatio: false, scales: { x: { grid: { display: false } }, y: { grid: { color: '#E4E4E7' }, ticks: { callback: v => 'R$ ' + v } } }, plugins: { legend: { display: false }, tooltip: { backgroundColor: '#0B0B0B', padding: 8, cornerRadius: 4, callbacks: { label: ctx => fmt(ctx.parsed.y) } } } } | |
| }); | |
| // Income Category Doughnut | |
| new Chart(document.getElementById('incCatChart'), { | |
| type: 'doughnut', | |
| data: { labels: incCategories.labels, datasets: [{ data: incCategories.values, backgroundColor: incCategories.colors, borderWidth: 0 }] }, | |
| options: { responsive: true, maintainAspectRatio: false, cutout: '55%', plugins: { legend: { display: false }, tooltip: { backgroundColor: '#0B0B0B', padding: 8, cornerRadius: 4, callbacks: { label: ctx => fmt(ctx.parsed) } } } } | |
| }); | |
| // Income Evolution | |
| new Chart(document.getElementById('incEvolutionChart'), { | |
| type: 'bar', | |
| data: { labels: dayLabels, datasets: [{ label: 'Receitas', data: dailyInc, backgroundColor: '#1DE199', borderRadius: 3 }] }, | |
| options: { responsive: true, maintainAspectRatio: false, scales: { x: { grid: { display: false } }, y: { grid: { color: '#E4E4E7' }, ticks: { callback: v => 'R$ ' + (v/1000).toFixed(0) + 'k' } } }, plugins: { legend: { display: false }, tooltip: { backgroundColor: '#0B0B0B', padding: 8, cornerRadius: 4, callbacks: { label: ctx => fmt(ctx.parsed.y) } } } } | |
| }); | |
| </script> | |
| </body> | |
| </html> |