feat: setup wizard page for first-deploy initialization

- SetupWizardPage: step-by-step config (DB, admin, project)
- Connects directly to AbstractWizard via SSH tunnel (127.0.0.1)
- App.tsx: detect backend health, show wizard if not ready
- Auto-switch wizard to readonly after setup
- Add VITE_WIZARD_PORT build arg
- Add vite-env.d.ts for type safety
This commit is contained in:
zhi
2026-03-06 13:46:46 +00:00
parent 54d4c4379a
commit f8fac48fcc
5 changed files with 288 additions and 0 deletions

View File

@@ -160,3 +160,26 @@ dd { font-size: .9rem; }
/* Text dim helper */
.text-dim { color: var(--text-dim); font-size: .85rem; }
/* Setup Wizard */
.setup-wizard { display: flex; align-items: center; justify-content: center; min-height: 100vh; padding: 24px; }
.setup-container { background: var(--bg-card); border: 1px solid var(--border); border-radius: 16px; padding: 40px; max-width: 600px; width: 100%; }
.setup-header { text-align: center; margin-bottom: 32px; }
.setup-header h1 { font-size: 1.5rem; margin-bottom: 20px; }
.setup-steps { display: flex; justify-content: center; gap: 8px; flex-wrap: wrap; }
.setup-step { font-size: .8rem; color: var(--text-dim); padding: 4px 10px; border-radius: 12px; border: 1px solid var(--border); }
.setup-step.active { color: var(--accent); border-color: var(--accent); background: rgba(59,130,246,.1); }
.setup-step.done { color: var(--success); border-color: var(--success); }
.setup-step-content { animation: fadeIn .2s ease; }
.setup-step-content h2 { margin-bottom: 8px; font-size: 1.2rem; }
.setup-form { display: flex; flex-direction: column; gap: 12px; margin: 20px 0; }
.setup-form label { display: flex; flex-direction: column; gap: 4px; font-size: .85rem; color: var(--text-dim); }
.setup-form input { padding: 10px 12px; border: 1px solid var(--border); border-radius: 6px; background: var(--bg); color: var(--text); font-size: .95rem; }
.setup-nav { display: flex; justify-content: space-between; margin-top: 24px; }
.setup-error { background: rgba(239,68,68,.1); border: 1px solid var(--danger); color: var(--danger); padding: 12px 16px; border-radius: 8px; margin-bottom: 16px; font-size: .9rem; white-space: pre-line; }
.setup-info { background: rgba(59,130,246,.08); border: 1px solid rgba(59,130,246,.2); padding: 16px; border-radius: 8px; margin: 16px 0; }
.setup-info code { display: block; background: var(--bg); padding: 8px 12px; border-radius: 4px; margin-top: 8px; font-size: .85rem; color: var(--accent); word-break: break-all; }
.setup-hint { color: var(--warning); font-size: .85rem; margin-top: 8px; }
.setup-done { text-align: center; }
.setup-done h2 { color: var(--success); margin-bottom: 12px; }
@keyframes fadeIn { from { opacity: 0; transform: translateY(8px); } to { opacity: 1; transform: none; } }