mirror of
https://github.com/Anil-matcha/Open-Generative-AI.git
synced 2026-05-07 01:17:18 +00:00
- Migrated from Vite/vanilla JS to Next.js App Router - Added packages/studio: shared React component library (ImageStudio, VideoStudio, LipSyncStudio, CinemaStudio) - BYOK flow via StandaloneShell (localStorage API key) + ApiKeyModal - Fixed dropdown clipping caused by overflow-x-auto on controls row - Fixed dropdown background to solid bg-[#111] (was glass/transparent) - Renamed Cinema tab to Cinema Studio - Updated README: architecture, quick start port (3000), tech stack Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
111 lines
3.7 KiB
JavaScript
111 lines
3.7 KiB
JavaScript
'use client';
|
|
|
|
import { useState, useEffect, useCallback } from 'react';
|
|
import { ImageStudio, VideoStudio, LipSyncStudio, CinemaStudio } from 'studio';
|
|
import ApiKeyModal from './ApiKeyModal';
|
|
|
|
const TABS = [
|
|
{ id: 'image', label: 'Image Studio' },
|
|
{ id: 'video', label: 'Video Studio' },
|
|
{ id: 'lipsync', label: 'Lip Sync' },
|
|
{ id: 'cinema', label: 'Cinema Studio' },
|
|
];
|
|
|
|
const STORAGE_KEY = 'muapi_key';
|
|
|
|
export default function StandaloneShell() {
|
|
const [apiKey, setApiKey] = useState(null);
|
|
const [activeTab, setActiveTab] = useState('image');
|
|
const [showSettings, setShowSettings] = useState(false);
|
|
|
|
useEffect(() => {
|
|
const stored = localStorage.getItem(STORAGE_KEY);
|
|
if (stored) setApiKey(stored);
|
|
}, []);
|
|
|
|
const handleKeySave = useCallback((key) => {
|
|
localStorage.setItem(STORAGE_KEY, key);
|
|
setApiKey(key);
|
|
}, []);
|
|
|
|
const handleKeyChange = useCallback(() => {
|
|
localStorage.removeItem(STORAGE_KEY);
|
|
setApiKey(null);
|
|
}, []);
|
|
|
|
if (!apiKey) {
|
|
return <ApiKeyModal onSave={handleKeySave} />;
|
|
}
|
|
|
|
return (
|
|
<div className="h-screen bg-[#050505] flex flex-col overflow-hidden">
|
|
{/* Header */}
|
|
<header className="flex-shrink-0 flex items-center justify-between px-4 pt-4 pb-0 border-b border-white/5">
|
|
<div className="flex items-center gap-3">
|
|
<span className="text-white font-black text-lg tracking-wider uppercase">
|
|
Open Higgsfield AI
|
|
</span>
|
|
</div>
|
|
|
|
{/* Tabs */}
|
|
<nav className="flex items-center gap-1">
|
|
{TABS.map((tab) => (
|
|
<button
|
|
key={tab.id}
|
|
onClick={() => setActiveTab(tab.id)}
|
|
className={`px-4 py-2 text-sm font-medium rounded-t-lg transition-colors ${
|
|
activeTab === tab.id
|
|
? 'bg-[#d9ff00] text-black'
|
|
: 'text-white/50 hover:text-white'
|
|
}`}
|
|
>
|
|
{tab.label}
|
|
</button>
|
|
))}
|
|
</nav>
|
|
|
|
{/* Settings */}
|
|
<button
|
|
onClick={() => setShowSettings(true)}
|
|
className="text-white/40 hover:text-white text-sm transition-colors"
|
|
>
|
|
⚙ Settings
|
|
</button>
|
|
</header>
|
|
|
|
{/* Studio Content */}
|
|
<div className="flex-1">
|
|
{activeTab === 'image' && <ImageStudio apiKey={apiKey} />}
|
|
{activeTab === 'video' && <VideoStudio apiKey={apiKey} />}
|
|
{activeTab === 'lipsync' && <LipSyncStudio apiKey={apiKey} />}
|
|
{activeTab === 'cinema' && <CinemaStudio apiKey={apiKey} />}
|
|
</div>
|
|
|
|
{/* Settings Modal */}
|
|
{showSettings && (
|
|
<div className="fixed inset-0 bg-black/80 flex items-center justify-center z-50">
|
|
<div className="bg-[#111] border border-white/10 rounded-2xl p-8 w-full max-w-md">
|
|
<h2 className="text-white font-bold text-xl mb-6">Settings</h2>
|
|
<p className="text-white/50 text-sm mb-4">
|
|
Current API key: <span className="text-white/80 font-mono">{apiKey.slice(0, 8)}••••••••</span>
|
|
</p>
|
|
<div className="flex gap-3">
|
|
<button
|
|
onClick={handleKeyChange}
|
|
className="flex-1 py-2 rounded-lg bg-red-500/20 text-red-400 hover:bg-red-500/30 text-sm transition-colors"
|
|
>
|
|
Change API Key
|
|
</button>
|
|
<button
|
|
onClick={() => setShowSettings(false)}
|
|
className="flex-1 py-2 rounded-lg bg-white/5 text-white hover:bg-white/10 text-sm transition-colors"
|
|
>
|
|
Close
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|