mirror of
https://github.com/davegallant/rfd-fyi.git
synced 2026-03-03 09:36:35 +00:00
Switch to a card-based view
This commit is contained in:
565
src/App.vue
565
src/App.vue
@@ -16,10 +16,6 @@ export default {
|
|||||||
filter: window.location.href.split("filter=")[1] || "",
|
filter: window.location.href.split("filter=")[1] || "",
|
||||||
sortColumn: this.sortColumn,
|
sortColumn: this.sortColumn,
|
||||||
topics: [],
|
topics: [],
|
||||||
hoveredTopicId: null,
|
|
||||||
tooltipData: {},
|
|
||||||
loadingTooltip: {},
|
|
||||||
tooltipPosition: { x: 0, y: 0 },
|
|
||||||
isMobile: false,
|
isMobile: false,
|
||||||
currentTheme: 'dark',
|
currentTheme: 'dark',
|
||||||
mediaQueryListener: null,
|
mediaQueryListener: null,
|
||||||
@@ -81,13 +77,22 @@ export default {
|
|||||||
this.darkModeQuery = darkModeQuery;
|
this.darkModeQuery = darkModeQuery;
|
||||||
},
|
},
|
||||||
applyTheme(theme) {
|
applyTheme(theme) {
|
||||||
// Apply theme using Vuetify's theme API
|
// Apply theme using Vuetify's theme API (using .value for reactive ref)
|
||||||
this.$vuetify.theme.global.name = theme;
|
this.$vuetify.theme.global.name.value = theme;
|
||||||
this.currentTheme = theme;
|
this.currentTheme = theme;
|
||||||
localStorage.setItem('vuetify-theme', theme);
|
localStorage.setItem('vuetify-theme', theme);
|
||||||
|
|
||||||
// Also update data-bs-theme for any custom CSS that uses it
|
// Update data-bs-theme attribute for CSS variables to work
|
||||||
document.documentElement.setAttribute('data-bs-theme', theme === 'dark' ? 'dark' : 'light');
|
document.documentElement.setAttribute('data-bs-theme', theme === 'dark' ? 'dark' : 'light');
|
||||||
|
|
||||||
|
// Update HTML class for theme-based CSS selectors
|
||||||
|
if (theme === 'dark') {
|
||||||
|
document.documentElement.classList.add('dark-theme');
|
||||||
|
document.documentElement.classList.remove('light-theme');
|
||||||
|
} else {
|
||||||
|
document.documentElement.classList.add('light-theme');
|
||||||
|
document.documentElement.classList.remove('dark-theme');
|
||||||
|
}
|
||||||
},
|
},
|
||||||
toggleTheme() {
|
toggleTheme() {
|
||||||
const newTheme = this.currentTheme === 'dark' ? 'light' : 'dark';
|
const newTheme = this.currentTheme === 'dark' ? 'light' : 'dark';
|
||||||
@@ -121,63 +126,7 @@ export default {
|
|||||||
this.$refs.filter.focus();
|
this.$refs.filter.focus();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
handleTitleHover(topic, event) {
|
|
||||||
// Don't load tooltips on mobile devices
|
|
||||||
if (this.isMobile) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this.hoveredTopicId = topic.topic_id;
|
|
||||||
this.tooltipPosition = {
|
|
||||||
x: event.clientX,
|
|
||||||
y: event.clientY,
|
|
||||||
};
|
|
||||||
this.loadTopicDetails(topic.topic_id);
|
|
||||||
},
|
|
||||||
handleTitleLeave() {
|
|
||||||
if (this.isMobile) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this.hoveredTopicId = null;
|
|
||||||
},
|
|
||||||
handleMouseMove(event) {
|
|
||||||
if (this.isMobile) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (this.hoveredTopicId !== null) {
|
|
||||||
this.tooltipPosition = {
|
|
||||||
x: event.clientX,
|
|
||||||
y: event.clientY,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
},
|
|
||||||
loadTopicDetails(topicId) {
|
|
||||||
if (!topicId) {
|
|
||||||
console.warn("Topic ID is undefined");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.tooltipData[topicId]) {
|
|
||||||
return; // Already loaded
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.loadingTooltip[topicId]) {
|
|
||||||
return; // Already loading
|
|
||||||
}
|
|
||||||
|
|
||||||
this.loadingTooltip[topicId] = true;
|
|
||||||
|
|
||||||
axios
|
|
||||||
.get(`api/v1/topics/${topicId}`)
|
|
||||||
.then((response) => {
|
|
||||||
this.tooltipData[topicId] = response.data;
|
|
||||||
})
|
|
||||||
.catch((err) => {
|
|
||||||
console.log("Error loading topic details:", err);
|
|
||||||
})
|
|
||||||
.finally(() => {
|
|
||||||
this.loadingTooltip[topicId] = false;
|
|
||||||
});
|
|
||||||
},
|
|
||||||
createFilterRoute(params) {
|
createFilterRoute(params) {
|
||||||
this.$refs.filter.blur();
|
this.$refs.filter.blur();
|
||||||
history.pushState(
|
history.pushState(
|
||||||
@@ -205,16 +154,18 @@ export default {
|
|||||||
};
|
};
|
||||||
},
|
},
|
||||||
filteredTopics() {
|
filteredTopics() {
|
||||||
return this.topics.filter((row) => {
|
return this.topics
|
||||||
const titles = (
|
.filter((row) => {
|
||||||
row.title.toString() +
|
const titles = (
|
||||||
" [" +
|
row.title.toString() +
|
||||||
row.Offer.dealer_name +
|
" [" +
|
||||||
"]"
|
row.Offer.dealer_name +
|
||||||
).toLowerCase();
|
"]"
|
||||||
const filterTerm = this.filter.toLowerCase();
|
).toLowerCase();
|
||||||
return titles.includes(filterTerm);
|
const filterTerm = this.filter.toLowerCase();
|
||||||
});
|
return titles.includes(filterTerm);
|
||||||
|
})
|
||||||
|
.sort((a, b) => b.score - a.score); // Always sort by score descending
|
||||||
},
|
},
|
||||||
highlightMatches() {
|
highlightMatches() {
|
||||||
return (v) => {
|
return (v) => {
|
||||||
@@ -226,46 +177,14 @@ export default {
|
|||||||
return v.replace(re, (matchedText) => `<mark>${matchedText}</mark>`);
|
return v.replace(re, (matchedText) => `<mark>${matchedText}</mark>`);
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
visibleHeaders() {
|
highlightDealerName() {
|
||||||
const baseHeaders = [
|
return (dealerName) => {
|
||||||
{ title: "Deal", value: "title", align: "center" },
|
if (this.filter == "") return dealerName;
|
||||||
{ title: "Score", value: "score", align: "center" },
|
const matchExists = dealerName.toLowerCase().includes(this.filter.toLowerCase());
|
||||||
];
|
if (!matchExists) return dealerName;
|
||||||
|
|
||||||
// Only show Last Post column on desktop
|
const re = new RegExp(this.filter, "ig");
|
||||||
if (!this.isMobile) {
|
return dealerName.replace(re, (matchedText) => `<mark>${matchedText}</mark>`);
|
||||||
baseHeaders.push({
|
|
||||||
title: "Last Post",
|
|
||||||
value: "last_post_time",
|
|
||||||
align: "center",
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return baseHeaders;
|
|
||||||
},
|
|
||||||
tooltipStyle() {
|
|
||||||
if (this.hoveredTopicId === null || !this.tooltipData[this.hoveredTopicId]) {
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
let top = this.tooltipPosition.y + 10;
|
|
||||||
let left = this.tooltipPosition.x + 10;
|
|
||||||
const tooltipWidth = 420;
|
|
||||||
|
|
||||||
// Check if tooltip would go off right side of screen
|
|
||||||
if (left + tooltipWidth > window.innerWidth) {
|
|
||||||
// Position to the left of cursor instead
|
|
||||||
left = Math.max(10, this.tooltipPosition.x - tooltipWidth - 10);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Keep tooltip within vertical bounds, allowing scrolling of content
|
|
||||||
top = Math.max(10, Math.min(top, window.innerHeight - 100));
|
|
||||||
|
|
||||||
return {
|
|
||||||
position: 'fixed',
|
|
||||||
left: Math.max(10, left) + 'px',
|
|
||||||
top: top + 'px',
|
|
||||||
zIndex: 9999,
|
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -280,195 +199,281 @@ const sortBy = ref([{ key: "score", order: "desc" }]);
|
|||||||
<v-app>
|
<v-app>
|
||||||
<v-main>
|
<v-main>
|
||||||
<link rel="shortcut icon" type="image/png" href="/favicon.png" />
|
<link rel="shortcut icon" type="image/png" href="/favicon.png" />
|
||||||
<body>
|
<div class="container">
|
||||||
<v-text-field
|
<div class="header">
|
||||||
v-model="filter"
|
<v-text-field
|
||||||
label="Filter"
|
v-model="filter"
|
||||||
ref="filter"
|
label="Filter deals"
|
||||||
@keyup.enter="createFilterRoute(filter.toString())"
|
ref="filter"
|
||||||
@keyup.esc="$refs.filter.blur()"
|
@keyup.enter="createFilterRoute(filter.toString())"
|
||||||
hide-details="true"
|
@keyup.esc="$refs.filter.blur()"
|
||||||
/>
|
hide-details="true"
|
||||||
<v-data-table
|
class="search-input"
|
||||||
:headers="visibleHeaders"
|
/>
|
||||||
:items="filteredTopics"
|
</div>
|
||||||
:sort-by="sortBy"
|
|
||||||
:items-per-page="50"
|
|
||||||
>
|
|
||||||
<template #item.title="{ item }">
|
|
||||||
<a
|
|
||||||
:href="`https://forums.redflagdeals.com${item.web_path}`"
|
|
||||||
target="_blank"
|
|
||||||
@mouseenter="handleTitleHover(item, $event)"
|
|
||||||
@mouseleave="handleTitleLeave"
|
|
||||||
@mousemove="handleMouseMove"
|
|
||||||
v-html="
|
|
||||||
highlightMatches(
|
|
||||||
item.title
|
|
||||||
)
|
|
||||||
"
|
|
||||||
></a>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<template #item.score="{ item }">
|
<div class="cards-grid">
|
||||||
<span v-if="item.score > 0" class="green-score"
|
<div
|
||||||
>+{{ item.score }}</span
|
v-for="topic in filteredTopics"
|
||||||
>
|
:key="topic.topic_id"
|
||||||
<span v-else-if="item.score < 0" class="red-score">{{
|
class="deal-card"
|
||||||
item.score
|
>
|
||||||
}}</span>
|
<div class="card-header">
|
||||||
<span v-else>{{ item.score }}</span>
|
<a
|
||||||
</template>
|
:href="`https://forums.redflagdeals.com${topic.web_path}`"
|
||||||
|
target="_blank"
|
||||||
<template #item.last_post_time="{ item }">
|
class="deal-title"
|
||||||
{{ formatDate(item.last_post_time) }}
|
@click.stop
|
||||||
</template>
|
v-html="highlightMatches(topic.title)"
|
||||||
|
></a>
|
||||||
<template #loading>
|
<div class="score-bubble" :class="{ positive: topic.score > 0, negative: topic.score < 0, neutral: topic.score === 0 }">
|
||||||
<v-progress-linear indeterminate color="grey" />
|
<span v-if="topic.score > 0">+{{ topic.score }}</span>
|
||||||
</template>
|
<span v-else>{{ topic.score }}</span>
|
||||||
</v-data-table>
|
|
||||||
|
|
||||||
<!-- Tooltip for deal details -->
|
|
||||||
<div
|
|
||||||
v-if="hoveredTopicId !== null && tooltipData[hoveredTopicId]"
|
|
||||||
class="deal-tooltip"
|
|
||||||
:style="tooltipStyle"
|
|
||||||
>
|
|
||||||
<div class="tooltip-content">
|
|
||||||
<div class="tooltip-stats">
|
|
||||||
<span class="stat-item">
|
|
||||||
<span class="material-symbols-outlined">visibility</span>
|
|
||||||
{{ tooltipData[hoveredTopicId].topic.total_views }} views
|
|
||||||
</span>
|
|
||||||
<span class="stat-item">
|
|
||||||
<span class="material-symbols-outlined">chat</span>
|
|
||||||
{{ tooltipData[hoveredTopicId].topic.total_replies }} replies
|
|
||||||
</span>
|
|
||||||
</div>
|
</div>
|
||||||
<div v-if="tooltipData[hoveredTopicId].description" class="tooltip-description">
|
|
||||||
<strong>Description:</strong>
|
|
||||||
{{ tooltipData[hoveredTopicId].description }}
|
|
||||||
</div>
|
</div>
|
||||||
<div class="tooltip-dealer">
|
|
||||||
{{ tooltipData[hoveredTopicId].topic.Offer.dealer_name }}
|
<div class="card-meta">
|
||||||
|
<span class="dealer-name" v-html="highlightDealerName(topic.Offer.dealer_name)"></span>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="tooltipData[hoveredTopicId].first_post" class="tooltip-first-post">
|
|
||||||
<strong>First Post:</strong>
|
<!-- Card details -->
|
||||||
{{ tooltipData[hoveredTopicId].first_post }}
|
<div class="card-details">
|
||||||
</div>
|
<div class="details-stats">
|
||||||
<div class="tooltip-times">
|
<div class="stat">
|
||||||
<div>Posted: {{ formatDate(tooltipData[hoveredTopicId].topic.post_time) }}</div>
|
<span class="material-symbols-outlined">visibility</span>
|
||||||
<div>Last Post: {{ formatDate(tooltipData[hoveredTopicId].topic.last_post_time) }}</div>
|
<span class="stat-value">{{ topic.total_views }} views</span>
|
||||||
|
</div>
|
||||||
|
<div class="stat">
|
||||||
|
<span class="material-symbols-outlined">chat</span>
|
||||||
|
<span class="stat-value">{{ topic.total_replies }} replies</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="card-timestamp">
|
||||||
|
Last post: {{ formatDate(topic.last_post_time) }}
|
||||||
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</body>
|
</div>
|
||||||
</v-main>
|
</v-main>
|
||||||
</v-app>
|
</v-app>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
#app {
|
.container {
|
||||||
font-family: Avenir, Helvetica, Arial, sans-serif;
|
max-width: 1200px;
|
||||||
-webkit-font-smoothing: antialiased;
|
margin: 0 auto;
|
||||||
-moz-osx-font-smoothing: grayscale;
|
padding: 20px;
|
||||||
text-align: center;
|
background-color: var(--bg-primary);
|
||||||
color: #2c3e50;
|
color: var(--text-primary);
|
||||||
|
min-height: 100vh;
|
||||||
}
|
}
|
||||||
|
|
||||||
.fixed-bottom {
|
.header {
|
||||||
background: #ffc;
|
margin-bottom: 30px;
|
||||||
color: black;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.deal-tooltip {
|
.search-input {
|
||||||
pointer-events: none;
|
width: 100%;
|
||||||
max-width: 400px;
|
max-width: 500px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip-content {
|
.cards-grid {
|
||||||
background: var(--tooltip-bg);
|
display: grid;
|
||||||
border: 2px solid var(--tooltip-border);
|
grid-template-columns: repeat(auto-fill, minmax(350px, 1fr));
|
||||||
border-radius: 8px;
|
gap: 20px;
|
||||||
|
margin-top: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.deal-card {
|
||||||
|
background-color: var(--bg-secondary);
|
||||||
|
border: 1.5px solid #aaaaaa;
|
||||||
|
border-radius: 12px;
|
||||||
padding: 16px;
|
padding: 16px;
|
||||||
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
|
display: flex;
|
||||||
font-size: 13px;
|
flex-direction: column;
|
||||||
color: var(--text-primary);
|
transition: all 0.2s ease;
|
||||||
text-align: left;
|
cursor: pointer;
|
||||||
transition: background-color 0.3s ease, color 0.3s ease, border-color 0.3s ease;
|
min-height: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip-header {
|
.deal-card:hover {
|
||||||
font-weight: bold;
|
transform: translateY(-4px);
|
||||||
font-size: 14px;
|
box-shadow: 0 8px 16px rgba(0, 0, 0, 0.2);
|
||||||
color: var(--text-primary);
|
border-color: #888888;
|
||||||
margin-bottom: 8px;
|
|
||||||
white-space: normal;
|
|
||||||
word-wrap: break-word;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip-dealer {
|
.card-header {
|
||||||
font-size: 12px;
|
|
||||||
color: var(--text-secondary);
|
|
||||||
margin-bottom: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.tooltip-stats {
|
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 12px;
|
gap: 12px;
|
||||||
margin-bottom: 8px;
|
align-items: flex-start;
|
||||||
font-size: 12px;
|
margin-bottom: 12px;
|
||||||
color: var(--text-secondary);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.stat-item {
|
.deal-title {
|
||||||
|
color: var(--link-color);
|
||||||
|
text-decoration: none;
|
||||||
|
font-weight: 500;
|
||||||
|
font-size: 15px;
|
||||||
|
line-height: 1.4;
|
||||||
|
flex: 1;
|
||||||
|
transition: color 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.deal-title:visited {
|
||||||
|
color: var(--link-visited);
|
||||||
|
}
|
||||||
|
|
||||||
|
.deal-title:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card-meta {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
flex-direction: column;
|
||||||
gap: 4px;
|
gap: 6px;
|
||||||
|
font-size: 13px;
|
||||||
|
margin-bottom: 12px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.stat-item .material-symbols-outlined {
|
.dealer-name {
|
||||||
font-size: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.tooltip-description {
|
|
||||||
margin-bottom: 8px;
|
|
||||||
padding: 8px;
|
|
||||||
background: var(--bg-secondary);
|
|
||||||
border-left: 2px solid var(--tooltip-border);
|
|
||||||
border-radius: 2px;
|
|
||||||
font-size: 12px;
|
|
||||||
white-space: normal;
|
|
||||||
word-wrap: break-word;
|
|
||||||
max-height: 60px;
|
|
||||||
overflow-y: auto;
|
|
||||||
color: var(--text-primary);
|
|
||||||
}
|
|
||||||
|
|
||||||
.tooltip-first-post {
|
|
||||||
margin-bottom: 8px;
|
|
||||||
padding: 8px;
|
|
||||||
background: var(--bg-secondary);
|
|
||||||
border-left: 2px solid var(--tooltip-border);
|
|
||||||
border-radius: 2px;
|
|
||||||
font-size: 12px;
|
|
||||||
white-space: normal;
|
|
||||||
word-wrap: break-word;
|
|
||||||
max-height: 60px;
|
|
||||||
overflow-y: auto;
|
|
||||||
color: var(--text-primary);
|
|
||||||
}
|
|
||||||
|
|
||||||
.tooltip-times {
|
|
||||||
font-size: 11px;
|
|
||||||
color: var(--text-secondary);
|
color: var(--text-secondary);
|
||||||
border-top: 1px solid var(--border-color);
|
font-weight: 500;
|
||||||
padding-top: 8px;
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card-timestamp {
|
||||||
|
color: var(--text-secondary);
|
||||||
|
font-size: 12px;
|
||||||
margin-top: 8px;
|
margin-top: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Filter input styling */
|
.score-bubble {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
min-width: 50px;
|
||||||
|
height: 40px;
|
||||||
|
border-radius: 6px;
|
||||||
|
font-weight: 700;
|
||||||
|
font-size: 12px;
|
||||||
|
flex-shrink: 0;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
padding: 0 8px;
|
||||||
|
box-shadow: 0 1px 2px rgba(0, 0, 0, 0.08);
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.score-bubble.positive {
|
||||||
|
background-color: rgb(34, 139, 34);
|
||||||
|
color: white;
|
||||||
|
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 {
|
||||||
|
background-color: rgb(247, 118, 142);
|
||||||
|
color: white;
|
||||||
|
box-shadow: 0 1px 3px rgba(247, 118, 142, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.score-bubble.neutral {
|
||||||
|
background-color: var(--text-secondary);
|
||||||
|
color: var(--bg-primary);
|
||||||
|
box-shadow: 0 1px 2px rgba(0, 0, 0, 0.08);
|
||||||
|
}
|
||||||
|
|
||||||
|
.card-details {
|
||||||
|
margin-top: 12px;
|
||||||
|
padding-top: 12px;
|
||||||
|
border-top: 1px solid var(--border-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
.details-stats {
|
||||||
|
display: flex;
|
||||||
|
gap: 16px;
|
||||||
|
margin-bottom: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.stat {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 6px;
|
||||||
|
font-size: 13px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.stat .material-symbols-outlined {
|
||||||
|
font-size: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.stat-value {
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.details-section {
|
||||||
|
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);
|
||||||
|
font-size: 12px;
|
||||||
|
line-height: 1.4;
|
||||||
|
word-wrap: break-word;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Mobile responsive */
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.cards-grid {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
|
||||||
|
.container {
|
||||||
|
padding: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-input {
|
||||||
|
max-width: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Mark highlighting */
|
||||||
|
:deep(mark) {
|
||||||
|
background-color: rgba(255, 193, 7, 0.3);
|
||||||
|
color: inherit;
|
||||||
|
font-weight: 600;
|
||||||
|
border-radius: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark-theme :deep(mark) {
|
||||||
|
background-color: rgba(255, 193, 7, 0.4);
|
||||||
|
color: inherit;
|
||||||
|
font-weight: 600;
|
||||||
|
border-radius: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Vuetify overrides */
|
||||||
:deep(.v-text-field) {
|
:deep(.v-text-field) {
|
||||||
--v-field-border-color: #cccccc;
|
--v-field-border-color: #cccccc;
|
||||||
}
|
}
|
||||||
@@ -488,4 +493,30 @@ html[data-bs-theme="light"] :deep(.v-field--focused .v-field__input) {
|
|||||||
html[data-bs-theme="dark"] :deep(.v-text-field) {
|
html[data-bs-theme="dark"] :deep(.v-text-field) {
|
||||||
--v-field-border-color: #555555;
|
--v-field-border-color: #555555;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
html.light-theme :deep(.v-text-field) {
|
||||||
|
--v-field-border-color: #cccccc;
|
||||||
|
}
|
||||||
|
|
||||||
|
html.light-theme :deep(.v-field__input) {
|
||||||
|
background-color: #d0d0d0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
html.light-theme :deep(.v-field--focused .v-field__input) {
|
||||||
|
background-color: #e8e8e8 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark-theme :deep(.v-text-field) {
|
||||||
|
--v-field-border-color: #555555;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ensure v-app and v-main use theme colors */
|
||||||
|
:deep(.v-app) {
|
||||||
|
background-color: var(--bg-primary) !important;
|
||||||
|
color: var(--text-primary) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
:deep(.v-main) {
|
||||||
|
background-color: var(--bg-primary) !important;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
176
src/theme.css
176
src/theme.css
@@ -15,16 +15,12 @@
|
|||||||
/* Theme-aware CSS variables */
|
/* Theme-aware CSS variables */
|
||||||
:root {
|
:root {
|
||||||
/* Light theme (default) */
|
/* Light theme (default) */
|
||||||
--bg-primary: #ffffff;
|
--bg-primary: #dddddd;
|
||||||
--bg-secondary: #f5f5f5;
|
--bg-secondary: #eeeeee;
|
||||||
--text-primary: #212529;
|
--text-primary: #212529;
|
||||||
--text-secondary: #6c757d;
|
--text-secondary: #6c757d;
|
||||||
--border-color: #dee2e6;
|
--border-color: #d0d0d0;
|
||||||
--tooltip-bg: #f8f9fa;
|
--link-color: #212529;
|
||||||
--tooltip-border: #dee2e6;
|
|
||||||
--tooltip-text: #212529;
|
|
||||||
--link-color: #0d6efd;
|
|
||||||
--link-visited: #990000;
|
|
||||||
--footer-bg: #f8f9fa;
|
--footer-bg: #f8f9fa;
|
||||||
--footer-text: #212529;
|
--footer-text: #212529;
|
||||||
}
|
}
|
||||||
@@ -37,11 +33,7 @@
|
|||||||
--text-primary: #e0e0e0;
|
--text-primary: #e0e0e0;
|
||||||
--text-secondary: #a0a0a0;
|
--text-secondary: #a0a0a0;
|
||||||
--border-color: #3a3a3a;
|
--border-color: #3a3a3a;
|
||||||
--tooltip-bg: #2a2a2a;
|
--link-color: #212529;
|
||||||
--tooltip-border: #444444;
|
|
||||||
--tooltip-text: #e0e0e0;
|
|
||||||
--link-color: #5b9cf5;
|
|
||||||
--link-visited: #990000;
|
|
||||||
--footer-bg: #1a1a1a;
|
--footer-bg: #1a1a1a;
|
||||||
--footer-text: #e0e0e0;
|
--footer-text: #e0e0e0;
|
||||||
}
|
}
|
||||||
@@ -54,42 +46,39 @@ html[data-bs-theme="dark"] {
|
|||||||
--text-primary: #e0e0e0;
|
--text-primary: #e0e0e0;
|
||||||
--text-secondary: #a0a0a0;
|
--text-secondary: #a0a0a0;
|
||||||
--border-color: #3a3a3a;
|
--border-color: #3a3a3a;
|
||||||
--tooltip-bg: #2a2a2a;
|
--link-color: ##e0e0e0;
|
||||||
--tooltip-border: #444444;
|
|
||||||
--tooltip-text: #e0e0e0;
|
|
||||||
--link-color: #e8e8e8;
|
|
||||||
--link-visited: #990000;
|
|
||||||
--footer-bg: #1a1a1a;
|
--footer-bg: #1a1a1a;
|
||||||
--footer-text: #e0e0e0;
|
--footer-text: #e0e0e0;
|
||||||
}
|
}
|
||||||
|
|
||||||
html[data-bs-theme="light"] {
|
html[data-bs-theme="light"],
|
||||||
--bg-primary: #ffffff;
|
html.light-theme {
|
||||||
--bg-secondary: #f5f5f5;
|
--bg-primary: #dddddd;
|
||||||
|
--bg-secondary: #eeeeee;
|
||||||
--text-primary: #212529;
|
--text-primary: #212529;
|
||||||
--text-secondary: #6c757d;
|
--text-secondary: #6c757d;
|
||||||
--border-color: #dee2e6;
|
--border-color: #d0d0d0;
|
||||||
--tooltip-bg: #f8f9fa;
|
--link-color: #212529;
|
||||||
--tooltip-border: #dee2e6;
|
|
||||||
--tooltip-text: #212529;
|
|
||||||
--link-color: #333333;
|
|
||||||
--link-visited: #990000;
|
|
||||||
--footer-bg: #f8f9fa;
|
--footer-bg: #f8f9fa;
|
||||||
--footer-text: #212529;
|
--footer-text: #212529;
|
||||||
}
|
}
|
||||||
|
|
||||||
body {
|
|
||||||
max-width: 100%;
|
|
||||||
background-color: var(--bg-primary);
|
|
||||||
color: var(--text-primary);
|
|
||||||
transition: background-color 0.3s ease, color 0.3s ease;
|
|
||||||
}
|
|
||||||
|
|
||||||
html {
|
html {
|
||||||
font-family: sans-serif;
|
font-family: 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%;
|
||||||
|
background-color: var(--bg-primary);
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
max-width: 100%;
|
||||||
|
background-color: var(--bg-primary);
|
||||||
|
color: var(--text-primary);
|
||||||
|
transition: background-color 0.3s ease, color 0.3s ease;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
footer {
|
footer {
|
||||||
@@ -130,10 +119,8 @@ a {
|
|||||||
transition: color 0.2s ease;
|
transition: color 0.2s ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
a:visited {
|
a:visited {
|
||||||
color: var(--link-visited);
|
color: var(--link-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (min-width: 769px) {
|
@media (min-width: 769px) {
|
||||||
@@ -144,89 +131,70 @@ a:visited {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Tooltip theme support */
|
/* Vuetify table dark theme overrides */
|
||||||
.deal-tooltip {
|
html.dark-theme .v-data-table {
|
||||||
pointer-events: none;
|
background-color: #1a1a1a !important;
|
||||||
max-width: 400px;
|
color: #e0e0e0 !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip-content {
|
html.dark-theme .v-data-table__tr {
|
||||||
background: var(--tooltip-bg);
|
background-color: #1a1a1a !important;
|
||||||
border: 2px solid var(--tooltip-border);
|
color: #e0e0e0 !important;
|
||||||
border-radius: 8px;
|
|
||||||
padding: 16px;
|
|
||||||
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
|
|
||||||
font-size: 13px;
|
|
||||||
color: var(--tooltip-text);
|
|
||||||
text-align: left;
|
|
||||||
transition: background-color 0.3s ease, color 0.3s ease, border-color 0.3s ease;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip-header {
|
html.dark-theme .v-data-table__td {
|
||||||
font-weight: bold;
|
background-color: #1a1a1a !important;
|
||||||
font-size: 14px;
|
color: #e0e0e0 !important;
|
||||||
color: var(--text-primary);
|
border-color: #888888 !important;
|
||||||
margin-bottom: 8px;
|
|
||||||
white-space: normal;
|
|
||||||
word-wrap: break-word;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip-dealer {
|
html.dark-theme .v-data-table-header {
|
||||||
font-size: 12px;
|
background-color: #2a2a2a !important;
|
||||||
color: var(--text-secondary);
|
|
||||||
margin-bottom: 8px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip-stats {
|
html.dark-theme .v-data-table-header th {
|
||||||
display: flex;
|
background-color: #2a2a2a !important;
|
||||||
gap: 12px;
|
color: #e0e0e0 !important;
|
||||||
margin-bottom: 8px;
|
border-color: #888888 !important;
|
||||||
font-size: 12px;
|
|
||||||
color: var(--text-secondary);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.stat-item {
|
html.dark-theme .v-data-footer {
|
||||||
display: flex;
|
background-color: #2a2a2a !important;
|
||||||
align-items: center;
|
color: #e0e0e0 !important;
|
||||||
gap: 4px;
|
border-color: #888888 !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.stat-item .material-symbols-outlined {
|
html.dark-theme .v-data-table__divider {
|
||||||
font-size: 16px;
|
border-color: #888888 !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip-description {
|
html.dark-theme .v-table__wrapper {
|
||||||
margin-bottom: 8px;
|
background-color: #1a1a1a !important;
|
||||||
padding: 8px;
|
|
||||||
background: var(--bg-secondary);
|
|
||||||
border-left: 2px solid var(--tooltip-border);
|
|
||||||
border-radius: 2px;
|
|
||||||
font-size: 12px;
|
|
||||||
white-space: normal;
|
|
||||||
word-wrap: break-word;
|
|
||||||
max-height: 60px;
|
|
||||||
overflow-y: auto;
|
|
||||||
color: var(--text-primary);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip-first-post {
|
/* Light theme table overrides */
|
||||||
margin-bottom: 8px;
|
html.light-theme .v-data-table {
|
||||||
padding: 8px;
|
background-color: #ffffff !important;
|
||||||
background: var(--bg-secondary);
|
color: #212529 !important;
|
||||||
border-left: 2px solid var(--tooltip-border);
|
|
||||||
border-radius: 2px;
|
|
||||||
font-size: 12px;
|
|
||||||
white-space: normal;
|
|
||||||
word-wrap: break-word;
|
|
||||||
max-height: 60px;
|
|
||||||
overflow-y: auto;
|
|
||||||
color: var(--text-primary);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip-times {
|
html.light-theme .v-data-table__tr {
|
||||||
font-size: 11px;
|
background-color: #ffffff !important;
|
||||||
color: var(--text-secondary);
|
color: #212529 !important;
|
||||||
border-top: 1px solid var(--border-color);
|
}
|
||||||
padding-top: 8px;
|
|
||||||
margin-top: 8px;
|
html.light-theme .v-data-table__td {
|
||||||
|
background-color: #ffffff !important;
|
||||||
|
color: #212529 !important;
|
||||||
|
border-color: #dee2e6 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
html.light-theme .v-data-table-header {
|
||||||
|
background-color: #f5f5f5 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
html.light-theme .v-data-table-header th {
|
||||||
|
background-color: #f5f5f5 !important;
|
||||||
|
color: #212529 !important;
|
||||||
|
border-color: #dee2e6 !important;
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user