Fixed project loading to restore all track properties and added inline project name editing in header. **AudioEditor.tsx:** - Added `loadTracks` from useMultiTrack hook - Fixed `handleLoadProject` to use `loadTracks()` instead of recreating tracks - Now properly restores all track properties (effects, automation, volume, pan, etc.) - Shows track count in success toast message - Added editable project name input in header - Positioned between logo and track actions - Auto-sizes based on text length - Saves on blur (triggers auto-save) - Smooth hover/focus transitions - Muted color that brightens on interaction **useMultiTrack.ts:** - Added `loadTracks()` method to replace all tracks at once - Enables proper project loading with full state restoration - Maintains all track properties during load **Fixes:** - Projects now load correctly with all tracks and their audio buffers - Track properties (effects, automation, volume, pan, etc.) fully restored - Project name can be edited inline in header - Auto-save triggers when project name changes 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
139 lines
4.5 KiB
TypeScript
139 lines
4.5 KiB
TypeScript
import { useState, useCallback, useEffect } from 'react';
|
|
import type { Track } from '@/types/track';
|
|
import { createTrack, createTrackFromBuffer } from '@/lib/audio/track-utils';
|
|
import { createEffectChain } from '@/lib/audio/effects/chain';
|
|
import { DEFAULT_TRACK_HEIGHT } from '@/types/track';
|
|
|
|
const STORAGE_KEY = 'audio-ui-multi-track';
|
|
|
|
export function useMultiTrack() {
|
|
const [tracks, setTracks] = useState<Track[]>(() => {
|
|
if (typeof window === 'undefined') return [];
|
|
|
|
try {
|
|
const saved = localStorage.getItem(STORAGE_KEY);
|
|
if (saved) {
|
|
const parsed = JSON.parse(saved);
|
|
|
|
// Clear corrupted data immediately if we detect issues
|
|
const hasInvalidData = parsed.some((t: any) =>
|
|
typeof t.name !== 'string' || t.name === '[object Object]'
|
|
);
|
|
|
|
if (hasInvalidData) {
|
|
console.warn('Detected corrupted track data in localStorage, clearing...');
|
|
localStorage.removeItem(STORAGE_KEY);
|
|
return [];
|
|
}
|
|
|
|
// Note: AudioBuffers can't be serialized, but EffectChains and Automation can
|
|
return parsed.map((t: any) => ({
|
|
...t,
|
|
name: String(t.name || 'Untitled Track'), // Ensure name is always a string
|
|
height: t.height && t.height >= DEFAULT_TRACK_HEIGHT ? t.height : DEFAULT_TRACK_HEIGHT, // Migrate old heights
|
|
audioBuffer: null, // Will need to be reloaded
|
|
effectChain: t.effectChain || createEffectChain(`${t.name} Effects`), // Restore effect chain or create new
|
|
automation: t.automation || { lanes: [], showAutomation: false }, // Restore automation or create new
|
|
selection: t.selection || null, // Initialize selection
|
|
showEffects: t.showEffects || false, // Restore showEffects state
|
|
}));
|
|
}
|
|
} catch (error) {
|
|
console.error('Failed to load tracks from localStorage:', error);
|
|
// Clear corrupted data
|
|
localStorage.removeItem(STORAGE_KEY);
|
|
}
|
|
|
|
return [];
|
|
});
|
|
|
|
// Save tracks to localStorage (without audio buffers)
|
|
useEffect(() => {
|
|
if (typeof window === 'undefined') return;
|
|
|
|
try {
|
|
// Only save serializable fields, excluding audioBuffer and any DOM references
|
|
const trackData = tracks.map((track) => ({
|
|
id: track.id,
|
|
name: String(track.name || 'Untitled Track'),
|
|
color: track.color,
|
|
height: track.height,
|
|
volume: track.volume,
|
|
pan: track.pan,
|
|
mute: track.mute,
|
|
solo: track.solo,
|
|
recordEnabled: track.recordEnabled,
|
|
collapsed: track.collapsed,
|
|
selected: track.selected,
|
|
showEffects: track.showEffects, // Save effects panel state
|
|
effectChain: track.effectChain, // Save effect chain
|
|
automation: track.automation, // Save automation data
|
|
}));
|
|
localStorage.setItem(STORAGE_KEY, JSON.stringify(trackData));
|
|
} catch (error) {
|
|
console.error('Failed to save tracks to localStorage:', error);
|
|
}
|
|
}, [tracks]);
|
|
|
|
const addTrack = useCallback((name?: string) => {
|
|
const track = createTrack(name);
|
|
setTracks((prev) => [...prev, track]);
|
|
return track;
|
|
}, []);
|
|
|
|
const addTrackFromBuffer = useCallback((buffer: AudioBuffer, name?: string) => {
|
|
const track = createTrackFromBuffer(buffer, name);
|
|
setTracks((prev) => [...prev, track]);
|
|
return track;
|
|
}, []);
|
|
|
|
const removeTrack = useCallback((trackId: string) => {
|
|
setTracks((prev) => prev.filter((t) => t.id !== trackId));
|
|
}, []);
|
|
|
|
const updateTrack = useCallback((trackId: string, updates: Partial<Track>) => {
|
|
setTracks((prev) =>
|
|
prev.map((track) =>
|
|
track.id === trackId ? { ...track, ...updates } : track
|
|
)
|
|
);
|
|
}, []);
|
|
|
|
const clearTracks = useCallback(() => {
|
|
setTracks([]);
|
|
}, []);
|
|
|
|
const reorderTracks = useCallback((fromIndex: number, toIndex: number) => {
|
|
setTracks((prev) => {
|
|
const newTracks = [...prev];
|
|
const [movedTrack] = newTracks.splice(fromIndex, 1);
|
|
newTracks.splice(toIndex, 0, movedTrack);
|
|
return newTracks;
|
|
});
|
|
}, []);
|
|
|
|
const setTrackBuffer = useCallback((trackId: string, buffer: AudioBuffer) => {
|
|
setTracks((prev) =>
|
|
prev.map((track) =>
|
|
track.id === trackId ? { ...track, audioBuffer: buffer } : track
|
|
)
|
|
);
|
|
}, []);
|
|
|
|
const loadTracks = useCallback((tracksToLoad: Track[]) => {
|
|
setTracks(tracksToLoad);
|
|
}, []);
|
|
|
|
return {
|
|
tracks,
|
|
addTrack,
|
|
addTrackFromBuffer,
|
|
removeTrack,
|
|
updateTrack,
|
|
clearTracks,
|
|
reorderTracks,
|
|
setTrackBuffer,
|
|
loadTracks,
|
|
};
|
|
}
|