Merge v0.14.0rc into refactor/store-ui-slice

Resolve merge conflict in app-store.ts by keeping UI slice implementation
of getEffectiveFontSans/getEffectiveFontMono (already provided by ui-slice.ts)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
Kacper
2026-01-27 17:48:41 +01:00
16 changed files with 252 additions and 159 deletions

View File

@@ -43,7 +43,7 @@ export function createCreateHandler(featureLoader: FeatureLoader, events?: Event
if (events) { if (events) {
events.emit('feature:created', { events.emit('feature:created', {
featureId: created.id, featureId: created.id,
featureName: created.name, featureName: created.title || 'Untitled Feature',
projectPath, projectPath,
}); });
} }

View File

@@ -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)

View File

@@ -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

View File

@@ -4597,21 +4597,54 @@ This mock response was generated because AUTOMAKER_MOCK_AGENT=true was set.
planVersion, planVersion,
}); });
// Build revision prompt // Build revision prompt using customizable template
let revisionPrompt = `The user has requested revisions to the plan/specification. const revisionPrompts = await getPromptCustomization(
this.settingsService,
'[AutoMode]'
);
## Previous Plan (v${planVersion - 1}) // Get task format example based on planning mode
${hasEdits ? approvalResult.editedPlan : currentPlanContent} const taskFormatExample =
planningMode === 'full'
? `\`\`\`tasks
## Phase 1: Foundation
- [ ] T001: [Description] | File: [path/to/file]
- [ ] T002: [Description] | File: [path/to/file]
## User Feedback ## Phase 2: Core Implementation
${approvalResult.feedback || 'Please revise the plan based on the edits above.'} - [ ] T003: [Description] | File: [path/to/file]
- [ ] T004: [Description] | File: [path/to/file]
\`\`\``
: `\`\`\`tasks
- [ ] T001: [Description] | File: [path/to/file]
- [ ] T002: [Description] | File: [path/to/file]
- [ ] T003: [Description] | File: [path/to/file]
\`\`\``;
## Instructions let revisionPrompt = revisionPrompts.taskExecution.planRevisionTemplate;
Please regenerate the specification incorporating the user's feedback. revisionPrompt = revisionPrompt.replace(
Keep the same format with the \`\`\`tasks block for task definitions. /\{\{planVersion\}\}/g,
After generating the revised spec, output: String(planVersion - 1)
"[SPEC_GENERATED] Please review the revised specification above." );
`; revisionPrompt = revisionPrompt.replace(
/\{\{previousPlan\}\}/g,
hasEdits
? approvalResult.editedPlan || currentPlanContent
: currentPlanContent
);
revisionPrompt = revisionPrompt.replace(
/\{\{userFeedback\}\}/g,
approvalResult.feedback ||
'Please revise the plan based on the edits above.'
);
revisionPrompt = revisionPrompt.replace(
/\{\{planningMode\}\}/g,
planningMode
);
revisionPrompt = revisionPrompt.replace(
/\{\{taskFormatExample\}\}/g,
taskFormatExample
);
// Update status to regenerating // Update status to regenerating
await this.updateFeaturePlanSpec(projectPath, featureId, { await this.updateFeaturePlanSpec(projectPath, featureId, {
@@ -4663,6 +4696,26 @@ After generating the revised spec, output:
const revisedTasks = parseTasksFromSpec(currentPlanContent); const revisedTasks = parseTasksFromSpec(currentPlanContent);
logger.info(`Revised plan has ${revisedTasks.length} tasks`); logger.info(`Revised plan has ${revisedTasks.length} tasks`);
// Warn if no tasks found in spec/full mode - this may cause fallback to single-agent
if (
revisedTasks.length === 0 &&
(planningMode === 'spec' || planningMode === 'full')
) {
logger.warn(
`WARNING: Revised plan in ${planningMode} mode has no tasks! ` +
`This will cause fallback to single-agent execution. ` +
`The AI may have omitted the required \`\`\`tasks block.`
);
this.emitAutoModeEvent('plan_revision_warning', {
featureId,
projectPath,
branchName,
planningMode,
warning:
'Revised plan missing tasks block - will use single-agent execution',
});
}
// Update planSpec with revised content // Update planSpec with revised content
await this.updateFeaturePlanSpec(projectPath, featureId, { await this.updateFeaturePlanSpec(projectPath, featureId, {
status: 'generated', status: 'generated',

View File

@@ -169,9 +169,10 @@ export class EventHookService {
} }
// Build context for variable substitution // Build context for variable substitution
// Use loaded featureName (from feature.title) or fall back to payload.featureName
const context: HookContext = { const context: HookContext = {
featureId: payload.featureId, featureId: payload.featureId,
featureName: payload.featureName, featureName: featureName || payload.featureName,
projectPath: payload.projectPath, projectPath: payload.projectPath,
projectName: payload.projectPath ? this.extractProjectName(payload.projectPath) : undefined, projectName: payload.projectPath ? this.extractProjectName(payload.projectPath) : undefined,
error: payload.error || payload.message, error: payload.error || payload.message,

View File

@@ -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>

View File

@@ -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',

View File

@@ -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

View File

@@ -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)

View File

@@ -1,8 +1,11 @@
import { useCallback, useEffect, useRef } from 'react'; import { useCallback, useEffect, useRef } from 'react';
import type { NavigateOptions } from '@tanstack/react-router'; import type { NavigateOptions } from '@tanstack/react-router';
import { ChevronDown, Wrench, Github } from 'lucide-react'; import { ChevronDown, Wrench, Github, Folder } from 'lucide-react';
import * as LucideIcons from 'lucide-react';
import type { LucideIcon } from 'lucide-react';
import { cn } from '@/lib/utils'; import { cn } from '@/lib/utils';
import { formatShortcut, useAppStore } from '@/store/app-store'; import { formatShortcut, useAppStore } from '@/store/app-store';
import { getAuthenticatedImageUrl } from '@/lib/api-fetch';
import type { NavSection } from '../types'; import type { NavSection } from '../types';
import type { Project } from '@/lib/electron'; import type { Project } from '@/lib/electron';
import type { SidebarStyle } from '@automaker/types'; import type { SidebarStyle } from '@automaker/types';
@@ -97,6 +100,17 @@ export function SidebarNavigation({
return !!currentProject; return !!currentProject;
}); });
// Get the icon component for the current project
const getProjectIcon = (): LucideIcon => {
if (currentProject?.icon && currentProject.icon in LucideIcons) {
return (LucideIcons as unknown as Record<string, LucideIcon>)[currentProject.icon];
}
return Folder;
};
const ProjectIcon = getProjectIcon();
const hasCustomIcon = !!currentProject?.customIconPath;
return ( return (
<nav <nav
ref={navRef} ref={navRef}
@@ -106,6 +120,27 @@ export function SidebarNavigation({
sidebarStyle === 'discord' ? 'pt-3' : 'mt-1' sidebarStyle === 'discord' ? 'pt-3' : 'mt-1'
)} )}
> >
{/* Project name display for classic/discord mode */}
{sidebarStyle === 'discord' && currentProject && sidebarOpen && (
<div className="mb-3">
<div className="flex items-center gap-2.5 px-3 py-2">
{hasCustomIcon ? (
<img
src={getAuthenticatedImageUrl(currentProject.customIconPath!, currentProject.path)}
alt={currentProject.name}
className="w-5 h-5 rounded object-cover"
/>
) : (
<ProjectIcon className="w-5 h-5 text-brand-500 shrink-0" />
)}
<span className="text-sm font-medium text-foreground truncate">
{currentProject.name}
</span>
</div>
<div className="h-px bg-border/40 mx-1 mt-1" />
</div>
)}
{/* Navigation sections */} {/* Navigation sections */}
{visibleSections.map((section, sectionIdx) => { {visibleSections.map((section, sectionIdx) => {
const isCollapsed = section.label ? collapsedNavSections[section.label] : false; const isCollapsed = section.label ? collapsedNavSections[section.label] : false;

View File

@@ -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>
); );
}); });

View File

@@ -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>

View File

@@ -19,6 +19,7 @@ import type { StoredEventSummary, StoredEvent, EventHookTrigger } from '@automak
import { EVENT_HOOK_TRIGGER_LABELS } from '@automaker/types'; import { EVENT_HOOK_TRIGGER_LABELS } from '@automaker/types';
import { getHttpApiClient } from '@/lib/http-api-client'; import { getHttpApiClient } from '@/lib/http-api-client';
import { ConfirmDialog } from '@/components/ui/confirm-dialog'; import { ConfirmDialog } from '@/components/ui/confirm-dialog';
import { toast } from 'sonner';
export function EventHistoryView() { export function EventHistoryView() {
const currentProject = useAppStore((state) => state.currentProject); const currentProject = useAppStore((state) => state.currentProject);
@@ -85,16 +86,18 @@ export function EventHistoryView() {
const failCount = hookResults.filter((r) => !r.success).length; const failCount = hookResults.filter((r) => !r.success).length;
if (hooksTriggered === 0) { if (hooksTriggered === 0) {
alert('No matching hooks found for this event trigger.'); toast.info('No matching hooks found for this event trigger.');
} else if (failCount === 0) { } else if (failCount === 0) {
alert(`Successfully ran ${successCount} hook(s).`); toast.success(`Successfully ran ${successCount} hook(s).`);
} else { } else {
alert(`Ran ${hooksTriggered} hook(s): ${successCount} succeeded, ${failCount} failed.`); toast.warning(
`Ran ${hooksTriggered} hook(s): ${successCount} succeeded, ${failCount} failed.`
);
} }
} }
} catch (error) { } catch (error) {
console.error('Failed to replay event:', error); console.error('Failed to replay event:', error);
alert('Failed to replay event. Check console for details.'); toast.error('Failed to replay event. Check console for details.');
} finally { } finally {
setReplayingEvent(null); setReplayingEvent(null);
} }

View File

@@ -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}`, {

View File

@@ -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
@@ -419,7 +419,7 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
const trashedProject: TrashedProject = { const trashedProject: TrashedProject = {
...project, ...project,
trashedAt: Date.now(), trashedAt: new Date().toISOString(),
}; };
set((state) => ({ set((state) => ({
@@ -428,12 +428,9 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
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) => {
@@ -449,12 +446,9 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
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) => {
@@ -462,21 +456,15 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
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) => {
@@ -533,14 +521,10 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
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;
}, },
@@ -623,11 +607,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
), ),
})); }));
// 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) => {
@@ -635,27 +616,31 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
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) => {
@@ -668,11 +653,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
: 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 - provided by UI slice // View actions - provided by UI slice
@@ -699,11 +681,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
); );
} }
// Persist to Electron store if available // Persist to storage
const electronAPI = getElectronAPI(); saveProjects(get().projects);
if (electronAPI) {
electronAPI.projects.setProjects(get().projects);
}
}, },
// Font actions (setFontSans, setFontMono, getEffectiveFontSans, getEffectiveFontMono provided by UI slice) // Font actions (setFontSans, setFontMono, getEffectiveFontSans, getEffectiveFontMono provided by UI slice)
@@ -719,11 +698,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
: 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) => ({
@@ -737,11 +713,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
: 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);
}
}, },
// Claude API Profile actions (per-project override) // Claude API Profile actions (per-project override)
@@ -757,11 +730,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
: 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
@@ -794,11 +764,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
}; };
}); });
// 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) => {
@@ -817,11 +784,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
}; };
}); });
// 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
@@ -843,11 +807,8 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
}; };
}); });
// 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
@@ -858,7 +819,7 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
})), })),
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;
}, },
@@ -2346,8 +2307,7 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
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;
@@ -2359,7 +2319,7 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
isDefault: boolean; isDefault: boolean;
}>; }>;
error?: string; error?: string;
}; }>('/api/codex/models');
if (data.success && data.models) { if (data.success && data.models) {
set({ set({
@@ -2417,8 +2377,7 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
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<{
@@ -2428,7 +2387,7 @@ export const useAppStore = create<AppState & AppActions>()((set, get, store) =>
authMethod?: string; authMethod?: string;
}>; }>;
error?: string; error?: string;
}; }>('/api/setup/opencode/models');
if (data.success && data.models) { if (data.success && data.models) {
// Filter out Bedrock models // Filter out Bedrock models

View File

@@ -965,8 +965,20 @@ export const DEFAULT_PLAN_REVISION_TEMPLATE = `The user has requested revisions
## Instructions ## Instructions
Please regenerate the specification incorporating the user's feedback. Please regenerate the specification incorporating the user's feedback.
Keep the same format with the \`\`\`tasks block for task definitions. **Current planning mode: {{planningMode}}**
After generating the revised spec, output:
**CRITICAL REQUIREMENT**: Your revised specification MUST include a \`\`\`tasks code block containing task definitions in the EXACT format shown below. This is MANDATORY - without the tasks block, the system cannot track or execute tasks properly.
### Required Task Format
{{taskFormatExample}}
**IMPORTANT**:
1. The \`\`\`tasks block must appear in your response
2. Each task MUST start with "- [ ] T###:" where ### is a sequential number (T001, T002, T003, etc.)
3. Each task MUST include "| File:" followed by the primary file path
4. Tasks should be ordered by dependencies (foundational tasks first)
After generating the revised spec with the tasks block, output:
"[SPEC_GENERATED] Please review the revised specification above."`; "[SPEC_GENERATED] Please review the revised specification above."`;
export const DEFAULT_CONTINUATION_AFTER_APPROVAL_TEMPLATE = `The plan/specification has been approved. Now implement it. export const DEFAULT_CONTINUATION_AFTER_APPROVAL_TEMPLATE = `The plan/specification has been approved. Now implement it.