Files
filestash/public/assets/pages/adminpage/ctrl_backend_component_storage.js
2023-11-27 20:58:54 +11:00

127 lines
5.7 KiB
JavaScript

import { createElement } from "../../lib/skeleton/index.js";
import rxjs, { effect, applyMutations, onClick } from "../../lib/rx.js";
import { createForm } from "../../lib/form.js";
import { qs, qsa } from "../../lib/dom.js";
import { formTmpl } from "../../components/form.js";
import { generateSkeleton } from "../../components/skeleton.js";
import { initStorage, getState, getBackendAvailable, getBackendEnabled, addBackendEnabled, removeBackendEnabled } from "./ctrl_backend_state.js";
import { save as saveConfig } from "./model_config.js";
import "./component_box-item.js";
export default async function(render) {
const $page = createElement(`
<div class="component_storagebackend">
<h2>Storage Backend</h2>
<div class="box-container" data-bind="backend-available">
${generateSkeleton(10)}
</div>
<form data-bind="backend-enabled"></form>
</div>
`);
render($page);
await initStorage();
// feature: setup the buttons
const init$ = getBackendAvailable().pipe(
rxjs.tap(() => qs($page, "[data-bind=\"backend-available\"]").innerHTML = ""),
rxjs.mergeMap((specs) => Promise.all(Object.keys(specs).map((label) => createElement(`
<div is="box-item" data-label="${label}"></div>
`)))),
applyMutations(qs($page, "[data-bind=\"backend-available\"]"), "appendChild"),
rxjs.share(),
);
effect(init$);
// feature: state of buttons
effect(init$.pipe(
rxjs.mergeMap(() => getBackendEnabled()),
rxjs.map((enabled) => {
const enabledSet = new Set();
enabled.forEach(({ type }) => {
enabledSet.add(type);
});
return enabledSet;
}),
rxjs.tap((backends) => qsa($page, "[is=\"box-item\"]").forEach(($button) => {
backends.has($button.getAttribute("data-label"))
? $button.classList.add("active")
: $button.classList.remove("active");
})),
));
// feature: click to select a backend
effect(init$.pipe(
rxjs.mergeMap(($nodes) => $nodes),
rxjs.mergeMap(($node) => onClick($node)),
rxjs.map(($node) => addBackendEnabled($node.getAttribute("data-label"))),
saveConnections(),
));
// feature: setup form
const setupForm$ = getBackendEnabled().pipe(
// initialise the forms
rxjs.mergeMap((enabled) => Promise.all(enabled.map(({ type, label }) => createForm({
[type]: {
"": { type: "text", placeholder: "Label", value: label },
}
}, formTmpl({
renderLeaf: () => createElement("<label></label>"),
renderNode: ({ label, format }) => {
const $fieldset = createElement(`
<fieldset>
<legend class="no-select">
${format(label)}
</legend>
<div data-bind="children"></div>
</fieldset>
`);
const $remove = createElement(`
<div class="icons no-select">
<img class="component_icon" draggable="false" src="" alt="close">
</div>
`);
$fieldset.appendChild($remove);
return $fieldset;
},
}))))),
rxjs.map((nodeList) => {
if (nodeList.length === 0) { return [createElement(`
<div class="alert">
You need to select at least 1 storage backend
</div>
`)]; }
return nodeList;
}),
rxjs.tap(() => qs($page, "[data-bind=\"backend-enabled\"]").innerHTML = ""),
applyMutations(qs($page, "[data-bind=\"backend-enabled\"]"), "appendChild"),
rxjs.share(),
);
effect(setupForm$);
// feature: remove an existing backend
effect(setupForm$.pipe(
rxjs.mergeMap(($nodes) => $nodes),
rxjs.mergeMap(($node) => onClick($node.querySelector(".icons"))),
rxjs.map(($node) => qs($node.parentElement, "input").value),
rxjs.map((label) => removeBackendEnabled(label)),
saveConnections(),
));
// feature: form input change handler
effect(setupForm$.pipe(
rxjs.mergeMap((forms) => forms),
rxjs.mergeMap(($el) => rxjs.fromEvent($el, "input")),
saveConnections(),
));
}
const saveConnections = () => rxjs.pipe(
rxjs.mergeMap((connections) => getState().pipe(rxjs.map((config) => {
if (Array.isArray(connections)) config.connections = connections;
return config;
}))),
saveConfig(),
);