feat(qn): added linking between feeds links and notes

This commit is contained in:
Nicola Zangrandi 2025-02-28 12:32:45 +01:00
parent 1b46c7810b
commit a20cb2964b
Signed by: wasp
GPG key ID: 43C1470D890F23ED
9 changed files with 86 additions and 11 deletions

View file

@ -224,11 +224,11 @@ func (s *Service) MarkEntryAsRead(id string) error {
// MarkAllEntriesAsRead marks all entries as read, optionally filtered by feed ID // MarkAllEntriesAsRead marks all entries as read, optionally filtered by feed ID
func (s *Service) MarkAllEntriesAsRead(feedID string) error { func (s *Service) MarkAllEntriesAsRead(feedID string) error {
query := s.db.Model(&Entry{}).Where("read_at IS NULL") query := s.db.Model(&Entry{}).Where("read_at IS NULL")
if feedID != "" { if feedID != "" {
query = query.Where("feed_id = ?", feedID) query = query.Where("feed_id = ?", feedID)
} }
return query.Update("read_at", time.Now()).Error return query.Update("read_at", time.Now()).Error
} }

View file

@ -19,7 +19,7 @@
}); });
</script> </script>
<nav class="navbar is-primary" role="navigation" aria-label="main navigation"> <nav class="navbar is-primary" aria-label="main navigation">
<div class="container"> <div class="container">
<div class="navbar-brand"> <div class="navbar-brand">
<a class="navbar-item" href="/"> <a class="navbar-item" href="/">
@ -28,11 +28,12 @@
<a <a
role="button" role="button"
href="#top"
class="navbar-burger" class="navbar-burger"
class:is-active={isActive} class:is-active={isActive}
aria-label="menu" aria-label="menu"
aria-expanded="false" aria-expanded="false"
on:click|stopPropagation={toggleMenu} onclick={toggleMenu}
> >
<span aria-hidden="true"></span> <span aria-hidden="true"></span>
<span aria-hidden="true"></span> <span aria-hidden="true"></span>

View file

