dragonflight/services/editor/apps/web/src/mam-bridge.ts
Zac 6a8e4ac250 fix(editor): show loading banner during auto-import so Edit feels responsive
Clicking Edit on the preview modal worked, but the user only saw an empty editor for ~25s while the recovery + format-chooser cycle ran and the bridge waited for a stable project. Looked broken. Now: a centered top banner appears the moment the bridge detects ?asset=, reads Loading clip from Z-AMPP MAM, switches to Clip ready in media bin on success, or surfaces the failure. Project-stability gate tightened from 1500ms to 600ms so the import lands sooner.
2026-05-17 22:44:08 -04:00

139 lines
8.1 KiB
TypeScript

// Z-AMPP MAM bridge. On boot, reads ?asset=<uuid> from URL and imports it.
// Also exports pickFromMAM() for an in-app picker.
type MAMAsset = { id: string; display_name?: string; filename?: string; };
async function getStreamURL(assetId: string): Promise<string> {
const r = await fetch('/api/v1/assets/' + assetId + '/stream', { credentials: 'omit' });
if (!r.ok) throw new Error('stream lookup failed: ' + r.status);
const j = await r.json();
if (!j || !j.url) throw new Error('stream lookup: no url');
return j.url;
}
async function listAssets(limit = 200): Promise<MAMAsset[]> {
const r = await fetch('/api/v1/assets?limit=' + limit, { credentials: 'omit' });
if (!r.ok) throw new Error('list assets failed: ' + r.status);
const j = await r.json();
return (j && j.assets) || [];
}
function getMediaBridge(): any {
const w = window as any;
return w.__mediaBridge || w.mediaBridge || w.MediaBridge || (w.editor && w.editor.mediaBridge) || null;
}
async function importAsset(assetId: string, name?: string): Promise<void> {
const url = await getStreamURL(assetId);
const safeName = (name || (assetId + '.mp4')).replace(/[^\w.\-]+/g, '_');
const store: any = (window as any).__projectStore;
if (store && typeof store.getState === 'function' && typeof store.getState().importMedia === 'function') {
const res = await fetch(url, { credentials: 'omit' });
if (!res.ok) throw new Error('fetch failed: ' + res.status);
const blob = await res.blob();
const inferred = (safeName.toLowerCase().endsWith('.webm') ? 'video/webm' : safeName.toLowerCase().endsWith('.mov') ? 'video/quicktime' : safeName.toLowerCase().endsWith('.mp3') ? 'audio/mpeg' : safeName.toLowerCase().endsWith('.wav') ? 'audio/wav' : 'video/mp4'); const ct = (blob.type && blob.type !== 'application/octet-stream') ? blob.type : inferred; const file = new File([blob], safeName, { type: ct });
return store.getState().importMedia(file);
}
const bridge = getMediaBridge();
if (bridge && typeof bridge.importFromURL === 'function') return bridge.importFromURL(url, safeName);
throw new Error('No import target');
}
export async function pickFromMAM(): Promise<void> {
const assets = await listAssets(200);
const overlay = document.createElement('div');
Object.assign(overlay.style, { position:'fixed',inset:'0',background:'rgba(0,0,0,0.72)',zIndex:'999999',display:'flex',alignItems:'center',justifyContent:'center' } as CSSStyleDeclaration);
const modal = document.createElement('div');
Object.assign(modal.style, { width:'min(960px,92vw)',maxHeight:'84vh',background:'#161618',color:'#e8e8ea',borderRadius:'12px',padding:'16px',display:'flex',flexDirection:'column',gap:'12px',overflow:'hidden' } as CSSStyleDeclaration);
modal.innerHTML = '<div style="display:flex;justify-content:space-between;align-items:center"><strong>Import from Z-AMPP MAM</strong><button data-close style="background:#2a2a2e;color:#fff;border:0;border-radius:6px;padding:6px 10px;cursor:pointer">Close</button></div>';
const grid = document.createElement('div');
Object.assign(grid.style, { display:'grid',gridTemplateColumns:'repeat(auto-fill,minmax(180px,1fr))',gap:'10px',overflowY:'auto',padding:'4px' } as CSSStyleDeclaration);
for (const a of assets) {
const card = document.createElement('button');
Object.assign(card.style, { background:'#1f1f23',border:'1px solid #2c2c32',borderRadius:'8px',padding:'6px',color:'#e8e8ea',cursor:'pointer',display:'flex',flexDirection:'column',gap:'6px',textAlign:'left' } as CSSStyleDeclaration);
const label = a.display_name || a.filename || a.id;
card.innerHTML = '<img loading="lazy" src="/api/v1/assets/' + a.id + '/thumbnail" style="width:100%;aspect-ratio:16/9;object-fit:cover;background:#000;border-radius:4px" onerror="this.style.opacity=0.2"/><span style="font-size:12px;line-height:1.2;word-break:break-word">' + label + '</span>';
card.addEventListener('click', async () => { card.disabled = true; card.style.opacity='0.5'; try { await importAsset(a.id, label); } catch(e){ alert('Import failed: '+(e as Error).message); } finally { card.disabled=false; card.style.opacity='1'; } });
grid.appendChild(card);
}
modal.appendChild(grid);
overlay.appendChild(modal);
document.body.appendChild(overlay);
const close = () => overlay.remove();
overlay.addEventListener('click', (e) => { if (e.target === overlay) close(); });
modal.querySelector('[data-close]')?.addEventListener('click', close);
}
function showBanner(text: string): void {
let el = document.getElementById('zampp-import-banner');
if (!el) {
el = document.createElement('div');
el.id = 'zampp-import-banner';
el.style.cssText = 'position:fixed;top:14px;left:50%;transform:translateX(-50%);background:rgba(31,58,208,0.95);color:#fff;font:500 13px/1.4 system-ui,sans-serif;padding:10px 18px;border-radius:8px;box-shadow:0 8px 24px rgba(0,0,0,0.4);z-index:2147483647;display:flex;align-items:center;gap:10px;letter-spacing:0.02em';
document.body.appendChild(el);
}
el.innerHTML = '<span style="width:10px;height:10px;border-radius:50%;background:#fff;animation:zamppPulse 1.2s ease-in-out infinite;display:inline-block"></span>' + text;
if (!document.getElementById('zampp-banner-keyframes')) {
const st = document.createElement('style'); st.id = 'zampp-banner-keyframes';
st.textContent = '@keyframes zamppPulse{0%,100%{opacity:0.4}50%{opacity:1}}';
document.head.appendChild(st);
}
}
function hideBanner(): void { const el = document.getElementById('zampp-import-banner'); if (el) el.remove(); }
export function initMAMBridge(): void {
try {
const qs = new URLSearchParams(window.location.search);
const projectId = qs.get('project');
if (projectId) { try { window.localStorage.setItem('mamProjectId', projectId); } catch {} }
(window as any).__zamppPickFromMAM = pickFromMAM;
const assetId = qs.get('asset');
if (!assetId) return;
showBanner('Loading clip from Z-AMPP MAM…');
// Auto-skip the startup chooser so MediaBridge initializes immediately.
const clickMatching = (matches: string[]): boolean => {
const btns = document.querySelectorAll('button,[role="button"],a');
for (const el of Array.from(btns)) {
const t = (el.textContent || '').trim().toLowerCase();
if (matches.some(m => t === m || t.startsWith(m))) { (el as HTMLElement).click(); return true; }
}
return false;
};
let dismissTries = 0;
const dismissInterval = setInterval(() => {
const acted = clickMatching(['horizontal', 'open editor', 'start fresh', 'skip tour']);
if (dismissTries++ > 80) clearInterval(dismissInterval);
if (acted) dismissTries = Math.max(0, dismissTries - 4);
}, 250);
let lastProjectId: string | null = null;
let stableSince = 0;
let imported = false;
const tryImport = async (n = 0): Promise<void> => {
if (imported) return;
const w = window as any;
const st = w.__projectStore && w.__projectStore.getState && w.__projectStore.getState();
const p = st && st.project;
const ready = !!(p && p.mediaLibrary);
if (!ready) {
if (n < 80) { setTimeout(() => tryImport(n + 1), 750); return; }
console.error('[mam-bridge] project never loaded'); return;
}
if (p.id !== lastProjectId) { lastProjectId = p.id; stableSince = Date.now(); setTimeout(() => tryImport(n + 1), 750); return; }
if (Date.now() - stableSince < 600) { setTimeout(() => tryImport(n + 1), 500); return; }
try {
await importAsset(assetId);
imported = true;
console.log('[mam-bridge] imported', assetId, 'into project', p.id);
showBanner('Clip ready in media bin');
setTimeout(hideBanner, 2500);
} catch (e) {
if (n < 80) setTimeout(() => tryImport(n + 1), 1000);
else { console.error('[mam-bridge] import retries exhausted', e); showBanner('Could not auto-import clip. Use Add Media instead.'); setTimeout(hideBanner, 6000); }
}
};
if (document.readyState === 'complete') tryImport();
else window.addEventListener('load', () => tryImport());
} catch (e) { console.error('[mam-bridge] init failed', e); }
}
initMAMBridge();