New shared/zip-source.js: a ZipDirectoryHandle / ZipFileHandle pair that exposes a JSZip instance behind the File-System-Access surface (values/entries/keys, getDirectoryHandle/getFileHandle, getFile) — read-only, with a zip-slip guard. Mirrors shared/zddc-source.js's HTTP polyfill. Wired into archive's and browse's build.sh (both already bundle JSZip). archive: a .zip whose name minus ".zip" parses as a transmittal-folder name is now scanned as that transmittal folder. Offline, the zip is opened in the browser (ZipDirectoryHandle) and its members enumerated exactly like an uncompressed folder's files — table/export/hash paths are unchanged (they go through file.handle.getFile()). Online, the scanner recurses into the server's "<…>.zip/" virtual-directory listing, so members come back as "<…>.zip/<member>" URLs the server extracts on demand — no whole-zip download. browse: the offline (file://) zip path is migrated onto the shared adapter — expanding a .zip now opens it as a ZipDirectoryHandle and its members become ordinary dir/file nodes handled by the normal fetchFsChildren path (nested zips fall out by recursion). The bespoke flat-entry walker (loadZipChildren / setZipDirChildren / zipEntries / zipParentId / zipPath / _zipSyntheticDir) is gone — one zip implementation repo-wide. Markdown members inside a zip are flagged read-only (the ZipFileHandle refuses createWritable; server "<…>.zip/" URLs 405 on PUT). Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
460 lines
19 KiB
JavaScript
460 lines
19 KiB
JavaScript
// tree.js — in-memory tree model + DOM rendering.
|
|
//
|
|
// Nodes are stored flat in state.nodes (Map by id). The visible
|
|
// render is a depth-first walk starting from state.rootIds, skipping
|
|
// children of unexpanded folders. This decouples model from DOM and
|
|
// keeps re-renders linear in the visible-row count.
|
|
(function () {
|
|
'use strict';
|
|
|
|
var state = window.app.state;
|
|
var loader = window.app.modules.loader;
|
|
|
|
// ── Model helpers ────────────────────────────────────────────────────
|
|
|
|
function newNode(raw, parentId, depth) {
|
|
var id = state.nextId++;
|
|
// A .zip file is treated as a folder for tree purposes — the
|
|
// chevron expands it. On expand, server mode fetches the
|
|
// server's "<…>.zip/" virtual-directory listing; offline mode
|
|
// opens the zip with JSZip behind a ZipDirectoryHandle. Either
|
|
// way the zip's members become ordinary directory/file nodes.
|
|
var isZip = !raw.isDir && raw.ext === 'zip';
|
|
var node = {
|
|
id: id,
|
|
name: raw.name,
|
|
// displayName is the rendered label when set by the parent
|
|
// .zddc display: map. Sort + lookup continues to use .name
|
|
// (the on-disk basename) so URL composition stays canonical.
|
|
displayName: raw.displayName || '',
|
|
isDir: raw.isDir,
|
|
size: raw.size,
|
|
modTime: raw.modTime,
|
|
ext: raw.ext,
|
|
url: raw.url,
|
|
handle: raw.handle,
|
|
depth: depth,
|
|
parentId: parentId,
|
|
expanded: false,
|
|
loaded: false,
|
|
childIds: [],
|
|
isZip: isZip,
|
|
_zipDirHandle: null, // cached ZipDirectoryHandle (offline / nested zips)
|
|
// True when this entry was synthesized client-side (e.g.
|
|
// canonical project folders that don't exist on disk yet).
|
|
// Rendered with a muted style + an "(empty)" hint.
|
|
virtual: !!raw.virtual
|
|
};
|
|
state.nodes.set(id, node);
|
|
return node;
|
|
}
|
|
|
|
function clearTree() {
|
|
state.nodes.clear();
|
|
state.rootIds = [];
|
|
state.nextId = 1;
|
|
}
|
|
|
|
// Sort an array of nodes by current sort key. Folders always come
|
|
// first within a level (mimics common file managers).
|
|
function sortNodes(ids) {
|
|
var key = state.sort.key;
|
|
var dir = state.sort.dir;
|
|
ids.sort(function (a, b) {
|
|
var na = state.nodes.get(a);
|
|
var nb = state.nodes.get(b);
|
|
// Folders before files
|
|
if (na.isDir !== nb.isDir) return na.isDir ? -1 : 1;
|
|
var av, bv;
|
|
switch (key) {
|
|
case 'size':
|
|
av = na.size; bv = nb.size; break;
|
|
case 'ext':
|
|
av = na.ext; bv = nb.ext; break;
|
|
case 'date':
|
|
av = na.modTime ? na.modTime.getTime() : 0;
|
|
bv = nb.modTime ? nb.modTime.getTime() : 0;
|
|
break;
|
|
default:
|
|
av = na.name.toLowerCase();
|
|
bv = nb.name.toLowerCase();
|
|
}
|
|
if (av < bv) return -1 * dir;
|
|
if (av > bv) return 1 * dir;
|
|
return na.name.toLowerCase().localeCompare(nb.name.toLowerCase());
|
|
});
|
|
}
|
|
|
|
// Populate state with the root listing.
|
|
function setRoot(rawEntries) {
|
|
clearTree();
|
|
rawEntries.forEach(function (raw) {
|
|
var n = newNode(raw, null, 0);
|
|
state.rootIds.push(n.id);
|
|
});
|
|
sortNodes(state.rootIds);
|
|
}
|
|
|
|
// Populate a folder's children. Caller passes raw entries in any order.
|
|
function setChildren(parentId, rawEntries) {
|
|
var parent = state.nodes.get(parentId);
|
|
if (!parent) return;
|
|
// Drop any existing children first (re-load case).
|
|
parent.childIds.forEach(function (id) { state.nodes.delete(id); });
|
|
parent.childIds = [];
|
|
rawEntries.forEach(function (raw) {
|
|
var n = newNode(raw, parentId, parent.depth + 1);
|
|
parent.childIds.push(n.id);
|
|
});
|
|
sortNodes(parent.childIds);
|
|
parent.loaded = true;
|
|
}
|
|
|
|
// Walk nodes in render order. Skips the children of a collapsed
|
|
// expandable.
|
|
function visibleIds() {
|
|
var out = [];
|
|
function walk(ids) {
|
|
for (var i = 0; i < ids.length; i++) {
|
|
var n = state.nodes.get(ids[i]);
|
|
if (!n) continue;
|
|
out.push(ids[i]);
|
|
if ((n.isDir || n.isZip) && n.expanded) walk(n.childIds);
|
|
}
|
|
}
|
|
// Re-sort everything at all levels so a sort change reorders
|
|
// already-loaded children consistently.
|
|
sortNodes(state.rootIds);
|
|
state.nodes.forEach(function (n) {
|
|
if ((n.isDir || n.isZip) && n.loaded) sortNodes(n.childIds);
|
|
});
|
|
walk(state.rootIds);
|
|
return out;
|
|
}
|
|
|
|
// ── Rendering ────────────────────────────────────────────────────────
|
|
|
|
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';
|
|
}
|
|
|
|
function fmtDate(d) {
|
|
if (!d) return '';
|
|
var pad = function (n) { return n < 10 ? '0' + n : '' + n; };
|
|
return d.getFullYear() + '-' + pad(d.getMonth() + 1) + '-' + pad(d.getDate())
|
|
+ ' ' + pad(d.getHours()) + ':' + pad(d.getMinutes());
|
|
}
|
|
|
|
function escapeHtml(s) {
|
|
return String(s).replace(/&/g, '&').replace(/</g, '<')
|
|
.replace(/>/g, '>').replace(/"/g, '"');
|
|
}
|
|
|
|
// Render a single tree row as a flat <div>. Indentation via
|
|
// padding-left so the row's hover background spans the full
|
|
// pane width. Files are rendered as plain rows (no anchor) —
|
|
// the preview pane handles click navigation, and a Ctrl/Cmd-
|
|
// click can fall back to opening the file's url in a new tab
|
|
// via the events.js click handler (it sees the modifier key).
|
|
function rowHtml(node) {
|
|
var indent = 0.4 + node.depth * 1.0;
|
|
var expandable = node.isDir || node.isZip;
|
|
var iconChar = node.isDir ? '📁' : (node.isZip ? '🗜️' : '📄');
|
|
var chevronClass = 'tree-name__chevron'
|
|
+ (expandable ? '' : ' tree-name__chevron--leaf');
|
|
var selected = state.selectedId === node.id ? ' is-selected' : '';
|
|
var virtualCls = node.virtual ? ' tree-row--virtual' : '';
|
|
var virtualHint = node.virtual
|
|
? '<span class="tree-name__hint" title="Folder not yet created on disk — opens an empty workspace">(empty)</span>'
|
|
: '';
|
|
return ''
|
|
+ '<div class="tree-row ' + (node.expanded ? 'expanded' : '') + selected + virtualCls
|
|
+ '" data-id="' + node.id
|
|
+ '" data-isdir="' + node.isDir
|
|
+ '" data-iszip="' + node.isZip + '"'
|
|
+ (node.virtual ? ' data-virtual="true"' : '')
|
|
+ ' style="padding-left:' + indent + 'rem"'
|
|
+ ' role="treeitem" tabindex="-1">'
|
|
+ '<span class="' + chevronClass + '"></span>'
|
|
+ '<span class="tree-name__icon">' + iconChar + '</span>'
|
|
+ '<span class="tree-name__label" title="' + escapeHtml(node.name) + '">'
|
|
+ escapeHtml(node.displayName || node.name) + '</span>'
|
|
+ virtualHint
|
|
+ '</div>';
|
|
}
|
|
|
|
function render() {
|
|
var body = document.getElementById('treeBody');
|
|
if (!body) return;
|
|
var ids = visibleIds();
|
|
var html = '';
|
|
for (var i = 0; i < ids.length; i++) {
|
|
html += rowHtml(state.nodes.get(ids[i]));
|
|
}
|
|
body.innerHTML = html;
|
|
updateCount();
|
|
renderBreadcrumbs();
|
|
}
|
|
|
|
// Count nodes that render at the root + every expanded subtree.
|
|
function expandedSetSize() {
|
|
var n = 0;
|
|
function walk(ids) {
|
|
for (var i = 0; i < ids.length; i++) {
|
|
n++;
|
|
var node = state.nodes.get(ids[i]);
|
|
if (node && (node.isDir || node.isZip) && node.expanded) {
|
|
walk(node.childIds);
|
|
}
|
|
}
|
|
}
|
|
walk(state.rootIds);
|
|
return n;
|
|
}
|
|
|
|
function updateCount() {
|
|
var el = document.getElementById('entryCount');
|
|
if (!el) return;
|
|
var total = expandedSetSize();
|
|
el.textContent = total + ' item' + (total === 1 ? '' : 's');
|
|
}
|
|
|
|
// ── Breadcrumbs ──────────────────────────────────────────────────────
|
|
|
|
// Inline outline home icon. Stroke-based so it tints with the
|
|
// current text color rather than depending on emoji rendering.
|
|
var HOME_SVG = '<svg class="bc-home-icon" xmlns="http://www.w3.org/2000/svg" '
|
|
+ 'viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" '
|
|
+ 'stroke-linecap="round" stroke-linejoin="round" aria-hidden="true">'
|
|
+ '<path d="M3 12l9-9 9 9"/>'
|
|
+ '<path d="M5 10v10h14V10"/>'
|
|
+ '<path d="M10 20v-6h4v6"/></svg>';
|
|
|
|
function renderBreadcrumbs() {
|
|
var el = document.getElementById('breadcrumbs');
|
|
if (!el) return;
|
|
var html = '';
|
|
if (state.source === 'server') {
|
|
// Server mode: every segment links to its directory URL.
|
|
// The browser navigates → server returns embedded browse →
|
|
// the new instance auto-loads that directory's listing.
|
|
var path = state.currentPath || '/';
|
|
var parts = path.split('/').filter(Boolean);
|
|
html += '<a class="bc-link bc-root" href="/" title="Site root">'
|
|
+ HOME_SVG + '</a>';
|
|
var sofar = '';
|
|
for (var i = 0; i < parts.length; i++) {
|
|
sofar += '/' + parts[i];
|
|
var isLast = i === parts.length - 1;
|
|
html += '<span class="bc-sep">/</span>';
|
|
if (isLast) {
|
|
html += '<span class="bc-link bc-link--current">'
|
|
+ escapeHtml(parts[i]) + '</span>';
|
|
} else {
|
|
html += '<a class="bc-link" href="' + escapeHtml(sofar + '/') + '">'
|
|
+ escapeHtml(parts[i]) + '</a>';
|
|
}
|
|
}
|
|
html += '<span class="bc-sep">/</span>';
|
|
} else if (state.source === 'fs') {
|
|
// FS-API mode: ancestor handles weren't retained when the
|
|
// user picked the root, so we can't navigate up. Show the
|
|
// root icon + handle name without links.
|
|
var name = state.rootHandle ? state.rootHandle.name : '';
|
|
html += '<span class="bc-link bc-root" title="Local directory">'
|
|
+ HOME_SVG + '</span>';
|
|
if (name) {
|
|
html += '<span class="bc-sep">/</span>';
|
|
html += '<span class="bc-link bc-link--current">' + escapeHtml(name) + '</span>';
|
|
}
|
|
html += '<span class="bc-sep">/</span>';
|
|
}
|
|
el.innerHTML = html;
|
|
}
|
|
|
|
// Sort headers no longer exist in the DOM (the tree replaced the
|
|
// table); the tree.setSort() method still works but only via
|
|
// programmatic callers — there's no UI for changing sort yet.
|
|
|
|
// True when this .zip node lives inside another zip, so its bytes
|
|
// can't be fetched as a standalone server resource: we read them
|
|
// through the containing handle (offline / nested) or by fetching
|
|
// the inner-zip member URL. In server mode a zip-inside-a-zip's URL
|
|
// contains ".zip/"; offline it has a handle that is itself a zip
|
|
// entry.
|
|
function zipNestedInsideZip(node) {
|
|
if (state.source === 'server') {
|
|
return pathFor(node).toLowerCase().indexOf('.zip/') !== -1;
|
|
}
|
|
return !!(node.handle && node.handle.isZipEntry);
|
|
}
|
|
|
|
// Open a .zip node as a directory handle (a ZipDirectoryHandle over
|
|
// a JSZip instance), cached on the node. Bytes come from a real
|
|
// FileSystemFileHandle / ZipFileHandle when present (offline, or a
|
|
// zip nested in a zip), else from a server URL — zddc-server returns
|
|
// the raw .zip for "<…>.zip" and the inner-zip bytes for
|
|
// "<outer>.zip/inner.zip".
|
|
async function zipDirHandle(node) {
|
|
if (node._zipDirHandle) return node._zipDirHandle;
|
|
await loader.ensureJSZip();
|
|
var zh;
|
|
if (node.handle) {
|
|
zh = await window.zddc.zip.fromFileHandle(node.handle);
|
|
} else if (node.url) {
|
|
var resp = await fetch(node.url, { credentials: 'same-origin' });
|
|
if (!resp.ok) throw new Error('HTTP ' + resp.status + ' fetching ' + node.url);
|
|
zh = await window.zddc.zip.fromBlob(await resp.arrayBuffer(), node.name);
|
|
} else {
|
|
throw new Error('cannot open zip ' + node.name + ' (no handle or URL)');
|
|
}
|
|
node._zipDirHandle = zh;
|
|
return zh;
|
|
}
|
|
|
|
// Load a folder's children (lazy; idempotent re-loads). Dispatches
|
|
// by node kind:
|
|
// - regular folder → server JSON listing OR FS-API entries
|
|
// - top-level .zip, server mode → the server's "<…>.zip/" virtual-
|
|
// directory listing (no whole-zip
|
|
// download — zddc-server extracts a
|
|
// member only when one is requested)
|
|
// - .zip otherwise (offline, or a zip nested in a zip)
|
|
// → open it with JSZip and enumerate
|
|
// it as a directory handle; members
|
|
// become ordinary dir/file nodes
|
|
async function loadChildren(node) {
|
|
if (node.loaded) return;
|
|
try {
|
|
if (node.isZip && state.source === 'server' && !zipNestedInsideZip(node)) {
|
|
setChildren(node.id, await loader.fetchServerChildren(pathFor(node) + '/'));
|
|
} else if (node.isZip) {
|
|
setChildren(node.id, await loader.fetchFsChildren(await zipDirHandle(node)));
|
|
} else if (node.isDir) {
|
|
var raw;
|
|
if (state.source === 'server') {
|
|
raw = await loader.fetchServerChildren(pathFor(node) + '/');
|
|
} else if (state.source === 'fs') {
|
|
raw = await loader.fetchFsChildren(node.handle);
|
|
} else {
|
|
return;
|
|
}
|
|
setChildren(node.id, raw);
|
|
}
|
|
} catch (e) {
|
|
window.app.modules.events.statusError(
|
|
'Failed to load ' + node.name + ': ' + e.message);
|
|
}
|
|
}
|
|
|
|
// Toggle a folder's expanded state. Loads children on first expand.
|
|
// Treats "expandable" as either a real directory OR a zip file
|
|
// (zip files act like folders for tree purposes — the chevron
|
|
// expands them and the contents come from JSZip).
|
|
async function toggleFolder(nodeId) {
|
|
var n = state.nodes.get(nodeId);
|
|
if (!n || !(n.isDir || n.isZip)) return;
|
|
if (!n.expanded && !n.loaded) {
|
|
await loadChildren(n);
|
|
if (!n.loaded) return; // load failed (statusError already set)
|
|
}
|
|
n.expanded = !n.expanded;
|
|
render();
|
|
}
|
|
|
|
// Recursive expand: load + expand all descendants of nodeId. Used
|
|
// for Shift-click on a folder. Walks breadth-first, fanning out
|
|
// through children, grand-children, etc. until every reachable
|
|
// expandable node (folder OR zip) is loaded and marked expanded.
|
|
// Skips zip-EXPANSION recursion to avoid auto-loading every
|
|
// archive in the tree (those can be huge); plain folders only.
|
|
async function expandSubtree(nodeId) {
|
|
var root = state.nodes.get(nodeId);
|
|
if (!root || !(root.isDir || root.isZip)) return;
|
|
var status = window.app.modules.events.statusInfo;
|
|
status('Expanding subtree…');
|
|
var processed = 0;
|
|
var queue = [root];
|
|
while (queue.length) {
|
|
var batch = queue;
|
|
queue = [];
|
|
await Promise.all(batch.map(function (n) { return loadChildren(n); }));
|
|
for (var i = 0; i < batch.length; i++) {
|
|
var n = batch[i];
|
|
n.expanded = true;
|
|
processed++;
|
|
for (var j = 0; j < n.childIds.length; j++) {
|
|
var c = state.nodes.get(n.childIds[j]);
|
|
// Recurse into plain folders only — don't auto-
|
|
// expand zip archives during a subtree expand
|
|
// (they can be very large).
|
|
if (c && c.isDir && !c.isZip) queue.push(c);
|
|
}
|
|
}
|
|
render();
|
|
status('Expanding subtree… (' + processed + ' folders loaded)');
|
|
}
|
|
status('Expanded ' + processed + ' folder' + (processed === 1 ? '' : 's'));
|
|
}
|
|
|
|
function collapseSubtree(nodeId) {
|
|
var root = state.nodes.get(nodeId);
|
|
if (!root || !(root.isDir || root.isZip)) return;
|
|
function walk(n) {
|
|
n.expanded = false;
|
|
for (var i = 0; i < n.childIds.length; i++) {
|
|
var c = state.nodes.get(n.childIds[i]);
|
|
if (c && (c.isDir || c.isZip)) walk(c);
|
|
}
|
|
}
|
|
walk(root);
|
|
render();
|
|
}
|
|
|
|
// Compute the URL/path for a node by walking parents.
|
|
function pathFor(node) {
|
|
var parts = [];
|
|
var cur = node;
|
|
while (cur) {
|
|
parts.unshift(cur.name);
|
|
cur = cur.parentId == null ? null : state.nodes.get(cur.parentId);
|
|
}
|
|
if (state.source === 'server') {
|
|
// currentPath is the dir containing rootIds — root nodes
|
|
// sit DIRECTLY under it.
|
|
return state.currentPath.replace(/\/$/, '') + '/' + parts.join('/');
|
|
}
|
|
return parts.join('/');
|
|
}
|
|
|
|
// Public API
|
|
window.app.modules.tree = {
|
|
setRoot: setRoot,
|
|
setChildren: setChildren,
|
|
render: render,
|
|
toggleFolder: toggleFolder,
|
|
expandSubtree: expandSubtree,
|
|
collapseSubtree: collapseSubtree,
|
|
setSort: function (key) {
|
|
if (state.sort.key === key) {
|
|
state.sort.dir = -state.sort.dir;
|
|
} else {
|
|
state.sort.key = key;
|
|
state.sort.dir = 1;
|
|
}
|
|
render();
|
|
},
|
|
// Set both key and direction explicitly. dir: 1 (asc) or -1 (desc).
|
|
// Used by the toolbar's sort dropdown.
|
|
setSortExplicit: function (key, dir) {
|
|
state.sort.key = key;
|
|
state.sort.dir = (dir === -1 ? -1 : 1);
|
|
render();
|
|
},
|
|
pathFor: pathFor
|
|
};
|
|
})();
|