diff --git a/CLAUDE.md b/CLAUDE.md index 3286033..ad497e3 100644 --- a/CLAUDE.md +++ b/CLAUDE.md @@ -20,6 +20,10 @@ python generate_report.py --style brutalism --output report.html python generate_report.py --style neumorphism --output report.html python generate_report.py --style material --output report.html python generate_report.py --style flat --output report.html +python generate_report.py --style synthwave --output report.html +python generate_report.py --style vaporwave --output report.html +python generate_report.py --style terminal --output report.html +python generate_report.py --style highcontrast --output report.html ``` Generate report with legacy Platinum template: @@ -40,19 +44,19 @@ python generate_report.py --db pga.db --output report.html --top 10 --background - Loads HTML template from `templates/` folder (default: `templates/platinum.html`) **HTML templates (`templates/`):** -- **modern.html**: Unified template for modern styles (brutalism, glassmorphism, neumorphism, material, flat) +- **modern.html**: Unified template for modern styles (brutalism, glassmorphism, neumorphism, material, flat, synthwave, vaporwave, terminal, highcontrast) - **platinum.html**: Legacy Mac OS 9 Platinum visual style (separate template due to unique structure) - All templates use Chart.js doughnut charts and dynamic JavaScript filtering - Placeholder tokens like `__ALL_GAMES__`, `__BACKGROUND_IMAGE__` are replaced at generation time - Modern templates support light/dark/auto theme toggle button **Style system (`styles.py`):** -- CSS definitions for each modern style (brutalism, glassmorphism, neumorphism, material, flat) +- CSS definitions for each modern style (brutalism, glassmorphism, neumorphism, material, flat, synthwave, vaporwave, terminal, highcontrast) - Theme configurations (colors, fonts, chart options) injected via `__THEME_CSS__` - Use `--style` argument to select a modern style instead of `--template` **Javascript (`templates/script.js`):** -- A single common script is used for each modern style (brutalism, glassmorphism, neumorphism, material, flat) +- A single common script is used for each modern style (brutalism, glassmorphism, neumorphism, material, flat, synthwave, vaporwave, terminal, highcontrast) - Theme configurations (colors, fonts, chart options) injected via `__THEME_CONFIG__` - Data inserted via `__ALL_GAMES__` and `__TOP_N__` diff --git a/generate_report.py b/generate_report.py index 8b58546..13800d9 100644 --- a/generate_report.py +++ b/generate_report.py @@ -27,7 +27,7 @@ from styles import get_theme_css, get_theme_config SCRIPT_DIR = Path(__file__).parent # Modern styles that use the unified template -MODERN_STYLES = ["brutalism", "glassmorphism", "neumorphism", "material", "flat"] +MODERN_STYLES = ["brutalism", "glassmorphism", "neumorphism", "material", "flat", "synthwave", "vaporwave", "terminal", "highcontrast"] def load_template(template_file: str) -> str: @@ -225,9 +225,9 @@ def main(): ) parser.add_argument( "--style", - choices=["brutalism", "glassmorphism", "neumorphism", "material", "flat"], + choices=["brutalism", "glassmorphism", "neumorphism", "material", "flat", "synthwave", "vaporwave", "terminal", "highcontrast"], default=None, - help="Modern style to use (brutalism, glassmorphism, neumorphism, material, flat). Overrides --template." + help="Modern style to use (brutalism, glassmorphism, neumorphism, material, flat, synthwave, vaporwave, terminal, highcontrast). Overrides --template." ) args = parser.parse_args() diff --git a/styles.py b/styles.py index 959db26..a57e36d 100644 --- a/styles.py +++ b/styles.py @@ -131,6 +131,94 @@ THEME_CONFIGS = { "tooltipBorderWidth": 0, "tooltipCornerRadius": 12, "uppercaseTooltip": False + }, + "synthwave": { + "colors": [ + "#ff00ff", "#00ffff", "#ffff00", "#ff0055", "#00ff99", + "#7a5af8", "#ff8800", "#ff00aa", "#00ccff", "#ccff00", + "#999999" + ], + "fontFamily": "'Orbitron', 'Segoe UI', sans-serif", + "fontWeight": "bold", + "pointStyle": "triangle", + "textColorLight": "#2d004d", + "textColorDark": "#ff00ff", + "borderColorLight": "rgba(255, 0, 255, 0.2)", + "borderColorDark": "rgba(0, 255, 255, 0.2)", + "borderWidth": 2, + "tooltipBg": "rgba(45, 0, 77, 0.9)", + "tooltipTitleColor": "#00ffff", + "tooltipBodyColor": "#ffffff", + "tooltipBorderColor": "#ff00ff", + "tooltipBorderWidth": 1, + "tooltipCornerRadius": 0, + "uppercaseTooltip": True + }, + "vaporwave": { + "colors": [ + "#ff71ce", "#01cdfe", "#05ffa1", "#b967ff", "#fffb96", + "#ff99cc", "#99ccff", "#ccff99", "#ffcc99", "#ffffcc", + "#e0e0e0" + ], + "fontFamily": "'MS PGothic', 'Palatino Linotype', serif", + "fontWeight": "normal", + "pointStyle": "circle", + "textColorLight": "#ff71ce", + "textColorDark": "#01cdfe", + "borderColorLight": "rgba(255, 255, 255, 0.5)", + "borderColorDark": "rgba(255, 255, 255, 0.2)", + "borderWidth": 1, + "tooltipBg": "rgba(255, 113, 206, 0.8)", + "tooltipTitleColor": "#ffffff", + "tooltipBodyColor": "#ffffff", + "tooltipBorderColor": "#01cdfe", + "tooltipBorderWidth": 2, + "tooltipCornerRadius": 4, + "uppercaseTooltip": False + }, + "terminal": { + "colors": [ + "#00ff00", "#00cc00", "#009900", "#006600", "#003300", + "#33ff33", "#66ff66", "#99ff99", "#ccffcc", "#ffffff", + "#888888" + ], + "fontFamily": "'Courier New', Courier, monospace", + "fontWeight": "bold", + "pointStyle": "rectRot", + "textColorLight": "#006600", + "textColorDark": "#00ff00", + "borderColorLight": "#000000", + "borderColorDark": "#00ff00", + "borderWidth": 1, + "tooltipBg": "#000000", + "tooltipTitleColor": "#00ff00", + "tooltipBodyColor": "#00ff00", + "tooltipBorderColor": "#00ff00", + "tooltipBorderWidth": 1, + "tooltipCornerRadius": 0, + "uppercaseTooltip": True + }, + "highcontrast": { + "colors": [ + "#888888", "#444444", "#cccccc", "#666666", "#aaaaaa", + "#333333", "#dddddd", "#555555", "#bbbbbb", "#222222", + "#999999" + ], + "fontFamily": "Arial, Helvetica, sans-serif", + "fontWeight": "900", + "pointStyle": "rect", + "textColorLight": "#000000", + "textColorDark": "#ffffff", + "borderColorLight": "#000000", + "borderColorDark": "#ffffff", + "borderWidth": 4, + "tooltipBg": "#000000", + "tooltipTitleColor": "#ffffff", + "tooltipBodyColor": "#ffffff", + "tooltipBorderColor": "#ffffff", + "tooltipBorderWidth": 3, + "tooltipCornerRadius": 0, + "uppercaseTooltip": True } } diff --git a/templates/highcontrast.css b/templates/highcontrast.css new file mode 100644 index 0000000..9bb5b9f --- /dev/null +++ b/templates/highcontrast.css @@ -0,0 +1,435 @@ +/*************************************************************************************************** + * Copyright (C) 2026 by WallyHackenslacker wallyhackenslacker@noreply.git.hackenslacker.space * + * * + * Permission to use, copy, modify, and/or distribute this software for any purpose with or without * + * fee is hereby granted. * + * * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS * + * SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE * + * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, * + * NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE * + * OF THIS SOFTWARE. * + ****************************************************************************************************/ + +:root { + --bg-base: #ffffff; + --bg-primary: #ffffff; + --bg-secondary: #ffffff; + --bg-tertiary: #000000; + --text-primary: #000000; + --text-secondary: #000000; + --text-muted: #000000; + --border-color: #000000; + --accent-color: #000000; + --accent-hover: #000000; + --selection-bg: #000000; + --selection-text: #ffffff; + --border-width: 4px; +} + +[data-theme="dark"] { + --bg-base: #000000; + --bg-primary: #000000; + --bg-secondary: #000000; + --bg-tertiary: #ffffff; + --text-primary: #ffffff; + --text-secondary: #ffffff; + --text-muted: #ffffff; + --border-color: #ffffff; + --accent-color: #ffffff; + --accent-hover: #ffffff; + --selection-bg: #ffffff; + --selection-text: #000000; +} + +@media (prefers-color-scheme: dark) { + :root:not([data-theme="light"]) { + --bg-base: #000000; + --bg-primary: #000000; + --bg-secondary: #000000; + --bg-tertiary: #ffffff; + --text-primary: #ffffff; + --text-secondary: #ffffff; + --text-muted: #ffffff; + --border-color: #ffffff; + --accent-color: #ffffff; + --accent-hover: #ffffff; + --selection-bg: #ffffff; + --selection-text: #000000; + } +} + +* { + box-sizing: border-box; + margin: 0; + padding: 0; +} + +body { + font-family: Arial, Helvetica, sans-serif; + font-size: 16px; + font-weight: 700; + max-width: 1200px; + margin: 0 auto; + padding: 20px; + background-color: var(--bg-base); + background-image: __BACKGROUND_IMAGE_CUSTOM__; + background-size: cover; + background-position: center; + background-attachment: fixed; + min-height: 100vh; + color: var(--text-primary); + line-height: 1.2; +} + +/* Theme Toggle Button */ +.theme-toggle { + position: fixed; + top: 20px; + right: 20px; + z-index: 1000; + width: 60px; + height: 60px; + border: var(--border-width) solid var(--border-color); + background: var(--bg-primary); + cursor: pointer; + display: flex; + align-items: center; + justify-content: center; + font-size: 30px; + color: var(--text-primary); +} + +.theme-toggle:hover { + background: var(--text-primary); + color: var(--bg-primary); +} + +.theme-toggle .icon-auto { + position: relative; + width: 30px; + height: 30px; + display: flex; + align-items: center; + justify-content: center; +} + +.theme-toggle .icon-auto::before, +.theme-toggle .icon-auto::after { + position: absolute; + font-size: 24px; + line-height: 1; +} + +.theme-toggle .icon-auto::before { + content: 'L'; + clip-path: inset(0 50% 0 0); +} + +.theme-toggle .icon-auto::after { + content: 'D'; + clip-path: inset(0 0 0 50%); +} + +/* Card Style */ +.card { + background: var(--bg-primary); + border: var(--border-width) solid var(--border-color); + margin-bottom: 30px; +} + +.card-header { + padding: 20px; + border-bottom: var(--border-width) solid var(--border-color); + background: var(--bg-primary); +} + +.card-title { + font-size: 30px; + font-weight: 900; + text-transform: uppercase; +} + +.card-content { + padding: 20px; +} + +.summaries-content { + padding: 0; +} + +/* Filters */ +.filters { + display: flex; + justify-content: center; + gap: 20px; + flex-wrap: wrap; +} + +.filter-label { + display: flex; + align-items: center; + gap: 10px; + cursor: pointer; + user-select: none; + padding: 10px; + border: var(--border-width) solid var(--border-color); + text-transform: uppercase; +} + +.filter-label:hover { + background: var(--text-primary); + color: var(--bg-primary); +} + +.filter-label input[type="checkbox"] { + appearance: none; + -webkit-appearance: none; + width: 24px; + height: 24px; + border: var(--border-width) solid var(--border-color); + background: var(--bg-primary); + cursor: pointer; + position: relative; +} + +.filter-label input[type="checkbox"]:checked { + background: var(--text-primary); +} + +.filter-label input[type="checkbox"]:checked::after { + content: 'X'; + position: absolute; + left: 50%; + top: 50%; + transform: translate(-50%, -50%); + font-size: 16px; + font-weight: 900; + color: var(--bg-primary); +} + +/* Stats */ +.stats { + display: flex; + justify-content: center; + gap: 30px; + flex-wrap: wrap; +} + +.stat { + text-align: center; + padding: 20px; + border: var(--border-width) solid var(--border-color); + min-width: 200px; +} + +.stat-value { + font-size: 48px; + font-weight: 900; +} + +.stat-label { + font-size: 16px; + text-transform: uppercase; +} + +/* Charts */ +.charts-wrapper { + display: grid; + grid-auto-columns: minmax(300px, 450px); + grid-template-columns: repeat(auto-fill, minmax(300px, 450px)); + grid-gap: 30px; + justify-content: center; +} + +.chart-container { + padding: 20px; + border: var(--border-width) solid var(--border-color); +} + +.chart-title { + font-size: 20px; + font-weight: 900; + text-align: center; + margin-bottom: 20px; + text-transform: uppercase; +} + +@media (max-width: 700px) { + .charts-wrapper { + flex-direction: column; + align-items: center; + } + .chart-container { + max-width: 100%; + width: 100%; + } +} + +/* Tabs */ +.tabs { + display: flex; + gap: 0; +} + +.tab { + padding: 15px 30px; + cursor: pointer; + font-weight: 900; + text-transform: uppercase; + border: var(--border-width) solid var(--border-color); + flex: 1; + text-align: center; +} + +.tab.active { + background: var(--text-primary); + color: var(--bg-primary); +} + +.tab-content { + border: var(--border-width) solid var(--border-color); + border-top: none; +} + +.tab-panel { + display: none; +} + +.tab-panel.active { + display: block; +} + +/* Tables */ +.table-wrapper { + overflow-x: auto; +} + +table { + width: 100%; + border-collapse: collapse; +} + +th { + padding: 15px; + text-align: left; + border-bottom: var(--border-width) solid var(--border-color); + text-transform: uppercase; + font-size: 14px; +} + +td { + padding: 15px; + border-bottom: 2px solid var(--border-color); +} + +tr:hover td { + background: var(--selection-bg); + color: var(--selection-text); +} + +.color-box { + display: inline-block; + width: 20px; + height: 20px; + margin-right: 10px; + border: 2px solid var(--border-color); +} + +.service-badge { + padding: 2px 8px; + border: 2px solid var(--border-color); + margin-left: 10px; + text-transform: uppercase; +} + +.category-badge { + padding: 2px 8px; + background: var(--text-primary); + color: var(--bg-primary); + margin-left: 5px; + text-transform: uppercase; +} + +/* Scrollbar */ +::-webkit-scrollbar { + width: 20px; +} + +::-webkit-scrollbar-track { + background: var(--bg-primary); + border-left: var(--border-width) solid var(--border-color); +} + +::-webkit-scrollbar-thumb { + background: var(--text-primary); +} + +/* Scroll to Top Button */ +.scroll-top { + position: fixed; + bottom: 20px; + right: 20px; + z-index: 1000; + width: 60px; + height: 60px; + border: var(--border-width) solid var(--border-color); + background: var(--bg-primary); + cursor: pointer; + display: flex; + align-items: center; + justify-content: center; + color: var(--text-primary); + opacity: 0; + visibility: hidden; +} + +.scroll-top.visible { + opacity: 1; + visibility: visible; +} + +.scroll-top:hover { + background: var(--text-primary); + color: var(--bg-primary); +} + +.scroll-top-arrow { + width: 0; + height: 0; + border-left: 15px solid transparent; + border-right: 15px solid transparent; + border-bottom: 20px solid currentColor; +} + +/* Others row expansion */ +.others-row { + cursor: pointer; +} + +.others-row td:first-child::before { + content: '\25B6'; + display: inline-block; + margin-right: 10px; + font-size: 14px; + transition: transform 0.2s ease; + vertical-align: middle; +} + +.others-row.expanded td:first-child::before { + transform: rotate(90deg); +} + +.others-detail { + display: none; + background: var(--bg-secondary); +} + +.others-detail.visible { + display: table-row; +} + +.others-detail td { + padding-left: 48px; +} diff --git a/templates/synthwave.css b/templates/synthwave.css new file mode 100644 index 0000000..455069e --- /dev/null +++ b/templates/synthwave.css @@ -0,0 +1,583 @@ +/*************************************************************************************************** + * Copyright (C) 2026 by WallyHackenslacker wallyhackenslacker@noreply.git.hackenslacker.space * + * * + * Permission to use, copy, modify, and/or distribute this software for any purpose with or without * + * fee is hereby granted. * + * * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS * + * SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE * + * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, * + * NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE * + * OF THIS SOFTWARE. * + ****************************************************************************************************/ + +@import url('https://fonts.googleapis.com/css2?family=Orbitron:wght@400;700;900&display=swap'); + +:root { + --bg-base: #2d004d; + --bg-primary: #1a0033; + --bg-secondary: #240046; + --bg-tertiary: #3c096c; + --text-primary: #ff00ff; + --text-secondary: #00ffff; + --text-muted: #7a5af8; + --border-color: #ff00ff; + --accent-color: #00ffff; + --accent-hover: #ffffff; + --accent-secondary: #ffff00; + --selection-bg: rgba(255, 0, 255, 0.3); + --selection-text: #ffffff; + --shadow-color: rgba(255, 0, 255, 0.5); + --glow-text: 0 0 10px var(--text-primary); + --glow-border: 0 0 10px var(--border-color); + --card-radius: 0px; + --border-width: 2px; +} + +[data-theme="light"] { + --bg-base: #f0e6ff; + --bg-primary: #ffffff; + --bg-secondary: #f8f0ff; + --bg-tertiary: #e0ccff; + --text-primary: #9d00ff; + --text-secondary: #00bfff; + --text-muted: #7a5af8; + --border-color: #9d00ff; + --shadow-color: rgba(157, 0, 255, 0.2); +} + +@media (prefers-color-scheme: dark) { + :root:not([data-theme="light"]) { + --bg-base: #2d004d; + --bg-primary: #1a0033; + --bg-secondary: #240046; + --bg-tertiary: #3c096c; + --text-primary: #ff00ff; + --text-secondary: #00ffff; + --text-muted: #7a5af8; + --border-color: #ff00ff; + --accent-color: #00ffff; + --accent-hover: #ffffff; + --selection-bg: rgba(255, 0, 255, 0.3); + --selection-text: #ffffff; + } +} + +* { + box-sizing: border-box; + margin: 0; + padding: 0; +} + +body { + font-family: 'Orbitron', sans-serif; + font-size: 14px; + max-width: 1200px; + margin: 0 auto; + padding: 20px; + background-color: var(--bg-base); + background-image: __BACKGROUND_IMAGE_CUSTOM__; + background-size: cover; + background-position: center; + background-attachment: fixed; + min-height: 100vh; + color: var(--text-primary); + line-height: 1.5; + text-shadow: var(--glow-text); +} + +/* Theme Toggle Button */ +.theme-toggle { + position: fixed; + top: 20px; + right: 20px; + z-index: 1000; + width: 50px; + height: 50px; + border: var(--border-width) solid var(--border-color); + background: var(--bg-primary); + cursor: pointer; + display: flex; + align-items: center; + justify-content: center; + font-size: 20px; + box-shadow: var(--glow-border); + transition: all 0.3s ease; + color: var(--text-primary); +} + +.theme-toggle:hover { + transform: scale(1.1); + box-shadow: 0 0 20px var(--border-color); +} + +.theme-toggle .icon-auto { + position: relative; + width: 24px; + height: 24px; + display: flex; + align-items: center; + justify-content: center; +} + +.theme-toggle .icon-auto::before, +.theme-toggle .icon-auto::after { + position: absolute; + font-size: 20px; + line-height: 1; +} + +.theme-toggle .icon-auto::before { + content: '☀️'; + clip-path: inset(0 50% 0 0); +} + +.theme-toggle .icon-auto::after { + content: '🌙'; + clip-path: inset(0 0 0 50%); +} + +/* Card Style */ +.card { + background: var(--bg-primary); + border: var(--border-width) solid var(--border-color); + box-shadow: var(--glow-border); + margin-bottom: 30px; + overflow: hidden; + position: relative; +} + +.card::after { + content: ''; + position: absolute; + top: 0; + left: 0; + right: 0; + height: 2px; + background: linear-gradient(90deg, transparent, var(--accent-color), transparent); +} + +.card-header { + padding: 16px 24px; + border-bottom: var(--border-width) solid var(--border-color); + background: var(--bg-secondary); +} + +.card-title { + font-size: 20px; + font-weight: 900; + text-transform: uppercase; + letter-spacing: 3px; + color: var(--text-secondary); + text-shadow: 0 0 10px var(--text-secondary); +} + +.card-content { + padding: 24px; +} + +.summaries-content { + padding: 0; + padding-top: 16px; +} + +/* Filters */ +.filters { + display: flex; + justify-content: center; + gap: 16px; + flex-wrap: wrap; +} + +.filter-label { + display: flex; + align-items: center; + gap: 8px; + cursor: pointer; + user-select: none; + padding: 10px 20px; + background: var(--bg-secondary); + border: 1px solid var(--border-color); + transition: all 0.2s ease; + font-weight: 700; + text-transform: uppercase; + font-size: 12px; +} + +.filter-label:hover { + box-shadow: 0 0 15px var(--border-color); + transform: translateY(-2px); +} + +.filter-label input[type="checkbox"] { + appearance: none; + -webkit-appearance: none; + width: 18px; + height: 18px; + border: 1px solid var(--border-color); + background: var(--bg-primary); + cursor: pointer; + position: relative; + transition: all 0.2s ease; +} + +.filter-label input[type="checkbox"]:checked { + background: var(--accent-color); + box-shadow: 0 0 10px var(--accent-color); +} + +.filter-label input[type="checkbox"]:checked::after { + content: ''; + position: absolute; + left: 5px; + top: 2px; + width: 5px; + height: 9px; + border: solid var(--bg-primary); + border-width: 0 2px 2px 0; + transform: rotate(45deg); +} + +.filter-label .service-name { + color: var(--text-secondary); +} + +.filter-label .service-count { + color: var(--text-muted); + font-size: 11px; +} + +/* Stats */ +.stats { + display: flex; + justify-content: center; + gap: 24px; + flex-wrap: wrap; +} + +.stat { + text-align: center; + padding: 24px 32px; + background: var(--bg-secondary); + border: 1px solid var(--border-color); + min-width: 160px; + box-shadow: inset 0 0 10px var(--shadow-color); +} + +.stat-value { + font-size: 36px; + font-weight: 900; + color: var(--accent-color); + text-shadow: 0 0 15px var(--accent-color); + font-variant-numeric: tabular-nums; +} + +.stat-label { + font-size: 12px; + color: var(--text-muted); + margin-top: 4px; + text-transform: uppercase; + letter-spacing: 2px; +} + +/* Charts */ +.charts-wrapper { + display: grid; + grid-auto-columns: minmax(280px, 450px); + grid-template-columns: repeat(auto-fill, minmax(280px, 450px)); + grid-gap: 30px; + justify-content: center; +} + +.chart-container { + min-width: 280px; + max-width: 450px; + padding: 20px; + background: var(--bg-secondary); + border: 1px solid var(--border-color); + box-shadow: inset 0 0 20px var(--shadow-color); +} + +.chart-title { + font-size: 16px; + font-weight: 700; + text-align: center; + margin-bottom: 20px; + color: var(--text-secondary); + text-transform: uppercase; + letter-spacing: 2px; +} + +@media (max-width: 700px) { + .charts-wrapper { + flex-direction: column; + align-items: center; + } + .chart-container { + max-width: 100%; + width: 100%; + } +} + +/* Tabs */ +.tabs { + display: flex; + gap: 0; + padding: 0 24px; + margin-bottom: -1px; + position: relative; + z-index: 1; +} + +.tab { + padding: 14px 28px; + cursor: pointer; + color: var(--text-muted); + font-weight: 700; + text-transform: uppercase; + font-size: 13px; + letter-spacing: 2px; + background: var(--bg-secondary); + border: var(--border-width) solid var(--border-color); + border-bottom: none; + transition: all 0.3s ease; + margin-right: 4px; +} + +.tab:hover { + color: var(--text-secondary); + box-shadow: 0 -5px 10px var(--shadow-color); +} + +.tab.active { + color: var(--accent-color); + background: var(--bg-primary); + border-bottom-color: var(--bg-primary); + box-shadow: 0 -5px 15px var(--shadow-color); + position: relative; +} + +.tab-content { + background: var(--bg-primary); + border: var(--border-width) solid var(--border-color); + padding: 24px; + box-shadow: var(--glow-border); +} + +.tab-panel { + display: none; +} + +.tab-panel.active { + display: block; +} + +/* Tables */ +.table-wrapper { + overflow-x: auto; +} + +table { + width: 100%; + border-collapse: collapse; + font-size: 14px; +} + +th { + padding: 16px 24px; + text-align: left; + font-weight: 900; + color: var(--text-secondary); + border-bottom: var(--border-width) solid var(--border-color); + text-transform: uppercase; + letter-spacing: 2px; + font-size: 12px; +} + +td { + padding: 14px 24px; + border-bottom: 1px solid var(--bg-tertiary); + color: var(--text-primary); +} + +tr:hover td { + background: var(--selection-bg); + color: var(--selection-text); +} + +tr:last-child td { + border-bottom: none; +} + +.time { + font-variant-numeric: tabular-nums; + font-weight: 700; +} + +.percent { + font-variant-numeric: tabular-nums; + text-align: right; + color: var(--text-muted); +} + +.color-box { + display: inline-block; + width: 12px; + height: 12px; + margin-right: 12px; + vertical-align: middle; + box-shadow: 0 0 5px currentColor; +} + +.service-badge { + display: inline-block; + font-size: 10px; + padding: 3px 10px; + background: var(--bg-tertiary); + border: 1px solid var(--border-color); + color: var(--text-secondary); + margin-left: 10px; + text-transform: uppercase; + font-weight: 700; +} + +.category-badge { + display: inline-block; + font-size: 10px; + padding: 3px 10px; + background: var(--accent-color); + border: 1px solid var(--border-color); + color: var(--bg-primary); + margin-left: 6px; + text-transform: uppercase; + font-weight: 700; +} + +.no-data { + text-align: center; + padding: 40px; + color: var(--text-muted); +} + +/* Others row expansion */ +.others-row { + cursor: pointer; +} + +.others-row td:first-child::before { + content: '\25B6'; + display: inline-block; + margin-right: 10px; + font-size: 12px; + transition: transform 0.3s ease; +} + +.others-row.expanded td:first-child::before { + transform: rotate(90deg); +} + +.others-detail { + display: none; + background: var(--bg-secondary); +} + +.others-detail.visible { + display: table-row; +} + +.others-detail td { + padding-left: 48px; +} + +/* Scrollbar */ +::-webkit-scrollbar { + width: 12px; + height: 12px; +} + +::-webkit-scrollbar-track { + background: var(--bg-primary); +} + +::-webkit-scrollbar-thumb { + background: var(--border-color); + box-shadow: 0 0 10px var(--border-color); +} + +::-webkit-scrollbar-thumb:hover { + background: var(--accent-color); +} + +/* Responsive */ +@media (max-width: 600px) { + body { + padding: 12px; + } + + .stats { + gap: 16px; + } + + .stat { + padding: 20px; + min-width: 120px; + } + + .stat-value { + font-size: 28px; + } + + .filter-label { + padding: 8px 14px; + font-size: 11px; + } + + .tabs { + padding: 0 12px; + } + + .tab { + padding: 12px 16px; + font-size: 11px; + } +} + +/* Scroll to Top Button */ +.scroll-top { + position: fixed; + bottom: 20px; + right: 20px; + z-index: 1000; + width: 50px; + height: 50px; + border: var(--border-width) solid var(--border-color); + background: var(--bg-primary); + cursor: pointer; + display: flex; + align-items: center; + justify-content: center; + box-shadow: var(--glow-border); + opacity: 0; + visibility: hidden; + transition: all 0.3s ease; + color: var(--text-primary); +} + +.scroll-top.visible { + opacity: 1; + visibility: visible; +} + +.scroll-top:hover { + transform: scale(1.1); + box-shadow: 0 0 20px var(--border-color); +} + +.scroll-top-arrow { + width: 0; + height: 0; + border-left: 10px solid transparent; + border-right: 10px solid transparent; + border-bottom: 12px solid currentColor; +} diff --git a/templates/terminal.css b/templates/terminal.css new file mode 100644 index 0000000..2beefe6 --- /dev/null +++ b/templates/terminal.css @@ -0,0 +1,459 @@ +/*************************************************************************************************** + * Copyright (C) 2026 by WallyHackenslacker wallyhackenslacker@noreply.git.hackenslacker.space * + * * + * Permission to use, copy, modify, and/or distribute this software for any purpose with or without * + * fee is hereby granted. * + * * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS * + * SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE * + * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, * + * NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE * + * OF THIS SOFTWARE. * + ****************************************************************************************************/ + +:root { + --bg-base: #000000; + --bg-primary: #000000; + --bg-secondary: #0a0a0a; + --bg-tertiary: #1a1a1a; + --text-primary: #00ff00; + --text-secondary: #00cc00; + --text-muted: #006600; + --border-color: #00ff00; + --accent-color: #00ff00; + --accent-hover: #33ff33; + --selection-bg: #00ff00; + --selection-text: #000000; + --border-width: 1px; + --scanline-opacity: 0.1; +} + +[data-theme="light"] { + --bg-base: #f0f0f0; + --bg-primary: #ffffff; + --bg-secondary: #e8e8e8; + --bg-tertiary: #d0d0d0; + --text-primary: #006600; + --text-secondary: #004400; + --text-muted: #002200; + --border-color: #006600; + --accent-color: #006600; + --accent-hover: #008800; + --selection-bg: #006600; + --selection-text: #ffffff; +} + +@media (prefers-color-scheme: dark) { + :root:not([data-theme="light"]) { + --bg-base: #000000; + --bg-primary: #000000; + --bg-secondary: #0a0a0a; + --bg-tertiary: #1a1a1a; + --text-primary: #00ff00; + --text-secondary: #00cc00; + --text-muted: #006600; + --border-color: #00ff00; + --accent-color: #00ff00; + --accent-hover: #33ff33; + --selection-bg: #00ff00; + --selection-text: #000000; + } +} + +* { + box-sizing: border-box; + margin: 0; + padding: 0; +} + +body { + font-family: 'Courier New', Courier, monospace; + font-size: 14px; + max-width: 1200px; + margin: 0 auto; + padding: 20px; + background-color: var(--bg-base); + background-image: __BACKGROUND_IMAGE_CUSTOM__; + background-size: cover; + background-position: center; + background-attachment: fixed; + min-height: 100vh; + color: var(--text-primary); + line-height: 1.4; + position: relative; +} + +body::before { + content: " "; + display: block; + position: fixed; + top: 0; + left: 0; + bottom: 0; + right: 0; + background: linear-gradient(rgba(18, 16, 16, 0) 50%, rgba(0, 0, 0, 0.25) 50%), linear-gradient(90deg, rgba(255, 0, 0, 0.06), rgba(0, 255, 0, 0.02), rgba(0, 0, 255, 0.06)); + z-index: 2000; + background-size: 100% 2px, 3px 100%; + pointer-events: none; +} + +/* Theme Toggle Button */ +.theme-toggle { + position: fixed; + top: 20px; + right: 20px; + z-index: 2100; + width: 40px; + height: 40px; + border: var(--border-width) solid var(--border-color); + background: var(--bg-primary); + cursor: pointer; + display: flex; + align-items: center; + justify-content: center; + color: var(--text-primary); + transition: all 0.1s; +} + +.theme-toggle:hover { + background: var(--text-primary); + color: var(--bg-primary); +} + +.theme-toggle .icon-auto { + position: relative; + width: 24px; + height: 24px; + display: flex; + align-items: center; + justify-content: center; +} + +.theme-toggle .icon-auto::before, +.theme-toggle .icon-auto::after { + position: absolute; + font-size: 18px; + line-height: 1; +} + +.theme-toggle .icon-auto::before { + content: 'S'; + clip-path: inset(0 50% 0 0); +} + +.theme-toggle .icon-auto::after { + content: 'T'; + clip-path: inset(0 0 0 50%); +} + +/* Card Style */ +.card { + background: var(--bg-primary); + border: var(--border-width) solid var(--border-color); + margin-bottom: 20px; + position: relative; +} + +.card-header { + padding: 8px 16px; + border-bottom: var(--border-width) solid var(--border-color); + background: var(--bg-tertiary); +} + +.card-header::before { + content: "> "; +} + +.card-title { + display: inline; + font-size: 14px; + font-weight: bold; + text-transform: uppercase; + letter-spacing: 1px; +} + +.card-content { + padding: 16px; +} + +.summaries-content { + padding: 0; + padding-top: 10px; +} + +/* Filters */ +.filters { + display: flex; + justify-content: flex-start; + gap: 20px; + flex-wrap: wrap; +} + +.filter-label { + display: flex; + align-items: center; + gap: 8px; + cursor: pointer; + user-select: none; + padding: 4px; +} + +.filter-label:hover { + background: var(--text-muted); + color: var(--bg-primary); +} + +.filter-label input[type="checkbox"] { + appearance: none; + -webkit-appearance: none; + width: 14px; + height: 14px; + border: 1px solid var(--border-color); + background: var(--bg-primary); + cursor: pointer; + position: relative; +} + +.filter-label input[type="checkbox"]:checked { + background: var(--text-primary); +} + +.filter-label input[type="checkbox"]:checked::after { + content: 'X'; + position: absolute; + left: 50%; + top: 50%; + transform: translate(-50%, -50%); + font-size: 10px; + font-weight: bold; + color: var(--bg-primary); +} + +.filter-label .service-name { + text-transform: uppercase; +} + +/* Stats */ +.stats { + display: flex; + justify-content: flex-start; + gap: 40px; + flex-wrap: wrap; +} + +.stat { + min-width: 120px; +} + +.stat-value { + font-size: 24px; + font-weight: bold; +} + +.stat-label { + font-size: 12px; + color: var(--text-secondary); + text-transform: uppercase; +} + +/* Charts */ +.charts-wrapper { + display: grid; + grid-auto-columns: minmax(280px, 450px); + grid-template-columns: repeat(auto-fill, minmax(280px, 450px)); + grid-gap: 20px; + justify-content: center; +} + +.chart-container { + padding: 10px; + border: 1px dashed var(--border-color); +} + +.chart-title { + font-size: 12px; + text-align: center; + margin-bottom: 10px; + text-transform: uppercase; +} + +@media (max-width: 700px) { + .charts-wrapper { + flex-direction: column; + align-items: center; + } + .chart-container { + max-width: 100%; + width: 100%; + } +} + +/* Tabs */ +.tabs { + display: flex; + gap: 0; + padding: 0 16px; +} + +.tab { + padding: 8px 16px; + cursor: pointer; + color: var(--text-secondary); + text-transform: uppercase; + font-size: 12px; + border: 1px solid transparent; +} + +.tab:hover { + color: var(--text-primary); + text-decoration: underline; +} + +.tab.active { + color: var(--bg-primary); + background: var(--text-primary); + border: 1px solid var(--text-primary); +} + +.tab-content { + border-top: 1px solid var(--border-color); + padding: 16px; +} + +.tab-panel { + display: none; +} + +.tab-panel.active { + display: block; +} + +/* Tables */ +.table-wrapper { + overflow-x: auto; +} + +table { + width: 100%; + border-collapse: collapse; +} + +th { + padding: 8px; + text-align: left; + color: var(--text-primary); + border-bottom: 2px solid var(--border-color); + text-transform: uppercase; +} + +td { + padding: 8px; + border-bottom: 1px solid var(--text-muted); +} + +tr:hover td { + background: var(--text-primary); + color: var(--bg-primary); +} + +.color-box { + display: inline-block; + width: 10px; + height: 10px; + margin-right: 8px; + border: 1px solid currentColor; +} + +.service-badge { + font-size: 10px; + padding: 0 4px; + border: 1px solid currentColor; + margin-left: 8px; +} + +.category-badge { + font-size: 10px; + padding: 0 4px; + background: var(--text-muted); + color: var(--bg-primary); + margin-left: 4px; +} + +/* Others row expansion */ +.others-row { + cursor: pointer; +} + +.others-row td:first-child::before { + content: '[+] '; +} + +.others-row.expanded td:first-child::before { + content: '[-] '; +} + +.others-detail { + display: none; + background: var(--bg-tertiary); +} + +.others-detail.visible { + display: table-row; +} + +.others-detail td { + padding-left: 32px; +} + +/* Scrollbar */ +::-webkit-scrollbar { + width: 10px; +} + +::-webkit-scrollbar-track { + background: var(--bg-primary); +} + +::-webkit-scrollbar-thumb { + background: var(--text-muted); +} + +::-webkit-scrollbar-thumb:hover { + background: var(--text-primary); +} + +/* Scroll to Top Button */ +.scroll-top { + position: fixed; + bottom: 20px; + right: 20px; + z-index: 2100; + width: 40px; + height: 40px; + border: var(--border-width) solid var(--border-color); + background: var(--bg-primary); + cursor: pointer; + display: flex; + align-items: center; + justify-content: center; + color: var(--text-primary); + opacity: 0; + visibility: hidden; +} + +.scroll-top.visible { + opacity: 1; + visibility: visible; +} + +.scroll-top:hover { + background: var(--text-primary); + color: var(--bg-primary); +} + +.scroll-top-arrow::before { + content: "^"; + font-size: 20px; + font-weight: bold; +} diff --git a/templates/vaporwave.css b/templates/vaporwave.css new file mode 100644 index 0000000..1995d43 --- /dev/null +++ b/templates/vaporwave.css @@ -0,0 +1,474 @@ +/*************************************************************************************************** + * Copyright (C) 2026 by WallyHackenslacker wallyhackenslacker@noreply.git.hackenslacker.space * + * * + * Permission to use, copy, modify, and/or distribute this software for any purpose with or without * + * fee is hereby granted. * + * * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS * + * SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE * + * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, * + * NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE * + * OF THIS SOFTWARE. * + ****************************************************************************************************/ + +:root { + --bg-base: #ffb7ff; + --bg-primary: rgba(255, 255, 255, 0.8); + --bg-secondary: #01cdfe; + --bg-tertiary: #fffb96; + --text-primary: #ff71ce; + --text-secondary: #01cdfe; + --text-muted: #b967ff; + --border-color: #05ffa1; + --accent-color: #ff71ce; + --accent-hover: #b967ff; + --accent-secondary: #05ffa1; + --selection-bg: rgba(255, 113, 206, 0.3); + --selection-text: #ffffff; + --shadow-color: rgba(1, 205, 254, 0.3); + --card-radius: 4px; + --border-width: 3px; +} + +[data-theme="dark"] { + --bg-base: #120458; + --bg-primary: rgba(18, 4, 88, 0.9); + --bg-secondary: #ff71ce; + --bg-tertiary: #01cdfe; + --text-primary: #05ffa1; + --text-secondary: #ff71ce; + --text-muted: #b967ff; + --border-color: #01cdfe; + --shadow-color: rgba(255, 113, 206, 0.5); +} + +@media (prefers-color-scheme: dark) { + :root:not([data-theme="light"]) { + --bg-base: #120458; + --bg-primary: rgba(18, 4, 88, 0.9); + --bg-secondary: #ff71ce; + --bg-tertiary: #01cdfe; + --text-primary: #05ffa1; + --text-secondary: #ff71ce; + --text-muted: #b967ff; + --border-color: #01cdfe; + --shadow-color: rgba(255, 113, 206, 0.5); + } +} + +* { + box-sizing: border-box; + margin: 0; + padding: 0; +} + +body { + font-family: 'MS PGothic', 'Palatino Linotype', 'Book Antiqua', Palatino, serif; + font-size: 14px; + max-width: 1200px; + margin: 0 auto; + padding: 20px; + background-color: var(--bg-base); + background-image: linear-gradient(45deg, var(--bg-base) 25%, transparent 25%), + linear-gradient(-45deg, var(--bg-base) 25%, transparent 25%), + linear-gradient(45deg, transparent 75%, var(--bg-tertiary) 75%), + linear-gradient(-45deg, transparent 75%, var(--bg-tertiary) 75%); + background-size: 100px 100px; + background-image: __BACKGROUND_IMAGE_CUSTOM__; + background-attachment: fixed; + min-height: 100vh; + color: var(--text-primary); + line-height: 1.6; +} + +/* Theme Toggle Button */ +.theme-toggle { + position: fixed; + top: 20px; + right: 20px; + z-index: 1000; + width: 48px; + height: 48px; + border: var(--border-width) solid var(--border-color); + background: var(--bg-primary); + cursor: pointer; + display: flex; + align-items: center; + justify-content: center; + font-size: 20px; + box-shadow: 8px 8px 0 var(--shadow-color); + transition: all 0.2s ease; +} + +.theme-toggle:hover { + transform: translate(-4px, -4px); + box-shadow: 12px 12px 0 var(--shadow-color); +} + +.theme-toggle .icon-auto { + position: relative; + width: 24px; + height: 24px; + display: flex; + align-items: center; + justify-content: center; +} + +.theme-toggle .icon-auto::before, +.theme-toggle .icon-auto::after { + position: absolute; + font-size: 20px; + line-height: 1; +} + +.theme-toggle .icon-auto::before { + content: '☀️'; + clip-path: inset(0 50% 0 0); +} + +.theme-toggle .icon-auto::after { + content: '🌙'; + clip-path: inset(0 0 0 50%); +} + +/* Card Style */ +.card { + background: var(--bg-primary); + border: var(--border-width) solid var(--border-color); + box-shadow: 10px 10px 0 var(--shadow-color); + margin-bottom: 40px; + overflow: hidden; +} + +.card-header { + padding: 16px 24px; + border-bottom: var(--border-width) solid var(--border-color); + background: linear-gradient(90deg, var(--bg-secondary), var(--text-primary)); +} + +.card-title { + font-size: 24px; + font-weight: 400; + font-style: italic; + color: #ffffff; + text-shadow: 3px 3px 0 var(--text-muted); +} + +.card-content { + padding: 24px; +} + +.summaries-content { + padding: 0; + padding-top: 20px; +} + +/* Filters */ +.filters { + display: flex; + justify-content: center; + gap: 16px; + flex-wrap: wrap; +} + +.filter-label { + display: flex; + align-items: center; + gap: 8px; + cursor: pointer; + user-select: none; + padding: 8px 16px; + background: var(--bg-tertiary); + border: 2px solid var(--border-color); + transition: all 0.2s ease; + font-style: italic; +} + +.filter-label:hover { + transform: scale(1.05); + background: #ffffff; +} + +.filter-label input[type="checkbox"] { + appearance: none; + -webkit-appearance: none; + width: 20px; + height: 20px; + border: 2px solid var(--border-color); + background: #ffffff; + cursor: pointer; + position: relative; +} + +.filter-label input[type="checkbox"]:checked { + background: var(--text-primary); +} + +.filter-label input[type="checkbox"]:checked::after { + content: 'OK'; + position: absolute; + left: 50%; + top: 50%; + transform: translate(-50%, -50%); + font-size: 10px; + font-weight: bold; + color: #ffffff; +} + +.filter-label .service-name { + color: var(--text-muted); +} + +.filter-label .service-count { + color: var(--text-secondary); + font-size: 11px; +} + +/* Stats */ +.stats { + display: flex; + justify-content: center; + gap: 30px; + flex-wrap: wrap; +} + +.stat { + text-align: center; + padding: 20px 30px; + background: #ffffff; + border: 2px solid var(--border-color); + min-width: 150px; + box-shadow: 5px 5px 0 var(--text-muted); +} + +.stat-value { + font-size: 32px; + font-weight: 400; + color: var(--text-primary); + font-variant-numeric: tabular-nums; +} + +.stat-label { + font-size: 11px; + color: var(--text-secondary); + margin-top: 4px; + text-transform: lowercase; + font-style: italic; +} + +/* Charts */ +.charts-wrapper { + display: grid; + grid-auto-columns: minmax(280px, 450px); + grid-template-columns: repeat(auto-fill, minmax(280px, 450px)); + grid-gap: 30px; + justify-content: center; +} + +.chart-container { + min-width: 280px; + max-width: 450px; + padding: 20px; + background: #ffffff; + border: var(--border-width) solid var(--border-color); +} + +.chart-title { + font-size: 18px; + font-style: italic; + text-align: center; + margin-bottom: 20px; + color: var(--text-primary); +} + +@media (max-width: 700px) { + .charts-wrapper { + flex-direction: column; + align-items: center; + } + .chart-container { + max-width: 100%; + width: 100%; + } +} + +/* Tabs */ +.tabs { + display: flex; + gap: 10px; + padding: 0 24px; +} + +.tab { + padding: 12px 24px; + cursor: pointer; + color: var(--text-muted); + font-style: italic; + background: #ffffff; + border: 2px solid var(--border-color); + border-bottom: none; + transition: all 0.2s ease; +} + +.tab:hover { + background: var(--bg-tertiary); +} + +.tab.active { + color: var(--text-primary); + background: var(--bg-tertiary); + font-weight: bold; + transform: translateY(-5px); + box-shadow: 5px 0 0 var(--shadow-color); +} + +.tab-content { + background: #ffffff; + border: var(--border-width) solid var(--border-color); + padding: 24px; + box-shadow: 10px 10px 0 var(--shadow-color); +} + +.tab-panel { + display: none; +} + +.tab-panel.active { + display: block; +} + +/* Tables */ +.table-wrapper { + overflow-x: auto; +} + +table { + width: 100%; + border-collapse: collapse; +} + +th { + padding: 12px 20px; + text-align: left; + font-style: italic; + color: var(--text-secondary); + border-bottom: 2px solid var(--border-color); +} + +td { + padding: 12px 20px; + border-bottom: 1px solid #f0f0f0; + color: var(--text-muted); +} + +tr:hover td { + background: var(--selection-bg); + color: var(--text-primary); +} + +.color-box { + display: inline-block; + width: 14px; + height: 14px; + margin-right: 10px; + vertical-align: middle; + border: 1px solid #000; +} + +.service-badge { + font-size: 10px; + padding: 2px 8px; + background: var(--bg-secondary); + color: #ffffff; + margin-left: 8px; +} + +.category-badge { + font-size: 10px; + padding: 2px 8px; + background: var(--bg-tertiary); + color: var(--text-primary); + margin-left: 4px; + border: 1px solid var(--border-color); +} + +/* Scrollbar */ +::-webkit-scrollbar { + width: 15px; +} + +::-webkit-scrollbar-track { + background: var(--bg-tertiary); +} + +::-webkit-scrollbar-thumb { + background: var(--text-primary); + border: 3px solid var(--bg-tertiary); +} + +/* Scroll to Top Button */ +.scroll-top { + position: fixed; + bottom: 20px; + right: 20px; + z-index: 1000; + width: 50px; + height: 50px; + border: var(--border-width) solid var(--border-color); + background: var(--bg-primary); + cursor: pointer; + display: flex; + align-items: center; + justify-content: center; + box-shadow: 8px 8px 0 var(--shadow-color); + opacity: 0; + visibility: hidden; + transition: all 0.2s ease; +} + +.scroll-top.visible { + opacity: 1; + visibility: visible; +} + +.scroll-top-arrow { + width: 0; + height: 0; + border-left: 10px solid transparent; + border-right: 10px solid transparent; + border-bottom: 12px solid var(--text-primary); +} + +/* Others row expansion */ +.others-row { + cursor: pointer; +} + +.others-row td:first-child::before { + content: '\25B6'; + display: inline-block; + margin-right: 10px; + font-size: 10px; + transition: transform 0.2s ease; + vertical-align: middle; +} + +.others-row.expanded td:first-child::before { + transform: rotate(90deg); +} + +.others-detail { + display: none; + background: var(--bg-tertiary); +} + +.others-detail.visible { + display: table-row; +} + +.others-detail td { + padding-left: 48px; + font-size: 13px; + color: var(--text-muted); +}