mirror of
https://github.com/mickael-kerjean/filestash
synced 2025-12-06 08:22:24 +01:00
133 lines
7.2 KiB
JavaScript
133 lines
7.2 KiB
JavaScript
import { createElement, createRender, onDestroy } from "../lib/skeleton/index.js";
|
|
import rxjs, { effect, onClick } from "../lib/rx.js";
|
|
import { qs } from "../lib/dom.js";
|
|
import { settingsGet, settingsSave } from "../lib/store.js";
|
|
import { loadCSS } from "../helpers/loader.js";
|
|
import t from "../locales/index.js";
|
|
import { getCurrentPath } from "../pages/viewerpage/common.js";
|
|
import { generateSkeleton } from "./skeleton.js";
|
|
|
|
import ctrlNavigationPane from "./sidebar_files.js";
|
|
import ctrlTagPane from "./sidebar_tags.js";
|
|
|
|
export default async function ctrlSidebar(render, {}) {
|
|
if (new URL(location.toString()).searchParams.get("nav") === "false") return;
|
|
else if (window.self !== window.top) return;
|
|
else if (document.body.clientWidth < 850) return;
|
|
|
|
const $sidebar = render(createElement(`
|
|
<div class="component_sidebar"><div>
|
|
<h3 class="no-select">
|
|
<img src="" alt="close">
|
|
<input type="text" placeholder="${t("Your Files")}" />
|
|
</h3>
|
|
<div data-bind="your-files">
|
|
${generateSkeleton(2)}
|
|
</div>
|
|
<div data-bind="your-tags">
|
|
${generateSkeleton(2)}
|
|
</div>
|
|
</div>
|
|
`));
|
|
withInstantLoad($sidebar);
|
|
withResize($sidebar);
|
|
|
|
const path = getCurrentPath("(/view/|/files/)");
|
|
|
|
// fature: file navigation pane
|
|
const $files = qs($sidebar, `[data-bind="your-files"]`);
|
|
ctrlNavigationPane(createRender($files), { $sidebar, path });
|
|
|
|
// feature: tag viewer
|
|
const $tags = qs($sidebar, `[data-bind="your-tags"]`);
|
|
effect(rxjs.merge(
|
|
rxjs.of(null),
|
|
rxjs.fromEvent(window, "filestash::tag"),
|
|
).pipe(
|
|
rxjs.tap(() => ctrlTagPane(createRender($tags), {
|
|
tags: [...$tags.querySelectorAll("a")].map(($tag) => $tag.innerText.trim()),
|
|
path,
|
|
})),
|
|
));
|
|
|
|
// feature: visibility of the sidebar
|
|
const isVisible = () => settingsGet({ visible: true }, "sidebar").visible;
|
|
const forceRefresh = () => window.dispatchEvent(new Event("resize"));
|
|
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();
|
|
}),
|
|
));
|
|
}
|
|
|
|
const withResize = (function() {
|
|
let memory = null;
|
|
return ($sidebar) => {
|
|
const $resize = createElement(`<div class="resizer"></div>`);
|
|
effect(rxjs.fromEvent($resize, "mousedown").pipe(
|
|
rxjs.mergeMap((e0) => rxjs.fromEvent(document, "mousemove").pipe(
|
|
rxjs.takeUntil(rxjs.fromEvent(document, "mouseup")),
|
|
rxjs.startWith(e0),
|
|
rxjs.pairwise(),
|
|
rxjs.map(([prevX, currX]) => currX.clientX - prevX.clientX),
|
|
rxjs.scan((width, delta) => width + delta, $sidebar.clientWidth),
|
|
)),
|
|
rxjs.startWith(memory),
|
|
rxjs.filter((w) => !!w),
|
|
rxjs.map((w) => Math.min(Math.max(w, 250), 350)),
|
|
rxjs.tap((w) => {
|
|
$sidebar.style.width = `${w}px`;
|
|
memory = w;
|
|
}),
|
|
));
|
|
$sidebar.appendChild($resize);
|
|
};
|
|
}());
|
|
|
|
const withInstantLoad = (function() {
|
|
const state = { scrollTop: 0, $cache: null };
|
|
return ($sidebar) => {
|
|
if (state.$cache) {
|
|
$sidebar.replaceChildren(state.$cache);
|
|
$sidebar.firstElementChild.scrollTop = state.scrollTop;
|
|
}
|
|
onDestroy(() => {
|
|
state.$cache = $sidebar.firstElementChild?.cloneNode(true);
|
|
state.scrollTop = $sidebar.firstElementChild.scrollTop;
|
|
});
|
|
};
|
|
}());
|
|
|
|
export function init() {
|
|
return loadCSS(import.meta.url, "./sidebar.css");
|
|
}
|