Files
2025-02-04 02:35:23 +11:00

236 lines
33 KiB
JavaScript

import { createElement } from "../../lib/skeleton/index.js";
import { animate, opacityOut, opacityIn } from "../../lib/animate.js";
import assert from "../../lib/assert.js";
import { get as getConfig } from "../../model/config.js";
import { extractPath, isDir, isNativeFileUpload } from "./helper.js";
import { files$ } from "./ctrl_filesystem.js";
import { addSelection, isSelected, clearSelection } from "./state_selection.js";
import { mv as mv$ } from "./model_files.js";
import { mv as mvVL, withVirtualLayer } from "./model_virtual_layer.js";
const mv = (from, to) => withVirtualLayer(
mv$(from, to),
mvVL(from, to),
);
const IMAGE = {
FILE: "",
FOLDER: "",
LOADING: "",
THUMBNAIL_PLACEHOLDER: "",
};
let TYPES = null;
export function init() {
TYPES = {
MIME: getConfig("mime", {}),
THUMBNAILER: (function() {
const set = new Set();
const thumbnailers = getConfig("thumbnailer");
for (let i=0; i<thumbnailers.length; i++) {
set.add(thumbnailers[i]);
}
return set;
})(),
};
}
const $tmpl = createElement(`
<a href="__TEMPLATE__" class="component_thing no-select" draggable="false" data-link>
<div class="component_checkbox"><input name="select" type="checkbox"><span class="indicator"></span></div>
<img class="component_icon" loading="lazy" draggable="false" src="__TEMPLATE__" alt="directory">
<div class="info_extension"><span class="ellipsis"></span></div>
<span class="component_filename">
<span class="file-details"><span class="ellipsis">
<span class="component_filesize">(281B)</span>
</span></span>
</span>
<span class="component_datetime"></span>
<div class="selectionOverlay"></div>
</a>
`);
// a filesystem "thing" is typically either a file or folder which have a lot of behavior builtin.
// Probably one day we can rename that to something more clear but the gist is a thing can be
// displayed in list mode / grid mode, have some substate to enable loading state for upload,
// can toggle links, potentially includes a thumbnail, can be used as a source and target for
// drag and drop on other folders and many other non obvious stuff
export function createThing({
name = "",
type = "N/A",
time = 0,
path = "",
size = 0,
loading = false,
link = "",
view = "",
search = "",
n = 0,
permissions = {},
}) {
const [, ext] = formatFile(name);
const mime = TYPES.MIME[ext.toLowerCase()];
const $thing = assert.type($tmpl.cloneNode(true), HTMLElement);
// you might wonder why don't we use querySelector to nicely get the dom nodes? Well,
// we're in the hot path, better performance here is critical to get 60FPS.
const $link = $thing;
const $checkbox = $thing.children[0]; // = qs($thing, ".component_checkbox");
const $img = $thing.children[1]; // = qs($thing, "img")
const $extension = $thing.children[2].firstElementChild; // = qs($thing, ".info_extension > span");
const $label = $thing.children[3].firstElementChild.firstElementChild; // = qs($thing, ".component_filename .file-details > span");
const $time = $thing.children[4]; // = qs($thing, ".component_datetime");
$link.setAttribute("href", link + location.search);
$thing.setAttribute("data-droptarget", type === "directory");
$thing.setAttribute("data-n", n);
$thing.setAttribute("data-path", path);
$thing.classList.add("view-" + view);
$time.textContent = formatTime(time);
$img.setAttribute("src", (type === "file" ? IMAGE.FILE : IMAGE.FOLDER));
$label.textContent = name;
if (type === "file") {
$extension.textContent = ext;
const $filesize = document.createElement("span");
$filesize.classList.add("component_filesize");
$filesize.textContent = formatSize(size);
$label.appendChild($filesize);
}
if (mime && view === "grid" && TYPES.THUMBNAILER.has(mime)) {
$extension.classList.add("hidden");
$img.classList.add("thumbnail");
const $placeholder = $img.cloneNode(true);
$placeholder.classList.add("placeholder");
$placeholder.setAttribute("src", IMAGE.THUMBNAIL_PLACEHOLDER);
$img.parentElement.appendChild($placeholder);
$img.src = "api/files/cat?path=" + encodeURIComponent(path) + "&thumbnail=true" + location.search.replace("?", "&");
$img.loaded = false;
const t = new Date().getTime();
$img.onload = async() => {
const duration = new Date().getTime() - t;
$img.loaded = true;
await Promise.all([
animate($img, {
keyframes: opacityIn(),
time: duration > 1500 ? 300 : duration > 50 ? 200 : 0,
}),
animate($placeholder, {
keyframes: opacityOut(),
time: duration > 1500 ? 300 : duration > 50 ? 200 : 0,
}),
]);
$placeholder.remove();
};
const id = setInterval(() => { // cancellation when image is outside the viewport
if ($img.loaded === true) return clearInterval(id);
else if (typeof $thing.checkVisibility !== "function") return clearInterval(id);
else if ($thing.checkVisibility() === false) {
$img.src = "";
clearInterval(id);
}
}, 250);
}
if (loading) {
$img.setAttribute("src", IMAGE.LOADING);
$link.setAttribute("href", "#");
$extension.innerHTML = "";
return $thing;
} else if (search !== "") {
$checkbox.classList.add("hidden");
if (view === "list") $img.style.visibility = "initial";
return $thing;
} else if (type === "hidden") {
$thing.classList.add("hidden");
return $thing;
}
const checked = isSelected(n);
if (permissions && permissions.can_move !== false) $thing.setAttribute("draggable", "true");
$thing.classList.add(checked ? "selected" : "not-selected");
$checkbox.firstElementChild.checked = checked;
$checkbox.onclick = (e) => {
e.preventDefault();
e.stopPropagation();
addSelection({
n,
path: $thing.getAttribute("data-path"),
shift: (e.shiftKey || e.metaKey),
files: (files$.value || []),
});
};
$thing.ondragstart = (e) => {
clearSelection();
$thing.classList.add("hover");
e.dataTransfer.setData("path", path);
e.dataTransfer.setDragImage($thing, e.offsetX, -10);
};
$thing.ondrop = async(e) => {
$thing.classList.remove("hover");
const from = e.dataTransfer.getData("path");
let to = path;
if ($thing.getAttribute("data-droptarget") !== "true") return;
else if (from === to) return;
if (isDir(to)) {
const [, fromName] = extractPath(from);
to += fromName;
if (isDir(from)) to += "/";
}
await mv(from, to).toPromise();
};
$thing.ondragover = (e) => {
if (isNativeFileUpload(e)) return;
else if ($thing.getAttribute("data-droptarget") !== "true") return;
e.preventDefault();
$thing.classList.add("hover");
};
$thing.ondragleave = () => {
$thing.classList.remove("hover");
};
return $thing;
}
function formatTime(unixTime) {
if (unixTime <= 0) return "";
const date = new Date(unixTime);
if (!date) return "";
// Intl.DateTimeFormat is slow and in the hot path, so
// let's render date manually if possible
if (navigator.language.substr(0, 2) === "en") {
return date.getFullYear() + "/" +
(date.getMonth() + 1).toString().padStart(2, "0") + "/" +
date.getDate().toString().padStart(2, "0");
}
return new Intl.DateTimeFormat(navigator.language).format(date);
}
function formatFile(filename) {
const fname = filename.split(".");
if (fname.length < 2) {
return [filename, ""];
}
const ext = fname.pop();
return [fname.join("."), ext];
}
function formatSize(bytes) {
if (Number.isNaN(bytes) || bytes < 0 || bytes === undefined) {
return "";
} else if (bytes < 1024) {
return "("+bytes+"B)";
} else if (bytes < 1048576) {
return "("+Math.round(bytes/1024*10)/10+"KB)";
} else if (bytes < 1073741824) {
return "("+Math.round(bytes/(1024*1024)*10)/10+"MB)";
} else if (bytes < 1099511627776) {
return "("+Math.round(bytes/(1024*1024*1024)*10)/10+"GB)";
} else {
return "("+Math.round(bytes/(1024*1024*1024*1024))+"TB)";
}
}