feat(qn): added linking between feeds links and notes
This commit is contained in:
parent
1b46c7810b
commit
a20cb2964b
9 changed files with 86 additions and 11 deletions
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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>
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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>
|
||||||
|
|
|
@ -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}
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -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">
|
||||||
|
|
Loading…
Add table
Reference in a new issue