nopaque/app/static/js/app.js

206 lines
6.1 KiB
JavaScript
Raw Normal View History

2023-10-30 11:36:28 +01:00
nopaque.App = class App {
#promises;
2021-11-30 16:22:16 +01:00
constructor() {
this.data = {
users: {}
};
this.#promises = {
getUser: {},
subscribeUser: {}
};
this.sockets = {
users: io('/users', {transports: ['websocket'], upgrade: false})
};
this.sockets.users.on('patch_user', (patch) => {this.onPatch(patch);});
2021-11-30 16:22:16 +01:00
}
2023-07-24 10:02:35 +02:00
getUser(userId) {
if (userId in this.#promises.getUser) {
return this.#promises.getUser[userId];
}
let socket = this.sockets.users;
this.#promises.getUser[userId] = new Promise((resolve, reject) => {
socket.emit('get_user', userId, (response) => {
2023-07-24 10:02:35 +02:00
if (response.status === 200) {
this.data.users[userId] = response.body;
resolve(this.data.users[userId]);
} else {
reject(`[${response.status}] ${response.statusText}`);
}
});
});
return this.#promises.getUser[userId];
2021-12-01 14:15:20 +01:00
}
2021-11-30 16:22:16 +01:00
2022-07-04 14:09:17 +02:00
subscribeUser(userId) {
if (userId in this.#promises.subscribeUser) {
return this.#promises.subscribeUser[userId];
2021-12-01 14:15:20 +01:00
}
let socket = this.sockets.users;
this.#promises.subscribeUser[userId] = new Promise((resolve, reject) => {
socket.emit('subscribe_user', userId, (response) => {
if (response.status === 200) {
resolve(response);
} else {
2022-07-04 14:09:17 +02:00
reject(response);
}
});
});
return this.#promises.subscribeUser[userId];
2021-11-30 16:22:16 +01:00
}
flash(message, category) {
2022-09-02 13:07:30 +02:00
let iconPrefix = '';
2021-11-30 16:22:16 +01:00
switch (category) {
2022-09-02 13:07:30 +02:00
case 'corpus': {
2021-12-01 14:15:20 +01:00
iconPrefix = '<i class="left material-icons">book</i>';
2021-11-30 16:22:16 +01:00
break;
2022-09-02 13:07:30 +02:00
}
case 'error': {
2021-12-01 14:15:20 +01:00
iconPrefix = '<i class="error-color-text left material-icons">error</i>';
2021-11-30 16:22:16 +01:00
break;
2022-09-02 13:07:30 +02:00
}
case 'job': {
iconPrefix = '<i class="left nopaque-icons">J</i>';
2021-11-30 16:22:16 +01:00
break;
2022-09-02 13:07:30 +02:00
}
2023-04-11 15:03:12 +02:00
case 'settings': {
iconPrefix = '<i class="left material-icons">settings</i>';
break;
}
2022-09-02 13:07:30 +02:00
default: {
2021-12-01 14:15:20 +01:00
iconPrefix = '<i class="left material-icons">notifications</i>';
break;
2022-09-02 13:07:30 +02:00
}
2021-11-30 16:22:16 +01:00
}
2022-09-02 13:07:30 +02:00
let toast = M.toast(
2021-12-01 14:15:20 +01:00
{
html: `
<span>${iconPrefix}${message}</span>
2022-09-02 13:07:30 +02:00
<button class="action-button btn-flat toast-action white-text" data-action="close">
2021-12-01 14:15:20 +01:00
<i class="material-icons">close</i>
</button>
`.trim()
}
);
2022-09-02 13:07:30 +02:00
let toastCloseActionElement = toast.el.querySelector('.action-button[data-action="close"]');
2021-11-30 16:22:16 +01:00
toastCloseActionElement.addEventListener('click', () => {toast.dismiss();});
}
2022-09-02 13:07:30 +02:00
onPatch(patch) {
// Filter Patch to only include operations on users that are initialized
let regExp = new RegExp(`^/users/(${Object.keys(this.data.users).join('|')})`);
let filteredPatch = patch.filter(operation => regExp.test(operation.path));
// Handle job status updates
let subRegExp = new RegExp(`^/users/([A-Za-z0-9]*)/jobs/([A-Za-z0-9]*)/status$`);
let subFilteredPatch = filteredPatch
.filter((operation) => {return operation.op === 'replace';})
.filter((operation) => {return subRegExp.test(operation.path);});
for (let operation of subFilteredPatch) {
let [match, userId, jobId] = operation.path.match(subRegExp);
2024-05-27 09:24:40 +02:00
this.flash(`[<a href="/jobs/${jobId}">${this.data.users[userId].jobs[jobId].title}</a>] New status: <span class="job-status-text" data-job-status="${operation.value}"></span>`, 'job');
2022-09-02 13:07:30 +02:00
}
// Apply Patch
jsonpatch.applyPatch(this.data, filteredPatch);
}
init() {
this.initUi();
}
initUi() {
/* Pre-Initialization fixes */
// #region
// Flask-WTF sets the standard HTML maxlength Attribute on input/textarea
// elements to specify their maximum length (in characters). Unfortunatly
// Materialize won't recognize the maxlength Attribute, instead it uses
// the data-length Attribute. It's conversion time :)
for (let elem of document.querySelectorAll('input[maxlength], textarea[maxlength]')) {
elem.dataset.length = elem.getAttribute('maxlength');
elem.removeAttribute('maxlength');
}
// To work around some limitations with the Form setup of Flask-WTF.
// HTML option elements with an empty value are considered as placeholder
// elements. The user should not be able to actively select these options.
// So they get the disabled attribute.
for (let optionElement of document.querySelectorAll('option[value=""]')) {
optionElement.disabled = true;
}
// TODO: Check why we are doing this.
for (let optgroupElement of document.querySelectorAll('optgroup[label=""]')) {
for (let c of optgroupElement.children) {
optgroupElement.parentElement.insertAdjacentElement('afterbegin', c);
}
optgroupElement.remove();
}
// #endregion
2023-10-30 11:36:28 +01:00
/* Initialize Materialize Components */
// #region
// Automatically initialize Materialize Components that do not require
// additional configuration.
M.AutoInit();
// CharacterCounters
// Materialize didn't include the CharacterCounter plugin within the
// AutoInit method (maybe they forgot it?). Anyway... We do it here. :)
M.CharacterCounter.init(document.querySelectorAll('input[data-length]:not(.no-autoinit), textarea[data-length]:not(.no-autoinit)'));
2024-11-15 15:21:26 +01:00
// Header navigation processes and services Dropdown.
M.Dropdown.init(
document.querySelector('#nav-processes-and-services-dropdown-trigger'),
{
constrainWidth: false,
coverTrigger: false
}
);
2024-05-03 15:08:57 +02:00
// Header navigation account Dropdown.
M.Dropdown.init(
document.querySelector('#nav-account-dropdown-trigger'),
{
alignment: 'right',
constrainWidth: false,
coverTrigger: false
}
);
// Terms of use modal
M.Modal.init(
document.querySelector('#terms-of-use-modal'),
{
dismissible: false,
onCloseEnd: (modalElement) => {
nopaque.requests.users.entity.acceptTermsOfUse();
}
}
);
// #endregion
2023-10-30 11:36:28 +01:00
/* Initialize nopaque Components */
// #region
nopaque.resource_displays.AutoInit();
nopaque.resource_lists.AutoInit();
nopaque.forms.AutoInit();
2023-10-30 11:36:28 +01:00
// #endregion
}
2023-10-12 14:13:47 +02:00
};