feat: Implement data import and export functionality for notes and tags.

This commit is contained in:
2025-12-23 02:27:26 +01:00
parent 2ee9de866a
commit e4f021a68f
6 changed files with 200 additions and 8 deletions

View File

@@ -27,7 +27,7 @@ interface NoteFormProps {
export function NoteForm({ defaultValues, onSubmit, isLoading, submitLabel = "Save" }: NoteFormProps) {
const form = useForm<NoteFormValues>({
resolver: zodResolver(noteSchema),
resolver: zodResolver(noteSchema) as any,
defaultValues: {
title: "",
content: "",
@@ -40,9 +40,9 @@ export function NoteForm({ defaultValues, onSubmit, isLoading, submitLabel = "Sa
return (
<Form {...form}>
<form onSubmit={form.handleSubmit(onSubmit)} className="space-y-4">
<form onSubmit={form.handleSubmit(onSubmit as any)} className="space-y-4">
<FormField
control={form.control}
control={form.control as any}
name="title"
render={({ field }) => (
<FormItem>
@@ -55,7 +55,7 @@ export function NoteForm({ defaultValues, onSubmit, isLoading, submitLabel = "Sa
)}
/>
<FormField
control={form.control}
control={form.control as any}
name="content"
render={({ field }) => (
<FormItem>
@@ -68,7 +68,7 @@ export function NoteForm({ defaultValues, onSubmit, isLoading, submitLabel = "Sa
)}
/>
<FormField
control={form.control}
control={form.control as any}
name="tags"
render={({ field }) => (
<FormItem>
@@ -82,7 +82,7 @@ export function NoteForm({ defaultValues, onSubmit, isLoading, submitLabel = "Sa
/>
<FormField
control={form.control}
control={form.control as any}
name="color"
render={({ field }) => (
<FormItem>
@@ -111,7 +111,7 @@ export function NoteForm({ defaultValues, onSubmit, isLoading, submitLabel = "Sa
/>
<FormField
control={form.control}
control={form.control as any}
name="is_pinned"
render={({ field }) => (
<FormItem className="flex flex-row items-center space-x-3 space-y-0 rounded-md border p-4">

View File

@@ -1,9 +1,11 @@
import { useState, useEffect } from "react";
import { useRef, useState, useEffect } from "react";
import { Dialog, DialogContent, DialogDescription, DialogHeader, DialogTitle, DialogFooter } from "@/components/ui/dialog";
import { Button } from "@/components/ui/button";
import { Input } from "@/components/ui/input";
import { Label } from "@/components/ui/label";
import { toast } from "sonner";
import { api } from "@/lib/api";
import { Separator } from "@/components/ui/separator";
interface SettingsDialogProps {
open: boolean;
@@ -35,6 +37,42 @@ export function SettingsDialog({ open, onOpenChange }: SettingsDialogProps) {
}
};
const fileInputRef = useRef<HTMLInputElement>(null);
const handleExport = async () => {
try {
const blob = await api.exportData();
const url = window.URL.createObjectURL(blob);
const a = document.createElement('a');
a.href = url;
a.download = `k-notes-backup-${new Date().toISOString().split('T')[0]}.json`;
document.body.appendChild(a);
a.click();
window.URL.revokeObjectURL(url);
document.body.removeChild(a);
toast.success("Export successful");
} catch (e) {
toast.error("Export failed");
}
};
const handleImport = async (e: React.ChangeEvent<HTMLInputElement>) => {
const file = e.target.files?.[0];
if (!file) return;
try {
const text = await file.text();
const data = JSON.parse(text);
await api.importData(data);
toast.success("Import successful. Reloading...");
onOpenChange(false);
window.location.reload();
} catch (e) {
console.error(e);
toast.error("Import failed");
}
};
return (
<Dialog open={open} onOpenChange={onOpenChange}>
<DialogContent>
@@ -58,6 +96,32 @@ export function SettingsDialog({ open, onOpenChange }: SettingsDialogProps) {
/>
</div>
</div>
<Separator className="my-2" />
<div className="py-4 space-y-4">
<div className="flex flex-col space-y-2">
<h4 className="font-medium leading-none">Data Management</h4>
<p className="text-sm text-muted-foreground">
Export your notes for backup or import from a JSON file.
</p>
</div>
<div className="flex gap-4">
<Button variant="outline" onClick={handleExport}>
Export Data
</Button>
<Button variant="outline" onClick={() => fileInputRef.current?.click()}>
Import Data
</Button>
<input
type="file"
ref={fileInputRef}
className="hidden"
accept=".json"
onChange={handleImport}
/>
</div>
</div>
<DialogFooter>
<Button onClick={handleSave}>Save changes</Button>
</DialogFooter>

View File

@@ -73,4 +73,12 @@ export const api = {
body: JSON.stringify(body),
}),
delete: (endpoint: string) => fetchWithAuth(endpoint, { method: "DELETE" }),
exportData: async () => {
const response = await fetch(`${getApiUrl()}/export`, {
credentials: "include",
});
if (!response.ok) throw new ApiError(response.status, "Failed to export data");
return response.blob();
},
importData: (data: any) => api.post("/import", data),
};

View File

@@ -0,0 +1,28 @@
declare module "react-resizable-panels" {
import * as React from "react";
export interface PanelGroupProps extends React.HTMLAttributes<HTMLDivElement> {
direction: "horizontal" | "vertical";
autoSaveId?: string;
storage?: any;
}
export const PanelGroup: React.FC<PanelGroupProps>;
export interface PanelProps extends React.HTMLAttributes<HTMLDivElement> {
defaultSize?: number;
minSize?: number;
maxSize?: number;
order?: number;
collapsible?: boolean;
collapsedSize?: number;
onCollapse?: (collapsed: boolean) => void;
onResize?: (size: number) => void;
}
export const Panel: React.FC<PanelProps>;
export interface PanelResizeHandleProps extends React.HTMLAttributes<HTMLDivElement> {
disabled?: boolean;
hitAreaMargins?: { fine: number; coarse: number };
}
export const PanelResizeHandle: React.FC<PanelResizeHandleProps>;
}