filestash/public/assets/pages/filespage/modal_share.js
MickaelK 8aa33143d3 fix (sharelink): default path in shared link
regression where shared link are created without a path. This was
reported by a customer
2024-11-20 15:48:48 +11:00

369 lines
20 KiB
JavaScript

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(`
<div class="component_share">
<h2>${t("Create a New Link")}</h2>
<div class="share--content link-type no-select">
<div data-role="viewer">${t("Viewer")}</div>
<div data-role="editor">${t("Editor")}</div>
<div data-role="uploader" class="${isDir(path) ? "" : "hidden"}">${t("Uploader")}</div>
</div>
<div data-bind="share-body"></div>
</div>
`);
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(`
<div class="hidden">
<h2>${t("Existing Links")}</h2>
<div class="share--content existing-links" style="max-height: 90px;">
<component-icon name="loading"></component-icon>
</div>
</div>
`);
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(`
<div class="link-details no-select">
<div class="copy role ellipsis">${t(shareObjToRole(shareObj))}</div>
<div class="copy path ellipsis" title="${shareObj.path}">${shareObj.path}</div>
<div class="link-details--icons">
<img class="component_icon" draggable="false" src="${IMAGE.DELETE}" alt="delete">
<img class="component_icon" draggable="false" src="${IMAGE.EDIT}" alt="edit">
</div>
</div>
`);
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(`
<component-icon name="loading"></component-icon>
`));
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(`
<div>
<h2 class="no-select pointer">${t("Advanced")}<span><img class="component_icon" draggable="false" src="" alt="arrow_bottom"></span></h2>
<form class="share--content restrictions no-select"></form>
<div class="shared-link">
<input name="create" class="copy" type="text" readonly="" value="${location.origin}${toHref("/s/" + id)}">
<button title="Copy URL">
<img class="component_icon" draggable="false" src="${IMAGE.COPY}" alt="copy">
</button>
</div>
</div>
`);
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("<div></div>"),
renderLeaf: ({ label, type }) => {
if (type !== "enable") return createElement("<label></label>");
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(`
<div class="component_supercheckbox">
<label class="ellipsis">
<span data-bind="children"></span>
<span class="label">${title}</span>
</label>
</div>
`);
},
});
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();
}