|
|
|
|
@ -1,7 +1,7 @@
|
|
|
|
|
import { createElement, createRender, onDestroy } from "../lib/skeleton/index.js";
|
|
|
|
|
import rxjs, { effect, onClick } from "../lib/rx.js";
|
|
|
|
|
import ajax from "../lib/ajax.js";
|
|
|
|
|
import { fromHref, toHref } from "../lib/skeleton/router.js";
|
|
|
|
|
import { toHref } from "../lib/skeleton/router.js";
|
|
|
|
|
import { qs, qsa, safe } from "../lib/dom.js";
|
|
|
|
|
import { forwardURLParams } from "../lib/path.js";
|
|
|
|
|
import { settingsGet, settingsSave } from "../lib/store.js";
|
|
|
|
|
@ -14,16 +14,11 @@ import { extractPath, isDir, isNativeFileUpload } from "../pages/filespage/helpe
|
|
|
|
|
import { mv as mvVL, withVirtualLayer } from "../pages/filespage/model_virtual_layer.js";
|
|
|
|
|
import { getCurrentPath } from "../pages/viewerpage/common.js";
|
|
|
|
|
import { generateSkeleton } from "./skeleton.js";
|
|
|
|
|
import { onLogout } from "../pages/ctrl_logout.js";
|
|
|
|
|
|
|
|
|
|
const state = { scrollTop: 0, $cache: null };
|
|
|
|
|
const mv = (from, to) => withVirtualLayer(
|
|
|
|
|
mv$(from, to),
|
|
|
|
|
mvVL(from, to),
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
export default async function ctrlSidebar(render, { nRestart = 0 }) {
|
|
|
|
|
if (!shouldDisplay()) return;
|
|
|
|
|
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>
|
|
|
|
|
@ -31,15 +26,38 @@ export default async function ctrlSidebar(render, { nRestart = 0 }) {
|
|
|
|
|
<img src="" alt="close">
|
|
|
|
|
<input type="text" placeholder="${t("Your Files")}" />
|
|
|
|
|
</h3>
|
|
|
|
|
<div data-bind="your-files"></div>
|
|
|
|
|
<div data-bind="your-tags"></div>
|
|
|
|
|
<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 forceRefresh = () => window.dispatchEvent(new Event("resize"));
|
|
|
|
|
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(() => {
|
|
|
|
|
@ -51,36 +69,30 @@ export default async function ctrlSidebar(render, { nRestart = 0 }) {
|
|
|
|
|
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 {
|
|
|
|
|
).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");
|
|
|
|
|
$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
|
|
|
|
|
effect(rxjs.merge(
|
|
|
|
|
rxjs.of(null),
|
|
|
|
|
rxjs.fromEvent(window, "filestash::tag"),
|
|
|
|
|
).pipe(rxjs.tap(() => {
|
|
|
|
|
ctrlTagPane(createRender(qs($sidebar, `[data-bind="your-tags"]`)));
|
|
|
|
|
})));
|
|
|
|
|
forceRefresh();
|
|
|
|
|
}),
|
|
|
|
|
));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const withResize = (function() {
|
|
|
|
|
@ -107,62 +119,66 @@ const withResize = (function() {
|
|
|
|
|
};
|
|
|
|
|
}());
|
|
|
|
|
|
|
|
|
|
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 dirpath = 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], dirpath);
|
|
|
|
|
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;
|
|
|
|
|
const withInstantLoad = (function() {
|
|
|
|
|
const state = { scrollTop: 0, $cache: null };
|
|
|
|
|
return ($sidebar) => {
|
|
|
|
|
if (state.$cache) {
|
|
|
|
|
$sidebar.replaceChildren(state.$cache);
|
|
|
|
|
$sidebar.firstElementChild.scrollTop = state.scrollTop;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
$files.replaceChildren($tree);
|
|
|
|
|
$sidebar.firstElementChild.scrollTop = state.scrollTop;
|
|
|
|
|
onDestroy(() => {
|
|
|
|
|
state.$cache = $sidebar.firstElementChild?.cloneNode(true);
|
|
|
|
|
state.scrollTop = $sidebar.firstElementChild.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()));
|
|
|
|
|
async function ctrlNavigationPane(render, { $sidebar, path }) {
|
|
|
|
|
// feature: init dom
|
|
|
|
|
const $fs = document.createDocumentFragment();
|
|
|
|
|
const dirname = path.replace(new RegExp("[^\/]*$"), "");
|
|
|
|
|
const chunks = dirname.split("/");
|
|
|
|
|
for (let i=1; i<chunks.length; i++) {
|
|
|
|
|
const cpath = chunks.slice(0, i).join("/") + "/";
|
|
|
|
|
const $ul = await _createListOfFiles(cpath, {
|
|
|
|
|
basename: chunks[i],
|
|
|
|
|
dirname,
|
|
|
|
|
});
|
|
|
|
|
if (cpath === "/") $fs.appendChild($ul);
|
|
|
|
|
else qs($fs, `[data-path="${cpath}"] ul`).appendChild($ul);
|
|
|
|
|
}
|
|
|
|
|
render($fs);
|
|
|
|
|
|
|
|
|
|
// feature: listen for updates
|
|
|
|
|
effect(new rxjs.Observable((subscriber) => {
|
|
|
|
|
const cleaners = [
|
|
|
|
|
hooks.ls.listen(({ path }) => subscriber.next(path)),
|
|
|
|
|
hooks.mutation.listen(async({ op, path }) => {
|
|
|
|
|
if (["mv", "mkdir", "rm"].indexOf(op) === -1) return;
|
|
|
|
|
subscriber.next(path);
|
|
|
|
|
}),
|
|
|
|
|
];
|
|
|
|
|
return () => cleaners.map((fn) => fn());
|
|
|
|
|
}).pipe(
|
|
|
|
|
rxjs.tap(async(path) => {
|
|
|
|
|
const display = path === "/" ? render : createRender(qs($sidebar, `[data-path="${path}"] ul`));
|
|
|
|
|
display(await _createListOfFiles(path, {}));
|
|
|
|
|
}),
|
|
|
|
|
));
|
|
|
|
|
|
|
|
|
|
// feature: highlight current selection
|
|
|
|
|
try {
|
|
|
|
|
const $active = qs($sidebar, `[data-path="${chunk.toString()}"] a`);
|
|
|
|
|
const $active = qs($sidebar, `[data-path="${dirname}"] a`);
|
|
|
|
|
const 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);
|
|
|
|
|
};
|
|
|
|
|
$active.setAttribute("aria-selected", "true");
|
|
|
|
|
if (checkVisible($active) === false) {
|
|
|
|
|
const tags = new URLSearchParams(location.search).getAll("tag").length;
|
|
|
|
|
if (checkVisible($active) === false && tags === 0) {
|
|
|
|
|
$active.offsetTop < window.innerHeight
|
|
|
|
|
? $sidebar.firstChild.scrollTo({ top: 0, behavior: "smooth" })
|
|
|
|
|
: $active.scrollIntoView({ behavior: "smooth", block: "nearest" });
|
|
|
|
|
@ -186,13 +202,96 @@ async function ctrlNavigationPane(render, { $sidebar, nRestart }) {
|
|
|
|
|
: $li.classList.remove("hidden");
|
|
|
|
|
});
|
|
|
|
|
}),
|
|
|
|
|
rxjs.finalize(() => $sidebar.classList.remove("search")),
|
|
|
|
|
));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function _createListOfFiles(path, currentName, dirpath) {
|
|
|
|
|
async function ctrlTagPane(render, { tags, path }) {
|
|
|
|
|
if (!getConfig("enable_tags", false)) return;
|
|
|
|
|
const $page = createElement(`
|
|
|
|
|
<div>
|
|
|
|
|
<h3 class="no-select">
|
|
|
|
|
<img src="" alt="tag">
|
|
|
|
|
${t("Tags")}
|
|
|
|
|
</h3>
|
|
|
|
|
<ul>
|
|
|
|
|
<li data-bind="taglist"></li>
|
|
|
|
|
</ul>
|
|
|
|
|
</div>
|
|
|
|
|
`);
|
|
|
|
|
const renderTaglist = createRender(qs($page, `[data-bind="taglist"]`));
|
|
|
|
|
effect(rxjs.merge(
|
|
|
|
|
tags.length === 0 ? rxjs.EMPTY : rxjs.of(tags),
|
|
|
|
|
ajax({
|
|
|
|
|
url: forwardURLParams(`api/metadata/search`, ["share"]),
|
|
|
|
|
method: "POST",
|
|
|
|
|
responseType: "json",
|
|
|
|
|
body: {
|
|
|
|
|
tags: [],
|
|
|
|
|
path,
|
|
|
|
|
},
|
|
|
|
|
}).pipe(
|
|
|
|
|
rxjs.map(({ responseJSON }) => {
|
|
|
|
|
const tags = {};
|
|
|
|
|
Object.values(responseJSON.results).forEach((forms) => {
|
|
|
|
|
forms.forEach(({ id, value = "" }) => {
|
|
|
|
|
if (id !== "tags") return;
|
|
|
|
|
value.split(",").forEach((tag) => {
|
|
|
|
|
tags[tag.trim()] = null;
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
return Object.keys(tags).sort();
|
|
|
|
|
}),
|
|
|
|
|
rxjs.catchError(() => rxjs.of([])),
|
|
|
|
|
),
|
|
|
|
|
).pipe(
|
|
|
|
|
rxjs.distinct((tags) => tags.join(", ")),
|
|
|
|
|
rxjs.tap((tags) => {
|
|
|
|
|
render($page);
|
|
|
|
|
if (tags.length === 0) {
|
|
|
|
|
$page.classList.add("hidden");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
$page.classList.remove("hidden");
|
|
|
|
|
const $fragment = document.createDocumentFragment();
|
|
|
|
|
tags.forEach((name) => {
|
|
|
|
|
const $tag = createElement(`
|
|
|
|
|
<a data-link draggable="false" class="no-select">
|
|
|
|
|
<div class="ellipsis">
|
|
|
|
|
<span class="hash"></span>
|
|
|
|
|
${name}
|
|
|
|
|
</div>
|
|
|
|
|
<svg class="component_icon" draggable="false" alt="close" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
|
|
|
|
<line x1="18" y1="6" x2="6" y2="18"></line>
|
|
|
|
|
<line x1="6" y1="6" x2="18" y2="18"></line>
|
|
|
|
|
</svg>
|
|
|
|
|
</a>
|
|
|
|
|
`);
|
|
|
|
|
const url = new URL(location.href);
|
|
|
|
|
if (url.searchParams.getAll("tag").indexOf(name) === -1) {
|
|
|
|
|
$tag.setAttribute("href", forwardURLParams(toHref("/files" + path.replace(new RegExp("[^\/]+$"), "") + "?tag=" + name), ["share", "tag"]));
|
|
|
|
|
} else {
|
|
|
|
|
url.searchParams.delete("tag", name);
|
|
|
|
|
$tag.setAttribute("href", url.toString());
|
|
|
|
|
$tag.setAttribute("aria-selected", "true");
|
|
|
|
|
}
|
|
|
|
|
$fragment.appendChild($tag);
|
|
|
|
|
});
|
|
|
|
|
renderTaglist($fragment);
|
|
|
|
|
}),
|
|
|
|
|
));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const mv = (from, to) => withVirtualLayer(
|
|
|
|
|
mv$(from, to),
|
|
|
|
|
mvVL(from, to),
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
async function _createListOfFiles(path, { basename = null, dirname = null }) {
|
|
|
|
|
const r = await cache().get(path);
|
|
|
|
|
const whats = r === null
|
|
|
|
|
? (currentName ? [currentName] : [])
|
|
|
|
|
? (basename ? [basename] : [])
|
|
|
|
|
: r.files
|
|
|
|
|
.filter(({ type, name }) => type === "directory" && name[0] !== ".")
|
|
|
|
|
.map(({ name }) => name)
|
|
|
|
|
@ -210,11 +309,11 @@ async function _createListOfFiles(path, currentName, dirpath) {
|
|
|
|
|
<img class="component_icon" src="" alt="directory">
|
|
|
|
|
<div class="ellipsis">${safe(whats[i])}</div>
|
|
|
|
|
</a>
|
|
|
|
|
<ul></ul>
|
|
|
|
|
</li>
|
|
|
|
|
`);
|
|
|
|
|
|
|
|
|
|
const $link = qs($li, "a");
|
|
|
|
|
if ($li.getAttribute("data-path") === dirpath && location.pathname.startsWith(toHref("/files/"))) {
|
|
|
|
|
if ($link.getAttribute("href") === "/files" + dirname) {
|
|
|
|
|
$link.removeAttribute("href", "");
|
|
|
|
|
$link.removeAttribute("data-link");
|
|
|
|
|
} else {
|
|
|
|
|
@ -257,112 +356,6 @@ async function _createListOfFiles(path, currentName, dirpath) {
|
|
|
|
|
return $ul;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let tagcache = null;
|
|
|
|
|
onLogout(() => tagcache = null);
|
|
|
|
|
async function ctrlTagPane(render) {
|
|
|
|
|
if (!getConfig("enable_tags", false)) return;
|
|
|
|
|
render(createElement(`<div>${generateSkeleton(2)}</div>`));
|
|
|
|
|
|
|
|
|
|
const $page = createElement(`
|
|
|
|
|
<div>
|
|
|
|
|
<h3 class="no-select">
|
|
|
|
|
<img src="" alt="tag">
|
|
|
|
|
${t("Tags")}
|
|
|
|
|
</h3>
|
|
|
|
|
<ul>
|
|
|
|
|
<li data-bind="taglist"></li>
|
|
|
|
|
</ul>
|
|
|
|
|
</div>
|
|
|
|
|
`);
|
|
|
|
|
render($page);
|
|
|
|
|
|
|
|
|
|
const path = getCurrentPath("(/view/|/files/)");
|
|
|
|
|
effect(rxjs.merge(
|
|
|
|
|
rxjs.of(tagcache).pipe(rxjs.filter((cache) => cache)),
|
|
|
|
|
ajax({
|
|
|
|
|
url: forwardURLParams(`api/metadata/search`, ["share"]),
|
|
|
|
|
method: "POST",
|
|
|
|
|
responseType: "json",
|
|
|
|
|
body: JSON.stringify({
|
|
|
|
|
"tags": new URLSearchParams(location.search).getAll("tag"),
|
|
|
|
|
path,
|
|
|
|
|
}),
|
|
|
|
|
}).pipe(
|
|
|
|
|
rxjs.map(({ responseJSON }) =>
|
|
|
|
|
responseJSON.results
|
|
|
|
|
.filter(({ type }) => type === "folder")
|
|
|
|
|
.map(({ name }) => name)
|
|
|
|
|
.sort()
|
|
|
|
|
),
|
|
|
|
|
rxjs.tap((tags) => tagcache = tags),
|
|
|
|
|
rxjs.catchError(() => rxjs.of([])),
|
|
|
|
|
),
|
|
|
|
|
).pipe(rxjs.tap((tags) => {
|
|
|
|
|
if (tags.length === 0) {
|
|
|
|
|
$page.classList.add("hidden");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
$page.classList.remove("hidden");
|
|
|
|
|
const $fragment = document.createDocumentFragment();
|
|
|
|
|
tags.forEach((name) => {
|
|
|
|
|
const $tag = createElement(`
|
|
|
|
|
<a data-link draggable="false" class="no-select">
|
|
|
|
|
<div class="ellipsis">
|
|
|
|
|
<span class="hash">#</span>
|
|
|
|
|
${name}
|
|
|
|
|
</div>
|
|
|
|
|
<svg class="component_icon" draggable="false" alt="close" xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
|
|
|
|
<line x1="18" y1="6" x2="6" y2="18"></line>
|
|
|
|
|
<line x1="6" y1="6" x2="18" y2="18"></line>
|
|
|
|
|
</svg>
|
|
|
|
|
</a>
|
|
|
|
|
`);
|
|
|
|
|
const url = new URL(location.href);
|
|
|
|
|
if (url.searchParams.getAll("tag").indexOf(name) === -1) {
|
|
|
|
|
$tag.setAttribute("href", forwardURLParams(toHref("/files" + path.replace(new RegExp("[^\/]+$"), "") + "?tag=" + name), ["share", "tag"]));
|
|
|
|
|
} else {
|
|
|
|
|
url.searchParams.delete("tag", name);
|
|
|
|
|
$tag.setAttribute("href", url.toString());
|
|
|
|
|
$tag.setAttribute("aria-selected", "true");
|
|
|
|
|
}
|
|
|
|
|
$fragment.appendChild($tag);
|
|
|
|
|
});
|
|
|
|
|
qs($page, `[data-bind="taglist"]`).replaceChildren($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("/"));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|