feat(electrobun): project wizard phases 1-5 (WIP)

- sanitize.ts: input sanitization (trim, control chars, path traversal)
- provider-scanner.ts: detect Claude/Codex/Ollama/Gemini availability
- model-fetcher.ts: live model lists from 4 provider APIs
- ModelConfigPanel.svelte: per-provider config (thinking, effort, sandbox, temperature)
- WizardStep1-3.svelte: split wizard into composable steps
- CustomDropdown/Checkbox/Radio: themed UI components
- provider-handlers.ts: provider.scan + provider.models RPC
- Wire providers into wizard step 3 (live detection + model lists)
- Replace native selects in 5 settings panels with CustomDropdown
This commit is contained in:
Hibryda 2026-03-23 13:05:07 +01:00
parent b7fc3a0f9b
commit d4014a193d
25 changed files with 2112 additions and 759 deletions

View file

@ -0,0 +1,284 @@
<script lang="ts">
import { t } from './i18n.svelte.ts';
import { appRpc } from './rpc.ts';
import { sanitize, sanitizeUrl, sanitizePath, isValidGitUrl, isValidGithubRepo } from './sanitize.ts';
import PathBrowser from './PathBrowser.svelte';
type SourceType = 'local' | 'git-clone' | 'github' | 'template' | 'remote';
type AuthMethod = 'password' | 'key' | 'agent' | 'config';
type PathState = 'idle' | 'checking' | 'valid' | 'invalid' | 'not-dir';
interface Props {
sourceType: SourceType;
localPath: string;
repoUrl: string;
cloneTarget: string;
githubRepo: string;
selectedTemplate: string;
remoteHost: string;
remoteUser: string;
remotePath: string;
remoteAuthMethod: AuthMethod;
remotePassword: string;
remoteKeyPath: string;
remoteSshfs: boolean;
pathValid: PathState;
isGitRepo: boolean;
gitBranch: string;
gitProbeStatus: 'idle' | 'probing' | 'ok' | 'error';
gitProbeBranches: string[];
githubInfo: { stars: number; description: string; defaultBranch: string } | null;
githubLoading: boolean;
cloning: boolean;
templates: Array<{ id: string; name: string; description: string; icon: string }>;
templateTargetDir: string;
onUpdate: (field: string, value: unknown) => void;
}
let {
sourceType, localPath, repoUrl, cloneTarget, githubRepo,
selectedTemplate, remoteHost, remoteUser, remotePath,
remoteAuthMethod, remotePassword, remoteKeyPath, remoteSshfs,
pathValid, isGitRepo, gitBranch,
gitProbeStatus, gitProbeBranches, githubInfo, githubLoading,
cloning, templates, templateTargetDir, onUpdate,
}: Props = $props();
let showBrowser = $state(false);
let showCloneBrowser = $state(false);
let showTemplateBrowser = $state(false);
let showKeyBrowser = $state(false);
const SOURCE_TYPES: Array<{ value: SourceType; label: string; icon: string }> = [
{ value: 'local', label: 'Local Folder', icon: 'folder' },
{ value: 'git-clone', label: 'Git Clone', icon: 'git-branch' },
{ value: 'github', label: 'GitHub Repo', icon: 'github' },
{ value: 'template', label: 'Template', icon: 'layout-template' },
{ value: 'remote', label: 'SSH Remote', icon: 'monitor' },
];
const AUTH_METHODS: Array<{ value: AuthMethod; label: string }> = [
{ value: 'password', label: 'Password' },
{ value: 'key', label: 'SSH Key' },
{ value: 'agent', label: 'SSH Agent' },
{ value: 'config', label: 'SSH Config' },
];
function validationIcon(state: PathState): string {
switch (state) {
case 'valid': return '\u2713';
case 'invalid': return '\u2717';
case 'not-dir': return '\u26A0';
case 'checking': return '\u2026';
default: return '';
}
}
function validationColor(state: PathState): string {
switch (state) {
case 'valid': return 'var(--ctp-green)';
case 'invalid': return 'var(--ctp-red)';
case 'not-dir': return 'var(--ctp-peach)';
default: return 'var(--ctp-overlay0)';
}
}
async function handleNativeBrowse(target: 'local' | 'clone' | 'template' | 'key') {
try {
const start = target === 'local' ? localPath : target === 'clone' ? cloneTarget : target === 'template' ? templateTargetDir : remoteKeyPath;
const result = await appRpc.request['files.pickDirectory']({ startingFolder: start || '~/' });
if (result?.path) {
const field = target === 'local' ? 'localPath' : target === 'clone' ? 'cloneTarget' : target === 'template' ? 'templateTargetDir' : 'remoteKeyPath';
onUpdate(field, result.path);
}
} catch { /* native dialog not available */ }
}
function handleBrowserSelect(field: string, path: string) {
onUpdate(field, path);
showBrowser = false;
showCloneBrowser = false;
showTemplateBrowser = false;
showKeyBrowser = false;
}
</script>
<div class="wz-radios">
{#each SOURCE_TYPES as opt}
<label class="wz-radio" class:selected={sourceType === opt.value}>
<input type="radio" name="source" value={opt.value}
checked={sourceType === opt.value}
onchange={() => onUpdate('sourceType', opt.value)} />
<span>{opt.label}</span>
</label>
{/each}
</div>
<div class="wz-source-fields">
<!-- Local Folder -->
<div style:display={sourceType === 'local' ? 'flex' : 'none'} class="wz-field-col">
<label class="wz-label">Project directory</label>
<div class="wz-path-row">
<input class="wz-input" type="text" placeholder="/home/user/project"
value={localPath} oninput={(e) => onUpdate('localPath', (e.target as HTMLInputElement).value)} />
<button class="wz-browse-btn" onclick={() => handleNativeBrowse('local')} title="System picker">&#128194;</button>
<button class="wz-browse-btn" onclick={() => showBrowser = !showBrowser} title="In-app browser">&#128269;</button>
{#if pathValid !== 'idle'}
<span class="wz-validation" style:color={validationColor(pathValid)}>{validationIcon(pathValid)}</span>
{/if}
</div>
<div style:display={showBrowser ? 'block' : 'none'} class="wz-browser-wrap">
<PathBrowser onSelect={(p) => handleBrowserSelect('localPath', p)} onClose={() => showBrowser = false} />
</div>
{#if pathValid === 'valid' && isGitRepo}
<span class="wz-badge git-badge">Git repo ({gitBranch})</span>
{/if}
{#if pathValid === 'invalid'}<span class="wz-hint error">Path does not exist</span>{/if}
{#if pathValid === 'not-dir'}<span class="wz-hint warn">Not a directory</span>{/if}
</div>
<!-- Git Clone -->
<div style:display={sourceType === 'git-clone' ? 'flex' : 'none'} class="wz-field-col">
<label class="wz-label">Repository URL</label>
<div class="wz-path-row">
<input class="wz-input" type="text" placeholder="https://github.com/user/repo.git"
value={repoUrl} oninput={(e) => onUpdate('repoUrl', (e.target as HTMLInputElement).value)} />
{#if gitProbeStatus === 'probing'}<span class="wz-validation" style:color="var(--ctp-blue)">&hellip;</span>{/if}
{#if gitProbeStatus === 'ok'}<span class="wz-validation" style:color="var(--ctp-green)">&check;</span>{/if}
{#if gitProbeStatus === 'error'}<span class="wz-validation" style:color="var(--ctp-red)">&cross;</span>{/if}
</div>
{#if gitProbeStatus === 'ok' && gitProbeBranches.length > 0}
<span class="wz-hint" style="color: var(--ctp-subtext0);">{gitProbeBranches.length} branches found</span>
{/if}
<label class="wz-label">Target directory</label>
<div class="wz-path-row">
<input class="wz-input" type="text" placeholder="~/projects/my-repo"
value={cloneTarget} oninput={(e) => onUpdate('cloneTarget', (e.target as HTMLInputElement).value)} />
<button class="wz-browse-btn" onclick={() => handleNativeBrowse('clone')}>&#128194;</button>
</div>
</div>
<!-- GitHub -->
<div style:display={sourceType === 'github' ? 'flex' : 'none'} class="wz-field-col">
<label class="wz-label">GitHub repository (owner/repo)</label>
<input class="wz-input" type="text" placeholder="owner/repo"
value={githubRepo} oninput={(e) => onUpdate('githubRepo', (e.target as HTMLInputElement).value)} />
{#if githubLoading}<span class="wz-hint" style="color: var(--ctp-blue);">Checking&hellip;</span>{/if}
{#if githubInfo}
<div class="wz-github-info">
<span class="wz-hint" style="color: var(--ctp-yellow);">&starf; {githubInfo.stars}</span>
<span class="wz-hint" style="color: var(--ctp-subtext0);">{githubInfo.description}</span>
<span class="wz-hint" style="color: var(--ctp-green);">Default: {githubInfo.defaultBranch}</span>
</div>
{/if}
</div>
<!-- Template -->
<div style:display={sourceType === 'template' ? 'flex' : 'none'} class="wz-field-col">
<label class="wz-label">Target directory</label>
<div class="wz-path-row">
<input class="wz-input" type="text" placeholder="~/projects"
value={templateTargetDir} oninput={(e) => onUpdate('templateTargetDir', (e.target as HTMLInputElement).value)} />
<button class="wz-browse-btn" onclick={() => handleNativeBrowse('template')}>&#128194;</button>
</div>
<div class="wz-template-grid">
{#each templates as tmpl}
<button class="wz-template-card" class:selected={selectedTemplate === tmpl.id}
onclick={() => onUpdate('selectedTemplate', tmpl.id)}>
<span class="wz-template-icon">{tmpl.icon}</span>
<span class="wz-template-name">{tmpl.name}</span>
<span class="wz-template-desc">{tmpl.description}</span>
</button>
{/each}
</div>
</div>
<!-- SSH Remote -->
<div style:display={sourceType === 'remote' ? 'flex' : 'none'} class="wz-field-col">
<label class="wz-label">Host</label>
<input class="wz-input" type="text" placeholder="192.168.1.100"
value={remoteHost} oninput={(e) => onUpdate('remoteHost', (e.target as HTMLInputElement).value)} />
<label class="wz-label">User</label>
<input class="wz-input" type="text" placeholder="user"
value={remoteUser} oninput={(e) => onUpdate('remoteUser', (e.target as HTMLInputElement).value)} />
<label class="wz-label">Remote path</label>
<input class="wz-input" type="text" placeholder="/home/user/project"
value={remotePath} oninput={(e) => onUpdate('remotePath', (e.target as HTMLInputElement).value)} />
<label class="wz-label">Auth method</label>
<div class="wz-segmented">
{#each AUTH_METHODS as m}
<button class="wz-seg-btn" class:active={remoteAuthMethod === m.value}
onclick={() => onUpdate('remoteAuthMethod', m.value)}>{m.label}</button>
{/each}
</div>
<div style:display={remoteAuthMethod === 'password' ? 'flex' : 'none'} class="wz-field-col">
<label class="wz-label">Password</label>
<input class="wz-input" type="password" placeholder="&bull;&bull;&bull;&bull;&bull;&bull;"
value={remotePassword} oninput={(e) => onUpdate('remotePassword', (e.target as HTMLInputElement).value)} />
</div>
<div style:display={remoteAuthMethod === 'key' ? 'flex' : 'none'} class="wz-field-col">
<label class="wz-label">Key file path</label>
<div class="wz-path-row">
<input class="wz-input" type="text" placeholder="~/.ssh/id_ed25519"
value={remoteKeyPath} oninput={(e) => onUpdate('remoteKeyPath', (e.target as HTMLInputElement).value)} />
<button class="wz-browse-btn" onclick={() => handleNativeBrowse('key')}>&#128194;</button>
</div>
</div>
<label class="wz-toggle-row">
<input type="checkbox" checked={remoteSshfs}
onchange={(e) => onUpdate('remoteSshfs', (e.target as HTMLInputElement).checked)} />
<span>Mount via SSHFS</span>
</label>
</div>
</div>
<style>
.wz-radios { display: flex; flex-wrap: wrap; gap: 0.375rem; }
.wz-radio {
display: flex; align-items: center; gap: 0.375rem;
padding: 0.375rem 0.625rem; border-radius: 0.375rem;
border: 1px solid var(--ctp-surface1); cursor: pointer;
font-size: 0.75rem; color: var(--ctp-subtext0);
transition: border-color 0.12s, background 0.12s;
}
.wz-radio:hover { border-color: var(--ctp-overlay1); color: var(--ctp-text); }
.wz-radio.selected { border-color: var(--ctp-blue); background: color-mix(in srgb, var(--ctp-blue) 10%, transparent); color: var(--ctp-text); }
.wz-radio input[type="radio"] { display: none; }
.wz-source-fields { display: flex; flex-direction: column; gap: 0.375rem; margin-top: 0.25rem; }
.wz-field-col { display: flex; flex-direction: column; gap: 0.375rem; }
.wz-label { font-size: 0.75rem; font-weight: 500; color: var(--ctp-subtext0); }
.wz-input { padding: 0.375rem 0.5rem; background: var(--ctp-surface0); border: 1px solid var(--ctp-surface1); border-radius: 0.25rem; color: var(--ctp-text); font-size: 0.8125rem; font-family: var(--ui-font-family); width: 100%; }
.wz-input:focus { outline: none; border-color: var(--ctp-blue); }
.wz-input::placeholder { color: var(--ctp-overlay0); }
.wz-path-row { display: flex; gap: 0.375rem; align-items: center; }
.wz-path-row .wz-input { flex: 1; }
.wz-browse-btn { padding: 0.375rem; background: var(--ctp-surface0); border: 1px solid var(--ctp-surface1); border-radius: 0.25rem; cursor: pointer; font-size: 0.875rem; flex-shrink: 0; }
.wz-browse-btn:hover { background: var(--ctp-surface1); }
.wz-validation { font-size: 0.875rem; font-weight: 700; flex-shrink: 0; }
.wz-badge { display: inline-flex; align-items: center; gap: 0.25rem; padding: 0.125rem 0.5rem; border-radius: 0.25rem; font-size: 0.6875rem; width: fit-content; }
.git-badge { background: color-mix(in srgb, var(--ctp-green) 15%, transparent); color: var(--ctp-green); border: 1px solid color-mix(in srgb, var(--ctp-green) 30%, transparent); }
.wz-hint { font-size: 0.6875rem; }
.wz-hint.error { color: var(--ctp-red); }
.wz-hint.warn { color: var(--ctp-peach); }
.wz-browser-wrap { max-height: 16rem; max-width: 32rem; overflow-y: auto; border: 1px solid var(--ctp-surface1); border-radius: 0.375rem; margin-top: 0.5rem; }
.wz-template-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 0.375rem; }
.wz-template-card { display: flex; flex-direction: column; align-items: center; gap: 0.25rem; padding: 0.75rem 0.5rem; border-radius: 0.375rem; border: 1px solid var(--ctp-surface1); background: var(--ctp-surface0); cursor: pointer; text-align: center; }
.wz-template-card:hover { border-color: var(--ctp-overlay1); }
.wz-template-card.selected { border-color: var(--ctp-blue); background: color-mix(in srgb, var(--ctp-blue) 10%, transparent); }
.wz-template-icon { font-size: 1.5rem; }
.wz-template-name { font-size: 0.75rem; font-weight: 600; color: var(--ctp-text); }
.wz-template-desc { font-size: 0.625rem; color: var(--ctp-subtext0); }
.wz-segmented { display: flex; gap: 0; border-radius: 0.25rem; overflow: hidden; border: 1px solid var(--ctp-surface1); }
.wz-seg-btn { flex: 1; padding: 0.375rem 0.5rem; font-size: 0.75rem; background: var(--ctp-surface0); border: none; color: var(--ctp-subtext0); cursor: pointer; font-family: var(--ui-font-family); border-right: 1px solid var(--ctp-surface1); }
.wz-seg-btn:last-child { border-right: none; }
.wz-seg-btn:hover { color: var(--ctp-text); }
.wz-seg-btn.active { background: color-mix(in srgb, var(--ctp-blue) 20%, transparent); color: var(--ctp-blue); }
.wz-toggle-row { display: flex; align-items: center; gap: 0.5rem; font-size: 0.8125rem; color: var(--ctp-text); cursor: pointer; margin-top: 0.25rem; }
.wz-github-info { display: flex; flex-direction: column; gap: 0.125rem; margin-top: 0.25rem; }
input[type="checkbox"] { -webkit-appearance: none; appearance: none; width: 1rem; height: 1rem; border: 1px solid var(--ctp-surface2); border-radius: 0.1875rem; background: var(--ctp-surface0); cursor: pointer; position: relative; vertical-align: middle; flex-shrink: 0; }
input[type="checkbox"]:checked { background: var(--ctp-blue); border-color: var(--ctp-blue); }
input[type="checkbox"]:checked::after { content: ''; position: absolute; left: 0.25rem; top: 0.0625rem; width: 0.3125rem; height: 0.5625rem; border: solid var(--ctp-base); border-width: 0 2px 2px 0; transform: rotate(45deg); }
</style>