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"> <script lang="ts">
import CardList from '$lib/components/CardList.svelte'; import CardList from '$lib/components/CardList.svelte';
import SearchBar from '$lib/components/SearchBar.svelte';
import type { Note, Feed } from '$lib/types'; import type { Note, Feed } from '$lib/types';
import { onMount } from 'svelte'; import { onMount } from 'svelte';
import Navigation from '$lib/components/Navigation.svelte'; import Navigation from '$lib/components/Navigation.svelte';
@ -10,6 +11,7 @@
let error: string | null = null; let error: string | null = null;
let importCount = 0; let importCount = 0;
let obsidianFile: File | null = null; let obsidianFile: File | null = null;
let searchQuery = '';
// Subscribe to the notes store // Subscribe to the notes store
const unsubscribe = notes.subscribe((value) => { const unsubscribe = notes.subscribe((value) => {
@ -109,6 +111,15 @@
'updatedAt' in item '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> </script>
<Navigation /> <Navigation />
@ -183,6 +194,10 @@
</div> </div>
</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> </section>
</div> </div>

View file

@ -2,6 +2,7 @@
import { onMount } from 'svelte'; import { onMount } from 'svelte';
import { entries, feeds } from '$lib/feeds'; import { entries, feeds } from '$lib/feeds';
import CardList from '$lib/components/CardList.svelte'; import CardList from '$lib/components/CardList.svelte';
import SearchBar from '$lib/components/SearchBar.svelte';
import type { Feed, FeedEntry, Note, ReadLaterItem } from '$lib/types'; import type { Feed, FeedEntry, Note, ReadLaterItem } from '$lib/types';
// Define the CardProps interface to match what CardList expects // Define the CardProps interface to match what CardList expects
@ -23,6 +24,7 @@
let error: string | null = null; let error: string | null = null;
let showUnreadOnly = true; let showUnreadOnly = true;
let feedsMap: Map<string, Feed> = new Map(); let feedsMap: Map<string, Feed> = new Map();
let searchQuery = '';
onMount(async () => { onMount(async () => {
await loadData(); await loadData();
@ -153,6 +155,19 @@
} }
return 'No description available'; 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> </script>
<div class="container"> <div class="container">
@ -213,6 +228,13 @@
</div> </div>
</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} {#if isLoading && $entries.length === 0}
<div class="has-text-centered py-6"> <div class="has-text-centered py-6">
<span class="icon is-large"> <span class="icon is-large">
@ -221,7 +243,7 @@
</div> </div>
{:else} {:else}
<CardList <CardList
items={$entries} items={filteredEntries}
{renderCard} {renderCard}
emptyMessage={showUnreadOnly emptyMessage={showUnreadOnly
? 'No unread entries found. Try refreshing your feeds or viewing all entries.' ? 'No unread entries found. Try refreshing your feeds or viewing all entries.'

View file

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

View file

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