diff --git a/app/blueprints/users/events.py b/app/blueprints/users/events.py index 254a5ca6..4f1284a6 100644 --- a/app/blueprints/users/events.py +++ b/app/blueprints/users/events.py @@ -1,11 +1,51 @@ +from flask import current_app, Flask from flask_login import current_user from flask_socketio import join_room, leave_room -from app import hashids, socketio +from app import db, hashids, socketio from app.decorators import socketio_login_required from app.models import User -@socketio.on('users.get_user') +def _delete_user(app: Flask, user_id: int): + with app.app_context(): + user = User.query.get(user_id) + user.delete() + db.session.commit() + + +@socketio.on('users.delete') +@socketio_login_required +def delete_user(user_hashid: str) -> dict: + user_id = hashids.decode(user_hashid) + + if not isinstance(user_id, int): + return {'status': 400, 'statusText': 'Bad Request'} + + user = User.query.get(user_id) + + if user is None: + return {'status': 404, 'statusText': 'Not found'} + + if not ( + user == current_user + or current_user.is_administrator + ): + return {'status': 403, 'statusText': 'Forbidden'} + + socketio.start_background_task( + _delete_user, + current_app._get_current_object(), + user.id + ) + + return { + 'body': f'User "{user.username}" marked for deletion', + 'status': 202, + 'statusText': 'Accepted' + } + + +@socketio.on('users.get') @socketio_login_required def get_user(user_hashid: str) -> dict: user_id = hashids.decode(user_hashid) @@ -34,7 +74,7 @@ def get_user(user_hashid: str) -> dict: } -@socketio.on('users.subscribe_user') +@socketio.on('users.subscribe') @socketio_login_required def subscribe_user(user_hashid: str) -> dict: user_id = hashids.decode(user_hashid) @@ -58,9 +98,9 @@ def subscribe_user(user_hashid: str) -> dict: return {'status': 200, 'statusText': 'OK'} -@socketio.on('users.unsubscribe_user') +@socketio.on('users.unsubscribe') @socketio_login_required -def on_unsubscribe_user(user_hashid: str) -> dict: +def unsubscribe_user(user_hashid: str) -> dict: user_id = hashids.decode(user_hashid) if not isinstance(user_id, int): diff --git a/app/models/event_listeners.py b/app/models/event_listeners.py index 115f2a73..98abe9ff 100644 --- a/app/models/event_listeners.py +++ b/app/models/event_listeners.py @@ -43,7 +43,7 @@ def resource_after_delete(mapper, connection, resource): } ] room = f'/users/{resource.user_hashid}' - socketio.emit('patch_user', jsonpatch, room=room) + socketio.emit('users.patch', jsonpatch, room=room) def cfa_after_delete(mapper, connection, cfa): @@ -55,7 +55,7 @@ def cfa_after_delete(mapper, connection, cfa): } ] room = f'/users/{cfa.corpus.user.hashid}' - socketio.emit('patch_user', jsonpatch, room=room) + socketio.emit('users.patch', jsonpatch, room=room) def resource_after_insert(mapper, connection, resource): @@ -70,7 +70,7 @@ def resource_after_insert(mapper, connection, resource): } ] room = f'/users/{resource.user_hashid}' - socketio.emit('patch_user', jsonpatch, room=room) + socketio.emit('users.patch', jsonpatch, room=room) def cfa_after_insert(mapper, connection, cfa): @@ -84,7 +84,7 @@ def cfa_after_insert(mapper, connection, cfa): } ] room = f'/users/{cfa.corpus.user.hashid}' - socketio.emit('patch_user', jsonpatch, room=room) + socketio.emit('users.patch', jsonpatch, room=room) def resource_after_update(mapper, connection, resource): @@ -110,7 +110,7 @@ def resource_after_update(mapper, connection, resource): ) if jsonpatch: room = f'/users/{resource.user_hashid}' - socketio.emit('patch_user', jsonpatch, room=room) + socketio.emit('users.patch', jsonpatch, room=room) def job_after_update(mapper, connection, job): diff --git a/app/static/js/app.js b/app/static/js/app.js index c1aafc59..439bb9c1 100644 --- a/app/static/js/app.js +++ b/app/static/js/app.js @@ -1,201 +1,33 @@ nopaque.App = class App { - #promises; - constructor() { - this.data = { - users: {} - }; - - this.#promises = { - getUser: {}, - subscribeUser: {} - }; + this.data = {}; this.socket = io({transports: ['websocket'], upgrade: false}); - this.socket.on('patch_user', (patch) => {this.onPatch(patch);}); + this.ui = new nopaque.UIExtension(this); + this.users = new nopaque.UsersExtension(this); } - getUser(userId) { - if (userId in this.#promises.getUser) { - return this.#promises.getUser[userId]; - } + // 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)); - this.#promises.getUser[userId] = new Promise((resolve, reject) => { - this.socket.emit('users.get_user', userId, (response) => { - if (response.status === 200) { - this.data.users[userId] = response.body; - resolve(this.data.users[userId]); - } else { - reject(`[${response.status}] ${response.statusText}`); - } - }); - }); + // // 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); + // this.flash(`[${this.data.users[userId].jobs[jobId].title}] New status: `, 'job'); + // } - return this.#promises.getUser[userId]; - } - - subscribeUser(userId) { - if (userId in this.#promises.subscribeUser) { - return this.#promises.subscribeUser[userId]; - } - - this.#promises.subscribeUser[userId] = new Promise((resolve, reject) => { - this.socket.emit('users.subscribe_user', userId, (response) => { - if (response.status === 200) { - resolve(response); - } else { - reject(response); - } - }); - }); - - return this.#promises.subscribeUser[userId]; - } - - flash(message, category) { - let iconPrefix = ''; - switch (category) { - case 'corpus': { - iconPrefix = 'book'; - break; - } - case 'error': { - iconPrefix = 'error'; - break; - } - case 'job': { - iconPrefix = 'J'; - break; - } - case 'settings': { - iconPrefix = 'settings'; - break; - } - default: { - iconPrefix = 'notifications'; - break; - } - } - let toast = M.toast( - { - html: ` - ${iconPrefix}${message} - - `.trim() - } - ); - let toastCloseActionElement = toast.el.querySelector('.action-button[data-action="close"]'); - toastCloseActionElement.addEventListener('click', () => {toast.dismiss();}); - } - - 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); - this.flash(`[${this.data.users[userId].jobs[jobId].title}] New status: `, 'job'); - } - - // Apply Patch - jsonpatch.applyPatch(this.data, filteredPatch); - } + // // 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 - - - /* 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)')); - - // Header navigation processes and services Dropdown. - M.Dropdown.init( - document.querySelector('#navbar-data-processing-and-analysis-dropdown-trigger'), - { - constrainWidth: false, - container: document.querySelector('#dropdowns'), - coverTrigger: false - } - ); - - // Header navigation account Dropdown. - M.Dropdown.init( - document.querySelector('#navbar-account-dropdown-trigger'), - { - alignment: 'right', - constrainWidth: false, - container: document.querySelector('#dropdowns'), - 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 - - - /* Initialize nopaque Components */ - // #region - nopaque.resource_displays.AutoInit(); - nopaque.resource_lists.AutoInit(); - nopaque.forms.AutoInit(); - // #endregion + this.ui.init(); } }; diff --git a/app/static/js/app.ui.js b/app/static/js/app.ui.js new file mode 100644 index 00000000..150cccec --- /dev/null +++ b/app/static/js/app.ui.js @@ -0,0 +1,126 @@ +nopaque.UIExtension = class UIExtension { + constructor(app) { + this.app = app; + } + + init() { + /* 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 + + + /* 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)')); + + // Header navigation processes and services Dropdown. + M.Dropdown.init( + document.querySelector('#navbar-data-processing-and-analysis-dropdown-trigger'), + { + constrainWidth: false, + container: document.querySelector('#dropdowns'), + coverTrigger: false + } + ); + + // Header navigation account Dropdown. + M.Dropdown.init( + document.querySelector('#navbar-account-dropdown-trigger'), + { + alignment: 'right', + constrainWidth: false, + container: document.querySelector('#dropdowns'), + 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 + + + /* Initialize nopaque Components */ + // #region + nopaque.resource_displays.AutoInit(); + nopaque.resource_lists.AutoInit(); + nopaque.forms.AutoInit(); + // #endregion + } + + flash(message, category) { + let iconPrefix; + + switch (category) { + case 'corpus': { + iconPrefix = 'book'; + break; + } + case 'job': { + iconPrefix = 'J'; + break; + } + case 'error': { + iconPrefix = 'error'; + break; + } + default: { + iconPrefix = 'notifications'; + break; + } + } + + let toast = M.toast( + { + html: ` + ${iconPrefix}${message} + + `.trim() + } + ); + let dismissToastElement = toast.el.querySelector('.toast-action[data-toast-action="dismiss"]'); + dismissToastElement.addEventListener('click', () => {toast.dismiss();}); + } +} diff --git a/app/static/js/app.users.js b/app/static/js/app.users.js new file mode 100644 index 00000000..fc3f0d3e --- /dev/null +++ b/app/static/js/app.users.js @@ -0,0 +1,53 @@ +nopaque.UsersExtension = class UsersExtension { + #data; + #promises; + + constructor(app) { + this.app = app; + + this.#data = {}; + this.app.data.users = this.#data; + + this.#promises = { + get: {}, + subscribe: {} + }; + } + + async #get(userId) { + const response = await this.app.socket.emitWithAck('users.get', userId); + + if (response.status != 200) { + throw new Error(`[${response.status}] ${response.statusText}`); + } + + this.#data[userId] = response.body; + return this.#data[userId]; + } + + get(userId) { + if (userId in this.#promises.get) { + return this.#promises.get[userId]; + } + + this.#promises.get[userId] = this.#get(userId); + return this.#promises.get[userId]; + } + + async #subscribe(userId) { + const response = await this.app.socket.emitWithAck('users.subscribe', userId); + + if (response.status != 200) { + throw new Error(`[${response.status}] ${response.statusText}`); + } + } + + subscribe(userId) { + if (userId in this.#promises.subscribe) { + return this.#promises.subscribe[userId]; + } + + this.#promises.subscribe[userId] = this.#subscribe(userId); + return this.#promises.subscribe[userId]; + } +} diff --git a/app/static/js/corpus-analysis/concordance-extension.js b/app/static/js/corpus-analysis/concordance-extension.js index f32d6781..7137a37d 100644 --- a/app/static/js/corpus-analysis/concordance-extension.js +++ b/app/static/js/corpus-analysis/concordance-extension.js @@ -66,7 +66,7 @@ nopaque.corpus_analysis.ConcordanceExtension = class ConcordanceExtension { errorString += `${error.constructor.name}`; this.elements.error.innerText = errorString; this.elements.error.classList.remove('hide'); - app.flash(errorString, 'error'); + app.ui.flash(errorString, 'error'); this.elements.progress.classList.add('hide'); } this.app.enableActionElements(); @@ -239,7 +239,7 @@ nopaque.corpus_analysis.ConcordanceExtension = class ConcordanceExtension { if (subcorpus.selectedItems.size === 0) { this.elements.progress.classList.add('hide'); this.app.enableActionElements(); - app.flash('No matches selected', 'error'); + app.ui.flash('No matches selected', 'error'); return; } promise = subcorpus.o.partialExport([...subcorpus.selectedItems], 50); @@ -298,7 +298,7 @@ nopaque.corpus_analysis.ConcordanceExtension = class ConcordanceExtension { let subcorpus = this.data.subcorpora[this.settings.selectedSubcorpus]; subcorpus.o.drop().then( (cQiStatus) => { - app.flash(`${subcorpus.o.name} deleted`, 'corpus'); + app.ui.flash(`${subcorpus.o.name} deleted`, 'corpus'); delete this.data.subcorpora[subcorpus.o.name]; this.settings.selectedSubcorpus = undefined; for (let subcorpusName in this.data.subcorpora) { @@ -320,7 +320,7 @@ nopaque.corpus_analysis.ConcordanceExtension = class ConcordanceExtension { }, (cqiError) => { let errorString = `${cqiError.code}: ${cqiError.constructor.name}`; - app.flash(errorString, 'error'); + app.ui.flash(errorString, 'error'); } ); }); diff --git a/app/static/js/corpus-analysis/reader-extension.js b/app/static/js/corpus-analysis/reader-extension.js index 3c7fec91..bca2a10d 100644 --- a/app/static/js/corpus-analysis/reader-extension.js +++ b/app/static/js/corpus-analysis/reader-extension.js @@ -46,7 +46,7 @@ nopaque.corpus_analysis.ReaderExtension = class ReaderExtension { if ('description' in error) {errorString += `: ${error.description}`;} this.elements.error.innerText = errorString; this.elements.error.classList.remove('hide'); - app.flash(errorString, 'error'); + app.ui.flash(errorString, 'error'); this.elements.progress.classList.add('hide'); } this.app.enableActionElements(); @@ -205,7 +205,7 @@ nopaque.corpus_analysis.ReaderExtension = class ReaderExtension { ` ); this.elements.corpusPagination.appendChild(pageElement); - + for (let paginateTriggerElement of this.elements.corpusPagination.querySelectorAll('.pagination-trigger[data-target]')) { paginateTriggerElement.addEventListener('click', (event) => { event.preventDefault(); diff --git a/app/static/js/forms/base-form.js b/app/static/js/forms/base-form.js index 5b4b9e41..29551306 100644 --- a/app/static/js/forms/base-form.js +++ b/app/static/js/forms/base-form.js @@ -101,7 +101,7 @@ nopaque.forms.BaseForm = class BaseForm { } } if (request.status === 500) { - app.flash('Internal Server Error', 'error'); + app.ui.flash('Internal Server Error', 'error'); } modal.close(); }); diff --git a/app/static/js/requests/index.js b/app/static/js/requests/index.js index 177eca88..a61d04ce 100644 --- a/app/static/js/requests/index.js +++ b/app/static/js/requests/index.js @@ -18,23 +18,23 @@ nopaque.requests.JSONfetch = (input, init={}) => { } if (response.status === 204) { return; - } + } response.json() .then( (json) => { let message = json.message; let category = json.category || 'message'; if (message) { - app.flash(message, category); + app.ui.flash(message, category); } }, (error) => { - app.flash(`[${response.status}]: ${response.statusText}`, 'error'); + app.ui.flash(`[${response.status}]: ${response.statusText}`, 'error'); } ); }, (response) => { - app.flash('Something went wrong', 'error'); + app.ui.flash('Something went wrong', 'error'); reject(response); } ); diff --git a/app/static/js/resource-displays/resource-display.js b/app/static/js/resource-displays/resource-display.js index 9d327131..e1036ef9 100644 --- a/app/static/js/resource-displays/resource-display.js +++ b/app/static/js/resource-displays/resource-display.js @@ -6,13 +6,13 @@ nopaque.resource_displays.ResourceDisplay = class ResourceDisplay { this.userId = this.displayElement.dataset.userId; this.isInitialized = false; if (this.userId) { - app.subscribeUser(this.userId) + app.users.subscribe(this.userId) .then((response) => { - app.socket.on('patch_user', (patch) => { + app.socket.on('users.patch', (patch) => { if (this.isInitialized) {this.onPatch(patch);} }); }); - app.getUser(this.userId) + app.users.get(this.userId) .then((user) => { this.init(user); this.isInitialized = true; diff --git a/app/static/js/resource-lists/corpus-file-list.js b/app/static/js/resource-lists/corpus-file-list.js index fd3acea1..3d819398 100644 --- a/app/static/js/resource-lists/corpus-file-list.js +++ b/app/static/js/resource-lists/corpus-file-list.js @@ -14,12 +14,12 @@ nopaque.resource_lists.CorpusFileList = class CorpusFileList extends nopaque.res this.hasPermissionView = listContainerElement.dataset?.hasPermissionView == 'true' || false; this.hasPermissionManageFiles = listContainerElement.dataset?.hasPermissionManageFiles == 'true' || false; if (this.userId === undefined || this.corpusId === undefined) {return;} - app.subscribeUser(this.userId).then((response) => { - app.socket.on('patch_user', (patch) => { + app.users.subscribe(this.userId).then((response) => { + app.socket.on('users.patch', (patch) => { if (this.isInitialized) {this.onPatch(patch);} }); }); - app.getUser(this.userId).then((user) => { + app.users.get(this.userId).then((user) => { this.add(Object.values(user.corpora[this.corpusId].files || user.followed_corpora[this.corpusId].files)); this.isInitialized = true; }); diff --git a/app/static/js/resource-lists/corpus-follower-list.js b/app/static/js/resource-lists/corpus-follower-list.js index 0a0679d0..98c78605 100644 --- a/app/static/js/resource-lists/corpus-follower-list.js +++ b/app/static/js/resource-lists/corpus-follower-list.js @@ -12,12 +12,12 @@ nopaque.resource_lists.CorpusFollowerList = class CorpusFollowerList extends nop this.userId = listContainerElement.dataset.userId; this.corpusId = listContainerElement.dataset.corpusId; if (this.userId === undefined || this.corpusId === undefined) {return;} - app.subscribeUser(this.userId).then((response) => { - app.socket.on('patch_user', (patch) => { + app.users.subscribe(this.userId).then((response) => { + app.socket.on('users.patch', (patch) => { if (this.isInitialized) {this.onPatch(patch);} }); }); - app.getUser(this.userId).then((user) => { + app.users.get(this.userId).then((user) => { // let corpusFollowerAssociations = Object.values(user.corpora[this.corpusId].corpus_follower_associations); // let filteredList = corpusFollowerAssociations.filter(association => association.follower.id != currentUserId); // this.add(filteredList); diff --git a/app/static/js/resource-lists/corpus-list.js b/app/static/js/resource-lists/corpus-list.js index b362bd78..6fac836f 100644 --- a/app/static/js/resource-lists/corpus-list.js +++ b/app/static/js/resource-lists/corpus-list.js @@ -11,12 +11,12 @@ nopaque.resource_lists.CorpusList = class CorpusList extends nopaque.resource_li this.selectedItemIds = new Set(); this.userId = listContainerElement.dataset.userId; if (this.userId === undefined) {return;} - app.subscribeUser(this.userId).then((response) => { - app.socket.on('patch_user', (patch) => { + app.users.subscribe(this.userId).then((response) => { + app.socket.on('users.patch', (patch) => { if (this.isInitialized) {this.onPatch(patch);} }); }); - app.getUser(this.userId).then((user) => { + app.users.get(this.userId).then((user) => { this.add(this.aggregateData(user)); this.isInitialized = true; }); diff --git a/app/static/js/resource-lists/job-input-list.js b/app/static/js/resource-lists/job-input-list.js index 270c6d2d..7de10e45 100644 --- a/app/static/js/resource-lists/job-input-list.js +++ b/app/static/js/resource-lists/job-input-list.js @@ -8,8 +8,8 @@ nopaque.resource_lists.JobInputList = class JobInputList extends nopaque.resourc this.userId = listContainerElement.dataset.userId; this.jobId = listContainerElement.dataset.jobId; if (this.userId === undefined || this.jobId === undefined) {return;} - app.subscribeUser(this.userId); - app.getUser(this.userId).then((user) => { + app.users.subscribe(this.userId); + app.users.get(this.userId).then((user) => { this.add(Object.values(user.jobs[this.jobId].inputs)); this.isInitialized = true; }); diff --git a/app/static/js/resource-lists/job-list.js b/app/static/js/resource-lists/job-list.js index 336562ff..a3fb87f1 100644 --- a/app/static/js/resource-lists/job-list.js +++ b/app/static/js/resource-lists/job-list.js @@ -12,12 +12,12 @@ nopaque.resource_lists.JobList = class JobList extends nopaque.resource_lists.Re this.selectedItemIds = new Set(); this.userId = listContainerElement.dataset.userId; if (this.userId === undefined) {return;} - app.subscribeUser(this.userId).then((response) => { - app.socket.on('patch_user', (patch) => { + app.users.subscribe(this.userId).then((response) => { + app.socket.on('users.patch', (patch) => { if (this.isInitialized) {this.onPatch(patch);} }); }); - app.getUser(this.userId).then((user) => { + app.users.get(this.userId).then((user) => { this.add(Object.values(user.jobs)); this.isInitialized = true; }); diff --git a/app/static/js/resource-lists/job-result-list.js b/app/static/js/resource-lists/job-result-list.js index f361306c..bc48e9a2 100644 --- a/app/static/js/resource-lists/job-result-list.js +++ b/app/static/js/resource-lists/job-result-list.js @@ -8,12 +8,12 @@ nopaque.resource_lists.JobResultList = class JobResultList extends nopaque.resou this.userId = listContainerElement.dataset.userId; this.jobId = listContainerElement.dataset.jobId; if (this.userId === undefined || this.jobId === undefined) {return;} - app.subscribeUser(this.userId).then((response) => { - app.socket.on('patch_user', (patch) => { + app.users.subscribe(this.userId).then((response) => { + app.socket.on('users.patch', (patch) => { if (this.isInitialized) {this.onPatch(patch);} }); }); - app.getUser(this.userId).then((user) => { + app.users.get(this.userId).then((user) => { this.add(Object.values(user.jobs[this.jobId].results)); this.isInitialized = true; }); diff --git a/app/static/js/resource-lists/spacy-nlp-pipeline-model-list.js b/app/static/js/resource-lists/spacy-nlp-pipeline-model-list.js index eb340a78..6b9f5e2a 100644 --- a/app/static/js/resource-lists/spacy-nlp-pipeline-model-list.js +++ b/app/static/js/resource-lists/spacy-nlp-pipeline-model-list.js @@ -8,12 +8,12 @@ nopaque.resource_lists.SpaCyNLPPipelineModelList = class SpaCyNLPPipelineModelLi this.isInitialized = false; this.userId = listContainerElement.dataset.userId; if (this.userId === undefined) {return;} - app.subscribeUser(this.userId).then((response) => { - app.socket.on('patch_user', (patch) => { + app.users.subscribe(this.userId).then((response) => { + app.socket.on('users.patch', (patch) => { if (this.isInitialized) {this.onPatch(patch);} }); }); - app.getUser(this.userId).then((user) => { + app.users.get(this.userId).then((user) => { this.add(Object.values(user.spacy_nlp_pipeline_models)); this.isInitialized = true; }); diff --git a/app/static/js/resource-lists/tesseract-ocr-pipeline-model-list.js b/app/static/js/resource-lists/tesseract-ocr-pipeline-model-list.js index 86d673d9..609dc1d1 100644 --- a/app/static/js/resource-lists/tesseract-ocr-pipeline-model-list.js +++ b/app/static/js/resource-lists/tesseract-ocr-pipeline-model-list.js @@ -8,12 +8,12 @@ nopaque.resource_lists.TesseractOCRPipelineModelList = class TesseractOCRPipelin this.isInitialized = false; this.userId = listContainerElement.dataset.userId; if (this.userId === undefined) {return;} - app.subscribeUser(this.userId).then((response) => { - app.socket.on('patch_user', (patch) => { + app.users.subscribe(this.userId).then((response) => { + app.socket.on('users.patch', (patch) => { if (this.isInitialized) {this.onPatch(patch);} }); }); - app.getUser(this.userId).then((user) => { + app.users.get(this.userId).then((user) => { this.add(Object.values(user.tesseract_ocr_pipeline_models)); for (let uncheckedCheckbox of this.listjs.list.querySelectorAll('input[data-checked="True"]')) { uncheckedCheckbox.setAttribute('checked', ''); diff --git a/app/templates/_base/scripts.html.j2 b/app/templates/_base/scripts.html.j2 index 1f6d5af0..fd37daef 100644 --- a/app/templates/_base/scripts.html.j2 +++ b/app/templates/_base/scripts.html.j2 @@ -9,6 +9,8 @@ output='gen/nopaque.%(version)s.js', 'js/index.js', 'js/app.js', + 'js/app.ui.js', + 'js/app.users.js', 'js/utils.js', 'js/forms/index.js', @@ -82,11 +84,11 @@ const currentUserId = {{ current_user.hashid|tojson }}; // Subscribe to the current user's data events - app.subscribeUser(currentUserId) + app.users.subscribe(currentUserId) .catch((error) => {throw JSON.stringify(error);}); // Get the current user's data - app.getUser(currentUserId, true, true) + app.users.get(currentUserId, true, true) .catch((error) => {throw JSON.stringify(error);}); {% if not current_user.terms_of_use_accepted -%} @@ -96,7 +98,7 @@ // Display flashed messages for (let [category, message] of {{ get_flashed_messages(with_categories=True)|tojson }}) { - app.flash(message, message); + app.ui.flash(message, message); } diff --git a/app/templates/corpora/corpus.html.j2 b/app/templates/corpora/corpus.html.j2 index 740f08b5..56dba7cb 100644 --- a/app/templates/corpora/corpus.html.j2 +++ b/app/templates/corpora/corpus.html.j2 @@ -364,8 +364,8 @@ shareLinkModalCreateButtonElement.addEventListener('click', (event) => { shareLinkModalOutputCopyButtonElement.addEventListener('click', (event) => { navigator.clipboard.writeText(shareLinkModalOutputFieldElement.value) .then( - () => {app.flash('Copied!');}, - () => {app.flash('Could not copy to clipboard. Please copy manually.', 'error');} + () => {app.ui.flash('Copied!');}, + () => {app.ui.flash('Could not copy to clipboard. Please copy manually.', 'error');} ); }); diff --git a/app/templates/corpora/public_corpus.html.j2 b/app/templates/corpora/public_corpus.html.j2 index 9a835601..a765526f 100644 --- a/app/templates/corpora/public_corpus.html.j2 +++ b/app/templates/corpora/public_corpus.html.j2 @@ -396,8 +396,8 @@ shareLinkModalCreateButtonElement.addEventListener('click', (event) => { shareLinkModalOutputCopyButtonElement.addEventListener('click', (event) => { navigator.clipboard.writeText(shareLinkModalOutputFieldElement.value) .then( - () => {app.flash('Copied!');}, - () => {app.flash('Could not copy to clipboard. Please copy manually.', 'error');} + () => {app.ui.flash('Copied!');}, + () => {app.ui.flash('Could not copy to clipboard. Please copy manually.', 'error');} ); });