mirror of
https://github.com/mickael-kerjean/filestash.git
synced 2025-11-01 10:56:31 +08:00
138 lines
8.2 KiB
JavaScript
138 lines
8.2 KiB
JavaScript
import { onDestroy, createElement, createFragment, nop } from "../../lib/skeleton/index.js";
|
|
import rxjs, { effect, applyMutation, onClick, preventDefault } from "../../lib/rx.js";
|
|
import { animate } from "../../lib/animate.js";
|
|
import { loadCSS } from "../../helpers/loader.js";
|
|
import { qs } from "../../lib/dom.js";
|
|
import { getSelection$, clearSelection } from "./model_files.js";
|
|
import { setAction } from "./model_action.js";
|
|
|
|
import componentShare from "./modal_share.js";
|
|
import componentEmbed from "./modal_embed.js";
|
|
import componentTag from "./modal_tag.js";
|
|
import componentRename from "./modal_rename.js";
|
|
import componentDelete from "./modal_delete.js";
|
|
|
|
import "../../components/dropdown.js";
|
|
import "../../components/icon.js";
|
|
|
|
import { createModal } from "../../components/modal.js";
|
|
const modalOpt = {
|
|
withButtonsRight: "OK",
|
|
withButtonsLeft: "CANCEL",
|
|
};
|
|
|
|
export default async function(render) {
|
|
const $page = createElement(`
|
|
<div class="component_submenu container">
|
|
</div>
|
|
`);
|
|
render($page);
|
|
onDestroy(() => clearSelection());
|
|
|
|
// feature1: layout base case
|
|
effect(getSelection$().pipe(
|
|
rxjs.filter((selections) => selections.length === 0),
|
|
rxjs.map(() => createFragment(`
|
|
<div class="action left no-select" style="margin-left:2px;">
|
|
<button data-action="new-file">New File</button>
|
|
<button data-action="new-folder">New Folder</button>
|
|
</div>
|
|
<div class="action right no-select" style="margin-right:2px;">
|
|
<button data-action="search">
|
|
<input style="
|
|
display: none;
|
|
background: transparent;
|
|
border: none;
|
|
border-bottom: 2px solid #e2e2e2;
|
|
margin-right: 10px;
|
|
color: var(--color);
|
|
font-size: 0.8rem;
|
|
">
|
|
<img class="component_icon" draggable="false" src="data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCA1MTIgNTEyIj4KICA8cGF0aCBzdHlsZT0iZmlsbDojNjI2NDY5O2ZpbGwtb3BhY2l0eToxIiBkPSJNNTA1IDQ0Mi43TDQwNS4zIDM0M2MtNC41LTQuNS0xMC42LTctMTctN0gzNzJjMjcuNi0zNS4zIDQ0LTc5LjcgNDQtMTI4QzQxNiA5My4xIDMyMi45IDAgMjA4IDBTMCA5My4xIDAgMjA4czkzLjEgMjA4IDIwOCAyMDhjNDguMyAwIDkyLjctMTYuNCAxMjgtNDR2MTYuM2MwIDYuNCAyLjUgMTIuNSA3IDE3bDk5LjcgOTkuN2M5LjQgOS40IDI0LjYgOS40IDMzLjkgMGwyOC4zLTI4LjNjOS40LTkuNCA5LjQtMjQuNi4xLTM0ek0yMDggMzM2Yy03MC43IDAtMTI4LTU3LjItMTI4LTEyOCAwLTcwLjcgNTcuMi0xMjggMTI4LTEyOCA3MC43IDAgMTI4IDU3LjIgMTI4IDEyOCAwIDcwLjctNTcuMiAxMjgtMTI4IDEyOHoiIC8+Cjwvc3ZnPgo=" alt="search" />
|
|
</button>
|
|
<button data-action="view">
|
|
<img class="component_icon" draggable="false" src="data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCA1MTIgNTEyIj4KICA8cGF0aCBzdHlsZT0iZmlsbDojNjI2NDY5O2ZpbGwtb3BhY2l0eToxIiBkPSJtIDEzMy4zMzMsNTYgdiA2NCBjIDAsMTMuMjU1IC0xMC43NDUsMjQgLTI0LDI0IEggMjQgQyAxMC43NDUsMTQ0IDAsMTMzLjI1NSAwLDEyMCBWIDU2IEMgMCw0Mi43NDUgMTAuNzQ1LDMyIDI0LDMyIGggODUuMzMzIGMgMTMuMjU1LDAgMjQsMTAuNzQ1IDI0LDI0IHogbSAzNzkuMzM0LDIzMiB2IC02NCBjIDAsLTEzLjI1NSAtMTAuNzQ1LC0yNCAtMjQsLTI0IEggMjEzLjMzMyBjIC0xMy4yNTUsMCAtMjQsMTAuNzQ1IC0yNCwyNCB2IDY0IGMgMCwxMy4yNTUgMTAuNzQ1LDI0IDI0LDI0IGggMjc1LjMzMyBjIDEzLjI1NiwwIDI0LjAwMSwtMTAuNzQ1IDI0LjAwMSwtMjQgeiBtIDAsLTE2OCBWIDU2IGMgMCwtMTMuMjU1IC0xMC43NDUsLTI0IC0yNCwtMjQgSCAyMTMuMzMzIGMgLTEzLjI1NSwwIC0yNCwxMC43NDUgLTI0LDI0IHYgNjQgYyAwLDEzLjI1NSAxMC43NDUsMjQgMjQsMjQgaCAyNzUuMzMzIGMgMTMuMjU2LDAgMjQuMDAxLC0xMC43NDUgMjQuMDAxLC0yNCB6IE0gMTA5LjMzMywyMDAgSCAyNCBDIDEwLjc0NSwyMDAgMCwyMTAuNzQ1IDAsMjI0IHYgNjQgYyAwLDEzLjI1NSAxMC43NDUsMjQgMjQsMjQgaCA4NS4zMzMgYyAxMy4yNTUsMCAyNCwtMTAuNzQ1IDI0LC0yNCB2IC02NCBjIDAsLTEzLjI1NSAtMTAuNzQ1LC0yNCAtMjQsLTI0IHogTSAwLDM5MiB2IDY0IGMgMCwxMy4yNTUgMTAuNzQ1LDI0IDI0LDI0IGggODUuMzMzIGMgMTMuMjU1LDAgMjQsLTEwLjc0NSAyNCwtMjQgdiAtNjQgYyAwLC0xMy4yNTUgLTEwLjc0NSwtMjQgLTI0LC0yNCBIIDI0IEMgMTAuNzQ1LDM2OCAwLDM3OC43NDUgMCwzOTIgWiBtIDE4OS4zMzMsMCB2IDY0IGMgMCwxMy4yNTUgMTAuNzQ1LDI0IDI0LDI0IGggMjc1LjMzMyBjIDEzLjI1NSwwIDI0LC0xMC43NDUgMjQsLTI0IHYgLTY0IGMgMCwtMTMuMjU1IC0xMC43NDUsLTI0IC0yNCwtMjQgSCAyMTMuMzMzIGMgLTEzLjI1NSwwIC0yNCwxMC43NDUgLTI0LDI0IHoiIC8+Cjwvc3ZnPgo=" alt="list" />
|
|
</button>
|
|
<button>
|
|
<img class="component_icon" draggable="false" src="data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCAzMjAgNTEyIj4KICA8cGF0aCBzdHlsZT0iZmlsbDojNjI2NDY5O2ZpbGwtb3BhY2l0eToxIiBkPSJNNDEgMjg4aDIzOGMyMS40IDAgMzIuMSAyNS45IDE3IDQxTDE3NyA0NDhjLTkuNCA5LjQtMjQuNiA5LjQtMzMuOSAwTDI0IDMyOWMtMTUuMS0xNS4xLTQuNC00MSAxNy00MXptMjU1LTEwNUwxNzcgNjRjLTkuNC05LjQtMjQuNi05LjQtMzMuOSAwTDI0IDE4M2MtMTUuMSAxNS4xLTQuNCA0MSAxNyA0MWgyMzhjMjEuNCAwIDMyLjEtMjUuOSAxNy00MXoiIC8+Cjwvc3ZnPgo=" alt="sort" />
|
|
</button>
|
|
<!--<div is="component-dropdown"></div>-->
|
|
</div>
|
|
`)),
|
|
applyMutation($page, "replaceChildren"),
|
|
rxjs.mergeMap(() => rxjs.merge(
|
|
rxjs.merge(
|
|
onClick(qs($page, `[data-action="new-file"]`)).pipe(rxjs.mapTo("NEW_FILE")),
|
|
onClick(qs($page, `[data-action="new-folder"]`)).pipe(rxjs.mapTo("NEW_FOLDER")),
|
|
).pipe(rxjs.mergeMap((actionName) => {
|
|
$scroll.scrollTo({top: 0, behavior: "smooth"});
|
|
setAction(actionName);
|
|
return rxjs.EMPTY;
|
|
})),
|
|
onClick(qs($page, `[data-action="search"]`)).pipe(rxjs.mapTo("SEARCH")),
|
|
onClick(qs($page, `[data-action="view"]`)).pipe(rxjs.mapTo("VIEW")),
|
|
)),
|
|
));
|
|
|
|
// feature2: update when selection is preset
|
|
effect(getSelection$().pipe(
|
|
rxjs.filter((selections) => selections.length > 0),
|
|
rxjs.tap((selections) => selections.length === 1 && animate($page)),
|
|
rxjs.map((selections) => createFragment(`
|
|
<div class="action left no-select">
|
|
<button data-action="download">Download</button>
|
|
<button data-action="share">Share</button>
|
|
<button data-action="embed">Embed</button>
|
|
<button data-action="tag">Tag</button>
|
|
<button data-action="rename">Rename</button>
|
|
<button data-action="delete">Delete</button>
|
|
</div>
|
|
<div class="action right no-select">
|
|
<button data-bind="clear">
|
|
${selections.length} <component-icon name="close"></component-icon>
|
|
</button>
|
|
</div>
|
|
`)),
|
|
applyMutation($page, "replaceChildren"),
|
|
rxjs.mergeMap(() => rxjs.merge(
|
|
onClick(qs($page, `[data-bind="clear"]`)).pipe(
|
|
rxjs.tap(() => clearSelection()),
|
|
rxjs.mergeMap(() => rxjs.EMPTY),
|
|
),
|
|
onClick(qs($page, `[data-action="download"]`)).pipe(
|
|
rxjs.mergeMap(() => rxjs.EMPTY),
|
|
),
|
|
onClick(qs($page, `[data-action="share"]`)).pipe(rxjs.tap(() => {
|
|
componentShare(createModal(modalOpt));
|
|
})),
|
|
onClick(qs($page, `[data-action="embed"]`)).pipe(rxjs.tap(() => {
|
|
componentEmbed(createModal(modalOpt));
|
|
})),
|
|
onClick(qs($page, `[data-action="tag"]`)).pipe(rxjs.tap(() => {
|
|
componentTag(createModal(modalOpt))
|
|
})),
|
|
onClick(qs($page, `[data-action="rename"]`)).pipe(rxjs.tap(() => {
|
|
componentRename(createModal(modalOpt));
|
|
})),
|
|
onClick(qs($page, `[data-action="delete"]`)).pipe(rxjs.tap(() => {
|
|
componentDelete(createModal(modalOpt));
|
|
})),
|
|
)),
|
|
));
|
|
|
|
// feature3: effect on scroll
|
|
const $scroll = $page.closest(".scroll-y");
|
|
effect(rxjs.fromEvent($scroll, "scroll", { passive: true }).pipe(
|
|
rxjs.map((e) => e.target.scrollTop > 30),
|
|
rxjs.distinctUntilChanged(),
|
|
rxjs.startWith(false),
|
|
rxjs.tap((scrolling) => scrolling
|
|
? $scroll.classList.add("scrolling")
|
|
: $scroll.classList.remove("scrolling")),
|
|
));
|
|
}
|
|
|
|
export function init() {
|
|
return loadCSS(import.meta.url, "./ctrl_submenu.css");
|
|
}
|