Files
2025-08-26 10:38:26 +10:00

217 lines
11 KiB
JavaScript

import { toHref } from "../lib/skeleton/router.js";
import { animate, slideYOut, slideYIn, opacityOut } from "../lib/animate.js";
import { forwardURLParams } from "../lib/path.js";
import { safe } from "../lib/dom.js";
import assert from "../lib/assert.js";
import { settingsSave } from "../lib/store.js";
import { get as getConfig } from "../model/config.js";
import { loadCSS } from "../helpers/loader.js";
import { extractPath, isDir, isNativeFileUpload } from "../pages/filespage/helper.js";
import { mv as mv$ } from "../pages/filespage/model_files.js";
import { mv as mvVL, withVirtualLayer } from "../pages/filespage/model_virtual_layer.js";
const mv = (from, to) => withVirtualLayer(
mv$(from, to),
mvVL(from, to),
);
class ComponentBreadcrumb extends HTMLElement {
constructor() {
super();
if (new URL(location.href).searchParams.get("nav") === "false") {
this.disabled = true;
return;
}
this.__init();
}
async __init() {
this.innerHTML = `
<nav class="component_breadcrumb container" aria-label="Breadcrumb">
<div class="breadcrumb no-select">
<div class="ul">
<img alt="sidebar-open" class="hidden" src="">
<span data-bind="path" role="status"></span>
<div class="li component_logout">${this.__htmlLogout()}</div>
</div>
</div>
</nav>`;
assert.type(this.querySelector("img[alt=\"sidebar-open\"]"), HTMLElement).onclick = () => {
settingsSave({ visible: true }, "sidebar");
window.dispatchEvent(new Event("resize"));
};
}
attributeChangedCallback(name, oldValue, newValue) {
if (this.disabled === true) return;
else if (oldValue === newValue) return;
switch (name) {
case "path":
if (newValue === "") return;
return this.renderPath({ path: newValue, previous: oldValue || null });
case "indicator":
return this.renderIndicator();
}
throw new Error("component::breadcrumb.js unknow attribute name: "+ name);
}
static get observedAttributes() {
return ["path", "indicator"];
}
async renderPath({ path = "", previous }) {
path = this.__normalised(path);
previous = this.__normalised(previous);
const pathChunks = path.split("/");
// STEP1: leaving animation on elements that will be removed
if (previous !== null && previous.indexOf(path) >= 0) {
const previousChunks = previous.split("/");
const nToAnimate = previousChunks.length - pathChunks.length;
const tasks = [];
for (let i=0; i<nToAnimate; i++) {
const n = previousChunks.length - i - 1;
const $chunk = assert.type(this.querySelector(`.component_path-element.n${n}`), HTMLElement);
tasks.push(animate($chunk, { time: 100, keyframes: slideYOut(-10) }));
}
await Promise.all(tasks);
}
// STEP2: setup the actual content
assert.type(this.querySelector(`[data-bind="path"]`), HTMLElement).innerHTML = pathChunks.map((chunk, idx) => {
const label = idx === 0 ? getConfig("name", "Filestash") : chunk;
const link = pathChunks.slice(0, idx + 1).join("/") + "/";
const limitSize = (word, highlight = false) => {
if (highlight === true && word.length > 30) {
return word.substring(0, 12).trim() + "..." +
word.substring(word.length - 10, word.length).trim();
}
else if (word.length > 27) return word.substring(0, 20).trim() + "...";
return word;
};
const isLast = idx === pathChunks.length - 1;
if (isLast) return `
<div class="component_path-element n${idx}">
<div class="li component_path-element-wrapper">
<div class="label">
<div aria-current="location">${safe(limitSize(label))}</div><span></span>
</div>
</div>
</div>`;
const minify = (() => {
if (idx === 0) return false;
else if (pathChunks.length <= (document.body.clientWidth > 800 ? 5 : 4)) return false;
else if (idx > pathChunks.length - (document.body.clientWidth > 1000 ? 4 : 3)) return false;
return true;
})();
const tmpl = (() => {
if (minify) return `
...
<span class="title">
${safe(limitSize(label, true))}
</span>
`;
return `<div>${safe(limitSize(label))}</div>`;
})();
return `
<div class="component_path-element n${idx}" data-path="${safe(pathChunks.slice(0, idx+1).join("/")) + "/"}">
<div class="li component_path-element-wrapper">
<a class="label" aria-label="${safe(label)}" href="${forwardURLParams(toHref("/files" + encodeURIComponent(link).replaceAll("%2F", "/")), ["share", "canary"])}" data-link draggable="false">
${tmpl}
</a>
<div class="component_separator">
<img alt="path_separator" width="16" height="16" src="">
</div>
</div>
</div>`;
}).join("");
this.setupDragDropTarget();
// STEP3: entering animation for elements that got added in
if (previous !== null && path.indexOf(previous) >= 0) {
const previousChunks = previous.split("/");
const nToAnimate = pathChunks.length - previousChunks.length;
for (let i=0; i<nToAnimate; i++) {
const n = pathChunks.length - i - 1;
const $chunk = assert.type(this.querySelector(`.component_path-element.n${n}`), HTMLElement);
await animate($chunk, { time: 100, keyframes: slideYIn(-5) });
}
}
}
async renderIndicator() {
let state = this.hasAttribute("indicator");
if (state && this.getAttribute("indicator") !== "false") state = true;
let $indicator = assert.type(this.querySelector(`[data-bind="path"]`), HTMLElement);
$indicator = assert.type($indicator.lastChild.querySelector("span"), HTMLElement);
if (state) {
$indicator.style.opacity = 1;
$indicator.innerHTML = `<div class="component_saving">*</div>`;
await animate($indicator, {
time: 500,
keyframes: [
{ transform: "scale(0)", offset: 0 },
{ transform: "scale(1.5)", offset: 0.3 },
{ transform: "scale(1)", offset: 1 },
],
fill: "none"
});
} else {
$indicator.style.opacity = 0;
await animate($indicator, { time: 200, keyframes: opacityOut(), fill: "none" });
}
}
setupDragDropTarget() {
this.querySelectorAll("a.label").forEach(($elmnt) => {
const $folder = assert.type($elmnt, HTMLElement);
const $path = assert.truthy($folder.closest(".component_path-element"));
$folder.ondrop = async(e) => {
$path.classList.remove("highlight");
const from = e.dataTransfer.getData("path");
let to = $path.getAttribute("data-path");
const [, fromName] = extractPath(from);
to += fromName;
if (isDir(from)) to += "/";
await mv(from, to).toPromise();
};
$folder.ondragover = (e) => {
if (isNativeFileUpload(e)) return;
e.preventDefault();
$path.classList.add("highlight");
};
$folder.ondragleave = () => {
$path.classList.remove("highlight");
};
});
}
__htmlLogout() {
if (window.self !== window.top) return ""; // no logout button from an iframe
return `
<a href="${toHref("/logout")}" data-link draggable="false">
<img class="component_icon" draggable="false" src="" alt="power">
</a>
`;
}
__normalised(path) {
if (path === null) return null;
else if (path.endsWith("/") === false) return path;
return path.replace(new RegExp("/$"), "");
}
}
export function init() {
return loadCSS(import.meta.url, "./breadcrumb.css");
}
customElements.define("component-breadcrumb", ComponentBreadcrumb);