feat(notes): Add obsidian style note linking with [[link]]
This commit is contained in:
parent
694af7ae91
commit
dba9535813
4 changed files with 61 additions and 63 deletions
|
@ -15,5 +15,7 @@ export function stripMarkdown(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);
|
||||
}
|
||||
|
|
|
@ -12,21 +12,18 @@
|
|||
|
||||
$: id = $page.params.id;
|
||||
|
||||
onMount(() => {
|
||||
const unsubscribe = notes.subscribe(allNotes => {
|
||||
if (!allNotes || !id) return;
|
||||
|
||||
note = allNotes.find(n => n.id === id);
|
||||
$: {
|
||||
if ($notes && id) {
|
||||
note = $notes.find(n => n.id === id);
|
||||
if (note) {
|
||||
editedTitle = note.title;
|
||||
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
|
||||
goto('/');
|
||||
}
|
||||
});
|
||||
return () => unsubscribe();
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
async function handleSave() {
|
||||
if (!editedTitle || !editedContent) return;
|
||||
|
@ -41,9 +38,28 @@
|
|||
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>
|
||||
|
||||
<div class="container">
|
||||
<div class="container" on:click={handleLinkClick}>
|
||||
<section class="section">
|
||||
{#if note}
|
||||
{#if isEditing}
|
||||
|
@ -72,6 +88,12 @@
|
|||
required
|
||||
></textarea>
|
||||
</div>
|
||||
<div class="mt-4 box">
|
||||
<label class="label">Preview</label>
|
||||
<div class="content">
|
||||
{@html renderMarkdown(editedContent)}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="field is-grouped">
|
||||
|
|
|
@ -1,27 +1,22 @@
|
|||
<script lang="ts">
|
||||
import { notes } from '$lib';
|
||||
import { goto } from '$app/navigation';
|
||||
export let data;
|
||||
|
||||
let title = '';
|
||||
let title = data.props.prefilledTitle;
|
||||
let content = '';
|
||||
let saving = false;
|
||||
let error = '';
|
||||
|
||||
async function handleSubmit() {
|
||||
|
||||
async function handleSave() {
|
||||
if (!title || !content) return;
|
||||
|
||||
saving = true;
|
||||
error = '';
|
||||
|
||||
|
||||
try {
|
||||
await notes.add({ title, content });
|
||||
await notes.load(); // Reload notes before navigation
|
||||
await goto('/');
|
||||
} catch (err) {
|
||||
console.error('Failed to create note:', err);
|
||||
error = 'Failed to save note. Please try again.';
|
||||
} finally {
|
||||
saving = false;
|
||||
await notes.add({
|
||||
title,
|
||||
content
|
||||
});
|
||||
goto('/');
|
||||
} catch (error) {
|
||||
console.error('Failed to create note:', error);
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
@ -29,8 +24,7 @@
|
|||
<div class="container">
|
||||
<section class="section">
|
||||
<h1 class="title">New Note</h1>
|
||||
|
||||
<form on:submit|preventDefault={handleSubmit}>
|
||||
<form on:submit|preventDefault={handleSave}>
|
||||
<div class="field">
|
||||
<label class="label" for="title">Title</label>
|
||||
<div class="control">
|
||||
|
@ -43,7 +37,7 @@
|
|||
>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
||||
<div class="field">
|
||||
<label class="label" for="content">Content</label>
|
||||
<div class="control">
|
||||
|
@ -52,30 +46,15 @@
|
|||
class="textarea"
|
||||
bind:value={content}
|
||||
rows="10"
|
||||
placeholder="Write your note in markdown..."
|
||||
required
|
||||
></textarea>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{#if error}
|
||||
<div class="notification is-danger">
|
||||
{error}
|
||||
</div>
|
||||
{/if}
|
||||
|
||||
<div class="field is-grouped">
|
||||
|
||||
<div class="field">
|
||||
<div class="control">
|
||||
<button
|
||||
type="submit"
|
||||
class="button is-primary"
|
||||
disabled={saving}
|
||||
>
|
||||
{saving ? 'Saving...' : 'Save'}
|
||||
</button>
|
||||
</div>
|
||||
<div class="control">
|
||||
<a href="/" class="button is-light" disabled={saving}>Cancel</a>
|
||||
<button type="submit" class="button is-primary">Create</button>
|
||||
<button type="button" class="button is-light" on:click={() => goto('/')}>Cancel</button>
|
||||
</div>
|
||||
</div>
|
||||
</form>
|
||||
|
|
|
@ -1,17 +1,12 @@
|
|||
import { notes } from '$lib';
|
||||
import type { PageLoad } from './$types';
|
||||
|
||||
export const load: PageLoad = async () => {
|
||||
try {
|
||||
await notes.load();
|
||||
return {
|
||||
status: 'success'
|
||||
};
|
||||
} catch (error) {
|
||||
console.error('Failed to load notes:', error);
|
||||
return {
|
||||
status: 'error',
|
||||
error: 'Failed to load notes'
|
||||
};
|
||||
}
|
||||
export const load: PageLoad = async ({ url }) => {
|
||||
const title = url.searchParams.get('title') || '';
|
||||
return {
|
||||
status: 'success',
|
||||
props: {
|
||||
prefilledTitle: title
|
||||
}
|
||||
};
|
||||
};
|
||||
|
|
Loading…
Add table
Reference in a new issue