Open-Generative-AI/components/StandaloneShell.js
Anil Matcha df7d2dd9a7 Convert to Next.js monorepo with shared studio component library
- 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>
2026-03-21 10:47:00 +05:30

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