@ -172,7 +172,7 @@ function createEntriesStore() {
if (!response.ok) throw new Error('Failed to load entries'); if (!response.ok) throw new Error('Failed to load entries');
const data = await response.json(); const data = await response.json();
const entries = data.map((entry: any) => ({ const entries = data.map((entry: FeedEntry) => ({
...entry, ...entry,
published: entry.published ? new Date(entry.published) : null, published: entry.published ? new Date(entry.published) : null,
updated: entry.updated ? new Date(entry.updated) : null, updated: entry.updated ? new Date(entry.updated) : null,

View file

@ -20,9 +20,20 @@
description: item.content, description: item.content,
timestamp: new Date(item.createdAt), timestamp: new Date(item.createdAt),
actions: [ actions: [
{
icon: 'eye',
label: 'View',
href: `/notes/${item.id}`
},
{
icon: 'edit',
label: 'Edit',
href: `/notes/${item.id}?edit=true`
},
{ {
icon: 'trash', icon: 'trash',
label: 'Delete', label: 'Delete',
isDangerous: true,
onClick: async () => { onClick: async () => {
await fetch(`/api/notes/${item.id}`, { method: 'DELETE' }); await fetch(`/api/notes/${item.id}`, { method: 'DELETE' });
notes = notes.filter((n) => n.id !== item.id); notes = notes.filter((n) => n.id !== item.id);

View file

@ -171,7 +171,9 @@
<ul> <ul>
<li><a href="/feeds">Feeds</a></li> <li><a href="/feeds">Feeds</a></li>
<li><a href="/feeds/list">Manage 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> <li class="is-active">
<a href="#top" aria-current="page">{feed?.title || 'Feed'}</a>
</li>
</ul> </ul>
</nav> </nav>
</div> </div>

View file

@ -3,6 +3,7 @@
import { entries, feeds } from '$lib/feeds'; import { entries, feeds } from '$lib/feeds';
import type { FeedEntry, Feed } from '$lib/types'; import type { FeedEntry, Feed } from '$lib/types';
import { page } from '$app/stores'; import { page } from '$app/stores';
import { goto } from '$app/navigation';
let entry: FeedEntry | null = null; let entry: FeedEntry | null = null;
let feed: Feed | null = null; let feed: Feed | null = null;
@ -29,7 +30,9 @@
// Load feed info // Load feed info
const feedsList = await feeds.load(); const feedsList = await feeds.load();
feed = feedsList.find((f) => f.id === entry.feedId) || null; // We've already checked that entry is not null
const currentEntry = entry;
feed = feedsList.find((f) => f.id === currentEntry.feedId) || null;
} }
} catch (e) { } catch (e) {
error = e instanceof Error ? e.message : 'Failed to load entry'; error = e instanceof Error ? e.message : 'Failed to load entry';
@ -53,6 +56,20 @@
isLoadingFullContent = false; isLoadingFullContent = false;
} }
} }
function createNote() {
if (!entry) return;
const title = `Highlights from ${entry.title}`;
const content = `[Original Feed Entry](/feeds/entries/${entry.id})\n\n${entry.fullContent || entry.summary || ''}`;
const params = new URLSearchParams({
title,
content
});
goto(`/notes/new?${params.toString()}`);
}
</script> </script>
<div class="container mt-4"> <div class="container mt-4">
@ -62,7 +79,7 @@
{#if feed} {#if feed}
<li><a href="/feeds/{feed.id}">{feed.title}</a></li> <li><a href="/feeds/{feed.id}">{feed.title}</a></li>
{/if} {/if}
<li class="is-active"><a href="#" aria-current="page">{entry?.title || 'Entry'}</a></li> <li class="is-active"><a href="#top" aria-current="page">{entry?.title || 'Entry'}</a></li>
</ul> </ul>
</nav> </nav>
@ -136,11 +153,20 @@
</button> </button>
</div> </div>
{/if} {/if}
<div class="level-item">
<button class="button is-small is-info" on:click={createNote}>
<span class="icon">
<i class="fas fa-sticky-note"></i>
</span>
<span>Create Note</span>
</button>
</div>
</div> </div>
</div> </div>
{#if entry.summary && !entry.fullContent} {#if entry.summary && !entry.fullContent}
<div class="content mt-4"> <div class="content mt-4">
<!-- eslint-disable-next-line svelte/no-at-html-tags -->
{@html entry.summary} {@html entry.summary}
</div> </div>
<div class="notification is-info is-light"> <div class="notification is-info is-light">
@ -148,6 +174,7 @@
</div> </div>
{:else if entry.fullContent} {:else if entry.fullContent}
<div class="content mt-4"> <div class="content mt-4">
<!-- eslint-disable-next-line svelte/no-at-html-tags -->
{@html entry.fullContent} {@html entry.fullContent}
</div> </div>
{:else} {:else}

View file

@ -3,10 +3,18 @@
import { notes } from '$lib'; import { notes } from '$lib';
import { goto } from '$app/navigation'; import { goto } from '$app/navigation';
import Navigation from '$lib/components/Navigation.svelte'; import Navigation from '$lib/components/Navigation.svelte';
let { data } = $props();
interface PageData {
props: {
prefilledTitle: string;
prefilledContent?: string;
};
}
let { data } = $props<{ data: PageData }>();
let title = $state(data.props.prefilledTitle); let title = $state(data.props.prefilledTitle);
let content = $state(''); let content = $state(data.props.prefilledContent || '');
async function handleSave() { async function handleSave() {
if (!title || !content) return; if (!title || !content) return;

View file

@ -2,10 +2,13 @@ import type { PageLoad } from './$types';
export const load: PageLoad = async ({ url }) => { export const load: PageLoad = async ({ url }) => {
const title = url.searchParams.get('title') || ''; const title = url.searchParams.get('title') || '';
const content = url.searchParams.get('content') || '';
return { return {
status: 'success', status: 'success',
props: { props: {
prefilledTitle: title prefilledTitle: title,
prefilledContent: content
} }
}; };
}; };

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 type { ReadLaterItem } from '$lib/types'; import type { ReadLaterItem } from '$lib/types';
import { goto } from '$app/navigation';
export let data: { id: string }; export let data: { id: string };
let id = data.id; let id = data.id;
@ -32,6 +33,20 @@
error = e instanceof Error ? e.message : 'Failed to load item'; error = e instanceof Error ? e.message : 'Failed to load item';
} }
}); });
function createNote() {
if (!item) return;
const title = `Highlights from ${item.title}`;
const content = `[Original Link](/readlist/${item.id})\n\n${item.content || ''}`;
const params = new URLSearchParams({
title,
content
});
goto(`/notes/new?${params.toString()}`);
}
</script> </script>
<div class="container"> <div class="container">
@ -78,6 +93,14 @@
</div> </div>
</div> </div>
<div class="level-right"> <div class="level-right">
<div class="level-item">
<button class="button is-info" onclick={() => createNote()}>
<span class="icon">
<i class="fas fa-sticky-note"></i>
</span>
<span>Create Note</span>
</button>
</div>
<div class="level-item"> <div class="level-item">
<button class="button is-danger" onclick={() => item && readlist.delete(item.id)}> <button class="button is-danger" onclick={() => item && readlist.delete(item.id)}>
<span class="icon"> <span class="icon">