mirror of
https://github.com/AutoMaker-Org/automaker.git
synced 2026-02-04 09:13:08 +00:00
fix(ui,server): Fix project icon updates and image upload issues
- Fix setProjectCustomIcon using wrong property name (customIcon -> customIconPath) - Add currentProject state update to setProjectIcon and setProjectCustomIcon - Fix data URL regex to handle all formats (e.g., charset=utf-8 in GIFs) - Increase project icon size limit from 2MB to 5MB for animated GIFs - Add toast notifications for upload validation errors - Add image error fallback to folder icon in project switcher - Make HttpApiClient get/put methods public for store access - Fix TypeScript errors in app-store.ts (trashedAt type, font properties) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
@@ -31,7 +31,9 @@ export function createSaveBoardBackgroundHandler() {
|
|||||||
await secureFs.mkdir(boardDir, { recursive: true });
|
await secureFs.mkdir(boardDir, { recursive: true });
|
||||||
|
|
||||||
// Decode base64 data (remove data URL prefix if present)
|
// Decode base64 data (remove data URL prefix if present)
|
||||||
const base64Data = data.replace(/^data:image\/\w+;base64,/, '');
|
// Use a regex that handles all data URL formats including those with extra params
|
||||||
|
// e.g., data:image/gif;charset=utf-8;base64,R0lGOD...
|
||||||
|
const base64Data = data.replace(/^data:[^,]+,/, '');
|
||||||
const buffer = Buffer.from(base64Data, 'base64');
|
const buffer = Buffer.from(base64Data, 'base64');
|
||||||
|
|
||||||
// Use a fixed filename for the board background (overwrite previous)
|
// Use a fixed filename for the board background (overwrite previous)
|
||||||
|
|||||||
@@ -31,7 +31,9 @@ export function createSaveImageHandler() {
|
|||||||
await secureFs.mkdir(imagesDir, { recursive: true });
|
await secureFs.mkdir(imagesDir, { recursive: true });
|
||||||
|
|
||||||
// Decode base64 data (remove data URL prefix if present)
|
// Decode base64 data (remove data URL prefix if present)
|
||||||
const base64Data = data.replace(/^data:image\/\w+;base64,/, '');
|
// Use a regex that handles all data URL formats including those with extra params
|
||||||
|
// e.g., data:image/gif;charset=utf-8;base64,R0lGOD...
|
||||||
|
const base64Data = data.replace(/^data:[^,]+,/, '');
|
||||||
const buffer = Buffer.from(base64Data, 'base64');
|
const buffer = Buffer.from(base64Data, 'base64');
|
||||||
|
|
||||||
// Generate unique filename with timestamp
|
// Generate unique filename with timestamp
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ import { getAuthenticatedImageUrl } from '@/lib/api-fetch';
|
|||||||
import { getHttpApiClient } from '@/lib/http-api-client';
|
import { getHttpApiClient } from '@/lib/http-api-client';
|
||||||
import type { Project } from '@/lib/electron';
|
import type { Project } from '@/lib/electron';
|
||||||
import { IconPicker } from './icon-picker';
|
import { IconPicker } from './icon-picker';
|
||||||
|
import { toast } from 'sonner';
|
||||||
|
|
||||||
interface EditProjectDialogProps {
|
interface EditProjectDialogProps {
|
||||||
project: Project;
|
project: Project;
|
||||||
@@ -52,11 +53,18 @@ export function EditProjectDialog({ project, open, onOpenChange }: EditProjectDi
|
|||||||
// Validate file type
|
// Validate file type
|
||||||
const validTypes = ['image/jpeg', 'image/png', 'image/gif', 'image/webp'];
|
const validTypes = ['image/jpeg', 'image/png', 'image/gif', 'image/webp'];
|
||||||
if (!validTypes.includes(file.type)) {
|
if (!validTypes.includes(file.type)) {
|
||||||
|
toast.error(
|
||||||
|
`Invalid file type: ${file.type || 'unknown'}. Please use JPG, PNG, GIF or WebP.`
|
||||||
|
);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Validate file size (max 2MB for icons)
|
// Validate file size (max 5MB for icons - allows animated GIFs)
|
||||||
if (file.size > 2 * 1024 * 1024) {
|
const maxSize = 5 * 1024 * 1024;
|
||||||
|
if (file.size > maxSize) {
|
||||||
|
toast.error(
|
||||||
|
`File too large (${(file.size / 1024 / 1024).toFixed(2)} MB). Maximum size is 5 MB.`
|
||||||
|
);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -72,15 +80,24 @@ export function EditProjectDialog({ project, open, onOpenChange }: EditProjectDi
|
|||||||
file.type,
|
file.type,
|
||||||
project.path
|
project.path
|
||||||
);
|
);
|
||||||
|
|
||||||
if (result.success && result.path) {
|
if (result.success && result.path) {
|
||||||
setCustomIconPath(result.path);
|
setCustomIconPath(result.path);
|
||||||
// Clear the Lucide icon when custom icon is set
|
// Clear the Lucide icon when custom icon is set
|
||||||
setIcon(null);
|
setIcon(null);
|
||||||
|
toast.success('Icon uploaded successfully');
|
||||||
|
} else {
|
||||||
|
toast.error('Failed to upload icon');
|
||||||
}
|
}
|
||||||
setIsUploadingIcon(false);
|
setIsUploadingIcon(false);
|
||||||
};
|
};
|
||||||
|
reader.onerror = () => {
|
||||||
|
toast.error('Failed to read file');
|
||||||
|
setIsUploadingIcon(false);
|
||||||
|
};
|
||||||
reader.readAsDataURL(file);
|
reader.readAsDataURL(file);
|
||||||
} catch {
|
} catch {
|
||||||
|
toast.error('Failed to upload icon');
|
||||||
setIsUploadingIcon(false);
|
setIsUploadingIcon(false);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@@ -162,7 +179,7 @@ export function EditProjectDialog({ project, open, onOpenChange }: EditProjectDi
|
|||||||
{isUploadingIcon ? 'Uploading...' : 'Upload Custom Icon'}
|
{isUploadingIcon ? 'Uploading...' : 'Upload Custom Icon'}
|
||||||
</Button>
|
</Button>
|
||||||
<p className="text-xs text-muted-foreground mt-1">
|
<p className="text-xs text-muted-foreground mt-1">
|
||||||
PNG, JPG, GIF or WebP. Max 2MB.
|
PNG, JPG, GIF or WebP. Max 5MB.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -59,7 +59,7 @@ interface ThemeButtonProps {
|
|||||||
/** Handler for pointer leave events (used to clear preview) */
|
/** Handler for pointer leave events (used to clear preview) */
|
||||||
onPointerLeave: (e: React.PointerEvent) => void;
|
onPointerLeave: (e: React.PointerEvent) => void;
|
||||||
/** Handler for click events (used to select theme) */
|
/** Handler for click events (used to select theme) */
|
||||||
onClick: () => void;
|
onClick: (e: React.MouseEvent) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -77,6 +77,7 @@ const ThemeButton = memo(function ThemeButton({
|
|||||||
const Icon = option.icon;
|
const Icon = option.icon;
|
||||||
return (
|
return (
|
||||||
<button
|
<button
|
||||||
|
type="button"
|
||||||
onPointerEnter={onPointerEnter}
|
onPointerEnter={onPointerEnter}
|
||||||
onPointerLeave={onPointerLeave}
|
onPointerLeave={onPointerLeave}
|
||||||
onClick={onClick}
|
onClick={onClick}
|
||||||
@@ -145,7 +146,10 @@ const ThemeColumn = memo(function ThemeColumn({
|
|||||||
isSelected={selectedTheme === option.value}
|
isSelected={selectedTheme === option.value}
|
||||||
onPointerEnter={() => onPreviewEnter(option.value)}
|
onPointerEnter={() => onPreviewEnter(option.value)}
|
||||||
onPointerLeave={onPreviewLeave}
|
onPointerLeave={onPreviewLeave}
|
||||||
onClick={() => onSelect(option.value)}
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
onSelect(option.value);
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
@@ -193,7 +197,6 @@ export function ProjectContextMenu({
|
|||||||
const {
|
const {
|
||||||
moveProjectToTrash,
|
moveProjectToTrash,
|
||||||
theme: globalTheme,
|
theme: globalTheme,
|
||||||
setTheme,
|
|
||||||
setProjectTheme,
|
setProjectTheme,
|
||||||
setPreviewTheme,
|
setPreviewTheme,
|
||||||
} = useAppStore();
|
} = useAppStore();
|
||||||
@@ -316,13 +319,24 @@ export function ProjectContextMenu({
|
|||||||
|
|
||||||
const handleThemeSelect = useCallback(
|
const handleThemeSelect = useCallback(
|
||||||
(value: ThemeMode | typeof USE_GLOBAL_THEME) => {
|
(value: ThemeMode | typeof USE_GLOBAL_THEME) => {
|
||||||
|
// Clear any pending close timeout to prevent race conditions
|
||||||
|
if (closeTimeoutRef.current) {
|
||||||
|
clearTimeout(closeTimeoutRef.current);
|
||||||
|
closeTimeoutRef.current = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close menu first
|
||||||
|
setShowThemeSubmenu(false);
|
||||||
|
onClose();
|
||||||
|
|
||||||
|
// Then apply theme changes
|
||||||
setPreviewTheme(null);
|
setPreviewTheme(null);
|
||||||
const isUsingGlobal = value === USE_GLOBAL_THEME;
|
const isUsingGlobal = value === USE_GLOBAL_THEME;
|
||||||
setTheme(isUsingGlobal ? globalTheme : value);
|
// Only set project theme - don't change global theme
|
||||||
|
// The UI uses getEffectiveTheme() which handles: previewTheme ?? projectTheme ?? globalTheme
|
||||||
setProjectTheme(project.id, isUsingGlobal ? null : value);
|
setProjectTheme(project.id, isUsingGlobal ? null : value);
|
||||||
setShowThemeSubmenu(false);
|
|
||||||
},
|
},
|
||||||
[globalTheme, project.id, setPreviewTheme, setProjectTheme, setTheme]
|
[onClose, project.id, setPreviewTheme, setProjectTheme]
|
||||||
);
|
);
|
||||||
|
|
||||||
const handleConfirmRemove = useCallback(() => {
|
const handleConfirmRemove = useCallback(() => {
|
||||||
@@ -426,9 +440,13 @@ export function ProjectContextMenu({
|
|||||||
<div className="p-2">
|
<div className="p-2">
|
||||||
{/* Use Global Option */}
|
{/* Use Global Option */}
|
||||||
<button
|
<button
|
||||||
|
type="button"
|
||||||
onPointerEnter={() => handlePreviewEnter(globalTheme)}
|
onPointerEnter={() => handlePreviewEnter(globalTheme)}
|
||||||
onPointerLeave={handlePreviewLeave}
|
onPointerLeave={handlePreviewLeave}
|
||||||
onClick={() => handleThemeSelect(USE_GLOBAL_THEME)}
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
handleThemeSelect(USE_GLOBAL_THEME);
|
||||||
|
}}
|
||||||
className={cn(
|
className={cn(
|
||||||
'w-full flex items-center gap-2 px-3 py-2 rounded-md',
|
'w-full flex items-center gap-2 px-3 py-2 rounded-md',
|
||||||
'text-sm font-medium text-left',
|
'text-sm font-medium text-left',
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import { useState } from 'react';
|
||||||
import { Folder, LucideIcon } from 'lucide-react';
|
import { Folder, LucideIcon } from 'lucide-react';
|
||||||
import * as LucideIcons from 'lucide-react';
|
import * as LucideIcons from 'lucide-react';
|
||||||
import { cn, sanitizeForTestId } from '@/lib/utils';
|
import { cn, sanitizeForTestId } from '@/lib/utils';
|
||||||
@@ -19,6 +20,8 @@ export function ProjectSwitcherItem({
|
|||||||
onClick,
|
onClick,
|
||||||
onContextMenu,
|
onContextMenu,
|
||||||
}: ProjectSwitcherItemProps) {
|
}: ProjectSwitcherItemProps) {
|
||||||
|
const [imageError, setImageError] = useState(false);
|
||||||
|
|
||||||
// Convert index to hotkey label: 0 -> "1", 1 -> "2", ..., 8 -> "9", 9 -> "0"
|
// Convert index to hotkey label: 0 -> "1", 1 -> "2", ..., 8 -> "9", 9 -> "0"
|
||||||
const hotkeyLabel =
|
const hotkeyLabel =
|
||||||
hotkeyIndex !== undefined && hotkeyIndex >= 0 && hotkeyIndex <= 9
|
hotkeyIndex !== undefined && hotkeyIndex >= 0 && hotkeyIndex <= 9
|
||||||
@@ -35,7 +38,7 @@ export function ProjectSwitcherItem({
|
|||||||
};
|
};
|
||||||
|
|
||||||
const IconComponent = getIconComponent();
|
const IconComponent = getIconComponent();
|
||||||
const hasCustomIcon = !!project.customIconPath;
|
const hasCustomIcon = !!project.customIconPath && !imageError;
|
||||||
|
|
||||||
// Combine project.id with sanitized name for uniqueness and readability
|
// Combine project.id with sanitized name for uniqueness and readability
|
||||||
// Format: project-switcher-{id}-{sanitizedName}
|
// Format: project-switcher-{id}-{sanitizedName}
|
||||||
@@ -74,6 +77,7 @@ export function ProjectSwitcherItem({
|
|||||||
'w-8 h-8 rounded-lg object-cover transition-all duration-200',
|
'w-8 h-8 rounded-lg object-cover transition-all duration-200',
|
||||||
isActive ? 'ring-1 ring-brand-500/50' : 'group-hover:scale-110'
|
isActive ? 'ring-1 ring-brand-500/50' : 'group-hover:scale-110'
|
||||||
)}
|
)}
|
||||||
|
onError={() => setImageError(true)}
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<IconComponent
|
<IconComponent
|
||||||
|
|||||||
@@ -100,14 +100,8 @@ export function ProjectSelectorWithOptions({
|
|||||||
|
|
||||||
const { sensors, handleDragEnd } = useDragAndDrop({ projects, reorderProjects });
|
const { sensors, handleDragEnd } = useDragAndDrop({ projects, reorderProjects });
|
||||||
|
|
||||||
const {
|
const { globalTheme, setProjectTheme, setPreviewTheme, handlePreviewEnter, handlePreviewLeave } =
|
||||||
globalTheme,
|
useProjectTheme();
|
||||||
setTheme,
|
|
||||||
setProjectTheme,
|
|
||||||
setPreviewTheme,
|
|
||||||
handlePreviewEnter,
|
|
||||||
handlePreviewLeave,
|
|
||||||
} = useProjectTheme();
|
|
||||||
|
|
||||||
if (!sidebarOpen || projects.length === 0) {
|
if (!sidebarOpen || projects.length === 0) {
|
||||||
return null;
|
return null;
|
||||||
@@ -281,11 +275,8 @@ export function ProjectSelectorWithOptions({
|
|||||||
onValueChange={(value) => {
|
onValueChange={(value) => {
|
||||||
if (currentProject) {
|
if (currentProject) {
|
||||||
setPreviewTheme(null);
|
setPreviewTheme(null);
|
||||||
if (value !== '') {
|
// Only set project theme - don't change global theme
|
||||||
setTheme(value as ThemeMode);
|
// The UI uses getEffectiveTheme() which handles: previewTheme ?? projectTheme ?? globalTheme
|
||||||
} else {
|
|
||||||
setTheme(globalTheme);
|
|
||||||
}
|
|
||||||
setProjectTheme(
|
setProjectTheme(
|
||||||
currentProject.id,
|
currentProject.id,
|
||||||
value === '' ? null : (value as ThemeMode)
|
value === '' ? null : (value as ThemeMode)
|
||||||
|
|||||||
@@ -9,19 +9,15 @@ export const ThemeMenuItem = memo(function ThemeMenuItem({
|
|||||||
}: ThemeMenuItemProps) {
|
}: ThemeMenuItemProps) {
|
||||||
const Icon = option.icon;
|
const Icon = option.icon;
|
||||||
return (
|
return (
|
||||||
<div
|
|
||||||
key={option.value}
|
|
||||||
onPointerEnter={() => onPreviewEnter(option.value)}
|
|
||||||
onPointerLeave={onPreviewLeave}
|
|
||||||
>
|
|
||||||
<DropdownMenuRadioItem
|
<DropdownMenuRadioItem
|
||||||
value={option.value}
|
value={option.value}
|
||||||
data-testid={`project-theme-${option.value}`}
|
data-testid={`project-theme-${option.value}`}
|
||||||
className="text-xs py-1.5"
|
className="text-xs py-1.5"
|
||||||
|
onPointerEnter={() => onPreviewEnter(option.value)}
|
||||||
|
onPointerLeave={onPreviewLeave}
|
||||||
>
|
>
|
||||||
<Icon className="w-3.5 h-3.5 mr-1.5" style={{ color: option.color }} />
|
<Icon className="w-3.5 h-3.5 mr-1.5" style={{ color: option.color }} />
|
||||||
<span>{option.label}</span>
|
<span>{option.label}</span>
|
||||||
</DropdownMenuRadioItem>
|
</DropdownMenuRadioItem>
|
||||||
</div>
|
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -68,10 +68,10 @@ export function ProjectIdentitySection({ project }: ProjectIdentitySectionProps)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Validate file size (max 2MB for icons)
|
// Validate file size (max 5MB for icons - allows animated GIFs)
|
||||||
if (file.size > 2 * 1024 * 1024) {
|
if (file.size > 5 * 1024 * 1024) {
|
||||||
toast.error('File too large', {
|
toast.error('File too large', {
|
||||||
description: 'Please upload an image smaller than 2MB.',
|
description: 'Please upload an image smaller than 5MB.',
|
||||||
});
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -208,7 +208,7 @@ export function ProjectIdentitySection({ project }: ProjectIdentitySectionProps)
|
|||||||
{isUploadingIcon ? 'Uploading...' : 'Upload Custom Icon'}
|
{isUploadingIcon ? 'Uploading...' : 'Upload Custom Icon'}
|
||||||
</Button>
|
</Button>
|
||||||
<p className="text-xs text-muted-foreground mt-1">
|
<p className="text-xs text-muted-foreground mt-1">
|
||||||
PNG, JPG, GIF or WebP. Max 2MB.
|
PNG, JPG, GIF or WebP. Max 5MB.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -946,7 +946,7 @@ export class HttpApiClient implements ElectronAPI {
|
|||||||
return response.json();
|
return response.json();
|
||||||
}
|
}
|
||||||
|
|
||||||
private async get<T>(endpoint: string): Promise<T> {
|
async get<T>(endpoint: string): Promise<T> {
|
||||||
// Ensure API key is initialized before making request
|
// Ensure API key is initialized before making request
|
||||||
await waitForApiKeyInit();
|
await waitForApiKeyInit();
|
||||||
const response = await fetch(`${this.serverUrl}${endpoint}`, {
|
const response = await fetch(`${this.serverUrl}${endpoint}`, {
|
||||||
@@ -976,7 +976,7 @@ export class HttpApiClient implements ElectronAPI {
|
|||||||
return response.json();
|
return response.json();
|
||||||
}
|
}
|
||||||
|
|
||||||
private async put<T>(endpoint: string, body?: unknown): Promise<T> {
|
async put<T>(endpoint: string, body?: unknown): Promise<T> {
|
||||||
// Ensure API key is initialized before making request
|
// Ensure API key is initialized before making request
|
||||||
await waitForApiKeyInit();
|
await waitForApiKeyInit();
|
||||||
const response = await fetch(`${this.serverUrl}${endpoint}`, {
|
const response = await fetch(`${this.serverUrl}${endpoint}`, {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { create } from 'zustand';
|
import { create } from 'zustand';
|
||||||
// Note: persist middleware removed - settings now sync via API (use-settings-sync.ts)
|
// Note: persist middleware removed - settings now sync via API (use-settings-sync.ts)
|
||||||
import type { Project, TrashedProject } from '@/lib/electron';
|
import type { Project, TrashedProject } from '@/lib/electron';
|
||||||
import { getElectronAPI } from '@/lib/electron';
|
import { saveProjects, saveTrashedProjects } from '@/lib/electron';
|
||||||
import { getHttpApiClient } from '@/lib/http-api-client';
|
import { getHttpApiClient } from '@/lib/http-api-client';
|
||||||
import { createLogger } from '@automaker/utils/logger';
|
import { createLogger } from '@automaker/utils/logger';
|
||||||
// Note: setItem/getItem moved to ./utils/theme-utils.ts
|
// Note: setItem/getItem moved to ./utils/theme-utils.ts
|
||||||
@@ -360,7 +360,7 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
|
|
||||||
const trashedProject: TrashedProject = {
|
const trashedProject: TrashedProject = {
|
||||||
...project,
|
...project,
|
||||||
trashedAt: Date.now(),
|
trashedAt: new Date().toISOString(),
|
||||||
};
|
};
|
||||||
|
|
||||||
set((state) => ({
|
set((state) => ({
|
||||||
@@ -369,12 +369,9 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
currentProject: state.currentProject?.id === projectId ? null : state.currentProject,
|
currentProject: state.currentProject?.id === projectId ? null : state.currentProject,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveProjects(get().projects);
|
||||||
if (electronAPI) {
|
saveTrashedProjects(get().trashedProjects);
|
||||||
electronAPI.projects.setProjects(get().projects);
|
|
||||||
electronAPI.projects.setTrashedProjects(get().trashedProjects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
restoreTrashedProject: (projectId: string) => {
|
restoreTrashedProject: (projectId: string) => {
|
||||||
@@ -390,12 +387,9 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
trashedProjects: state.trashedProjects.filter((p) => p.id !== projectId),
|
trashedProjects: state.trashedProjects.filter((p) => p.id !== projectId),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveProjects(get().projects);
|
||||||
if (electronAPI) {
|
saveTrashedProjects(get().trashedProjects);
|
||||||
electronAPI.projects.setProjects(get().projects);
|
|
||||||
electronAPI.projects.setTrashedProjects(get().trashedProjects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
deleteTrashedProject: (projectId: string) => {
|
deleteTrashedProject: (projectId: string) => {
|
||||||
@@ -403,21 +397,15 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
trashedProjects: state.trashedProjects.filter((p) => p.id !== projectId),
|
trashedProjects: state.trashedProjects.filter((p) => p.id !== projectId),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveTrashedProjects(get().trashedProjects);
|
||||||
if (electronAPI) {
|
|
||||||
electronAPI.projects.setTrashedProjects(get().trashedProjects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
emptyTrash: () => {
|
emptyTrash: () => {
|
||||||
set({ trashedProjects: [] });
|
set({ trashedProjects: [] });
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveTrashedProjects([]);
|
||||||
if (electronAPI) {
|
|
||||||
electronAPI.projects.setTrashedProjects([]);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
setCurrentProject: (project) => {
|
setCurrentProject: (project) => {
|
||||||
@@ -474,14 +462,10 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
get().addProject(newProject);
|
get().addProject(newProject);
|
||||||
get().setCurrentProject(newProject);
|
get().setCurrentProject(newProject);
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage (small delay to ensure state is updated)
|
||||||
const electronAPI = getElectronAPI();
|
|
||||||
if (electronAPI) {
|
|
||||||
// Small delay to ensure state is updated before persisting
|
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
electronAPI.projects.setProjects(get().projects);
|
saveProjects(get().projects);
|
||||||
}, 0);
|
}, 0);
|
||||||
}
|
|
||||||
|
|
||||||
return newProject;
|
return newProject;
|
||||||
},
|
},
|
||||||
@@ -564,11 +548,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
),
|
),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveProjects(get().projects);
|
||||||
if (electronAPI) {
|
|
||||||
electronAPI.projects.setProjects(get().projects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
setProjectIcon: (projectId: string, icon: string | null) => {
|
setProjectIcon: (projectId: string, icon: string | null) => {
|
||||||
@@ -576,27 +557,31 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
projects: state.projects.map((p) =>
|
projects: state.projects.map((p) =>
|
||||||
p.id === projectId ? { ...p, icon: icon ?? undefined } : p
|
p.id === projectId ? { ...p, icon: icon ?? undefined } : p
|
||||||
),
|
),
|
||||||
|
// Also update currentProject if it's the one being modified
|
||||||
|
currentProject:
|
||||||
|
state.currentProject?.id === projectId
|
||||||
|
? { ...state.currentProject, icon: icon ?? undefined }
|
||||||
|
: state.currentProject,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveProjects(get().projects);
|
||||||
if (electronAPI) {
|
|
||||||
electronAPI.projects.setProjects(get().projects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
setProjectCustomIcon: (projectId: string, customIconPath: string | null) => {
|
setProjectCustomIcon: (projectId: string, customIconPath: string | null) => {
|
||||||
set((state) => ({
|
set((state) => ({
|
||||||
projects: state.projects.map((p) =>
|
projects: state.projects.map((p) =>
|
||||||
p.id === projectId ? { ...p, customIcon: customIconPath ?? undefined } : p
|
p.id === projectId ? { ...p, customIconPath: customIconPath ?? undefined } : p
|
||||||
),
|
),
|
||||||
|
// Also update currentProject if it's the one being modified
|
||||||
|
currentProject:
|
||||||
|
state.currentProject?.id === projectId
|
||||||
|
? { ...state.currentProject, customIconPath: customIconPath ?? undefined }
|
||||||
|
: state.currentProject,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveProjects(get().projects);
|
||||||
if (electronAPI) {
|
|
||||||
electronAPI.projects.setProjects(get().projects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
setProjectName: (projectId: string, name: string) => {
|
setProjectName: (projectId: string, name: string) => {
|
||||||
@@ -609,11 +594,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
: state.currentProject,
|
: state.currentProject,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveProjects(get().projects);
|
||||||
if (electronAPI) {
|
|
||||||
electronAPI.projects.setProjects(get().projects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
// View actions
|
// View actions
|
||||||
@@ -659,11 +641,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveProjects(get().projects);
|
||||||
if (electronAPI) {
|
|
||||||
electronAPI.projects.setProjects(get().projects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
getEffectiveTheme: () => {
|
getEffectiveTheme: () => {
|
||||||
const state = get();
|
const state = get();
|
||||||
@@ -696,11 +675,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
: state.currentProject,
|
: state.currentProject,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveProjects(get().projects);
|
||||||
if (electronAPI) {
|
|
||||||
electronAPI.projects.setProjects(get().projects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
setProjectFontMono: (projectId: string, fontFamily: string | null) => {
|
setProjectFontMono: (projectId: string, fontFamily: string | null) => {
|
||||||
set((state) => ({
|
set((state) => ({
|
||||||
@@ -714,20 +690,17 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
: state.currentProject,
|
: state.currentProject,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveProjects(get().projects);
|
||||||
if (electronAPI) {
|
|
||||||
electronAPI.projects.setProjects(get().projects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
getEffectiveFontSans: () => {
|
getEffectiveFontSans: () => {
|
||||||
const state = get();
|
const state = get();
|
||||||
const projectFont = state.currentProject?.fontSans;
|
const projectFont = state.currentProject?.fontFamilySans;
|
||||||
return getEffectiveFont(projectFont, state.fontFamilySans, UI_SANS_FONT_OPTIONS);
|
return getEffectiveFont(projectFont, state.fontFamilySans, UI_SANS_FONT_OPTIONS);
|
||||||
},
|
},
|
||||||
getEffectiveFontMono: () => {
|
getEffectiveFontMono: () => {
|
||||||
const state = get();
|
const state = get();
|
||||||
const projectFont = state.currentProject?.fontMono;
|
const projectFont = state.currentProject?.fontFamilyMono;
|
||||||
return getEffectiveFont(projectFont, state.fontFamilyMono, UI_MONO_FONT_OPTIONS);
|
return getEffectiveFont(projectFont, state.fontFamilyMono, UI_MONO_FONT_OPTIONS);
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -744,11 +717,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
: state.currentProject,
|
: state.currentProject,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveProjects(get().projects);
|
||||||
if (electronAPI) {
|
|
||||||
electronAPI.projects.setProjects(get().projects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
// Project Phase Model Overrides
|
// Project Phase Model Overrides
|
||||||
@@ -781,11 +751,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveProjects(get().projects);
|
||||||
if (electronAPI) {
|
|
||||||
electronAPI.projects.setProjects(get().projects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
clearAllProjectPhaseModelOverrides: (projectId: string) => {
|
clearAllProjectPhaseModelOverrides: (projectId: string) => {
|
||||||
@@ -804,11 +771,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveProjects(get().projects);
|
||||||
if (electronAPI) {
|
|
||||||
electronAPI.projects.setProjects(get().projects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
// Project Default Feature Model Override
|
// Project Default Feature Model Override
|
||||||
@@ -830,11 +794,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
// Persist to Electron store if available
|
// Persist to storage
|
||||||
const electronAPI = getElectronAPI();
|
saveProjects(get().projects);
|
||||||
if (electronAPI) {
|
|
||||||
electronAPI.projects.setProjects(get().projects);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
// Feature actions
|
// Feature actions
|
||||||
@@ -845,7 +806,7 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
})),
|
})),
|
||||||
addFeature: (feature) => {
|
addFeature: (feature) => {
|
||||||
const id = feature.id ?? `feature-${Date.now()}-${Math.random().toString(36).slice(2)}`;
|
const id = feature.id ?? `feature-${Date.now()}-${Math.random().toString(36).slice(2)}`;
|
||||||
const newFeature: Feature = { ...feature, id };
|
const newFeature = { ...feature, id } as Feature;
|
||||||
set((state) => ({ features: [...state.features, newFeature] }));
|
set((state) => ({ features: [...state.features, newFeature] }));
|
||||||
return newFeature;
|
return newFeature;
|
||||||
},
|
},
|
||||||
@@ -2471,8 +2432,7 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const httpApi = getHttpApiClient();
|
const httpApi = getHttpApiClient();
|
||||||
const response = await httpApi.get('/api/codex/models');
|
const data = await httpApi.get<{
|
||||||
const data = response.data as {
|
|
||||||
success: boolean;
|
success: boolean;
|
||||||
models?: Array<{
|
models?: Array<{
|
||||||
id: string;
|
id: string;
|
||||||
@@ -2484,7 +2444,7 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
isDefault: boolean;
|
isDefault: boolean;
|
||||||
}>;
|
}>;
|
||||||
error?: string;
|
error?: string;
|
||||||
};
|
}>('/api/codex/models');
|
||||||
|
|
||||||
if (data.success && data.models) {
|
if (data.success && data.models) {
|
||||||
set({
|
set({
|
||||||
@@ -2542,8 +2502,7 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const httpApi = getHttpApiClient();
|
const httpApi = getHttpApiClient();
|
||||||
const response = await httpApi.get('/api/opencode/models');
|
const data = await httpApi.get<{
|
||||||
const data = response.data as {
|
|
||||||
success: boolean;
|
success: boolean;
|
||||||
models?: ModelDefinition[];
|
models?: ModelDefinition[];
|
||||||
providers?: Array<{
|
providers?: Array<{
|
||||||
@@ -2553,7 +2512,7 @@ export const useAppStore = create<AppState & AppActions>()((set, get) => ({
|
|||||||
authMethod?: string;
|
authMethod?: string;
|
||||||
}>;
|
}>;
|
||||||
error?: string;
|
error?: string;
|
||||||
};
|
}>('/api/opencode/models');
|
||||||
|
|
||||||
if (data.success && data.models) {
|
if (data.success && data.models) {
|
||||||
// Filter out Bedrock models
|
// Filter out Bedrock models
|
||||||
|
|||||||
Reference in New Issue
Block a user