Cleanup frontend code

This commit is contained in:
2026-02-15 20:27:12 -05:00
parent b6d6c23eeb
commit 20f294b8d7
4 changed files with 310 additions and 389 deletions

2
.eslintignore Normal file
View File

@@ -0,0 +1,2 @@
dist/
node_modules/

View File

@@ -3,157 +3,178 @@ import axios from "axios";
import dayjs from "dayjs"; import dayjs from "dayjs";
import utc from "dayjs/plugin/utc"; import utc from "dayjs/plugin/utc";
import "vue-loading-overlay/dist/css/index.css";
import "./theme.css"; import "./theme.css";
// Configure day.js with UTC support
dayjs.extend(utc); dayjs.extend(utc);
export default { export default {
data() { data() {
return { return {
ascending: this.ascending,
filter: decodeURIComponent(window.location.href.split("filter=")[1] || ""), filter: decodeURIComponent(window.location.href.split("filter=")[1] || ""),
sortColumn: this.sortColumn, sortMethod: "score",
sortMethod: 'score',
topics: [], topics: [],
isMobile: false, isMobile: false,
currentTheme: typeof localStorage !== 'undefined' ? (localStorage.getItem('theme') || 'auto') : 'auto', currentTheme: "auto",
mediaQueryListener: null,
vuetifyTheme: null,
darkModeQuery: null, darkModeQuery: null,
themeChangeHandler: null, themeChangeHandler: null,
}; };
}, },
mounted() { mounted() {
window.addEventListener("keydown", this.handleKeyDown); window.addEventListener("keydown", this.handleKeyDown);
window.addEventListener("resize", this.handleResize);
this.detectMobile(); this.detectMobile();
this.fetchDeals(); this.fetchDeals();
// Initialize sort method from local storage
this.initializeSortMethod(); this.initializeSortMethod();
// Initialize theme immediately to prevent flash
this.initializeTheme(); this.initializeTheme();
this.setupThemeListener(); this.setupThemeListener();
}, },
beforeUnmount() { beforeUnmount() {
window.removeEventListener("keydown", this.handleKeyDown); window.removeEventListener("keydown", this.handleKeyDown);
window.removeEventListener("resize", this.detectMobile); window.removeEventListener("resize", this.handleResize);
if (this.darkModeQuery && this.themeChangeHandler) { if (this.darkModeQuery && this.themeChangeHandler) {
this.darkModeQuery.removeEventListener('change', this.themeChangeHandler); this.darkModeQuery.removeEventListener("change", this.themeChangeHandler);
} }
}, },
computed: {
filteredTopics() {
const filterTerm = this.filter.toLowerCase();
const filtered = this.topics.filter((row) => {
const searchText = `${row.title} [${row.Offer.dealer_name}]`.toLowerCase();
return searchText.includes(filterTerm);
});
const sortFns = {
score: (a, b) => b.score - a.score,
views: (a, b) => b.total_views - a.total_views,
recency: (a, b) => new Date(b.last_post_time) - new Date(a.last_post_time),
};
return filtered.sort(sortFns[this.sortMethod] || sortFns.score);
},
themeIcon() {
const icons = { auto: "brightness_auto", dark: "light_mode", light: "dark_mode" };
return icons[this.currentTheme];
},
themeTitle() {
const titles = {
auto: "Theme: Auto (click for Light)",
light: "Theme: Light (click for Dark)",
dark: "Theme: Dark (click for Auto)",
};
return titles[this.currentTheme];
},
sortIcon() {
const icons = { score: "trending_up", views: "visibility", recency: "schedule" };
return icons[this.sortMethod];
},
sortTitle() {
const titles = {
score: "Sort by Score (click for Views)",
views: "Sort by Views (click for Recency)",
recency: "Sort by Recency (click for Score)",
};
return titles[this.sortMethod];
},
},
methods: { methods: {
initializeTheme() { formatDate(dateString) {
// If no saved preference, default to auto return dayjs(String(dateString)).format("YYYY-MM-DD hh:mm A");
const savedTheme = localStorage.getItem('theme');
if (!savedTheme) {
this.currentTheme = 'auto';
this.applyTheme('auto', true); // skipSave=true to avoid redundant write
} else {
this.currentTheme = savedTheme;
// Apply saved theme (skipSave=true since it's already saved)
this.applyTheme(savedTheme, true);
}
}, },
highlightText(text) {
if (!this.filter) return text;
const lowerText = text.toLowerCase();
const lowerFilter = this.filter.toLowerCase();
if (!lowerText.includes(lowerFilter)) return text;
const regex = new RegExp(this.filter, "ig");
return text.replace(regex, (match) => `<mark>${match}</mark>`);
},
initializeTheme() {
const savedTheme = localStorage.getItem("theme") || "auto";
this.currentTheme = savedTheme;
this.applyTheme(savedTheme, true);
},
setupThemeListener() { setupThemeListener() {
// Listen for system theme preference changes this.darkModeQuery = window.matchMedia("(prefers-color-scheme: dark)");
const darkModeQuery = window.matchMedia('(prefers-color-scheme: dark)');
this.mediaQueryListener = darkModeQuery; this.themeChangeHandler = (e) => {
const savedTheme = localStorage.getItem("theme");
// Use arrow function to preserve 'this' context if (savedTheme === "auto" || !savedTheme) {
const themeChangeHandler = (e) => { this.applyThemeActual(e.matches ? "dark" : "light");
// Only auto-update theme if set to 'auto'
const savedTheme = localStorage.getItem('theme');
if (savedTheme === 'auto' || !savedTheme) {
const newTheme = e.matches ? 'dark' : 'light';
console.log('System theme changed to:', newTheme);
this.applyThemeActual(newTheme);
} }
}; };
darkModeQuery.addEventListener('change', themeChangeHandler); this.darkModeQuery.addEventListener("change", this.themeChangeHandler);
// Store the handler so we can remove it later if needed
this.themeChangeHandler = themeChangeHandler;
this.darkModeQuery = darkModeQuery;
}, },
applyTheme(theme, skipSave = false) { applyTheme(theme, skipSave = false) {
this.currentTheme = theme; this.currentTheme = theme;
if (!skipSave) { if (!skipSave) {
localStorage.setItem('theme', theme); localStorage.setItem("theme", theme);
} }
// Determine actual theme to apply
let actualTheme = theme; let actualTheme = theme;
if (theme === 'auto') { if (theme === "auto") {
const prefersDark = window.matchMedia('(prefers-color-scheme: dark)').matches; actualTheme = window.matchMedia("(prefers-color-scheme: dark)").matches ? "dark" : "light";
actualTheme = prefersDark ? 'dark' : 'light';
} }
this.applyThemeActual(actualTheme); this.applyThemeActual(actualTheme);
}, },
applyThemeActual(actualTheme) {
// Update data-bs-theme attribute for CSS variables to work
document.documentElement.setAttribute('data-bs-theme', actualTheme === 'dark' ? 'dark' : 'light');
// Update HTML class for theme-based CSS selectors applyThemeActual(theme) {
if (actualTheme === 'dark') { document.documentElement.setAttribute("data-bs-theme", theme);
document.documentElement.classList.add('dark-theme'); document.documentElement.classList.toggle("dark-theme", theme === "dark");
document.documentElement.classList.remove('light-theme'); document.documentElement.classList.toggle("light-theme", theme === "light");
} else {
document.documentElement.classList.add('light-theme');
document.documentElement.classList.remove('dark-theme');
}
}, },
toggleTheme() { toggleTheme() {
// Cycle through: auto -> light -> dark -> auto const cycle = { auto: "light", light: "dark", dark: "auto" };
let newTheme; this.applyTheme(cycle[this.currentTheme]);
if (this.currentTheme === 'auto') {
newTheme = 'light';
} else if (this.currentTheme === 'light') {
newTheme = 'dark';
} else {
newTheme = 'auto';
}
this.applyTheme(newTheme);
}, },
detectMobile() { detectMobile() {
// Detect if device is mobile/tablet based on touch capability and screen size const hasTouch =
const hasTouch = () => { "ontouchstart" in window ||
return (
(typeof window !== "undefined" &&
("ontouchstart" in window ||
navigator.maxTouchPoints > 0 || navigator.maxTouchPoints > 0 ||
navigator.msMaxTouchPoints > 0)) || navigator.msMaxTouchPoints > 0;
false
);
};
const isMobileScreen = () => { const isMobileScreen = window.innerWidth <= 1024;
return window.innerWidth <= 1024; this.isMobile = hasTouch || isMobileScreen;
};
this.isMobile = hasTouch() || isMobileScreen();
window.addEventListener("resize", this.detectMobile);
}, },
handleResize() {
this.detectMobile();
},
handleKeyDown(event) { handleKeyDown(event) {
const isInput = ["INPUT", "TEXTAREA"].includes( const isInput = ["INPUT", "TEXTAREA"].includes(document.activeElement.tagName);
document.activeElement.tagName
);
if (event.key === "/" && !isInput) { if (event.key === "/" && !isInput) {
event.preventDefault(); // prevent typing `/` into whatever is focused event.preventDefault();
this.$refs.filter.focus(); this.$refs.filter.focus();
} }
}, },
createFilterRoute(params) { createFilterRoute(params) {
this.$refs.filter.blur(); this.$refs.filter.blur();
history.pushState( history.pushState({}, null, `${window.location.origin}#/filter=${encodeURIComponent(params)}`);
{},
null,
`${window.location.origin}#/filter=${encodeURIComponent(params)}`
);
}, },
fetchDeals() { fetchDeals() {
axios axios
.get("api/v1/topics") .get("api/v1/topics")
@@ -161,111 +182,21 @@ export default {
this.topics = response.data; this.topics = response.data;
}) })
.catch((err) => { .catch((err) => {
console.log(err.response); console.error("Failed to fetch deals:", err.response || err);
}); });
}, },
},
computed: {
formatDate() {
return (v) => {
const date = dayjs(String(v));
return date.format("YYYY-MM-DD hh:mm A");
};
},
filteredTopics() {
let filtered = this.topics
.filter((row) => {
const titles = (
row.title.toString() +
" [" +
row.Offer.dealer_name +
"]"
).toLowerCase();
const filterTerm = this.filter.toLowerCase();
return titles.includes(filterTerm);
});
// Sort based on selected method
if (this.sortMethod === 'score') {
return filtered.sort((a, b) => b.score - a.score);
} else if (this.sortMethod === 'views') {
return filtered.sort((a, b) => b.total_views - a.total_views);
} else {
return filtered.sort((a, b) => new Date(b.last_post_time) - new Date(a.last_post_time));
}
},
highlightMatches() {
return (v) => {
if (this.filter == "") return v;
const matchExists = v.toLowerCase().includes(this.filter.toLowerCase());
if (!matchExists) return v;
const re = new RegExp(this.filter, "ig");
return v.replace(re, (matchedText) => `<mark>${matchedText}</mark>`);
};
},
highlightDealerName() {
return (dealerName) => {
if (this.filter == "") return dealerName;
const matchExists = dealerName.toLowerCase().includes(this.filter.toLowerCase());
if (!matchExists) return dealerName;
const re = new RegExp(this.filter, "ig");
return dealerName.replace(re, (matchedText) => `<mark>${matchedText}</mark>`);
};
},
getThemeIcon() {
if (this.currentTheme === 'auto') {
return 'brightness_auto';
} else if (this.currentTheme === 'dark') {
return 'light_mode';
} else {
return 'dark_mode';
}
},
getThemeTitle() {
if (this.currentTheme === 'auto') {
return 'Theme: Auto (click for Light)';
} else if (this.currentTheme === 'light') {
return 'Theme: Light (click for Dark)';
} else {
return 'Theme: Dark (click for Auto)';
}
},
initializeSortMethod() { initializeSortMethod() {
const saved = localStorage.getItem('sortMethod'); const saved = localStorage.getItem("sortMethod");
if (saved) { if (saved) {
this.sortMethod = saved; this.sortMethod = saved;
} }
}, },
toggleSort() { toggleSort() {
// Cycle through: score -> views -> recency -> score const cycle = { score: "views", views: "recency", recency: "score" };
if (this.sortMethod === 'score') { this.sortMethod = cycle[this.sortMethod];
this.sortMethod = 'views'; localStorage.setItem("sortMethod", this.sortMethod);
} else if (this.sortMethod === 'views') {
this.sortMethod = 'recency';
} else {
this.sortMethod = 'score';
}
localStorage.setItem('sortMethod', this.sortMethod);
},
getSortIcon() {
if (this.sortMethod === 'score') {
return 'trending_up';
} else if (this.sortMethod === 'views') {
return 'visibility';
} else {
return 'schedule';
}
},
getSortTitle() {
if (this.sortMethod === 'score') {
return 'Sort by Score (click for Views)';
} else if (this.sortMethod === 'views') {
return 'Sort by Views (click for Recency)';
} else {
return 'Sort by Recency (click for Score)';
}
}, },
}, },
}; };
@@ -273,42 +204,36 @@ export default {
<template> <template>
<div id="app"> <div id="app">
<link rel="shortcut icon" type="image/png" href="/favicon.png" />
<div class="container"> <div class="container">
<div class="header"> <div class="header">
<div class="header-controls"> <div class="header-controls">
<input <input
ref="filter"
v-model="filter" v-model="filter"
type="text" type="text"
placeholder="Filter deals" placeholder="Filter deals"
ref="filter" class="search-input"
@keyup.enter="createFilterRoute(filter.toString())" @keyup.enter="createFilterRoute(filter.toString())"
@keyup.esc="$refs.filter.blur()" @keyup.esc="$refs.filter.blur()"
class="search-input"
/> />
<button @click="toggleSort" class="sort-toggle" :title="getSortTitle"> <button class="icon-button" :title="sortTitle" @click="toggleSort">
<span class="material-symbols-outlined">{{ getSortIcon }}</span> <span class="material-symbols-outlined">{{ sortIcon }}</span>
</button> </button>
<button @click="toggleTheme" class="theme-toggle" :title="getThemeTitle"> <button class="icon-button" :title="themeTitle" @click="toggleTheme">
<span class="material-symbols-outlined">{{ getThemeIcon }}</span> <span class="material-symbols-outlined">{{ themeIcon }}</span>
</button> </button>
</div> </div>
</div> </div>
<div class="cards-grid"> <div class="cards-grid">
<div <div v-for="topic in filteredTopics" :key="topic.topic_id" class="deal-card">
v-for="topic in filteredTopics"
:key="topic.topic_id"
class="deal-card"
>
<div class="card-header"> <div class="card-header">
<div class="title-with-link"> <div class="title-with-link">
<a <a
:href="`https://forums.redflagdeals.com${topic.web_path}`" :href="`https://forums.redflagdeals.com${topic.web_path}`"
target="_blank" target="_blank"
class="deal-title" class="deal-title"
@click.stop v-html="highlightText(topic.title)"
v-html="highlightMatches(topic.title)"
></a> ></a>
<a <a
v-if="topic.Offer.url" v-if="topic.Offer.url"
@@ -320,14 +245,21 @@ export default {
<span class="material-symbols-outlined">open_in_new</span> <span class="material-symbols-outlined">open_in_new</span>
</a> </a>
</div> </div>
<div class="score-bubble" :class="{ positive: topic.score > 0, negative: topic.score < 0, neutral: topic.score === 0 }"> <div
class="score-bubble"
:class="{
positive: topic.score > 0,
negative: topic.score < 0,
neutral: topic.score === 0,
}"
>
<span v-if="topic.score > 0">+{{ topic.score }}</span> <span v-if="topic.score > 0">+{{ topic.score }}</span>
<span v-else>{{ topic.score }}</span> <span v-else>{{ topic.score }}</span>
</div> </div>
</div> </div>
<div class="card-meta"> <div class="card-meta">
<span class="dealer-name" v-html="highlightDealerName(topic.Offer.dealer_name)"></span> <span class="dealer-name" v-html="highlightText(topic.Offer.dealer_name)"></span>
</div> </div>
<div class="card-details"> <div class="card-details">
@@ -342,10 +274,7 @@ export default {
</div> </div>
</div> </div>
<div class="card-timestamp"> <div class="card-timestamp">Last post: {{ formatDate(topic.last_post_time) }}</div>
Last post: {{ formatDate(topic.last_post_time) }}
</div>
</div> </div>
</div> </div>
</div> </div>

View File

@@ -2,8 +2,6 @@ import { createApp } from "vue";
import App from "./App.vue"; import App from "./App.vue";
import { createRouter, createWebHashHistory } from "vue-router"; import { createRouter, createWebHashHistory } from "vue-router";
import "./theme.css";
const routes = [ const routes = [
{ {
path: '/:pathMatch(.*)*', path: '/:pathMatch(.*)*',

View File

@@ -1,3 +1,4 @@
/* Material Symbols Icon Font */
.material-symbols-outlined { .material-symbols-outlined {
font-family: 'Material Symbols Outlined'; font-family: 'Material Symbols Outlined';
font-weight: normal; font-weight: normal;
@@ -12,51 +13,95 @@
direction: ltr; direction: ltr;
} }
/* Theme-aware CSS variables */ /* ============================================
Theme Variables
============================================ */
:root { :root {
/* Light theme (default) */ /* Light theme (default) */
--bg-primary: #dddddd; --bg-primary: #dddddd;
--bg-secondary: #e8e8e8; --bg-secondary: #e8e8e8;
--bg-input: #f5f5f5;
--bg-input-focus: #ffffff;
--text-primary: #212529; --text-primary: #212529;
--text-secondary: #6c757d; --text-secondary: #6c757d;
--border-color: #d0d0d0; --border-color: #d0d0d0;
--border-color-light: #cccccc;
--border-color-hover: #999999;
--link-color: #212529; --link-color: #212529;
--score-positive-bg: rgb(34, 139, 34);
--score-positive-text: white;
--score-negative-bg: rgb(247, 118, 142);
--score-negative-text: white;
--mark-bg: rgba(255, 193, 7, 0.3);
--shadow-light: rgba(0, 0, 0, 0.05);
--shadow-medium: rgba(0, 0, 0, 0.1);
} }
/* Dark theme */ /* Dark theme via media query */
@media (prefers-color-scheme: dark) { @media (prefers-color-scheme: dark) {
:root { :root:not([data-bs-theme="light"]):not(.light-theme) {
--bg-primary: #1a1a1a; --bg-primary: #1a1a1a;
--bg-secondary: #2a2a2a; --bg-secondary: #2a2a2a;
--bg-input: #1a1a1a;
--bg-input-focus: #2a2a2a;
--text-primary: #e0e0e0; --text-primary: #e0e0e0;
--text-secondary: #a0a0a0; --text-secondary: #a0a0a0;
--border-color: #3a3a3a; --border-color: #3a3a3a;
--link-color: #212529; --border-color-light: #555555;
--border-color-hover: #777777;
--link-color: #e0e0e0;
--score-positive-bg: rgb(158, 206, 106);
--score-positive-text: #1a1a1a;
--mark-bg: rgba(255, 193, 7, 0.4);
--shadow-light: rgba(255, 255, 255, 0.1);
} }
} }
/* Support for explicit data-bs-theme attribute (Bootstrap override) */ /* Explicit dark theme */
html[data-bs-theme="dark"] { html[data-bs-theme="dark"],
html.dark-theme {
--bg-primary: #1a1a1a; --bg-primary: #1a1a1a;
--bg-secondary: #2a2a2a; --bg-secondary: #2a2a2a;
--bg-input: #1a1a1a;
--bg-input-focus: #2a2a2a;
--text-primary: #e0e0e0; --text-primary: #e0e0e0;
--text-secondary: #a0a0a0; --text-secondary: #a0a0a0;
--border-color: #3a3a3a; --border-color: #3a3a3a;
--border-color-light: #555555;
--border-color-hover: #777777;
--link-color: #e0e0e0; --link-color: #e0e0e0;
--score-positive-bg: rgb(158, 206, 106);
--score-positive-text: #1a1a1a;
--mark-bg: rgba(255, 193, 7, 0.4);
--shadow-light: rgba(255, 255, 255, 0.1);
} }
/* Explicit light theme */
html[data-bs-theme="light"], html[data-bs-theme="light"],
html.light-theme { html.light-theme {
--bg-primary: #dddddd; --bg-primary: #dddddd;
--bg-secondary: #e8e8e8; --bg-secondary: #e8e8e8;
--bg-input: #f5f5f5;
--bg-input-focus: #ffffff;
--text-primary: #212529; --text-primary: #212529;
--text-secondary: #6c757d; --text-secondary: #6c757d;
--border-color: #d0d0d0; --border-color: #d0d0d0;
--border-color-light: #cccccc;
--border-color-hover: #999999;
--link-color: #212529; --link-color: #212529;
--score-positive-bg: rgb(34, 139, 34);
--score-positive-text: white;
--mark-bg: rgba(255, 193, 7, 0.3);
--shadow-light: rgba(0, 0, 0, 0.05);
} }
/* ============================================
Base Styles
============================================ */
html { html {
font-family: sans-serif; font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, sans-serif;
min-width: 100%; min-width: 100%;
-webkit-text-size-adjust: 100%; -webkit-text-size-adjust: 100%;
-ms-text-size-adjust: 100%; -ms-text-size-adjust: 100%;
@@ -73,22 +118,6 @@ body {
padding: 0; padding: 0;
} }
.green-score {
color: rgb(34, 139, 34) !important;
}
html[data-bs-theme="light"] .green-score {
color: rgb(34, 139, 34) !important;
}
html[data-bs-theme="dark"] .green-score {
color: rgb(158, 206, 106) !important;
}
.red-score {
color: rgb(247, 118, 142) !important;
}
a { a {
color: var(--link-color); color: var(--link-color);
transition: color 0.2s ease; transition: color 0.2s ease;
@@ -98,7 +127,9 @@ a:visited {
color: var(--link-color); color: var(--link-color);
} }
/* App styles */ /* ============================================
Layout
============================================ */
.container { .container {
max-width: 1200px; max-width: 1200px;
@@ -119,14 +150,18 @@ a:visited {
align-items: center; align-items: center;
} }
/* ============================================
Search Input
============================================ */
.search-input { .search-input {
flex: 1; flex: 1;
max-width: 500px; max-width: 500px;
padding: 10px 12px; padding: 10px 12px;
font-size: 14px; font-size: 14px;
border: 1px solid #cccccc; border: 1px solid var(--border-color-light);
border-radius: 4px; border-radius: 4px;
background-color: #f5f5f5; background-color: var(--bg-input);
color: var(--text-primary); color: var(--text-primary);
transition: all 0.2s ease; transition: all 0.2s ease;
font-family: inherit; font-family: inherit;
@@ -134,36 +169,28 @@ a:visited {
.search-input:focus { .search-input:focus {
outline: none; outline: none;
border-color: #999999; border-color: var(--border-color-hover);
background-color: #ffffff; background-color: var(--bg-input-focus);
box-shadow: 0 0 0 2px rgba(0, 0, 0, 0.05); box-shadow: 0 0 0 2px var(--shadow-light);
}
html.dark-theme .search-input {
border-color: #555555;
background-color: #1a1a1a;
color: #e0e0e0;
}
html.dark-theme .search-input:focus {
background-color: #2a2a2a;
border-color: #777777;
box-shadow: 0 0 0 2px rgba(255, 255, 255, 0.1);
} }
.search-input::placeholder { .search-input::placeholder {
color: var(--text-secondary); color: var(--text-secondary);
} }
.sort-toggle { /* ============================================
Icon Buttons (Theme & Sort Toggle)
============================================ */
.icon-button {
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
width: 40px; width: 40px;
height: 40px; height: 40px;
border: 1px solid #cccccc; border: 1px solid var(--border-color-light);
border-radius: 4px; border-radius: 4px;
background-color: #f5f5f5; background-color: var(--bg-input);
color: var(--text-primary); color: var(--text-primary);
cursor: pointer; cursor: pointer;
transition: all 0.2s ease; transition: all 0.2s ease;
@@ -172,35 +199,26 @@ html.dark-theme .search-input:focus {
flex-shrink: 0; flex-shrink: 0;
} }
.sort-toggle:hover { .icon-button:hover {
background-color: #e8e8e8; background-color: var(--bg-secondary);
border-color: #999999; border-color: var(--border-color-hover);
} }
.sort-toggle:active { .icon-button:active {
transform: scale(0.95); transform: scale(0.95);
} }
html.dark-theme .sort-toggle { /* Legacy class names for compatibility */
border-color: #555555; .sort-toggle,
background-color: #1a1a1a;
color: #e0e0e0;
}
html.dark-theme .sort-toggle:hover {
background-color: #2a2a2a;
border-color: #777777;
}
.theme-toggle { .theme-toggle {
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
width: 40px; width: 40px;
height: 40px; height: 40px;
border: 1px solid #cccccc; border: 1px solid var(--border-color-light);
border-radius: 4px; border-radius: 4px;
background-color: #f5f5f5; background-color: var(--bg-input);
color: var(--text-primary); color: var(--text-primary);
cursor: pointer; cursor: pointer;
transition: all 0.2s ease; transition: all 0.2s ease;
@@ -209,25 +227,20 @@ html.dark-theme .sort-toggle:hover {
flex-shrink: 0; flex-shrink: 0;
} }
.sort-toggle:hover,
.theme-toggle:hover { .theme-toggle:hover {
background-color: #e8e8e8; background-color: var(--bg-secondary);
border-color: #999999; border-color: var(--border-color-hover);
} }
.sort-toggle:active,
.theme-toggle:active { .theme-toggle:active {
transform: scale(0.95); transform: scale(0.95);
} }
html.dark-theme .theme-toggle { /* ============================================
border-color: #555555; Cards Grid
background-color: #1a1a1a; ============================================ */
color: #e0e0e0;
}
html.dark-theme .theme-toggle:hover {
background-color: #2a2a2a;
border-color: #777777;
}
.cards-grid { .cards-grid {
display: grid; display: grid;
@@ -248,11 +261,14 @@ html.dark-theme .theme-toggle:hover {
} }
.deal-card:hover { .deal-card:hover {
background-color: #15151515; box-shadow: 0 4px 8px var(--shadow-medium);
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1); border-color: var(--border-color-hover);
border-color: #999999;
} }
/* ============================================
Card Header
============================================ */
.card-header { .card-header {
display: flex; display: flex;
gap: 12px; gap: 12px;
@@ -286,26 +302,6 @@ html.dark-theme .theme-toggle:hover {
text-decoration: underline; text-decoration: underline;
} }
.card-meta {
display: flex;
flex-direction: column;
gap: 6px;
font-size: 13px;
margin-bottom: 12px;
}
.dealer-name {
color: var(--text-secondary);
font-weight: 500;
font-size: 13px;
}
.card-timestamp {
color: var(--text-secondary);
font-size: 12px;
margin-top: 8px;
}
.card-link { .card-link {
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
@@ -324,6 +320,10 @@ html.dark-theme .theme-toggle:hover {
font-size: 18px; font-size: 18px;
} }
/* ============================================
Score Bubble
============================================ */
.score-bubble { .score-bubble {
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
@@ -341,26 +341,14 @@ html.dark-theme .theme-toggle:hover {
} }
.score-bubble.positive { .score-bubble.positive {
background-color: rgb(34, 139, 34); background-color: var(--score-positive-bg);
color: white; color: var(--score-positive-text);
box-shadow: 0 1px 3px rgba(34, 139, 34, 0.2); box-shadow: 0 1px 3px rgba(34, 139, 34, 0.2);
} }
html.light-theme .score-bubble.positive {
background-color: rgb(34, 139, 34);
color: white;
box-shadow: 0 1px 3px rgba(34, 139, 34, 0.2);
}
html.dark-theme .score-bubble.positive {
background-color: rgb(158, 206, 106);
color: #1a1a1a;
box-shadow: 0 1px 3px rgba(158, 206, 106, 0.2);
}
.score-bubble.negative { .score-bubble.negative {
background-color: rgb(247, 118, 142); background-color: var(--score-negative-bg);
color: white; color: var(--score-negative-text);
box-shadow: 0 1px 3px rgba(247, 118, 142, 0.2); box-shadow: 0 1px 3px rgba(247, 118, 142, 0.2);
} }
@@ -370,6 +358,24 @@ html.dark-theme .score-bubble.positive {
box-shadow: 0 1px 2px rgba(0, 0, 0, 0.08); box-shadow: 0 1px 2px rgba(0, 0, 0, 0.08);
} }
/* ============================================
Card Meta & Details
============================================ */
.card-meta {
display: flex;
flex-direction: column;
gap: 6px;
font-size: 13px;
margin-bottom: 12px;
}
.dealer-name {
color: var(--text-secondary);
font-weight: 500;
font-size: 13px;
}
.card-details { .card-details {
margin-top: 12px; margin-top: 12px;
padding-top: 12px; padding-top: 12px;
@@ -398,26 +404,27 @@ html.dark-theme .score-bubble.positive {
font-weight: 500; font-weight: 500;
} }
.details-section { .card-timestamp {
margin-bottom: 12px;
}
.details-section strong {
display: block;
color: var(--text-primary);
margin-bottom: 4px;
font-size: 13px;
}
.details-section p {
margin: 0;
color: var(--text-secondary); color: var(--text-secondary);
font-size: 12px; font-size: 12px;
line-height: 1.4; margin-top: 8px;
word-wrap: break-word;
} }
/* Mobile responsive */ /* ============================================
Mark Highlighting
============================================ */
mark {
background-color: var(--mark-bg);
color: inherit;
font-weight: 600;
border-radius: 2px;
}
/* ============================================
Mobile Responsive
============================================ */
@media (max-width: 768px) { @media (max-width: 768px) {
.cards-grid { .cards-grid {
grid-template-columns: 1fr; grid-template-columns: 1fr;
@@ -431,18 +438,3 @@ html.dark-theme .score-bubble.positive {
max-width: 100%; max-width: 100%;
} }
} }
/* Mark highlighting */
mark {
background-color: rgba(255, 193, 7, 0.3);
color: inherit;
font-weight: 600;
border-radius: 2px;
}
html.dark-theme mark {
background-color: rgba(255, 193, 7, 0.4);
color: inherit;
font-weight: 600;
border-radius: 2px;
}