// preview.js — file-preview rendering for the browse tool's right pane. // // Default flow: showFilePreview(node) renders into the inline preview // pane (#previewBody). Popup flow: showFilePreview(node, {popup:true}) // opens a separate window — kept for users who want previews on a // second monitor. // // Rendering uses shared/preview-lib.js for content types it handles // (TIFF, ZIP listing, image-mime detection). PDF / HTML go in iframes; // text into a
; markdown into the dedicated markdown plugin
// (preview-markdown.js); unknown extensions show a download button.
(function () {
    'use strict';

    var state = window.app.state;
    var loader = window.app.modules.loader;
    var preview = window.zddc && window.zddc.preview;
    if (!preview) {
        console.error('[browse] zddc.preview not loaded — preview disabled.');
    }

    function escapeHtml(s) {
        return String(s).replace(/&/g, '&').replace(//g, '>').replace(/"/g, '"');
    }

    var MIME = {
        'pdf': 'application/pdf',
        'html': 'text/html', 'htm': 'text/html',
        'jpg': 'image/jpeg', 'jpeg': 'image/jpeg', 'png': 'image/png',
        'gif': 'image/gif', 'webp': 'image/webp', 'svg': 'image/svg+xml',
        'tif': 'image/tiff', 'tiff': 'image/tiff',
        'zip': 'application/zip',
        'txt': 'text/plain', 'md': 'text/markdown', 'json': 'application/json',
        'xml': 'application/xml', 'csv': 'text/csv', 'log': 'text/plain',
        'js': 'text/javascript', 'css': 'text/css',
        'docx': 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
        'xlsx': 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
        'xls': 'application/vnd.ms-excel'
    };

    function getMime(ext) { return MIME[ext] || 'application/octet-stream'; }

    function fmtSize(bytes) {
        if (bytes == null) return '';
        if (bytes < 1024) return bytes + ' B';
        if (bytes < 1024 * 1024) return (bytes / 1024).toFixed(1) + ' KB';
        if (bytes < 1024 * 1024 * 1024) return (bytes / (1024 * 1024)).toFixed(1) + ' MB';
        return (bytes / (1024 * 1024 * 1024)).toFixed(2) + ' GB';
    }

    async function getArrayBuffer(node) {
        // A zip member node carries a ZipFileHandle in node.handle, so
        // it falls through the same getFile() path as any local file.
        if (state.source === 'server' && node.url) {
            var resp = await fetch(node.url);
            if (!resp.ok) throw new Error('HTTP ' + resp.status);
            return await resp.arrayBuffer();
        }
        if (node.handle) {
            var f = await node.handle.getFile();
            return await f.arrayBuffer();
        }
        throw new Error('no source for file');
    }

    async function getBlobUrl(node) {
        // Server-served files (including zip members at "<…>.zip/"
        // URLs) load straight from the server — preserves Content-Type
        // and lets relative links inside HTML resolve back to the server.
        if (state.source === 'server' && node.url) {
            return { url: node.url, fromServer: true };
        }
        var buf = await getArrayBuffer(node);
        var blob = new Blob([buf], { type: getMime(node.ext) });
        return { url: URL.createObjectURL(blob), fromServer: false };
    }

    // ── Inline rendering ────────────────────────────────────────────────────

    function renderEmpty(container, msg) {
        container.innerHTML = '
' + escapeHtml(msg) + '
'; } function renderError(container, msg) { container.innerHTML = '
' + escapeHtml(msg) + '
'; } async function renderInline(node) { var container = document.getElementById('previewBody'); var titleEl = document.getElementById('previewTitle'); var metaEl = document.getElementById('previewMeta'); var popoutBtn = document.getElementById('previewPopout'); if (!container) return; if (titleEl) titleEl.textContent = node.name; if (metaEl) { var meta = []; if (!node.isDir && !node.isZip) meta.push(fmtSize(node.size)); if (node.ext) meta.push(node.ext.toUpperCase()); metaEl.textContent = meta.join(' · '); } if (popoutBtn) popoutBtn.classList.remove('hidden'); var ext = (node.ext || '').toLowerCase(); // Markdown plugin (if loaded) takes over for .md / .markdown. if ((ext === 'md' || ext === 'markdown') && window.app.modules.markdown && typeof window.app.modules.markdown.render === 'function') { try { await window.app.modules.markdown.render(node, container, { getArrayBuffer: getArrayBuffer }); } catch (e) { renderError(container, 'Markdown render failed: ' + (e.message || e)); } return; } // YAML plugin: .yaml / .yml / .zddc / *.zddc.yaml route to a // CodeMirror 5 editor with js-yaml linting; .zddc files also // get a schema-aware lint pass. var yamlMod = window.app.modules.yamledit; if (yamlMod && yamlMod.handles(node)) { try { await yamlMod.render(node, container, { getArrayBuffer: getArrayBuffer }); } catch (e) { renderError(container, 'YAML render failed: ' + (e.message || e)); } return; } // PDF / HTML → iframe. if (ext === 'pdf' || ext === 'html' || ext === 'htm') { try { var info = await getBlobUrl(node); var sandbox = (ext === 'pdf') ? '' : ' sandbox="allow-same-origin allow-popups allow-popups-to-escape-sandbox"'; container.innerHTML = ''; } catch (e) { renderError(container, e.message || String(e)); } return; } // Plain images (jpg/png/gif/webp/svg) → . TIFF goes through preview-lib. if (preview && preview.isImage(ext) && !preview.isTiff(ext)) { try { var imgInfo = await getBlobUrl(node); container.innerHTML = '' + escapeHtml(node.name)
                    + ''; } catch (e) { renderError(container, e.message || String(e)); } return; } if (preview && preview.isTiff(ext)) { try { var tiffBuf = await getArrayBuffer(node); container.innerHTML = ''; await preview.renderTiff(document, container, tiffBuf, { fileName: node.name }); } catch (e) { renderError(container, 'Failed to render TIFF: ' + (e.message || e)); } return; } if (preview && preview.isZip(ext)) { try { var zipBuf = await getArrayBuffer(node); container.innerHTML = ''; await preview.renderZipListing(document, container, zipBuf, { fileName: node.name }); } catch (e) { renderError(container, 'Failed to read ZIP: ' + (e.message || e)); } return; } // Office docs (.docx via docx-preview, .xlsx/.xls via SheetJS) → // shared/preview-lib renderers. .doc/.ppt etc. fall through to the // download fallback below. if (preview && preview.isOffice(ext)) { try { var officeBuf = await getArrayBuffer(node); container.innerHTML = ''; if (ext === 'docx') { await preview.renderDocx(document, container, officeBuf, { fileName: node.name }); } else { await preview.renderXlsx(document, container, officeBuf, { fileName: node.name }); } } catch (e) { renderError(container, 'Failed to render ' + ext.toUpperCase() + ': ' + (e.message || e)); } return; } if (preview && preview.isText(ext)) { try { var txtBuf = await getArrayBuffer(node); var text = new TextDecoder('utf-8', { fatal: false }).decode(txtBuf); var MAX = 200000; if (text.length > MAX) { text = text.substring(0, MAX) + '\n\n... (truncated, ' + (text.length - MAX) + ' more chars)'; } container.innerHTML = ''; var pre = document.createElement('pre'); pre.className = 'preview-text'; pre.textContent = text; container.appendChild(pre); } catch (e) { renderError(container, e.message || String(e)); } return; } // Unknown type — offer a download link. try { var fallbackInfo = await getBlobUrl(node); container.innerHTML = '
' + 'No inline preview for .' + escapeHtml(ext) + '. ' + '
' + 'Download ' + escapeHtml(node.name) + '' + '
'; } catch (e) { renderError(container, 'No source for ' + node.name); } } // ── Popup window (kept for "Pop out" button) ──────────────────────────── function popupShell(node, primaryUrl) { var safeName = escapeHtml(node.name); var safeHref = escapeHtml(primaryUrl); var ext = (node.ext || '').toLowerCase(); var contentHtml; if (ext === 'pdf') { contentHtml = ''; } else if (ext === 'html' || ext === 'htm') { contentHtml = ''; } else if (preview && preview.isImage(ext) && !preview.isTiff(ext)) { contentHtml = '' + safeName + ''; } else { contentHtml = '
Loading preview…
'; } return '' + '' + safeName + ' — preview' + '

' + safeName + '

' + '
' + contentHtml + '