mirror of
https://github.com/mickael-kerjean/filestash
synced 2025-12-24 01:04:39 +01:00
This contains a bunch of things packaged in 1: 1) UI improvements for the 3D viewer to support all sort of file types and create a nice rendering in a clean way with all sort of options 2) enable people to use Filestash as an SDK so we can embed the 3d viewer elsewhere
279 lines
15 KiB
JavaScript
279 lines
15 KiB
JavaScript
import { createElement, createRender, onDestroy } from "../lib/skeleton/index.js";
|
|
import rxjs, { effect, onClick } from "../lib/rx.js";
|
|
import assert from "../lib/assert.js";
|
|
import { fromHref, toHref } from "../lib/skeleton/router.js";
|
|
import { qs, qsa } from "../lib/dom.js";
|
|
import { forwardURLParams } from "../lib/path.js";
|
|
import { settingsGet, settingsSave } from "../lib/store.js";
|
|
import { loadCSS } from "../helpers/loader.js";
|
|
import t from "../locales/index.js";
|
|
import cache from "../pages/filespage/cache.js";
|
|
import { hooks, mv as mv$ } from "../pages/filespage/model_files.js";
|
|
import { extractPath, isDir, isNativeFileUpload } from "../pages/filespage/helper.js";
|
|
import { mv as mvVL, withVirtualLayer } from "../pages/filespage/model_virtual_layer.js";
|
|
|
|
const state = { scrollTop: 100, $cache: null };
|
|
const mv = (from, to) => withVirtualLayer(
|
|
mv$(from, to),
|
|
mvVL(from, to),
|
|
);
|
|
|
|
export default async function ctrlSidebar(render, nRestart = 0) {
|
|
if (!shouldDisplay()) return;
|
|
|
|
const $sidebar = render(createElement(`
|
|
<div class="component_sidebar"><div>
|
|
<h3>
|
|
<img src="" alt="close">
|
|
<input type="text" placeholder="${t("Your Files")}" />
|
|
</h3>
|
|
<div data-bind="your-files"></div>
|
|
|
|
<h3>
|
|
<img src="" alt="tag">
|
|
${t("Tags")}
|
|
</h3>
|
|
<div data-bind="your-tags"></div>
|
|
</div>
|
|
`));
|
|
|
|
// feature: visibility of the sidebar
|
|
const forceRefresh = () => window.dispatchEvent(new Event("resize"));
|
|
const isVisible = () => settingsGet({ visible: true }, "sidebar").visible;
|
|
effect(rxjs.merge(rxjs.fromEvent(window, "keydown")).pipe(
|
|
rxjs.filter((e) => e.key === "b" && e.ctrlKey === true),
|
|
rxjs.tap(() => {
|
|
settingsSave({ visible: $sidebar.classList.contains("hidden") }, "sidebar");
|
|
isVisible() ? $sidebar.classList.remove("hidden") : $sidebar.classList.add("hidden");
|
|
forceRefresh();
|
|
}),
|
|
));
|
|
effect(rxjs.merge(
|
|
rxjs.fromEvent(window, "resize"),
|
|
rxjs.of(null),
|
|
).pipe(rxjs.tap(() => {
|
|
const $breadcrumbButton = qs(document.body, "[alt=\"sidebar-open\"]");
|
|
if (document.body.clientWidth < 1100) $sidebar.classList.add("hidden");
|
|
else if (isVisible()) {
|
|
$sidebar.classList.remove("hidden");
|
|
$breadcrumbButton.classList.add("hidden");
|
|
} else {
|
|
$sidebar.classList.add("hidden");
|
|
$breadcrumbButton.classList.remove("hidden");
|
|
}
|
|
}), rxjs.catchError((err) => {
|
|
if (err instanceof DOMException) return rxjs.EMPTY;
|
|
throw err;
|
|
})));
|
|
effect(onClick(qs($sidebar, `img[alt="close"]`)).pipe(rxjs.tap(() => {
|
|
settingsSave({ visible: false }, "sidebar");
|
|
$sidebar.classList.add("hidden");
|
|
forceRefresh();
|
|
})));
|
|
|
|
// fature: navigation pane
|
|
ctrlNavigationPane(render, { $sidebar, nRestart });
|
|
|
|
// feature: tag viewer
|
|
ctrlTagPane(createRender(qs($sidebar, `[data-bind="your-tags"]`)));
|
|
}
|
|
|
|
async function ctrlNavigationPane(render, { $sidebar, nRestart }) {
|
|
// feature: setup the DOM
|
|
const $files = qs($sidebar, `[data-bind="your-files"]`);
|
|
if (state.$cache) {
|
|
$files.replaceChildren(state.$cache);
|
|
$sidebar.firstElementChild.scrollTop = state.scrollTop;
|
|
}
|
|
onDestroy(() => {
|
|
$sidebar.classList.remove("search");
|
|
state.$cache = $files.firstElementChild?.cloneNode(true);
|
|
state.scrollTop = $sidebar.firstElementChild.scrollTop;
|
|
});
|
|
const chunk = new PathChunk();
|
|
const arr = chunk.toArray();
|
|
const fullpath = chunk.toString();
|
|
const $tree = document.createDocumentFragment();
|
|
for (let i = 0; i<arr.length-1; i++) {
|
|
const path = chunk.toString(i);
|
|
try {
|
|
const $list = await _createListOfFiles(path, arr[i+1], fullpath);
|
|
const $anchor = i === 0 ? $tree : qs($tree, `[data-path="${chunk.toString(i)}"]`);
|
|
$anchor.appendChild($list);
|
|
} catch (err) {
|
|
await cache().remove("/", false);
|
|
if (err instanceof DOMException) return;
|
|
else if (nRestart < 2) ctrlSidebar(render, nRestart + 1);
|
|
else throw err;
|
|
}
|
|
}
|
|
$files.replaceChildren($tree);
|
|
$sidebar.firstElementChild.scrollTop = state.scrollTop;
|
|
|
|
// feature: smart refresh whenever something happen
|
|
const cleaners = [];
|
|
cleaners.push(hooks.ls.listen(async({ path }) => {
|
|
const $list = await _createListOfFiles(path);
|
|
try {
|
|
const $ul = qs($sidebar, `[data-path="${path}"] ul`);
|
|
$ul.replaceWith($list);
|
|
} catch (err) { $files.replaceChildren($list); }
|
|
}));
|
|
cleaners.push(hooks.mutation.listen(async({ op, path }) => {
|
|
if (["mv", "mkdir", "rm"].indexOf(op) === -1) return;
|
|
const $list = await _createListOfFiles(path);
|
|
try {
|
|
const $ul = qs($sidebar, `[data-path="${path}"] ul`);
|
|
$ul.replaceWith($list);
|
|
} catch (err) {}
|
|
}));
|
|
onDestroy(() => cleaners.map((fn) => fn()));
|
|
|
|
// feature: highlight current selection
|
|
try {
|
|
const $active = qs($sidebar, `[data-path="${chunk.toString()}"] a`);
|
|
$active.classList.add("active");
|
|
if (checkVisible($active) === false) {
|
|
$active.scrollIntoView({ behavior: "smooth" });
|
|
}
|
|
} catch (err) {}
|
|
|
|
// feature: quick search
|
|
effect(rxjs.fromEvent(qs($sidebar, "h3 input"), "keydown").pipe(
|
|
rxjs.debounceTime(200),
|
|
rxjs.tap((e) => {
|
|
const inputValue = e.target.value.toLowerCase();
|
|
qsa($sidebar, "[data-bind=\"your-files\"] li a").forEach(($li) => {
|
|
if (inputValue === "") {
|
|
$li.classList.remove("hidden");
|
|
$sidebar.classList.remove("search");
|
|
return;
|
|
}
|
|
$sidebar.classList.add("search");
|
|
qs($li, "div").textContent.toLowerCase().indexOf(inputValue) === -1
|
|
? $li.classList.add("hidden")
|
|
: $li.classList.remove("hidden");
|
|
});
|
|
}),
|
|
));
|
|
}
|
|
|
|
async function _createListOfFiles(path, currentName, fullpath) {
|
|
const r = await cache().get(path);
|
|
const whats = r === null
|
|
? (currentName ? [currentName] : [])
|
|
: r.files
|
|
.filter(({ type, name }) => type === "directory" && name[0] !== ".")
|
|
.map(({ name }) => name)
|
|
.sort();
|
|
const $ul = document.createElement("ul");
|
|
for (let i=0; i<whats.length; i++) {
|
|
const currpath = path + whats[i] + "/";
|
|
const $li = createElement(`
|
|
<li data-path="${currpath}" title="${currpath}" class="no-select">
|
|
<a data-link href="${forwardURLParams(toHref("/files" + encodeURIComponent(currpath).replaceAll("%2F", "/")), ["share", "canary"])}" draggable="false">
|
|
<img class="component_icon" src="" alt="directory">
|
|
<div class="ellipsis">${whats[i]}</div>
|
|
</a>
|
|
</li>
|
|
`);
|
|
$ul.appendChild($li);
|
|
const $link = qs($li, "a");
|
|
if ($li.getAttribute("data-path") === fullpath) {
|
|
$link.removeAttribute("href", "");
|
|
$link.removeAttribute("data-link");
|
|
continue;
|
|
}
|
|
$link.ondrop = async(e) => {
|
|
$link.classList.remove("highlight");
|
|
const from = e.dataTransfer.getData("path");
|
|
let to = $link.parentElement.getAttribute("data-path");
|
|
const [, fromName] = extractPath(from);
|
|
to += fromName;
|
|
if (isDir(from)) to += "/";
|
|
if (from === to) return;
|
|
await mv(from, to).toPromise();
|
|
};
|
|
$link.ondragover = (e) => {
|
|
if (isNativeFileUpload(e)) return;
|
|
e.preventDefault();
|
|
$link.classList.add("highlight");
|
|
};
|
|
$link.ondragleave = () => {
|
|
$link.classList.remove("highlight");
|
|
};
|
|
}
|
|
return $ul;
|
|
}
|
|
|
|
async function ctrlTagPane(render) {
|
|
const $page = createElement(`
|
|
<ul>
|
|
<li data-bind="taglist"></li>
|
|
</ul>
|
|
`);
|
|
render($page);
|
|
|
|
// only enable this pane in canary mode until it's actually ready
|
|
if (new URLSearchParams(location.search).get("canary") !== "true") {
|
|
$page.classList.add("hidden");
|
|
const orFail = (something) => assert.type(something, HTMLElement);
|
|
orFail(orFail($page.parentElement).previousElementSibling).classList.add("hidden");
|
|
return;
|
|
}
|
|
|
|
const tags = [
|
|
{ name: t("Bookmark"), color: "green" },
|
|
{ name: "important", color: "red" },
|
|
{ name: "foobar", color: "saddlebrown" },
|
|
];
|
|
const $tmpl = (name, color) => createElement(`
|
|
<a data-link href="/tags/${name}/?canary=true" draggable="false">
|
|
<svg class="component_icon" viewBox="0 0 100 100" xmlns="http://www.w3.org/2000/svg">
|
|
<circle cx="50" cy="50" r="50" style="opacity: 0.25; fill: ${color};" />
|
|
</svg>
|
|
<div class="ellipsis">${name}</div>
|
|
</a>
|
|
`);
|
|
const $fragment = document.createDocumentFragment();
|
|
tags.forEach(({ name, color }) => {
|
|
$fragment.appendChild($tmpl(name, color));
|
|
});
|
|
qs($page, `[data-bind="taglist"]`).appendChild($fragment);
|
|
}
|
|
|
|
export function init() {
|
|
return loadCSS(import.meta.url, "./sidebar.css");
|
|
}
|
|
|
|
function checkVisible($el) {
|
|
const rect = $el.getBoundingClientRect();
|
|
return rect.top >= 0 &&
|
|
rect.left >= 0 &&
|
|
rect.bottom <= (window.innerHeight || document.documentElement.clientHeight) &&
|
|
rect.right <= (window.innerWidth || document.documentElement.clientWidth);
|
|
}
|
|
|
|
function shouldDisplay() {
|
|
if (new URL(location.toString()).searchParams.get("nav") === "false") return false;
|
|
else if (window.self !== window.top) return false;
|
|
else if (document.body.clientWidth < 850) return false;
|
|
return true;
|
|
}
|
|
|
|
class PathChunk {
|
|
constructor() {
|
|
this.pathname = [""].concat(fromHref(
|
|
location.pathname.replace(new RegExp("[^/]*$"), "")
|
|
).split("/").slice(2));
|
|
}
|
|
|
|
toArray() {
|
|
return this.pathname;
|
|
}
|
|
|
|
toString(i) {
|
|
if (i >= 0) return decodeURIComponent(this.pathname.slice(0, i+1).join("/") + "/");
|
|
return decodeURIComponent(this.pathname.join("/"));
|
|
}
|
|
}
|