import { muapi } from '../lib/muapi.js'; import { t2iModels, getAspectRatiosForModel, getResolutionsForModel, getQualityFieldForModel, i2iModels, getAspectRatiosForI2IModel, getResolutionsForI2IModel, getQualityFieldForI2IModel, getMaxImagesForI2IModel } from '../lib/models.js'; import { AuthModal } from './AuthModal.js'; import { createUploadPicker } from './UploadPicker.js'; export function ImageStudio() { const container = document.createElement('div'); container.className = 'w-full h-full flex flex-col items-center justify-center bg-app-bg relative p-4 md:p-6 overflow-y-auto custom-scrollbar overflow-x-hidden'; // --- State --- const defaultModel = t2iModels[0]; let selectedModel = defaultModel.id; let selectedModelName = defaultModel.name; let selectedAr = defaultModel.inputs?.aspect_ratio?.default || '1:1'; let dropdownOpen = null; let uploadedImageUrls = []; // array of uploaded image URLs (multi-image support) let imageMode = false; // false = t2i models, true = i2i models const getCurrentModels = () => imageMode ? i2iModels : t2iModels; const getCurrentAspectRatios = (id) => imageMode ? getAspectRatiosForI2IModel(id) : getAspectRatiosForModel(id); const getCurrentResolutions = (id) => imageMode ? getResolutionsForI2IModel(id) : getResolutionsForModel(id); const getCurrentQualityField = (id) => imageMode ? getQualityFieldForI2IModel(id) : getQualityFieldForModel(id); // ========================================== // 1. HERO SECTION // ========================================== const hero = document.createElement('div'); hero.className = 'flex flex-col items-center mb-10 md:mb-20 animate-fade-in-up transition-all duration-700'; hero.innerHTML = `

Image Studio

Transform images with AI — upscale, stylize, animate and more

`; container.appendChild(hero); // ========================================== // 2. PROMPT BAR (Tailwind Refactor) // ========================================== const promptWrapper = document.createElement('div'); promptWrapper.className = 'w-full max-w-4xl relative z-40 animate-fade-in-up'; promptWrapper.style.animationDelay = '0.2s'; const bar = document.createElement('div'); bar.className = 'w-full bg-[#111]/90 backdrop-blur-xl border border-white/10 rounded-[1.5rem] md:rounded-[2.5rem] p-3 md:p-5 flex flex-col gap-3 md:gap-5 shadow-3xl'; // Top Row: Input const topRow = document.createElement('div'); topRow.className = 'flex items-start gap-5 px-2'; // --- Image Upload Picker (Image-to-Image) --- const picker = createUploadPicker({ anchorContainer: container, onSelect: ({ url, urls }) => { uploadedImageUrls = urls || [url]; if (!imageMode) { imageMode = true; selectedModel = i2iModels[0].id; selectedModelName = i2iModels[0].name; selectedAr = getAspectRatiosForI2IModel(selectedModel)[0]; document.getElementById('model-btn-label').textContent = selectedModelName; document.getElementById('ar-btn-label').textContent = selectedAr; const validResolutions = getResolutionsForI2IModel(selectedModel); qualityBtn.style.display = validResolutions.length > 0 ? 'flex' : 'none'; if (validResolutions.length > 0) document.getElementById('quality-btn-label').textContent = validResolutions[0]; picker.setMaxImages(getMaxImagesForI2IModel(selectedModel)); } textarea.placeholder = uploadedImageUrls.length > 1 ? `${uploadedImageUrls.length} images selected — describe the transformation (optional)` : 'Describe how to transform this image (optional)'; }, onClear: () => { uploadedImageUrls = []; imageMode = false; selectedModel = t2iModels[0].id; selectedModelName = t2iModels[0].name; selectedAr = getAspectRatiosForModel(selectedModel)[0]; document.getElementById('model-btn-label').textContent = selectedModelName; document.getElementById('ar-btn-label').textContent = selectedAr; const t2iResolutions = getResolutionsForModel(selectedModel); qualityBtn.style.display = t2iResolutions.length > 0 ? 'flex' : 'none'; if (t2iResolutions.length > 0) document.getElementById('quality-btn-label').textContent = t2iResolutions[0]; picker.setMaxImages(1); textarea.placeholder = 'Describe the image you want to create'; } }); topRow.appendChild(picker.trigger); container.appendChild(picker.panel); const textarea = document.createElement('textarea'); textarea.placeholder = 'Describe the image you want to create'; textarea.className = 'flex-1 bg-transparent border-none text-white text-base md:text-xl placeholder:text-muted focus:outline-none resize-none pt-2.5 leading-relaxed min-h-[40px] max-h-[150px] md:max-h-[250px] overflow-y-auto custom-scrollbar'; textarea.rows = 1; textarea.oninput = () => { textarea.style.height = 'auto'; const maxHeight = window.innerWidth < 768 ? 150 : 250; textarea.style.height = Math.min(textarea.scrollHeight, maxHeight) + 'px'; }; topRow.appendChild(textarea); bar.appendChild(topRow); // Bottom Row: Controls const bottomRow = document.createElement('div'); bottomRow.className = 'flex flex-col sm:flex-row items-stretch sm:items-center justify-between gap-4 px-2 pt-4 border-t border-white/5'; const controlsLeft = document.createElement('div'); controlsLeft.className = 'flex items-center gap-1.5 md:gap-2.5 relative overflow-x-auto no-scrollbar pb-1 md:pb-0'; const createControlBtn = (icon, label, id) => { const btn = document.createElement('button'); btn.id = id; btn.className = 'flex items-center gap-1.5 md:gap-2.5 px-3 md:px-4 py-2 md:py-2.5 bg-white/5 hover:bg-white/10 rounded-xl md:rounded-2xl transition-all border border-white/5 group whitespace-nowrap'; btn.innerHTML = ` ${icon} ${label} `; return btn; }; const modelBtn = createControlBtn(`
G
`, selectedModelName, 'model-btn'); const arBtn = createControlBtn(` `, selectedAr, 'ar-btn'); const qualityBtn = createControlBtn(` `, '720p', 'quality-btn'); controlsLeft.appendChild(modelBtn); controlsLeft.appendChild(arBtn); controlsLeft.appendChild(qualityBtn); // Show quality button if the default model has quality/resolution options const _initResolutions = getResolutionsForModel(defaultModel.id); qualityBtn.style.display = _initResolutions.length > 0 ? 'flex' : 'none'; if (_initResolutions.length > 0) document.getElementById('quality-btn-label').textContent = _initResolutions[0]; const generateBtn = document.createElement('button'); generateBtn.className = 'bg-primary text-black px-6 md:px-8 py-3 md:py-3.5 rounded-xl md:rounded-[1.5rem] font-black text-sm md:text-base hover:shadow-glow hover:scale-105 active:scale-95 transition-all flex items-center justify-center gap-2.5 w-full sm:w-auto shadow-lg'; generateBtn.innerHTML = `Generate ✨`; bottomRow.appendChild(controlsLeft); bottomRow.appendChild(generateBtn); bar.appendChild(bottomRow); promptWrapper.appendChild(bar); container.appendChild(promptWrapper); // ========================================== // 3. DROPDOWNS (Professional implementation) // ========================================== const dropdown = document.createElement('div'); dropdown.className = 'absolute bottom-[102%] left-2 z-50 transition-all opacity-0 pointer-events-none scale-95 origin-bottom-left glass rounded-3xl p-3 translate-y-2 w-[calc(100vw-3rem)] max-w-xs shadow-4xl border border-white/10 flex flex-col'; const showDropdown = (type, anchorBtn) => { dropdown.innerHTML = ''; dropdown.classList.remove('opacity-0', 'pointer-events-none'); dropdown.classList.add('opacity-100', 'pointer-events-auto'); if (type === 'model') { dropdown.classList.add('w-[calc(100vw-3rem)]', 'max-w-xs'); dropdown.classList.remove('max-w-[240px]', 'max-w-[200px]'); dropdown.innerHTML = `
Available models
`; const list = dropdown.querySelector('#model-list-container'); const renderModels = (filter = '') => { list.innerHTML = ''; const filtered = getCurrentModels().filter(m => m.name.toLowerCase().includes(filter.toLowerCase()) || m.id.toLowerCase().includes(filter.toLowerCase())); filtered.forEach(m => { const item = document.createElement('div'); item.className = `flex items-center justify-between p-3.5 hover:bg-white/5 rounded-2xl cursor-pointer transition-all border border-transparent hover:border-white/5 ${selectedModel === m.id ? 'bg-white/5 border-white/5' : ''}`; item.innerHTML = `
${m.name.charAt(0)}
${m.name}
${selectedModel === m.id ? '' : ''} `; item.onclick = (e) => { e.stopPropagation(); selectedModel = m.id; selectedModelName = m.name; const availableArs = getCurrentAspectRatios(selectedModel); selectedAr = availableArs[0]; document.getElementById('model-btn-label').textContent = selectedModelName; document.getElementById('ar-btn-label').textContent = selectedAr; const validResolutions = getCurrentResolutions(selectedModel); qualityBtn.style.display = validResolutions.length > 0 ? 'flex' : 'none'; if (validResolutions.length > 0) { document.getElementById('quality-btn-label').textContent = validResolutions[0]; } // Update picker's max images when switching i2i models if (imageMode) { picker.setMaxImages(getMaxImagesForI2IModel(selectedModel)); } closeDropdown(); }; list.appendChild(item); }); }; renderModels(); const searchInput = dropdown.querySelector('#model-search'); searchInput.onclick = (e) => e.stopPropagation(); searchInput.oninput = (e) => renderModels(e.target.value); } else if (type === 'ar') { dropdown.classList.add('max-w-[240px]'); dropdown.innerHTML = `
Aspect Ratio
`; const list = document.createElement('div'); list.className = 'flex flex-col gap-1'; const availableArs = getCurrentAspectRatios(selectedModel); availableArs.forEach(r => { const item = document.createElement('div'); item.className = 'flex items-center justify-between p-3.5 hover:bg-white/5 rounded-2xl cursor-pointer transition-all group'; item.innerHTML = `
${r}
${selectedAr === r ? '' : ''} `; item.onclick = (e) => { e.stopPropagation(); selectedAr = r; document.getElementById('ar-btn-label').textContent = r; closeDropdown(); }; list.appendChild(item); }); dropdown.appendChild(list); } else if (type === 'quality') { dropdown.classList.add('max-w-[200px]'); dropdown.innerHTML = `
Resolution
`; const list = document.createElement('div'); list.className = 'flex flex-col gap-1'; const options = getCurrentResolutions(selectedModel); options.forEach(opt => { const item = document.createElement('div'); item.className = 'flex items-center justify-between p-3.5 hover:bg-white/5 rounded-2xl cursor-pointer transition-all group'; item.innerHTML = ` ${opt} ${document.getElementById('quality-btn-label').textContent === opt ? '' : ''} `; item.onclick = (e) => { e.stopPropagation(); document.getElementById('quality-btn-label').textContent = opt; closeDropdown(); }; list.appendChild(item); }); dropdown.appendChild(list); } // Position dropdown const btnRect = anchorBtn.getBoundingClientRect(); const containerRect = container.getBoundingClientRect(); // Horizontal position if (window.innerWidth < 768) { // Center on mobile dropdown.style.left = '50%'; dropdown.style.transform = 'translateX(-50%) translate(0, 8px)'; } else { // Align with button on desktop dropdown.style.left = `${btnRect.left - containerRect.left}px`; dropdown.style.transform = 'translate(0, 8px)'; } // Vertical position (always above button) dropdown.style.bottom = `${containerRect.bottom - btnRect.top + 8}px`; }; const closeDropdown = () => { dropdown.classList.add('opacity-0', 'pointer-events-none'); dropdown.classList.remove('opacity-100', 'pointer-events-auto'); dropdownOpen = null; }; modelBtn.onclick = (e) => { e.stopPropagation(); if (dropdownOpen === 'model') closeDropdown(); else { dropdownOpen = 'model'; showDropdown('model', modelBtn); } }; arBtn.onclick = (e) => { e.stopPropagation(); if (dropdownOpen === 'ar') closeDropdown(); else { dropdownOpen = 'ar'; showDropdown('ar', arBtn); } }; qualityBtn.onclick = (e) => { e.stopPropagation(); if (dropdownOpen === 'quality') closeDropdown(); else { dropdownOpen = 'quality'; showDropdown('quality', qualityBtn); } }; window.onclick = () => closeDropdown(); container.appendChild(dropdown); // ========================================== // 4. CANVAS AREA + HISTORY // ========================================== const generationHistory = []; // History sidebar const historySidebar = document.createElement('div'); historySidebar.className = 'fixed right-0 top-0 h-full w-20 md:w-24 bg-black/60 backdrop-blur-xl border-l border-white/5 z-50 flex flex-col items-center py-4 gap-3 overflow-y-auto transition-all duration-500 translate-x-full opacity-0'; historySidebar.id = 'history-sidebar'; const historyLabel = document.createElement('div'); historyLabel.className = 'text-[9px] font-bold text-muted uppercase tracking-widest mb-2 rotate-0'; historyLabel.textContent = 'History'; historySidebar.appendChild(historyLabel); const historyList = document.createElement('div'); historyList.className = 'flex flex-col gap-2 w-full px-2'; historySidebar.appendChild(historyList); container.appendChild(historySidebar); // Main canvas const canvas = document.createElement('div'); canvas.className = 'absolute inset-0 flex flex-col items-center justify-center p-4 min-[800px]:p-16 z-10 opacity-0 pointer-events-none transition-all duration-1000 translate-y-10 scale-95'; const imageContainer = document.createElement('div'); imageContainer.className = 'relative group'; const resultImg = document.createElement('img'); resultImg.className = 'max-h-[60vh] max-w-[80vw] rounded-3xl shadow-3xl border border-white/10 interactive-glow object-contain'; imageContainer.appendChild(resultImg); // Canvas Controls const canvasControls = document.createElement('div'); canvasControls.className = 'mt-6 flex gap-3 opacity-0 transition-opacity delay-500 duration-500 justify-center'; const regenerateBtn = document.createElement('button'); regenerateBtn.className = 'bg-white/10 hover:bg-white/20 px-6 py-2.5 rounded-2xl text-xs font-bold transition-all border border-white/5 backdrop-blur-lg text-white'; regenerateBtn.textContent = '↻ Regenerate'; const downloadBtn = document.createElement('button'); downloadBtn.className = 'bg-primary text-black px-6 py-2.5 rounded-2xl text-xs font-bold transition-all shadow-glow active:scale-95'; downloadBtn.textContent = '↓ Download'; const newPromptBtn = document.createElement('button'); newPromptBtn.className = 'bg-white/10 hover:bg-white/20 px-6 py-2.5 rounded-2xl text-xs font-bold transition-all border border-white/5 backdrop-blur-lg text-white'; newPromptBtn.textContent = '+ New'; canvasControls.appendChild(regenerateBtn); canvasControls.appendChild(downloadBtn); canvasControls.appendChild(newPromptBtn); canvas.appendChild(imageContainer); canvas.appendChild(canvasControls); container.appendChild(canvas); // --- Helper: Show image in canvas --- const showImageInCanvas = (imageUrl) => { // Fully hide hero and prompt hero.classList.add('hidden'); promptWrapper.classList.add('hidden'); resultImg.src = imageUrl; resultImg.onload = () => { canvas.classList.remove('opacity-0', 'pointer-events-none', 'translate-y-10', 'scale-95'); canvas.classList.add('opacity-100', 'translate-y-0', 'scale-100'); canvasControls.classList.remove('opacity-0'); canvasControls.classList.add('opacity-100'); }; }; // --- Helper: Add to history --- const addToHistory = (entry) => { generationHistory.unshift(entry); // Save to localStorage localStorage.setItem('muapi_history', JSON.stringify(generationHistory.slice(0, 50))); // Show sidebar historySidebar.classList.remove('translate-x-full', 'opacity-0'); historySidebar.classList.add('translate-x-0', 'opacity-100'); renderHistory(); }; const renderHistory = () => { historyList.innerHTML = ''; generationHistory.forEach((entry, idx) => { const thumb = document.createElement('div'); thumb.className = `relative group/thumb cursor-pointer rounded-xl overflow-hidden border-2 transition-all duration-300 ${idx === 0 ? 'border-primary shadow-glow' : 'border-white/10 hover:border-white/30'}`; thumb.innerHTML = ` ${entry.prompt?.substring(0, 30) || 'Generated'}
`; thumb.onclick = (e) => { if (e.target.closest('.hist-download')) { downloadImage(entry.url, `muapi-${entry.id || idx}.jpg`); return; } showImageInCanvas(entry.url); // Update active border historyList.querySelectorAll('div').forEach(t => { t.classList.remove('border-primary', 'shadow-glow'); t.classList.add('border-white/10'); }); thumb.classList.remove('border-white/10'); thumb.classList.add('border-primary', 'shadow-glow'); }; historyList.appendChild(thumb); }); }; // --- Helper: Download image --- const downloadImage = async (url, filename) => { try { const response = await fetch(url); const blob = await response.blob(); const blobUrl = URL.createObjectURL(blob); const a = document.createElement('a'); a.href = blobUrl; a.download = filename; document.body.appendChild(a); a.click(); document.body.removeChild(a); URL.revokeObjectURL(blobUrl); } catch (err) { // Fallback: open in new tab window.open(url, '_blank'); } }; // --- Load history from localStorage --- try { const saved = JSON.parse(localStorage.getItem('muapi_history') || '[]'); if (saved.length > 0) { saved.forEach(e => generationHistory.push(e)); historySidebar.classList.remove('translate-x-full', 'opacity-0'); historySidebar.classList.add('translate-x-0', 'opacity-100'); renderHistory(); } } catch (e) { /* ignore */ } // --- Button Handlers --- downloadBtn.onclick = () => { const current = resultImg.src; if (current) { const entry = generationHistory.find(e => e.url === current); downloadImage(current, `muapi-${entry?.id || 'image'}.jpg`); } }; regenerateBtn.onclick = () => { generateBtn.click(); }; newPromptBtn.onclick = () => { // Reset to prompt view canvas.classList.add('opacity-0', 'pointer-events-none', 'translate-y-10', 'scale-95'); canvas.classList.remove('opacity-100', 'translate-y-0', 'scale-100'); canvasControls.classList.add('opacity-0'); canvasControls.classList.remove('opacity-100'); // Restore hero and prompt hero.classList.remove('hidden', 'opacity-0', 'scale-95', '-translate-y-10', 'pointer-events-none'); promptWrapper.classList.remove('hidden', 'opacity-40'); textarea.value = ''; picker.reset(); uploadedImageUrls = []; picker.setMaxImages(1); // Reset to t2i mode imageMode = false; selectedModel = t2iModels[0].id; selectedModelName = t2iModels[0].name; selectedAr = getAspectRatiosForModel(selectedModel)[0]; document.getElementById('model-btn-label').textContent = selectedModelName; document.getElementById('ar-btn-label').textContent = selectedAr; const resetResolutions = getResolutionsForModel(selectedModel); qualityBtn.style.display = resetResolutions.length > 0 ? 'flex' : 'none'; if (resetResolutions.length > 0) document.getElementById('quality-btn-label').textContent = resetResolutions[0]; textarea.placeholder = 'Describe the image you want to create'; textarea.focus(); }; // ========================================== // 5. GENERATION LOGIC // ========================================== generateBtn.onclick = async () => { const prompt = textarea.value.trim(); if (imageMode) { if (uploadedImageUrls.length === 0) { alert('Please upload a reference image first.'); return; } } else { if (!prompt) { alert('Please enter a prompt to generate an image.'); return; } } const apiKey = localStorage.getItem('muapi_key'); if (!apiKey) { AuthModal(() => generateBtn.click()); return; } hero.classList.add('opacity-0', 'scale-95', '-translate-y-10', 'pointer-events-none'); generateBtn.disabled = true; generateBtn.innerHTML = ` Generating...`; try { let res; const qualityLabel = document.getElementById('quality-btn-label')?.textContent; if (imageMode) { const genParams = { model: selectedModel, images_list: uploadedImageUrls, image_url: uploadedImageUrls[0], // backward compat for single-image models aspect_ratio: selectedAr }; if (prompt) genParams.prompt = prompt; const qualityField = getCurrentQualityField(selectedModel); if (qualityField && qualityLabel) genParams[qualityField] = qualityLabel; res = await muapi.generateI2I(genParams); } else { const genParams = { model: selectedModel, prompt, aspect_ratio: selectedAr }; const qualityField = getCurrentQualityField(selectedModel); if (qualityField && qualityLabel) genParams[qualityField] = qualityLabel; res = await muapi.generateImage(genParams); } console.log('[ImageStudio] Full response:', res); if (res && res.url) { // Add to history addToHistory({ id: res.id || Date.now().toString(), url: res.url, prompt: prompt, model: selectedModel, aspect_ratio: selectedAr, timestamp: new Date().toISOString() }); // Show image showImageInCanvas(res.url); } else { console.error('[ImageStudio] No image URL in response:', res); throw new Error('No image URL returned by API'); } } catch (e) { console.error(e); generateBtn.innerHTML = `Error: ${e.message.slice(0, 40)}`; setTimeout(() => { generateBtn.innerHTML = `Generate ✨`; generateBtn.disabled = false; }, 3000); } finally { generateBtn.disabled = false; generateBtn.innerHTML = `Generate ✨`; } }; return container; }