feat(frontend): add search functionality to all sections
This commit is contained in:
parent
6f189a6ee2
commit
7fd939a988
5 changed files with 112 additions and 4 deletions
34
frontend/src/lib/components/SearchBar.svelte
Normal file
34
frontend/src/lib/components/SearchBar.svelte
Normal 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>
|
|
@ -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>
|
||||
|
|
|
@ -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.'
|
||||
|
|
|
@ -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."
|
||||
/>
|
||||
|
|
|
@ -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,8 +61,17 @@
|
|||
}
|
||||
|
||||
function getFilteredItems() {
|
||||
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>
|
||||
|
||||
<div class="container">
|
||||
|
@ -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) => {
|
||||
|
|
Loading…
Add table
Reference in a new issue