import { createElement } from "../../lib/skeleton/index.js"; import { qs } from "../../lib/dom.js"; import assert from "../../lib/assert.js"; import { addSelection, isSelected } from "./state_selection.js"; const IMAGE = { FILE: "", FOLDER: "" }; const $tmpl = createElement(`
`); // 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 = null, type = "N/A", time = 0, path = null, // size = 0, // time = null, link = "", // permissions = {} view = "", n = 0, read_only = false, }) { const $thing = $tmpl.cloneNode(true); assert.type($thing, window.HTMLElement); const $label = qs($thing, ".component_filename .file-details > span"); const $time = qs($thing, ".component_datetime"); $label.textContent = name; $thing.querySelector("a").setAttribute("href", link); $thing.querySelector("img").setAttribute("src", (type === "file" ? IMAGE.FILE : IMAGE.FOLDER)); $thing.setAttribute("data-droptarget", type === "directory"); $thing.setAttribute("data-n", n); $thing.classList.add("view-" + view); $time.textContent = formatTime(new Date(time)); sideEffectSelection($thing, isSelected(n)); if (read_only === true) return $thing; else if (type === "hidden") { $thing.classList.add("hidden"); return $thing; } $thing.querySelector(".component_checkbox").onclick = function(e) { e.preventDefault(); e.stopPropagation(); addSelection({ name, type, shift: e.shiftKey, n, }); $thing.closest(".list").querySelectorAll(".component_thing").forEach(($el) => { sideEffectSelection($el, isSelected(parseInt($el.getAttribute("data-n")))); }); }; $thing.ondragstart = (e) => { e.dataTransfer.setData("path", path); $thing.classList.add("hover"); const crt = $thing.cloneNode(true); $thing.style.opacity = "0.7"; const $box = crt.querySelector(".box"); crt.style.opacity = "0.2"; crt.style.backgroundColor = "var(--border)"; $box.style.backgroundColor = "inherit"; $box.style.border = "none"; $box.style.borderRadius = "0"; $thing.closest("[data-target=\"list\"]").appendChild(crt); e.dataTransfer.setDragImage(crt, e.offsetX, -10); }; $thing.ondragover = (e) => { if ($thing.getAttribute("data-droptarget") !== "true") return; e.preventDefault(); $thing.classList.add("hover"); }; $thing.ondragleave = () => { $thing.classList.remove("hover"); }; $thing.ondrop = (e) => { $thing.classList.remove("hover"); console.log("DROPPED!", e.dataTransfer.getData("path")); }; return $thing; } function sideEffectSelection($el, checked) { $el.classList.add(checked ? "selected" : "not-selected"); $el.querySelector(`input[type="checkbox"]`).checked = checked; } function formatTime(date) { if (!date) return ""; return new Intl.DateTimeFormat(navigator.language || "en-US") .format(date) .split("/") .map((chunk) => chunk.padStart(2, "0")) .join("/"); }