feat: render inline image previews

This commit is contained in:
mirivlad 2026-06-29 03:16:16 +08:00
parent 7b249fcf48
commit f0c7b31c9e
3 changed files with 39 additions and 1 deletions

View File

@ -25,6 +25,8 @@
'.fp-btn{font-size:.75rem;padding:.28rem .58rem;border:1px solid #333;border-radius:4px;background:#1a1a2e;color:#ccc;cursor:pointer}', '.fp-btn{font-size:.75rem;padding:.28rem .58rem;border:1px solid #333;border-radius:4px;background:#1a1a2e;color:#ccc;cursor:pointer}',
'.fp-btn:hover{background:#2a2a4e;border-color:#4ecca3;color:#4ecca3}', '.fp-btn:hover{background:#2a2a4e;border-color:#4ecca3;color:#4ecca3}',
'.fp-body{flex:1;min-height:0;overflow:auto;padding:1rem 1.2rem}', '.fp-body{flex:1;min-height:0;overflow:auto;padding:1rem 1.2rem}',
'.fp-image-wrap{display:flex;align-items:center;justify-content:center;min-height:220px;margin:0 0 1rem;padding:1rem;background:#090914;border:1px solid #16213e;border-radius:6px}',
'.fp-image{display:block;max-width:100%;max-height:62vh;object-fit:contain}',
'.fp-meta{display:grid;grid-template-columns:max-content 1fr;gap:.45rem .8rem;max-width:760px;font-size:.86rem}', '.fp-meta{display:grid;grid-template-columns:max-content 1fr;gap:.45rem .8rem;max-width:760px;font-size:.86rem}',
'.fp-meta dt{color:#8b8ba8}.fp-meta dd{margin:0;color:#e0e0e0;word-break:break-word}', '.fp-meta dt{color:#8b8ba8}.fp-meta dd{margin:0;color:#e0e0e0;word-break:break-word}',
'.fp-kind{margin:0 0 1rem;color:#f0f0ff;font-size:1.1rem}', '.fp-kind{margin:0 0 1rem;color:#f0f0ff;font-size:1.1rem}',
@ -66,6 +68,26 @@
return (size / (1024 * 1024)).toFixed(1) + ' MB'; return (size / (1024 * 1024)).toFixed(1) + ' MB';
} }
function imageMime(ext, hint) {
if (hint && String(hint).indexOf('image/') === 0) return String(hint);
if (ext === 'jpg') return 'image/jpeg';
if (ext === 'svg') return 'image/svg+xml';
if (IMAGE_EXTS.indexOf(ext) !== -1) return 'image/' + ext;
return 'application/octet-stream';
}
function renderImage(body, path, meta, ext, bytes) {
var src = 'data:' + imageMime(ext, bytes && bytes.mimeHint || meta && meta.mimeHint) + ';base64,' + (bytes && bytes.dataBase64 || '');
var img = el('img', {
className: 'fp-image',
src: src,
alt: path,
'data-preview-image': 'true'
});
var wrap = el('div', { className: 'fp-image-wrap' }, [img]);
body.appendChild(wrap);
}
function renderMeta(body, path, meta, ext) { function renderMeta(body, path, meta, ext) {
body.className = 'fp-body'; body.className = 'fp-body';
body.innerHTML = ''; body.innerHTML = '';
@ -111,6 +133,11 @@
api.files.metadata(path).then(function (meta) { api.files.metadata(path).then(function (meta) {
renderMeta(body, path, meta || {}, ext); renderMeta(body, path, meta || {}, ext);
if (IMAGE_EXTS.indexOf(ext) === -1 || !api.files.readBytes) return null;
return api.files.readBytes(path).then(function (bytes) {
if (bytes && bytes.dataBase64) renderImage(body, path, meta || {}, ext, bytes);
return null;
});
}).catch(function (err) { }).catch(function (err) {
body.className = 'fp-error'; body.className = 'fp-error';
body.textContent = 'Preview error: ' + (err && err.message ? err.message : String(err)); body.textContent = 'Preview error: ' + (err && err.message ? err.message : String(err));

View File

@ -4,7 +4,7 @@
"name": "File Preview", "name": "File Preview",
"version": "0.1.0", "version": "0.1.0",
"apiVersion": "0.1.0", "apiVersion": "0.1.0",
"description": "Read-only image metadata preview provider for vault files.", "description": "Read-only inline image preview provider for vault files.",
"source": "official", "source": "official",
"icon": "eye", "icon": "eye",
"provides": [ "provides": [

View File

@ -123,17 +123,28 @@ async function flush() {
const imageContainer = new FakeNode('div'); const imageContainer = new FakeNode('div');
const opened = []; const opened = [];
const byteReads = [];
component.mount(imageContainer, { component.mount(imageContainer, {
request: { path: 'Images/logo.png', extension: '.png', mode: 'view' }, request: { path: 'Images/logo.png', extension: '.png', mode: 'view' },
}, { }, {
files: { files: {
metadata: async () => ({ type: 'file', size: 2048, isText: false, modifiedAt: '2026-06-27T00:00:00Z' }), metadata: async () => ({ type: 'file', size: 2048, isText: false, modifiedAt: '2026-06-27T00:00:00Z' }),
readBytes: async (relativePath) => {
byteReads.push(relativePath);
return { relativePath, size: 4, mimeHint: 'image/png', dataBase64: 'iVBORw==' };
},
readText: async () => { throw new Error('file preview should not read text content'); }, readText: async () => { throw new Error('file preview should not read text content'); },
openExternal: async (relativePath) => { opened.push(relativePath); }, openExternal: async (relativePath) => { opened.push(relativePath); },
}, },
}); });
await flush(); await flush();
if (!imageContainer.textContent.includes('Image Preview')) throw new Error('image preview metadata was not rendered'); if (!imageContainer.textContent.includes('Image Preview')) throw new Error('image preview metadata was not rendered');
if (byteReads[0] !== 'Images/logo.png') throw new Error(`expected readBytes path Images/logo.png, got ${byteReads[0] || '<none>'}`);
const previewImage = walk(imageContainer, (node) => node.tagName === 'IMG' && node.getAttribute('data-preview-image') === 'true');
if (!previewImage) throw new Error('preview image not rendered');
if (previewImage.getAttribute('src') !== 'data:image/png;base64,iVBORw==') {
throw new Error(`unexpected preview image src: ${previewImage.getAttribute('src') || '<none>'}`);
}
const openButton = walk(imageContainer, (node) => node.getAttribute && node.getAttribute('data-action') === 'open-external'); const openButton = walk(imageContainer, (node) => node.getAttribute && node.getAttribute('data-action') === 'open-external');
if (!openButton) throw new Error('open external button not found'); if (!openButton) throw new Error('open external button not found');
openButton.click(); openButton.click();