mirror of
https://github.com/mickael-kerjean/filestash
synced 2025-12-08 01:12:49 +01:00
369 lines
20 KiB
JavaScript
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();
|
|
}
|