2026-04-17 15:07:22 +02:00
|
|
|
<script lang="ts">
|
2026-04-17 15:28:22 +02:00
|
|
|
import { onMount } from 'svelte';
|
|
|
|
|
import type { SearchHit } from '$lib/server/recipes/search-local';
|
2026-04-17 17:47:26 +02:00
|
|
|
import type { WebHit } from '$lib/server/search/searxng';
|
2026-04-17 17:58:27 +02:00
|
|
|
import { randomQuote } from '$lib/quotes';
|
2026-04-17 15:28:22 +02:00
|
|
|
|
|
|
|
|
let query = $state('');
|
2026-04-17 17:58:27 +02:00
|
|
|
let quote = $state('');
|
2026-04-17 15:28:22 +02:00
|
|
|
let recent = $state<SearchHit[]>([]);
|
2026-04-17 17:41:10 +02:00
|
|
|
let hits = $state<SearchHit[]>([]);
|
2026-04-17 17:47:26 +02:00
|
|
|
let webHits = $state<WebHit[]>([]);
|
2026-04-17 17:41:10 +02:00
|
|
|
let searching = $state(false);
|
2026-04-17 17:47:26 +02:00
|
|
|
let webSearching = $state(false);
|
|
|
|
|
let webError = $state<string | null>(null);
|
2026-04-17 17:41:10 +02:00
|
|
|
let searchedFor = $state<string | null>(null);
|
2026-04-17 15:28:22 +02:00
|
|
|
|
|
|
|
|
onMount(async () => {
|
2026-04-17 17:58:27 +02:00
|
|
|
quote = randomQuote();
|
2026-04-17 15:28:22 +02:00
|
|
|
const res = await fetch('/api/recipes/search');
|
|
|
|
|
const body = await res.json();
|
|
|
|
|
recent = body.hits;
|
|
|
|
|
});
|
2026-04-17 17:41:10 +02:00
|
|
|
|
|
|
|
|
let debounceTimer: ReturnType<typeof setTimeout> | null = null;
|
|
|
|
|
|
2026-04-17 18:04:59 +02:00
|
|
|
async function runSearch(q: string) {
|
|
|
|
|
try {
|
|
|
|
|
const res = await fetch(`/api/recipes/search?q=${encodeURIComponent(q)}`);
|
|
|
|
|
const body = await res.json();
|
|
|
|
|
if (query.trim() !== q) return;
|
|
|
|
|
hits = body.hits;
|
|
|
|
|
searchedFor = q;
|
|
|
|
|
if (body.hits.length === 0) {
|
|
|
|
|
webSearching = true;
|
|
|
|
|
try {
|
|
|
|
|
const wres = await fetch(`/api/recipes/search/web?q=${encodeURIComponent(q)}`);
|
|
|
|
|
if (query.trim() !== q) return;
|
|
|
|
|
if (!wres.ok) {
|
|
|
|
|
const err = await wres.json().catch(() => ({}));
|
|
|
|
|
webError = err.message ?? `HTTP ${wres.status}`;
|
|
|
|
|
} else {
|
|
|
|
|
const wbody = await wres.json();
|
|
|
|
|
webHits = wbody.hits;
|
|
|
|
|
}
|
|
|
|
|
} finally {
|
|
|
|
|
if (query.trim() === q) webSearching = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} finally {
|
|
|
|
|
if (query.trim() === q) searching = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-17 17:41:10 +02:00
|
|
|
$effect(() => {
|
|
|
|
|
const q = query.trim();
|
|
|
|
|
if (debounceTimer) clearTimeout(debounceTimer);
|
|
|
|
|
if (q.length <= 3) {
|
|
|
|
|
hits = [];
|
2026-04-17 17:47:26 +02:00
|
|
|
webHits = [];
|
2026-04-17 17:41:10 +02:00
|
|
|
searchedFor = null;
|
|
|
|
|
searching = false;
|
2026-04-17 17:47:26 +02:00
|
|
|
webSearching = false;
|
|
|
|
|
webError = null;
|
2026-04-17 17:41:10 +02:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
searching = true;
|
2026-04-17 17:47:26 +02:00
|
|
|
webHits = [];
|
|
|
|
|
webSearching = false;
|
|
|
|
|
webError = null;
|
2026-04-17 18:04:59 +02:00
|
|
|
debounceTimer = setTimeout(() => {
|
|
|
|
|
void runSearch(q);
|
2026-04-17 17:41:10 +02:00
|
|
|
}, 300);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
function submit(e: SubmitEvent) {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
const q = query.trim();
|
2026-04-17 18:04:59 +02:00
|
|
|
if (q.length <= 3) return;
|
|
|
|
|
if (debounceTimer) clearTimeout(debounceTimer);
|
|
|
|
|
searching = true;
|
|
|
|
|
void runSearch(q);
|
2026-04-17 17:41:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const activeSearch = $derived(query.trim().length > 3);
|
2026-04-17 15:07:22 +02:00
|
|
|
</script>
|
|
|
|
|
|
2026-04-17 15:28:22 +02:00
|
|
|
<section class="hero">
|
2026-04-17 15:07:22 +02:00
|
|
|
<h1>Kochwas</h1>
|
2026-04-17 17:58:27 +02:00
|
|
|
<p class="tagline" aria-live="polite">{quote || '\u00a0'}</p>
|
2026-04-17 17:41:10 +02:00
|
|
|
<form onsubmit={submit}>
|
2026-04-17 15:07:22 +02:00
|
|
|
<input
|
|
|
|
|
type="search"
|
|
|
|
|
bind:value={query}
|
|
|
|
|
placeholder="Rezept suchen…"
|
|
|
|
|
autocomplete="off"
|
|
|
|
|
inputmode="search"
|
2026-04-17 15:28:22 +02:00
|
|
|
aria-label="Suchbegriff"
|
2026-04-17 15:07:22 +02:00
|
|
|
/>
|
|
|
|
|
</form>
|
2026-04-17 15:28:22 +02:00
|
|
|
</section>
|
|
|
|
|
|
2026-04-17 17:41:10 +02:00
|
|
|
{#if activeSearch}
|
|
|
|
|
<section class="results">
|
|
|
|
|
{#if searching && hits.length === 0}
|
|
|
|
|
<p class="muted">Suche läuft …</p>
|
|
|
|
|
{:else if hits.length > 0}
|
|
|
|
|
<ul class="cards">
|
|
|
|
|
{#each hits as r (r.id)}
|
|
|
|
|
<li>
|
|
|
|
|
<a href={`/recipes/${r.id}`} class="card">
|
|
|
|
|
{#if r.image_path}
|
|
|
|
|
<img src={`/images/${r.image_path}`} alt="" loading="lazy" />
|
|
|
|
|
{:else}
|
|
|
|
|
<div class="placeholder">🥘</div>
|
|
|
|
|
{/if}
|
|
|
|
|
<div class="card-body">
|
|
|
|
|
<div class="title">{r.title}</div>
|
|
|
|
|
{#if r.source_domain}
|
|
|
|
|
<div class="domain">{r.source_domain}</div>
|
|
|
|
|
{/if}
|
|
|
|
|
</div>
|
|
|
|
|
</a>
|
|
|
|
|
</li>
|
|
|
|
|
{/each}
|
|
|
|
|
</ul>
|
2026-04-17 17:47:26 +02:00
|
|
|
<a class="web-more" href={`/search/web?q=${encodeURIComponent(query.trim())}`}>
|
|
|
|
|
🌐 Im Internet weitersuchen
|
|
|
|
|
</a>
|
2026-04-17 17:41:10 +02:00
|
|
|
{:else if searchedFor === query.trim()}
|
2026-04-17 17:47:26 +02:00
|
|
|
<p class="muted no-local-msg">Keine lokalen Rezepte für „{searchedFor}" — Ergebnisse aus dem Internet:</p>
|
|
|
|
|
{#if webSearching}
|
|
|
|
|
<p class="muted">Suche im Internet läuft …</p>
|
|
|
|
|
{:else if webError}
|
|
|
|
|
<p class="error">Internet-Suche zurzeit nicht möglich: {webError}</p>
|
|
|
|
|
{:else if webHits.length > 0}
|
|
|
|
|
<ul class="cards">
|
|
|
|
|
{#each webHits as w (w.url)}
|
|
|
|
|
<li>
|
|
|
|
|
<a class="card" href={`/preview?url=${encodeURIComponent(w.url)}`}>
|
|
|
|
|
{#if w.thumbnail}
|
|
|
|
|
<img src={w.thumbnail} alt="" loading="lazy" />
|
|
|
|
|
{:else}
|
|
|
|
|
<div class="placeholder">🍽️</div>
|
|
|
|
|
{/if}
|
|
|
|
|
<div class="card-body">
|
|
|
|
|
<div class="title">{w.title}</div>
|
|
|
|
|
<div class="domain">{w.domain}</div>
|
|
|
|
|
</div>
|
|
|
|
|
</a>
|
|
|
|
|
</li>
|
|
|
|
|
{/each}
|
|
|
|
|
</ul>
|
|
|
|
|
{:else}
|
|
|
|
|
<p class="muted">Auch im Internet nichts gefunden.</p>
|
|
|
|
|
{/if}
|
2026-04-17 17:41:10 +02:00
|
|
|
{/if}
|
|
|
|
|
</section>
|
|
|
|
|
{:else if recent.length > 0}
|
2026-04-17 15:28:22 +02:00
|
|
|
<section class="recent">
|
|
|
|
|
<h2>Zuletzt hinzugefügt</h2>
|
|
|
|
|
<ul class="cards">
|
|
|
|
|
{#each recent as r (r.id)}
|
|
|
|
|
<li>
|
|
|
|
|
<a href={`/recipes/${r.id}`} class="card">
|
|
|
|
|
{#if r.image_path}
|
|
|
|
|
<img src={`/images/${r.image_path}`} alt="" loading="lazy" />
|
|
|
|
|
{:else}
|
|
|
|
|
<div class="placeholder">🥘</div>
|
|
|
|
|
{/if}
|
|
|
|
|
<div class="card-body">
|
|
|
|
|
<div class="title">{r.title}</div>
|
|
|
|
|
{#if r.source_domain}
|
|
|
|
|
<div class="domain">{r.source_domain}</div>
|
|
|
|
|
{/if}
|
|
|
|
|
</div>
|
|
|
|
|
</a>
|
|
|
|
|
</li>
|
|
|
|
|
{/each}
|
|
|
|
|
</ul>
|
|
|
|
|
</section>
|
|
|
|
|
{/if}
|
2026-04-17 15:07:22 +02:00
|
|
|
|
|
|
|
|
<style>
|
2026-04-17 15:28:22 +02:00
|
|
|
.hero {
|
2026-04-17 15:07:22 +02:00
|
|
|
text-align: center;
|
2026-04-17 15:28:22 +02:00
|
|
|
padding: 3rem 0 1.5rem;
|
|
|
|
|
}
|
|
|
|
|
.hero h1 {
|
|
|
|
|
font-size: clamp(2.2rem, 8vw, 3.5rem);
|
2026-04-17 17:58:27 +02:00
|
|
|
margin: 0 0 0.5rem;
|
2026-04-17 15:28:22 +02:00
|
|
|
color: #2b6a3d;
|
|
|
|
|
letter-spacing: -0.02em;
|
2026-04-17 15:07:22 +02:00
|
|
|
}
|
2026-04-17 17:58:27 +02:00
|
|
|
.tagline {
|
|
|
|
|
margin: 0 auto 1.5rem;
|
|
|
|
|
max-width: 36rem;
|
|
|
|
|
color: #6a7670;
|
|
|
|
|
font-style: italic;
|
|
|
|
|
font-size: 1rem;
|
|
|
|
|
line-height: 1.35;
|
|
|
|
|
min-height: 1.4rem;
|
|
|
|
|
}
|
2026-04-17 15:07:22 +02:00
|
|
|
form {
|
|
|
|
|
display: flex;
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
}
|
2026-04-17 15:28:22 +02:00
|
|
|
input[type='search'] {
|
2026-04-17 15:07:22 +02:00
|
|
|
flex: 1;
|
2026-04-17 15:28:22 +02:00
|
|
|
padding: 0.9rem 1rem;
|
2026-04-17 15:07:22 +02:00
|
|
|
font-size: 1.1rem;
|
2026-04-17 15:28:22 +02:00
|
|
|
border: 1px solid #cfd9d1;
|
|
|
|
|
border-radius: 10px;
|
|
|
|
|
background: white;
|
|
|
|
|
min-height: 48px;
|
|
|
|
|
}
|
|
|
|
|
input[type='search']:focus {
|
|
|
|
|
outline: 2px solid #2b6a3d;
|
|
|
|
|
outline-offset: 1px;
|
2026-04-17 15:07:22 +02:00
|
|
|
}
|
2026-04-17 17:41:10 +02:00
|
|
|
.results,
|
2026-04-17 15:28:22 +02:00
|
|
|
.recent {
|
2026-04-17 17:41:10 +02:00
|
|
|
margin-top: 1.5rem;
|
2026-04-17 15:07:22 +02:00
|
|
|
}
|
2026-04-17 15:28:22 +02:00
|
|
|
.recent h2 {
|
|
|
|
|
font-size: 1.05rem;
|
|
|
|
|
color: #444;
|
|
|
|
|
margin: 0 0 0.75rem;
|
|
|
|
|
}
|
2026-04-17 17:41:10 +02:00
|
|
|
.muted {
|
|
|
|
|
color: #888;
|
|
|
|
|
text-align: center;
|
2026-04-17 17:47:26 +02:00
|
|
|
padding: 1rem 0;
|
|
|
|
|
}
|
|
|
|
|
.no-local-msg {
|
|
|
|
|
font-size: 0.95rem;
|
|
|
|
|
padding: 0.25rem 0 1rem;
|
|
|
|
|
}
|
|
|
|
|
.error {
|
|
|
|
|
color: #c53030;
|
|
|
|
|
text-align: center;
|
|
|
|
|
padding: 1rem 0;
|
2026-04-17 17:41:10 +02:00
|
|
|
}
|
2026-04-17 15:28:22 +02:00
|
|
|
.cards {
|
|
|
|
|
list-style: none;
|
|
|
|
|
padding: 0;
|
|
|
|
|
margin: 0;
|
|
|
|
|
display: grid;
|
|
|
|
|
grid-template-columns: repeat(auto-fill, minmax(220px, 1fr));
|
|
|
|
|
gap: 0.75rem;
|
|
|
|
|
}
|
|
|
|
|
.card {
|
|
|
|
|
display: block;
|
|
|
|
|
background: white;
|
|
|
|
|
border: 1px solid #e4eae7;
|
|
|
|
|
border-radius: 14px;
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
text-decoration: none;
|
|
|
|
|
color: inherit;
|
|
|
|
|
transition: transform 0.1s;
|
|
|
|
|
}
|
|
|
|
|
.card:active {
|
|
|
|
|
transform: scale(0.98);
|
|
|
|
|
}
|
|
|
|
|
.card img,
|
|
|
|
|
.placeholder {
|
|
|
|
|
width: 100%;
|
|
|
|
|
aspect-ratio: 4 / 3;
|
|
|
|
|
object-fit: cover;
|
|
|
|
|
background: #eef3ef;
|
|
|
|
|
display: grid;
|
|
|
|
|
place-items: center;
|
|
|
|
|
font-size: 2rem;
|
|
|
|
|
}
|
|
|
|
|
.card-body {
|
|
|
|
|
padding: 0.6rem 0.75rem 0.75rem;
|
|
|
|
|
}
|
|
|
|
|
.title {
|
|
|
|
|
font-weight: 600;
|
|
|
|
|
font-size: 0.95rem;
|
|
|
|
|
line-height: 1.25;
|
|
|
|
|
}
|
|
|
|
|
.domain {
|
|
|
|
|
font-size: 0.8rem;
|
|
|
|
|
color: #888;
|
|
|
|
|
margin-top: 0.25rem;
|
|
|
|
|
}
|
2026-04-17 17:47:26 +02:00
|
|
|
.web-more {
|
2026-04-17 17:41:10 +02:00
|
|
|
display: inline-block;
|
2026-04-17 17:47:26 +02:00
|
|
|
margin-top: 1rem;
|
|
|
|
|
padding: 0.7rem 1.1rem;
|
|
|
|
|
border: 1px solid #b7d6c2;
|
2026-04-17 17:41:10 +02:00
|
|
|
border-radius: 10px;
|
2026-04-17 17:47:26 +02:00
|
|
|
text-decoration: none;
|
|
|
|
|
color: #2b6a3d;
|
|
|
|
|
background: #eaf4ed;
|
|
|
|
|
font-size: 0.95rem;
|
|
|
|
|
}
|
|
|
|
|
.web-more:hover {
|
|
|
|
|
background: #d8e8df;
|
2026-04-17 17:41:10 +02:00
|
|
|
}
|
2026-04-17 15:07:22 +02:00
|
|
|
</style>
|