Files
sLogos/src/components/Header.svelte

453 lines
12 KiB
Svelte

<script>
import { onMount } from "svelte";
import { location } from "svelte-spa-router";
import Filter from "./Filter.svelte";
import ThemeSwitcher from "./ThemeSwitcher.svelte";
import ListViewSwitcher from "./ListViewSwitcher.svelte";
import SearchBar from "./SearchBar.svelte";
import InlineSvg from "./InlineSvg.svelte";
import AchievementButton from "./AchievementButton.svelte";
import { collections } from "../collections.js";
export let displayLogos = [];
export let allLogos = []; // Add this to get total count
export let theme = "system";
export let setTheme = () => {};
export let viewMode;
export let setGridView;
export let setListView;
export let setCompactView;
export let searchQuery;
export let setSearchQuery;
export let allTags = [];
export let selectedTags = [];
export let selectedBrands = [];
export let selectedVariants = [];
export let tagDropdownOpen = false;
export let toggleDropdown = () => console.log("toggleDropdown not provided");
export let addTag = () => console.log("addTag not provided");
export let removeTag = () => console.log("removeTag not provided");
export let addBrand = () => console.log("addBrand not provided");
export let removeBrand = () => console.log("removeBrand not provided");
export let addVariant = () => console.log("addVariant not provided");
export let removeVariant = () => console.log("removeVariant not provided");
export let getTagObj = (tag) => ({ text: tag });
export let compactMode = false;
export let setCompactMode = () => {};
export let collection = "logos";
export let setCollection = () => {};
// Quiz-only data
export let score = null;
export let gameStats = null;
export let achievementCount = { unlocked: 0, total: 0 };
export let onAchievementClick = () => {};
let dropdownOpen = false;
// Check if we're in game mode
$: isGameMode = $location && $location.startsWith('/game');
$: isQuizPage = $location && $location.startsWith('/game/flags');
function handleTitleClick() {
dropdownOpen = !dropdownOpen;
}
function handleGameClick() {
if (isGameMode) {
// Go back to main page if we're in game mode
window.location.hash = '#/';
} else {
// Go to games page if we're not in game mode
window.location.hash = '#/game';
}
}
function handleCollectionSelect(name) {
setCollection(name);
dropdownOpen = false;
}
$: currentCollectionObj = collections.find(c => c.name === collection);
$: currentLabel = (currentCollectionObj?.title || currentCollectionObj?.label || "Logo Gallery").replace(/s$/, "");
$: displayLabel = isGameMode ? "Flag Quiz" : currentLabel;
</script>
<header class="main-header">
<div class="header-row">
<div class="header-left">
<div class="header-title">
<div class="header-icon">
<img src="favicon.svg" alt="Logo Gallery icon" />
</div>
<button class="collection-title-btn" on:click={handleTitleClick} aria-haspopup="listbox" aria-expanded={dropdownOpen}>
{displayLabel} <span class="triangle"></span>
</button>
{#if dropdownOpen}
<ul class="collection-dropdown" role="listbox">
{#each collections as c}
<li
class:active={c.name === collection}
role="option"
aria-selected={c.name === collection}
tabindex="0"
on:click={() => handleCollectionSelect(c.name)}
on:keydown={(e) => {
if (e.key === "Enter" || e.key === " ") {
e.preventDefault();
handleCollectionSelect(c.name);
}
}}
>{c.title} Gallery</li>
{/each}
</ul>
{/if}
</div>
{#if !isGameMode}
<span class="logo-count">
{#if (searchQuery && searchQuery.trim() !== "") || selectedTags.length > 0 || selectedBrands.length > 0 || selectedVariants.length > 0 || compactMode}
{displayLogos ? displayLogos.length : 0} of {allLogos ? allLogos.length : 0} images
displayed
{:else}
{allLogos ? allLogos.length : 0} images in gallery
{/if}
</span>
{/if}
</div>
<a href="#/game" class="game-button" class:active={isGameMode} title={isGameMode ? "Back to Main" : "Quiz Games"} on:click|preventDefault={handleGameClick}>
<span class="icon"><InlineSvg path="/icons/gamepad.svg" alt="Games" /></span>
</a>
<ThemeSwitcher {theme} {setTheme} />
</div>
{#if !isGameMode || isQuizPage}
<div class="header-row header-controls" class:quiz-mode={isQuizPage}>
{#if isQuizPage}
<div class="quiz-header-stats">
<div class="qh-block">
<span class="qh-label">Current:</span>
<span class="qh-value">{score ? `${score.correct}/${score.total}` : '0/0'} {score && score.skipped > 0 ? `(` : ''}{#if score && score.skipped > 0}<span class="quiz-icon quiz-skip"><InlineSvg path="/icons/skip-square.svg" alt="Skipped" /></span> {score.skipped}){/if}</span>
</div>
<div class="qh-block">
<span class="qh-label">All Time:</span>
<span class="qh-value"><span class="quiz-icon quiz-ok"><InlineSvg path="/icons/ok-square.svg" alt="Correct" /></span> {gameStats?.correct || 0} <span class="quiz-icon quiz-fail"><InlineSvg path="/icons/fail-square.svg" alt="Wrong" /></span> {gameStats?.wrong || 0} {gameStats?.skipped > 0 ? `` : ''}{#if gameStats?.skipped > 0}<span class="quiz-icon quiz-skip"><InlineSvg path="/icons/skip-square.svg" alt="Skipped" /></span> {gameStats.skipped}{/if}</span>
</div>
<div class="qh-block achievement-block">
<AchievementButton
{achievementCount}
on:click={onAchievementClick}
/>
</div>
</div>
{:else}
<div class="header-controls-left">
<SearchBar {searchQuery} {setSearchQuery} />
<Filter
{allLogos}
{allTags}
{selectedTags}
{selectedBrands}
{selectedVariants}
{tagDropdownOpen}
{toggleDropdown}
{addTag}
{removeTag}
{addBrand}
{removeBrand}
{addVariant}
{removeVariant}
{getTagObj}
{compactMode}
{setCompactMode}
collection={currentCollectionObj}
/>
</div>
<div class="header-controls-right">
<ListViewSwitcher
{viewMode}
{setGridView}
{setListView}
{setCompactView}
/>
</div>
{/if}
</div>
{/if}
</header>
<style>
.main-header {
padding: 0.5rem 1rem;
background: var(--color-card);
border-bottom: 1px solid var(--color-border);
border-radius: 0;
margin-bottom: 1rem;
}
.header-row {
display: grid;
grid-template-columns: 1fr auto 1fr;
grid-template-areas: "left center right";
align-items: center;
gap: 1rem;
}
.header-left {
display: flex;
align-items: center;
gap: 1rem;
grid-area: left;
justify-self: start;
}
.header-title {
display: flex;
align-items: center;
gap: 0.5em;
}
.logo-count {
font-family: system-ui, Arial, sans-serif;
font-size: 0.6rem;
font-weight: normal;
color: var(--color-text);
opacity: 0.7;
align-self: center;
white-space: nowrap;
}
.game-button {
display: inline-flex;
align-items: center;
justify-content: center;
width: 40px;
height: 40px;
border-radius: 8px;
background: var(--color-bg-secondary);
border: 1px solid var(--color-border);
text-decoration: none;
font-size: 1.2rem;
transition: all 0.3s ease;
cursor: pointer;
color: var(--color-text);
grid-area: center;
justify-self: center;
}
.game-button:hover {
background: var(--color-primary);
border-color: var(--color-primary);
}
.game-button.active {
background: var(--color-primary);
border-color: var(--color-primary);
color: white;
}
.game-button.active:hover {
background: var(--color-primary-dark, #0056b3);
border-color: var(--color-primary-dark, #0056b3);
}
.game-button .icon {
width: 22px;
height: 22px;
display: inline-flex;
}
.main-header :global(.theme-switcher) {
grid-area: right;
justify-self: end;
margin-left: 0;
}
.header-title {
display: flex;
align-items: center;
gap: 0.5em;
grid-area: left;
justify-self: start;
}
.header-icon {
width: 28px;
height: 28px;
border-radius: 4px;
}
.collection-title-btn {
background: none;
border: none;
font: inherit;
color: inherit;
font-size: 1.5rem;
font-weight: bold;
display: flex;
align-items: center;
gap: 0.3em;
cursor: pointer;
padding: 0;
margin: 0 1rem 0 0;
position: relative;
transition: color 0.15s;
}
.collection-title-btn:hover,
.collection-title-btn:focus {
color: var(--color-primary, #0070f3);
}
.triangle {
font-size: 1rem;
margin-left: 0.2em;
transition: transform 0.2s;
}
.collection-title-btn[aria-expanded="true"] .triangle {
transform: rotate(180deg);
}
.collection-dropdown {
position: absolute;
top: 2.2em;
left: 0;
background: var(--color-card);
border: 1px solid var(--color-border);
border-radius: 6px;
box-shadow: 0 2px 8px rgba(0,0,0,0.08);
z-index: 10;
min-width: 180px;
padding: 0.3em 0;
margin: 0;
list-style: none;
}
.collection-dropdown li {
padding: 0.5em 1.2em;
cursor: pointer;
font-size: 1.1rem;
color: var(--color-text);
transition: background 0.13s, color 0.13s;
}
.collection-dropdown li.active,
.collection-dropdown li:hover {
background: var(--color-primary, #0070f3);
color: #fff;
}
.header-controls {
display: flex;
align-items: center;
justify-content: space-between;
gap: 1rem;
margin: 1rem 0;
}
.header-controls.quiz-mode {
justify-content: center;
}
.header-controls-left {
display: flex;
align-items: center;
gap: 1rem;
}
.header-controls-right {
display: flex;
align-items: center;
gap: 1rem;
}
.quiz-header-stats {
display: flex;
align-items: center;
gap: 0.75rem;
flex-wrap: wrap;
justify-content: center;
margin: 0 auto;
}
.qh-block {
background: var(--color-bg-secondary);
border: 1px solid var(--color-border);
border-radius: 8px;
padding: 0.35rem 0.75rem;
font-size: 0.9rem;
color: var(--color-text);
}
.qh-label {
color: var(--color-text-secondary);
margin-right: 0.35rem;
}
.qh-value {
font-weight: 600;
}
.quiz-icon {
display: inline-flex;
width: 20px;
height: 20px;
vertical-align: middle;
margin-right: 0.25rem;
}
.quiz-icon.quiz-ok {
color: #22c55e; /* green */
}
.quiz-icon.quiz-fail {
color: #ef4444; /* red */
}
.quiz-icon.quiz-skip {
color: #6b7280; /* gray */
}
@media (max-width: 700px) {
.header-row {
grid-template-columns: 1fr auto auto;
grid-template-rows: auto auto;
grid-template-areas:
"left center right"
"count count count";
gap: 0.5rem;
}
.header-left {
flex-direction: column;
align-items: flex-start;
gap: 0.5rem;
}
.logo-count {
grid-area: count;
justify-self: start;
} .header-controls {
display: grid;
grid-template-columns: 1fr auto;
grid-template-rows: auto auto;
gap: 1rem;
margin: 1rem 0;
}
.header-controls-left {
grid-column: 1;
grid-row: 1 / -1;
display: grid;
grid-template-columns: 1fr;
grid-template-rows: auto auto;
gap: 1rem;
}
.header-controls-right {
grid-column: 2;
grid-row: 1;
}
/* Prevent header title from wrapping and reduce size on small screens */
.collection-title-btn {
font-size: 1.1rem;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
max-width: 50vw;
}
}
</style>