mirror of
https://github.com/mickael-kerjean/filestash
synced 2025-12-14 12:26:53 +01:00
119 lines
7.4 KiB
JavaScript
119 lines
7.4 KiB
JavaScript
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(`
|
|
<div class="component_thing" draggable="true">
|
|
<a href="__TEMPLATE__" data-link>
|
|
<div class="box">
|
|
<div class="component_checkbox"><input type="checkbox"><span class="indicator"></span></div>
|
|
<span>
|
|
<img class="component_icon" draggable="false" src="__TEMPLATE__" alt="directory">
|
|
</span>
|
|
<span class="component_filename">
|
|
<span class="file-details">
|
|
<span>__TEMPLATE__<span class="extension"></span></span>
|
|
</span>
|
|
</span>
|
|
<span class="component_datetime"></span>
|
|
<div class="component_action"></div>
|
|
<div class="selectionOverlay"></div>
|
|
</div>
|
|
</a>
|
|
</div>
|
|
`);
|
|
|
|
// 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,
|
|
}) {
|
|
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 (type === "hidden") $thing.classList.add("hidden");
|
|
|
|
$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("/");
|
|
}
|