import { createElement, createRender } from "../../lib/skeleton/index.js"; import { toHref } from "../../lib/skeleton/router.js"; import rxjs, { effect, onClick } from "../../lib/rx.js"; import assert from "../../lib/assert.js"; import ajax from "../../lib/ajax.js"; import { forwardURLParams, join } from "../../lib/path.js"; import { qs, qsa } from "../../lib/dom.js"; import { randomString } from "../../lib/random.js"; import { animate } from "../../lib/animate.js"; import { createForm, mutateForm } from "../../lib/form.js"; import { formTmpl } from "../../components/form.js"; import notification from "../../components/notification.js"; import t from "../../locales/index.js"; import { currentPath, isDir } from "./helper.js"; const IMAGE = { COPY: "", LOADING: "", DELETE: "", EDIT: "", }; export default function(render, { path }) { const $modal = createElement(`

${t("Create a New Link")}

`); render($modal); const ret = new rxjs.Subject(); const role$ = new rxjs.BehaviorSubject(null); const state = { /** @type {object} */ form: {}, /** @type {any[] | null} */ links: null, }; // feature: select const toggle = (val) => rxjs.mergeMap(() => { state.form = {}; role$.next(role$.value === val ? null : val); return rxjs.EMPTY; }); effect(rxjs.merge( onClick(qs($modal, `[data-role="viewer"]`)).pipe(toggle("viewer")), onClick(qs($modal, `[data-role="editor"]`)).pipe(toggle("editor")), onClick(qs($modal, `[data-role="uploader"]`)).pipe(toggle("uploader")), role$.asObservable(), ).pipe(rxjs.tap(() => { const ctrl = role$.value === null ? ctrlListShares : ctrlCreateShare; // feature: set active button for (const $button of qs($modal, ".share--content").children) { $button.getAttribute("data-role") === role$.value ? $button.classList.add("active") : $button.classList.remove("active"); } // feature: render body and associated events ctrl(createRender(qs($modal, `[data-bind="share-body"]`)), { formState: { path, ...state.form, }, formLinks: state.links, load: (data) => { const role = shareObjToRole(data); state.form = { ...data, url_enable: !!data.url, password_enable: !!data.password, expire_enable: !!data.expire, users_enable: !!data.users, }; role$.next(role); }, save: async({ id, ...data }) => { const body = { id, path, ...data, ...roleToShareObj(role$.value) }; await ajax({ method: "POST", body, url: `api/share/${id}`, }).toPromise(); assert.truthy(state.links).push({ ...body, path: body.path.substring(currentPath().length - 1), }); role$.next(null); }, remove: async({ id }) => { await ajax({ method: "DELETE", url: `api/share/${id}`, }).toPromise(); state.links = (state.links || []).filter((link) => link && link.id !== id); role$.next(null); }, all: async() => { const { responseJSON } = await ajax({ url: `api/share?path=` + encodeURIComponent(path), method: "GET", responseType: "json", }).toPromise(); const currentFolder = path.replace(new RegExp("/$"), "").split("/").pop(); const sharedLinkIsFolder = new RegExp("/$").test(path); state.links = responseJSON.results.map((obj) => { obj.path = sharedLinkIsFolder ? `./${currentFolder}${obj.path}` : `./${currentFolder}`; return obj; }).sort((a, b) => { if (a.path === b.path) return a.id > b.id ? 1 : -1; return a.path > b.path ? 1 : -1; }); return state.links; }, }); }))); return ret.toPromise(); } async function ctrlListShares(render, { load, remove, all, formLinks }) { const $page = createElement(` `); render($page); effect(rxjs.merge( rxjs.of(formLinks).pipe(rxjs.filter((val) => val !== null)), rxjs.from(all()), ).pipe(rxjs.tap((links) => { if (links.length === 0) { $page.classList.add("hidden"); return; } $page.classList.remove("hidden"); const $fragment = document.createDocumentFragment(); const $content = qs($page, ".share--content"); let length = links.length; links.forEach((shareObj) => { const $share = createElement(` `); qsa($share, ".copy").forEach(($el) => $el.onclick = () => { const link = location.origin + forwardURLParams(toHref(`/s/${shareObj.id}`), ["share"]); copyToClipboard(link); notification.info(t("The link was copied in the clipboard")); }); qs($share, `[alt="delete"]`).onclick = async() => { $share.remove(); length -= 1; if (length === 0) $content.replaceChildren(createElement(` `)); await remove(shareObj); }; qs($share, `[alt="edit"]`).onclick = () => load(shareObj); $fragment.appendChild($share); }); $content.replaceChildren($fragment); }))); } async function ctrlCreateShare(render, { save, formState }) { if (formState.path) formState.path = join( location.origin + currentPath(), formState.path, ); let id = formState.id || randomString(7); const $page = createElement(`

${t("Advanced")}arrow_bottom

`); render($page); const $body = qs($page, ".restrictions"); // feature1: setup the shared link form const formSpec = { users_enable: { type: "enable", label: t("Only for users"), target: ["users"], default: false, }, users: { id: "users", type: "text", placeholder: "name0@email.com,name1@email.com", }, password_enable: { label: t("Password"), type: "enable", target: ["password"], default: false, }, password: { id: "password", type: "text", placeholder: t("Password"), }, expire_enable: { label: t("Expiration"), type: "enable", target: ["expire"], default: false, }, expire: { id: "expire", type: "date", }, url_enable: { label: "link", type: "enable", target: ["link"], default: false, }, url: { id: "link", type: "text", }, path: { type: "hidden", }, }; const tmpl = formTmpl({ renderNode: () => createElement("
"), renderLeaf: ({ label, type }) => { if (type !== "enable") return createElement(""); const title = label === "users_enable" ? t("Only for users") : label === "expire_enable" ? t("Expiration") : label === "password_enable" ? t("Password") : label === "url_enable" ? t("Custom Link url") : assert.fail("unknown label"); return createElement(`
`); }, }); const $form = await createForm(mutateForm(formSpec, formState), tmpl); $body.replaceChildren($form); const clientHeight = $body.offsetHeight; $body.classList.add("hidden"); qs($page, "h2").onclick = async() => { // toggle advanced button if ($body.classList.contains("hidden")) { $body.classList.remove("hidden"); await animate($body, { time: 200, keyframes: [{ height: "0" }, { height: `${clientHeight}px` }], }); return; } await animate($body, { time: 100, keyframes: [{ height: `${clientHeight}px` }, { height: "0" }], }); $body.classList.add("hidden"); }; // sync editable custom link input with link id effect(rxjs.fromEvent(qs($form, `[name="url"]`), "keyup").pipe(rxjs.tap((e) => { id = e.target.value.replaceAll(" ", "-").replace(new RegExp("[^A-Za-z\-]"), ""); qs(assert.type($form.closest(".component_share"), HTMLElement), `input[name="create"]`).value = `${location.origin}${toHref("/s/" + id)}`; }))); // feature: create a shared link const $copy = qs($page, `[alt="copy"]`); effect(onClick(qs($page, ".shared-link")).pipe( rxjs.first(), rxjs.switchMap(async() => { const form = new FormData(assert.type(qs(document.body, ".component_share form"), HTMLFormElement)); const body = [...form].reduce((acc, [key, value]) => { if (form.has(`${key}_enable`)) acc[key] = value; return acc; }, { id, path: form.get("path") }); $copy.setAttribute("src", IMAGE.LOADING); const link = location.origin + forwardURLParams(toHref(`/s/${id}`), ["share"]); await save(body); copyToClipboard(link); notification.info(t("The link was copied in the clipboard")); }), rxjs.catchError((err) => { $copy.setAttribute("src", IMAGE.COPY); notification.error(t(err.message)); throw err; }), rxjs.retry(), )); } function roleToShareObj(role) { return { can_read: (function(r) { if (r === "viewer") return true; else if (r === "editor") return true; return false; }(role)), can_write: (function(r) { if (r === "editor") return true; return false; }(role)), can_upload: (function(r) { if (r === "uploader") return true; else if (r === "editor") return true; return false; }(role)), }; } function shareObjToRole({ can_read, can_write, can_upload }) { if (can_read === true && can_write === false && can_upload === false) { return "viewer"; } else if (can_read === false && can_write === false && can_upload === true) { return "uploader"; } else if (can_read === true && can_write === true && can_upload === true) { return "editor"; } return undefined; } export function copyToClipboard(str) { if (!str) return; const $input = document.createElement("input"); $input.setAttribute("type", "text"); $input.setAttribute("style", "position: absolute; top:0;left:0;background:red"); $input.setAttribute("display", "none"); document.body.appendChild($input); $input.value = str; $input.select(); document.execCommand("copy"); $input.remove(); }