feat: Implement data import and export functionality for notes and tags.
This commit is contained in:
@@ -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">
|
||||
|
||||
@@ -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>
|
||||
|
||||
@@ -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),
|
||||
};
|
||||
|
||||
28
k-notes-frontend/src/types/resizable-panels-override.d.ts
vendored
Normal file
28
k-notes-frontend/src/types/resizable-panels-override.d.ts
vendored
Normal 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>;
|
||||
}
|
||||
Reference in New Issue
Block a user