273 lines
6.6 KiB
Svelte
273 lines
6.6 KiB
Svelte
<script lang="ts">
|
|
import { onMount } from 'svelte';
|
|
import { entries, feeds } from '$lib/feeds';
|
|
import CardList from '$lib/components/CardList.svelte';
|
|
import type { Feed, FeedEntry, Note, ReadLaterItem } from '$lib/types';
|
|
import { page } from '$app/stores';
|
|
|
|
// Define the CardProps interface to match what CardList expects
|
|
interface CardProps {
|
|
title: string;
|
|
description: string;
|
|
timestamp: Date;
|
|
actions?: Array<{
|
|
icon?: string;
|
|
label: string;
|
|
href?: string;
|
|
target?: string;
|
|
onClick?: () => void;
|
|
}>;
|
|
}
|
|
|
|
let feed: Feed | null = null;
|
|
let isLoading = true;
|
|
let isRefreshing = false;
|
|
let showUnreadOnly = false;
|
|
let error: string | null = null;
|
|
|
|
onMount(async () => {
|
|
await loadData();
|
|
});
|
|
|
|
async function loadData() {
|
|
isLoading = true;
|
|
error = null;
|
|
try {
|
|
const feedId = $page.params.id;
|
|
|
|
// Load feed info
|
|
const feedsList = await feeds.load();
|
|
feed = feedsList.find((f) => f.id === feedId) || null;
|
|
|
|
if (!feed) {
|
|
error = 'Feed not found';
|
|
return;
|
|
}
|
|
|
|
// Load entries for this feed
|
|
await loadEntries();
|
|
} catch (e) {
|
|
error = e instanceof Error ? e.message : 'Failed to load feed';
|
|
} finally {
|
|
isLoading = false;
|
|
}
|
|
}
|
|
|
|
async function loadEntries() {
|
|
if (!feed) return;
|
|
|
|
isLoading = true;
|
|
try {
|
|
await entries.loadEntries(feed.id, showUnreadOnly);
|
|
} catch (e) {
|
|
error = e instanceof Error ? e.message : 'Failed to load entries';
|
|
} finally {
|
|
isLoading = false;
|
|
}
|
|
}
|
|
|
|
async function handleRefreshFeed() {
|
|
if (!feed) return;
|
|
|
|
isRefreshing = true;
|
|
error = null;
|
|
try {
|
|
await feeds.refresh(feed.id);
|
|
await loadEntries();
|
|
} catch (e) {
|
|
error = e instanceof Error ? e.message : 'Failed to refresh feed';
|
|
} finally {
|
|
isRefreshing = false;
|
|
}
|
|
}
|
|
|
|
async function handleMarkAllAsRead() {
|
|
if (!feed) return;
|
|
|
|
error = null;
|
|
try {
|
|
await entries.markAllAsRead(feed.id);
|
|
// Reload entries if showing unread only
|
|
if (showUnreadOnly) {
|
|
await loadEntries();
|
|
}
|
|
} catch (e) {
|
|
error = e instanceof Error ? e.message : 'Failed to mark all entries as read';
|
|
}
|
|
}
|
|
|
|
function toggleUnreadFilter() {
|
|
showUnreadOnly = !showUnreadOnly;
|
|
loadEntries();
|
|
}
|
|
|
|
// This function adapts any item type to the format expected by CardList
|
|
function renderCard(item: Feed | Note | ReadLaterItem): CardProps {
|
|
// We know we're only passing FeedEntry objects to this function in this component
|
|
if ('feedId' in item) {
|
|
const entry = item as FeedEntry;
|
|
|
|
return {
|
|
title: entry.title,
|
|
description: entry.summary || 'No summary available',
|
|
timestamp: entry.published,
|
|
actions: [
|
|
{
|
|
icon: 'fas fa-book-reader',
|
|
label: 'Read',
|
|
href: `/feeds/entries/${entry.id}`
|
|
},
|
|
{
|
|
icon: 'fas fa-external-link-alt',
|
|
label: 'Original',
|
|
href: entry.url,
|
|
target: '_blank'
|
|
},
|
|
{
|
|
icon: entry.readAt ? 'fas fa-check' : 'fas fa-bookmark',
|
|
label: entry.readAt ? 'Read' : 'Mark Read',
|
|
onClick: async () => {
|
|
if (!entry.readAt) {
|
|
try {
|
|
await entries.markAsRead(entry.id);
|
|
if (showUnreadOnly) {
|
|
await loadEntries();
|
|
}
|
|
} catch (e) {
|
|
error = e instanceof Error ? e.message : 'Failed to mark as read';
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
};
|
|
}
|
|
|
|
// For other item types (Feed, Note, ReadLaterItem) - this won't be used in this component
|
|
// but is needed to satisfy the type requirements
|
|
return {
|
|
title: item.title,
|
|
description: getDescription(item),
|
|
timestamp: item.createdAt
|
|
};
|
|
}
|
|
|
|
// Helper function to safely extract description from different item types
|
|
function getDescription(item: Feed | Note | ReadLaterItem): string {
|
|
if ('description' in item && item.description) {
|
|
return item.description;
|
|
} else if ('content' in item) {
|
|
return item.content;
|
|
}
|
|
return 'No description available';
|
|
}
|
|
</script>
|
|
|
|
<div class="container">
|
|
<div class="level mt-4">
|
|
<div class="level-left">
|
|
<div class="level-item">
|
|
<nav class="breadcrumb" aria-label="breadcrumbs">
|
|
<ul>
|
|
<li><a href="/feeds">Feeds</a></li>
|
|
<li><a href="/feeds/list">Manage Feeds</a></li>
|
|
<li class="is-active"><a href="#" aria-current="page">{feed?.title || 'Feed'}</a></li>
|
|
</ul>
|
|
</nav>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{#if error}
|
|
<div class="notification is-danger">
|
|
<p>{error}</p>
|
|
</div>
|
|
{/if}
|
|
|
|
{#if feed}
|
|
<div class="box">
|
|
<div class="level">
|
|
<div class="level-left">
|
|
<div class="level-item">
|
|
<h1 class="title">{feed.title}</h1>
|
|
</div>
|
|
</div>
|
|
<div class="level-right">
|
|
<div class="level-item">
|
|
<a href={feed.siteUrl || feed.url} target="_blank" class="button is-small">
|
|
<span class="icon">
|
|
<i class="fas fa-external-link-alt"></i>
|
|
</span>
|
|
<span>Visit Site</span>
|
|
</a>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{#if feed.description}
|
|
<div class="content">
|
|
<p>{feed.description}</p>
|
|
</div>
|
|
{/if}
|
|
|
|
<div class="level">
|
|
<div class="level-left">
|
|
<div class="level-item">
|
|
<button class="button" on:click={toggleUnreadFilter}>
|
|
<span class="icon">
|
|
<i class="fas fa-filter"></i>
|
|
</span>
|
|
<span>{showUnreadOnly ? 'Show All' : 'Show Unread Only'}</span>
|
|
</button>
|
|
</div>
|
|
<div class="level-item">
|
|
<button class="button is-primary" on:click={handleRefreshFeed} disabled={isRefreshing}>
|
|
<span class="icon">
|
|
<i class="fas fa-sync" class:fa-spin={isRefreshing}></i>
|
|
</span>
|
|
<span>Refresh Feed</span>
|
|
</button>
|
|
</div>
|
|
<div class="level-item">
|
|
<button class="button is-info" on:click={handleMarkAllAsRead} disabled={isLoading}>
|
|
<span class="icon">
|
|
<i class="fas fa-check-double"></i>
|
|
</span>
|
|
<span>Mark All as Read</span>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
<div class="level-right">
|
|
{#if feed.lastFetched}
|
|
<div class="level-item">
|
|
<span class="tag is-light">
|
|
Last updated: {feed.lastFetched.toLocaleString()}
|
|
</span>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{#if isLoading && $entries.length === 0}
|
|
<div class="has-text-centered py-6">
|
|
<span class="icon is-large">
|
|
<i class="fas fa-spinner fa-spin fa-2x"></i>
|
|
</span>
|
|
</div>
|
|
{:else}
|
|
<CardList
|
|
items={$entries}
|
|
{renderCard}
|
|
emptyMessage={showUnreadOnly
|
|
? 'No unread entries found. Try refreshing your feed or viewing all entries.'
|
|
: 'No entries found. Try refreshing your feed.'}
|
|
/>
|
|
{/if}
|
|
{:else if !isLoading}
|
|
<div class="notification is-warning">
|
|
<p>Feed not found.</p>
|
|
<a href="/feeds" class="button is-light mt-4">Back to Feeds</a>
|
|
</div>
|
|
{/if}
|
|
</div>
|