feat: Add obsidian style note linking with [[link]]

This commit is contained in:
Nicola Zangrandi 2025-02-17 16:03:31 +01:00
parent c449aa21df
commit f4df77b2f7
Signed by: wasp
GPG key ID: 43C1470D890F23ED
4 changed files with 61 additions and 63 deletions

View file

@ -15,5 +15,7 @@ export function stripMarkdown(text: string): string {
} }
export function renderMarkdown(text: string): string { export function renderMarkdown(text: string): string {
return marked(text); // Replace [[link]] with anchor tags having data-title attribute
const linkedText = text.replace(/\[\[([^\]]+)\]\]/g, '<a href="#" class="note-link" data-title="$1">$1</a>');
return marked(linkedText);
} }

View file

@ -12,21 +12,18 @@
$: id = $page.params.id; $: id = $page.params.id;
onMount(() => { $: {
const unsubscribe = notes.subscribe(allNotes => { if ($notes && id) {
if (!allNotes || !id) return; note = $notes.find(n => n.id === id);
note = allNotes.find(n => n.id === id);
if (note) { if (note) {
editedTitle = note.title; editedTitle = note.title;
editedContent = note.content; editedContent = note.content;
} else if (allNotes.length > 0) { } else if ($notes.length > 0) {
// Only redirect if we have loaded notes and didn't find this one // Only redirect if we have loaded notes and didn't find this one
goto('/'); goto('/');
} }
}); }
return () => unsubscribe(); }
});
async function handleSave() { async function handleSave() {
if (!editedTitle || !editedContent) return; if (!editedTitle || !editedContent) return;
@ -41,9 +38,28 @@
console.error('Failed to update note:', error); console.error('Failed to update note:', error);
} }
} }
async function handleLinkClick(event: MouseEvent) {
const target = event.target as HTMLElement;
if (target.tagName === 'A' && target.classList.contains('note-link')) {
event.preventDefault();
const title = target.getAttribute('data-title');
if (!title) return;
// Check if a note with this title exists
const existingNote = $notes.find(n => n.title === title);
if (existingNote) {
// Navigate to the existing note
goto(`/notes/${existingNote.id}`);
} else {
// Navigate to the new note page with the title pre-filled
goto(`/notes/new?title=${encodeURIComponent(title)}`);
}
}
}
</script> </script>
<div class="container"> <div class="container" on:click={handleLinkClick}>
<section class="section"> <section class="section">
{#if note} {#if note}
{#if isEditing} {#if isEditing}
@ -72,6 +88,12 @@
required required
></textarea> ></textarea>
</div> </div>
<div class="mt-4 box">
<label class="label">Preview</label>
<div class="content">
{@html renderMarkdown(editedContent)}
</div>
</div>
</div> </div>
<div class="field is-grouped"> <div class="field is-grouped">

View file

@ -1,27 +1,22 @@
<script lang="ts"> <script lang="ts">
import { notes } from '$lib'; import { notes } from '$lib';
import { goto } from '$app/navigation'; import { goto } from '$app/navigation';
export let data;
let title = ''; let title = data.props.prefilledTitle;
let content = ''; let content = '';
let saving = false;
let error = '';
async function handleSubmit() { async function handleSave() {
if (!title || !content) return; if (!title || !content) return;
saving = true;
error = '';
try { try {
await notes.add({ title, content }); await notes.add({
await notes.load(); // Reload notes before navigation title,
await goto('/'); content
} catch (err) { });
console.error('Failed to create note:', err); goto('/');
error = 'Failed to save note. Please try again.'; } catch (error) {
} finally { console.error('Failed to create note:', error);
saving = false;
} }
} }
</script> </script>
@ -29,8 +24,7 @@
<div class="container"> <div class="container">
<section class="section"> <section class="section">
<h1 class="title">New Note</h1> <h1 class="title">New Note</h1>
<form on:submit|preventDefault={handleSave}>
<form on:submit|preventDefault={handleSubmit}>
<div class="field"> <div class="field">
<label class="label" for="title">Title</label> <label class="label" for="title">Title</label>
<div class="control"> <div class="control">
@ -52,30 +46,15 @@
class="textarea" class="textarea"
bind:value={content} bind:value={content}
rows="10" rows="10"
placeholder="Write your note in markdown..."
required required
></textarea> ></textarea>
</div> </div>
</div> </div>
{#if error} <div class="field">
<div class="notification is-danger">
{error}
</div>
{/if}
<div class="field is-grouped">
<div class="control"> <div class="control">
<button <button type="submit" class="button is-primary">Create</button>
type="submit" <button type="button" class="button is-light" on:click={() => goto('/')}>Cancel</button>
class="button is-primary"
disabled={saving}
>
{saving ? 'Saving...' : 'Save'}
</button>
</div>
<div class="control">
<a href="/" class="button is-light" disabled={saving}>Cancel</a>
</div> </div>
</div> </div>
</form> </form>

View file

@ -1,17 +1,12 @@
import { notes } from '$lib'; import { notes } from '$lib';
import type { PageLoad } from './$types'; import type { PageLoad } from './$types';
export const load: PageLoad = async () => { export const load: PageLoad = async ({ url }) => {
try { const title = url.searchParams.get('title') || '';
await notes.load();
return { return {
status: 'success' status: 'success',
}; props: {
} catch (error) { prefilledTitle: title
console.error('Failed to load notes:', error);
return {
status: 'error',
error: 'Failed to load notes'
};
} }
}; };
};