feat(frontend): add search functionality to all sections

This commit is contained in:
Nicola Zangrandi 2025-02-28 17:12:48 +01:00
parent 6f189a6ee2
commit 7fd939a988
Signed by: wasp
GPG key ID: 43C1470D890F23ED
5 changed files with 112 additions and 4 deletions

View file

@ -0,0 +1,34 @@
<script lang="ts">
/**
* A reusable search bar component that emits search events
*/
export let placeholder = 'Search...';
export let value = '';
export let debounceTime = 300;
let timeoutId: ReturnType<typeof setTimeout> | null = null;
function handleInput(event: Event) {
const target = event.target as HTMLInputElement;
const searchValue = target.value;
// Clear any existing timeout
if (timeoutId) {
clearTimeout(timeoutId);
}
// Set a new timeout to debounce the search
timeoutId = setTimeout(() => {
value = searchValue;
}, debounceTime);
}
</script>
<div class="field">
<div class="control has-icons-left">
<input class="input" type="text" {placeholder} {value} on:input={handleInput} />
<span class="icon is-small is-left">
<i class="fas fa-search"></i>
</span>
</div>
</div>

View file

@ -1,5 +1,6 @@
<script lang="ts">
import CardList from '$lib/components/CardList.svelte';
import SearchBar from '$lib/components/SearchBar.svelte';
import type { Note, Feed } from '$lib/types';
import { onMount } from 'svelte';
import Navigation from '$lib/components/Navigation.svelte';
@ -10,6 +11,7 @@
let error: string | null = null;
let importCount = 0;
let obsidianFile: File | null = null;
let searchQuery = '';
// Subscribe to the notes store
const unsubscribe = notes.subscribe((value) => {
@ -109,6 +111,15 @@
'updatedAt' in item
);
}
// Filter notes based on search query
$: filteredNotes = searchQuery
? notesList.filter(
(note) =>
note.title.toLowerCase().includes(searchQuery.toLowerCase()) ||
note.content.toLowerCase().includes(searchQuery.toLowerCase())
)
: notesList;
</script>
<Navigation />
@ -183,6 +194,10 @@
</div>
</div>
<CardList items={notesList} {renderCard} emptyMessage="No notes found." />
<div class="mb-4">
<SearchBar placeholder="Search notes by title or content..." bind:value={searchQuery} />
</div>
<CardList items={filteredNotes} {renderCard} emptyMessage="No notes found." />
</section>
</div>

View file

@ -2,6 +2,7 @@
import { onMount } from 'svelte';
import { entries, feeds } from '$lib/feeds';
import CardList from '$lib/components/CardList.svelte';
import SearchBar from '$lib/components/SearchBar.svelte';
import type { Feed, FeedEntry, Note, ReadLaterItem } from '$lib/types';
// Define the CardProps interface to match what CardList expects
@ -23,6 +24,7 @@
let error: string | null = null;
let showUnreadOnly = true;
let feedsMap: Map<string, Feed> = new Map();
let searchQuery = '';
onMount(async () => {
await loadData();
@ -153,6 +155,19 @@
}
return 'No description available';
}
// Filter entries based on search query
$: filteredEntries = searchQuery
? $entries.filter((entry) => {
const feedName = feedsMap.get(entry.feedId)?.title || '';
return (
entry.title.toLowerCase().includes(searchQuery.toLowerCase()) ||
entry.summary?.toLowerCase().includes(searchQuery.toLowerCase()) ||
feedName.toLowerCase().includes(searchQuery.toLowerCase()) ||
entry.content?.toLowerCase().includes(searchQuery.toLowerCase())
);
})
: $entries;
</script>
<div class="container">
@ -213,6 +228,13 @@
</div>
</div>
<div class="mb-4">
<SearchBar
placeholder="Search entries by title, content, or feed name..."
bind:value={searchQuery}
/>
</div>
{#if isLoading && $entries.length === 0}
<div class="has-text-centered py-6">
<span class="icon is-large">
@ -221,7 +243,7 @@
</div>
{:else}
<CardList
items={$entries}
items={filteredEntries}
{renderCard}
emptyMessage={showUnreadOnly
? 'No unread entries found. Try refreshing your feeds or viewing all entries.'

View file

@ -2,6 +2,7 @@
import { onMount } from 'svelte';
import { feeds } from '$lib/feeds';
import CardList from '$lib/components/CardList.svelte';
import SearchBar from '$lib/components/SearchBar.svelte';
import type { Feed, Note, ReadLaterItem } from '$lib/types';
// Define the CardProps interface to match what CardList expects
@ -26,6 +27,7 @@
let isRefreshing = false;
let error: string | null = null;
let importCount = 0;
let searchQuery = '';
onMount(async () => {
await loadFeeds();
@ -156,6 +158,16 @@
timestamp: item.createdAt
};
}
// Filter feeds based on search query
$: filteredFeeds = searchQuery
? feedsList.filter(
(feed) =>
feed.title.toLowerCase().includes(searchQuery.toLowerCase()) ||
(feed.description || '').toLowerCase().includes(searchQuery.toLowerCase()) ||
feed.url.toLowerCase().includes(searchQuery.toLowerCase())
)
: feedsList;
</script>
<div class="container">
@ -266,6 +278,13 @@
</form>
</div>
<div class="mb-4">
<SearchBar
placeholder="Search feeds by title, description, or URL..."
bind:value={searchQuery}
/>
</div>
{#if isLoading && feedsList.length === 0}
<div class="has-text-centered py-6">
<span class="icon is-large">
@ -274,7 +293,7 @@
</div>
{:else}
<CardList
items={feedsList}
items={filteredFeeds}
{renderCard}
emptyMessage="No feeds found. Add your first feed above."
/>

View file

@ -2,6 +2,7 @@
import { onMount } from 'svelte';
import { readlist } from '$lib/readlist';
import CardList from '$lib/components/CardList.svelte';
import SearchBar from '$lib/components/SearchBar.svelte';
import type { ReadLaterItem } from '$lib/types';
let url = '';
@ -9,6 +10,7 @@
let error: string | null = null;
let showArchived = false;
let archivingItems: Record<string, boolean> = {};
let searchQuery = '';
onMount(() => {
loadItems();
@ -59,7 +61,16 @@
}
function getFilteredItems() {
return $readlist;
if (!searchQuery) {
return $readlist;
}
return $readlist.filter(
(item) =>
item.title.toLowerCase().includes(searchQuery.toLowerCase()) ||
item.description.toLowerCase().includes(searchQuery.toLowerCase()) ||
item.url.toLowerCase().includes(searchQuery.toLowerCase())
);
}
</script>
@ -107,6 +118,13 @@
</div>
</div>
<div class="mb-4">
<SearchBar
placeholder="Search saved links by title, description, or URL..."
bind:value={searchQuery}
/>
</div>
<CardList
items={getFilteredItems()}
renderCard={(item) => {