feat: implement Phase 12.3 - Project Export/Import
Added full export/import functionality for projects:
Export Features:
- Export button for each project in Projects dialog
- Downloads project as JSON file with all data
- Includes tracks, audio buffers, effects, automation, settings
- Filename format: {project_name}_{timestamp}.json
Import Features:
- Import button in Projects dialog header
- File picker for .json files
- Automatically generates new project ID to avoid conflicts
- Appends "(Imported)" to project name
- Preserves all project data
This enables:
- Backup of projects outside the browser
- Sharing projects with collaborators
- Migration between computers/browsers
- Version snapshots at different stages
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
@@ -1,7 +1,7 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import * as React from 'react';
|
import * as React from 'react';
|
||||||
import { X, Plus, Trash2, Copy, FolderOpen } from 'lucide-react';
|
import { X, Plus, Trash2, Copy, FolderOpen, Download, Upload } from 'lucide-react';
|
||||||
import { Button } from '@/components/ui/Button';
|
import { Button } from '@/components/ui/Button';
|
||||||
import type { ProjectMetadata } from '@/lib/storage/db';
|
import type { ProjectMetadata } from '@/lib/storage/db';
|
||||||
import { formatDuration } from '@/lib/audio/decoder';
|
import { formatDuration } from '@/lib/audio/decoder';
|
||||||
@@ -14,6 +14,8 @@ export interface ProjectsDialogProps {
|
|||||||
onLoadProject: (projectId: string) => void;
|
onLoadProject: (projectId: string) => void;
|
||||||
onDeleteProject: (projectId: string) => void;
|
onDeleteProject: (projectId: string) => void;
|
||||||
onDuplicateProject: (projectId: string) => void;
|
onDuplicateProject: (projectId: string) => void;
|
||||||
|
onExportProject: (projectId: string) => void;
|
||||||
|
onImportProject: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function ProjectsDialog({
|
export function ProjectsDialog({
|
||||||
@@ -24,6 +26,8 @@ export function ProjectsDialog({
|
|||||||
onLoadProject,
|
onLoadProject,
|
||||||
onDeleteProject,
|
onDeleteProject,
|
||||||
onDuplicateProject,
|
onDuplicateProject,
|
||||||
|
onExportProject,
|
||||||
|
onImportProject,
|
||||||
}: ProjectsDialogProps) {
|
}: ProjectsDialogProps) {
|
||||||
if (!open) return null;
|
if (!open) return null;
|
||||||
|
|
||||||
@@ -44,6 +48,15 @@ export function ProjectsDialog({
|
|||||||
<div className="flex items-center justify-between p-6 border-b border-border">
|
<div className="flex items-center justify-between p-6 border-b border-border">
|
||||||
<h2 className="text-lg font-semibold text-foreground">Projects</h2>
|
<h2 className="text-lg font-semibold text-foreground">Projects</h2>
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-2">
|
||||||
|
<Button
|
||||||
|
onClick={onImportProject}
|
||||||
|
variant="outline"
|
||||||
|
size="sm"
|
||||||
|
className="gap-2"
|
||||||
|
>
|
||||||
|
<Upload className="h-4 w-4" />
|
||||||
|
Import
|
||||||
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
onClick={onNewProject}
|
onClick={onNewProject}
|
||||||
variant="default"
|
variant="default"
|
||||||
@@ -111,6 +124,13 @@ export function ProjectsDialog({
|
|||||||
>
|
>
|
||||||
Open
|
Open
|
||||||
</button>
|
</button>
|
||||||
|
<button
|
||||||
|
onClick={() => onExportProject(project.id)}
|
||||||
|
className="p-1.5 text-muted-foreground hover:text-foreground hover:bg-accent rounded transition-colors"
|
||||||
|
title="Export project"
|
||||||
|
>
|
||||||
|
<Download className="h-4 w-4" />
|
||||||
|
</button>
|
||||||
<button
|
<button
|
||||||
onClick={() => onDuplicateProject(project.id)}
|
onClick={() => onDuplicateProject(project.id)}
|
||||||
className="p-1.5 text-muted-foreground hover:text-foreground hover:bg-accent rounded transition-colors"
|
className="p-1.5 text-muted-foreground hover:text-foreground hover:bg-accent rounded transition-colors"
|
||||||
|
|||||||
@@ -42,6 +42,8 @@ import {
|
|||||||
listProjects,
|
listProjects,
|
||||||
removeProject,
|
removeProject,
|
||||||
duplicateProject,
|
duplicateProject,
|
||||||
|
exportProjectAsJSON,
|
||||||
|
importProjectFromJSON,
|
||||||
type ProjectMetadata,
|
type ProjectMetadata,
|
||||||
} from '@/lib/storage/projects';
|
} from '@/lib/storage/projects';
|
||||||
import { getAudioContext } from '@/lib/audio/context';
|
import { getAudioContext } from '@/lib/audio/context';
|
||||||
@@ -1110,6 +1112,70 @@ export function AudioEditor() {
|
|||||||
}
|
}
|
||||||
}, [projects, loadProjectsList, addToast]);
|
}, [projects, loadProjectsList, addToast]);
|
||||||
|
|
||||||
|
// Export project
|
||||||
|
const handleExportProject = React.useCallback(async (projectId: string) => {
|
||||||
|
try {
|
||||||
|
const project = projects.find(p => p.id === projectId);
|
||||||
|
if (!project) return;
|
||||||
|
|
||||||
|
await exportProjectAsJSON(projectId);
|
||||||
|
|
||||||
|
addToast({
|
||||||
|
title: 'Project Exported',
|
||||||
|
description: `"${project.name}" exported successfully`,
|
||||||
|
variant: 'success',
|
||||||
|
duration: 2000,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to export project:', error);
|
||||||
|
addToast({
|
||||||
|
title: 'Export Failed',
|
||||||
|
description: 'Could not export project',
|
||||||
|
variant: 'error',
|
||||||
|
duration: 3000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [projects, addToast]);
|
||||||
|
|
||||||
|
// Import project
|
||||||
|
const handleImportProject = React.useCallback(async () => {
|
||||||
|
try {
|
||||||
|
// Create file input
|
||||||
|
const input = document.createElement('input');
|
||||||
|
input.type = 'file';
|
||||||
|
input.accept = '.json';
|
||||||
|
|
||||||
|
input.onchange = async (e) => {
|
||||||
|
const file = (e.target as HTMLInputElement).files?.[0];
|
||||||
|
if (!file) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const projectId = await importProjectFromJSON(file);
|
||||||
|
await loadProjectsList();
|
||||||
|
|
||||||
|
addToast({
|
||||||
|
title: 'Project Imported',
|
||||||
|
description: 'Project imported successfully',
|
||||||
|
variant: 'success',
|
||||||
|
duration: 2000,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to import project:', error);
|
||||||
|
addToast({
|
||||||
|
title: 'Import Failed',
|
||||||
|
description: 'Could not import project file',
|
||||||
|
variant: 'error',
|
||||||
|
duration: 3000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
input.click();
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to open file picker:', error);
|
||||||
|
}
|
||||||
|
}, [loadProjectsList, addToast]);
|
||||||
|
|
||||||
// Zoom controls
|
// Zoom controls
|
||||||
const handleZoomIn = () => {
|
const handleZoomIn = () => {
|
||||||
setZoom((prev) => Math.min(20, prev + 1));
|
setZoom((prev) => Math.min(20, prev + 1));
|
||||||
@@ -1586,6 +1652,8 @@ export function AudioEditor() {
|
|||||||
onLoadProject={handleLoadProject}
|
onLoadProject={handleLoadProject}
|
||||||
onDeleteProject={handleDeleteProject}
|
onDeleteProject={handleDeleteProject}
|
||||||
onDuplicateProject={handleDuplicateProject}
|
onDuplicateProject={handleDuplicateProject}
|
||||||
|
onExportProject={handleExportProject}
|
||||||
|
onImportProject={handleImportProject}
|
||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -210,3 +210,64 @@ export async function duplicateProject(sourceProjectId: string, newName: string)
|
|||||||
await saveProject(newProject);
|
await saveProject(newProject);
|
||||||
return newId;
|
return newId;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Export project as JSON file
|
||||||
|
*/
|
||||||
|
export async function exportProjectAsJSON(projectId: string): Promise<void> {
|
||||||
|
const project = await loadProject(projectId);
|
||||||
|
if (!project) throw new Error('Project not found');
|
||||||
|
|
||||||
|
// Convert the project to JSON
|
||||||
|
const json = JSON.stringify(project, null, 2);
|
||||||
|
|
||||||
|
// Create blob and download
|
||||||
|
const blob = new Blob([json], { type: 'application/json' });
|
||||||
|
const url = URL.createObjectURL(blob);
|
||||||
|
const a = document.createElement('a');
|
||||||
|
a.href = url;
|
||||||
|
a.download = `${project.metadata.name.replace(/[^a-z0-9]/gi, '_').toLowerCase()}_${Date.now()}.json`;
|
||||||
|
document.body.appendChild(a);
|
||||||
|
a.click();
|
||||||
|
document.body.removeChild(a);
|
||||||
|
URL.revokeObjectURL(url);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Import project from JSON file
|
||||||
|
*/
|
||||||
|
export async function importProjectFromJSON(file: File): Promise<string> {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
const reader = new FileReader();
|
||||||
|
|
||||||
|
reader.onload = async (e) => {
|
||||||
|
try {
|
||||||
|
const json = e.target?.result as string;
|
||||||
|
const project = JSON.parse(json) as ProjectData;
|
||||||
|
|
||||||
|
// Generate new ID to avoid conflicts
|
||||||
|
const newId = generateProjectId();
|
||||||
|
const now = Date.now();
|
||||||
|
|
||||||
|
const importedProject: ProjectData = {
|
||||||
|
...project,
|
||||||
|
metadata: {
|
||||||
|
...project.metadata,
|
||||||
|
id: newId,
|
||||||
|
name: `${project.metadata.name} (Imported)`,
|
||||||
|
createdAt: now,
|
||||||
|
updatedAt: now,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
await saveProject(importedProject);
|
||||||
|
resolve(newId);
|
||||||
|
} catch (error) {
|
||||||
|
reject(new Error('Failed to parse project file'));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
reader.onerror = () => reject(new Error('Failed to read file'));
|
||||||
|
reader.readAsText(file);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user