mirror of
https://github.com/hrfee/jfa-go.git
synced 2024-11-13 05:40:10 +00:00
Harvey Tindall
32b8ed4aa2
i wanted to split up the web ui components into multiple files, and figured it'd be a good chance to try out typescript. run make typescript to compile everything in ts/ and put it in data/static/. This is less of a rewrite and more of a refactoring, most of it still works the same but bits have been cleaned up too. Remaining javascript found in setup.js and form.html
336 lines
13 KiB
TypeScript
336 lines
13 KiB
TypeScript
const checkCheckboxes = (): void => {
|
|
const defaultsButton = document.getElementById('accountsTabSetDefaults');
|
|
const deleteButton = document.getElementById('accountsTabDelete');
|
|
const checkboxes: NodeListOf<HTMLInputElement> = document.getElementById('accountsList').querySelectorAll('input[type=checkbox]:checked');
|
|
let checked = checkboxes.length;
|
|
if (checked == 0) {
|
|
Unfocus(defaultsButton);
|
|
Unfocus(deleteButton);
|
|
} else {
|
|
Focus(defaultsButton);
|
|
Focus(deleteButton);
|
|
if (checked == 1) {
|
|
deleteButton.textContent = 'Delete User';
|
|
} else {
|
|
deleteButton.textContent = 'Delete Users';
|
|
}
|
|
}
|
|
}
|
|
|
|
const validateEmail = (email: string): boolean => /\S+@\S+\.\S+/.test(email);
|
|
|
|
function changeEmail(icon: HTMLElement, id: string): void {
|
|
const iconContent = icon.outerHTML;
|
|
icon.setAttribute('class', '');
|
|
const entry = icon.nextElementSibling as HTMLInputElement;
|
|
const ogEmail = entry.value;
|
|
entry.readOnly = false;
|
|
entry.classList.remove('form-control-plaintext');
|
|
entry.classList.add('form-control');
|
|
if (ogEmail == "") {
|
|
entry.placeholder = 'Address';
|
|
}
|
|
const tick = createEl(`
|
|
<i class="fa fa-check d-inline-block icon-button text-success" style="margin-left: 0.5rem; margin-right: 0.5rem;"></i>
|
|
`);
|
|
tick.onclick = (): void => {
|
|
const newEmail = entry.value;
|
|
if (!validateEmail(newEmail) || newEmail == ogEmail) {
|
|
return;
|
|
}
|
|
cross.remove();
|
|
const spinner = createEl(`
|
|
<div class="spinner-border spinner-border-sm" role="status" style="width: 1rem; height: 1rem; margin-left: 0.5rem;">
|
|
<span class="sr-only">Saving...</span>
|
|
</div>
|
|
`);
|
|
tick.replaceWith(spinner);
|
|
let send = {};
|
|
send[id] = newEmail;
|
|
_post("/modifyEmails", send, function (): void {
|
|
if (this.readyState == 4) {
|
|
if (this.status == 200 || this.status == 204) {
|
|
entry.nextElementSibling.remove();
|
|
} else {
|
|
entry.value = ogEmail;
|
|
}
|
|
}
|
|
});
|
|
icon.outerHTML = iconContent;
|
|
entry.readOnly = true;
|
|
entry.classList.remove('form-control');
|
|
entry.classList.add('form-control-plaintext');
|
|
entry.placeholder = '';
|
|
};
|
|
const cross = createEl(`
|
|
<i class="fa fa-close d-inline-block icon-button text-danger"></i>
|
|
`);
|
|
cross.onclick = (): void => {
|
|
tick.remove();
|
|
cross.remove();
|
|
icon.outerHTML = iconContent;
|
|
entry.readOnly = true;
|
|
entry.classList.remove('form-control');
|
|
entry.classList.add('form-control-plaintext');
|
|
entry.placeholder = '';
|
|
entry.value = ogEmail;
|
|
};
|
|
icon.parentNode.appendChild(tick);
|
|
icon.parentNode.appendChild(cross);
|
|
};
|
|
|
|
var jfUsers: Array<Object>;
|
|
|
|
function populateUsers(): void {
|
|
const acList = document.getElementById('accountsList');
|
|
acList.innerHTML = `
|
|
<div class="d-flex align-items-center">
|
|
<strong>Getting Users...</strong>
|
|
<div class="spinner-border ml-auto" role="status" aria-hidden="true"></div>
|
|
</div>
|
|
`;
|
|
Unfocus(acList.parentNode.querySelector('thead'));
|
|
const accountsList = document.createElement('tbody');
|
|
accountsList.id = 'accountsList';
|
|
const generateEmail = (id: string, name: string, email: string): string => {
|
|
let entry: HTMLDivElement = document.createElement('div');
|
|
entry.id = 'email_' + id;
|
|
let emailValue: string = email;
|
|
if (emailValue == undefined) {
|
|
emailValue = "";
|
|
}
|
|
entry.innerHTML = `
|
|
<i class="fa fa-edit d-inline-block icon-button" style="margin-right: 2%;" onclick="changeEmail(this, '${id}')"></i>
|
|
<input type="email" class="form-control-plaintext form-control-sm text-muted d-inline-block addressText" id="address_${id}" style="width: auto;" value="${emailValue}" readonly>
|
|
`;
|
|
return entry.outerHTML;
|
|
};
|
|
const template = (id: string, username: string, email: string, lastActive: string, admin: boolean): string => {
|
|
let isAdmin = "No";
|
|
if (admin) {
|
|
isAdmin = "Yes";
|
|
}
|
|
let fci = "form-check-input";
|
|
if (bsVersion != 5) {
|
|
fci = "";
|
|
}
|
|
return `
|
|
<td nowrap="nowrap" class="align-middle" scope="row"><input class="${fci}" type="checkbox" value="" id="select_${id}" onclick="checkCheckboxes();"></td>
|
|
<td nowrap="nowrap" class="align-middle">${username}</td>
|
|
<td nowrap="nowrap" class="align-middle">${generateEmail(id, name, email)}</td>
|
|
<td nowrap="nowrap" class="align-middle">${lastActive}</td>
|
|
<td nowrap="nowrap" class="align-middle">${isAdmin}</td>
|
|
`;
|
|
};
|
|
|
|
_get("/getUsers", null, function (): void {
|
|
if (this.readyState == 4 && this.status == 200) {
|
|
jfUsers = this.response['users'];
|
|
for (const user of jfUsers) {
|
|
let tr = document.createElement('tr');
|
|
tr.innerHTML = template(user['id'], user['name'], user['email'], user['last_active'], user['admin']);
|
|
accountsList.appendChild(tr);
|
|
}
|
|
Focus(acList.parentNode.querySelector('thead'));
|
|
acList.replaceWith(accountsList);
|
|
}
|
|
});
|
|
}
|
|
|
|
function populateRadios(): void {
|
|
const radioList = document.getElementById('defaultUserRadios');
|
|
radioList.textContent = '';
|
|
let first = true;
|
|
for (const i in jfUsers) {
|
|
const user = jfUsers[i];
|
|
const radio = document.createElement('div');
|
|
radio.classList.add('form-check');
|
|
let checked = '';
|
|
if (first) {
|
|
checked = 'checked';
|
|
first = false;
|
|
}
|
|
radio.innerHTML = `
|
|
<input class="form-check-input" type="radio" name="defaultRadios" id="default_${user['id']}" ${checked}>
|
|
<label class="form-check-label" for="default_${user['id']}">${user['name']}</label>`;
|
|
radioList.appendChild(radio);
|
|
}
|
|
}
|
|
|
|
(<HTMLInputElement>document.getElementById('selectAll')).onclick = function (): void {
|
|
const checkboxes: NodeListOf<HTMLInputElement> = document.getElementById('accountsList').querySelectorAll('input[type=checkbox]');
|
|
for (let i = 0; i < checkboxes.length; i++) {
|
|
checkboxes[i].checked = (<HTMLInputElement>this).checked;
|
|
}
|
|
checkCheckboxes();
|
|
};
|
|
|
|
(<HTMLInputElement>document.getElementById('deleteModalNotify')).onclick = function (): void {
|
|
const textbox: HTMLElement = document.getElementById('deleteModalReasonBox');
|
|
if ((<HTMLInputElement>this).checked) {
|
|
Focus(textbox);
|
|
} else {
|
|
Unfocus(textbox);
|
|
}
|
|
};
|
|
|
|
(<HTMLButtonElement>document.getElementById('accountsTabDelete')).onclick = function (): void {
|
|
const deleteButton = this as HTMLButtonElement;
|
|
const checkboxes: NodeListOf<HTMLInputElement> = document.getElementById('accountsList').querySelectorAll('input[type=checkbox]:checked');
|
|
let selected: Array<string> = new Array(checkboxes.length);
|
|
for (let i = 0; i < checkboxes.length; i++) {
|
|
selected[i] = checkboxes[i].id.replace("select_", "");
|
|
}
|
|
let title = " user";
|
|
let msg = "Notify user";
|
|
if (selected.length > 1) {
|
|
title += "s";
|
|
msg += "s";
|
|
}
|
|
title = `Delete ${selected.length} ${title}`;
|
|
msg += " of account deletion";
|
|
|
|
document.getElementById('deleteModalTitle').textContent = title;
|
|
const dmNotify = document.getElementById('deleteModalNotify') as HTMLInputElement;
|
|
dmNotify.checked = false;
|
|
document.getElementById('deleteModalNotifyLabel').textContent = msg;
|
|
const dmReason = document.getElementById('deleteModalReason') as HTMLTextAreaElement;
|
|
dmReason.value = '';
|
|
Unfocus(document.getElementById('deleteModalReasonBox'));
|
|
const dmSend = document.getElementById('deleteModalSend') as HTMLButtonElement;
|
|
dmSend.textContent = 'Delete';
|
|
dmSend.onclick = function (): void {
|
|
const button = this as HTMLButtonElement;
|
|
const send = {
|
|
'users': selected,
|
|
'notify': dmNotify.checked,
|
|
'reason': dmReason.value
|
|
};
|
|
_post("/deleteUser", send, function (): void {
|
|
if (this.readyState == 4) {
|
|
if (this.status == 500) {
|
|
if ("error" in this.reponse) {
|
|
button.textContent = 'Failed';
|
|
} else {
|
|
button.textContent = 'Partial fail (check console)';
|
|
console.log(this.response);
|
|
}
|
|
setTimeout((): void => {
|
|
Unfocus(deleteButton);
|
|
deleteModal.hide();
|
|
}, 4000);
|
|
} else {
|
|
Unfocus(deleteButton);
|
|
deleteModal.hide()
|
|
}
|
|
populateUsers();
|
|
checkCheckboxes();
|
|
}
|
|
});
|
|
};
|
|
deleteModal.show();
|
|
};
|
|
|
|
(<HTMLInputElement>document.getElementById('selectAll')).checked = false;
|
|
|
|
(<HTMLButtonElement>document.getElementById('accountsTabSetDefaults')).onclick = function (): void {
|
|
const checkboxes: NodeListOf<HTMLInputElement> = document.getElementById('accountsList').querySelectorAll('input[type=checkbox]:checked');
|
|
let userIDs: Array<string> = new Array(checkboxes.length);
|
|
for (let i = 0; i < checkboxes.length; i++){
|
|
userIDs[i] = checkboxes[i].id.replace("select_", "");
|
|
}
|
|
if (userIDs.length == 0) {
|
|
return;
|
|
}
|
|
populateRadios();
|
|
let userString = 'user';
|
|
if (userIDs.length > 1) {
|
|
userString += "s";
|
|
}
|
|
document.getElementById('defaultsTitle').textContent = `Apply settings to ${userIDs.length} ${userString}`;
|
|
document.getElementById('userDefaultsDescription').textContent = `
|
|
Create an account and configure it to your liking, then choose it from below to apply to your selected users.
|
|
`;
|
|
document.getElementById('storeHomescreenLabel').textContent = `Apply homescreen layout`;
|
|
Focus(document.getElementById('defaultsSourceSection'));
|
|
(<HTMLSelectElement>document.getElementById('defaultsSource')).value = 'userTemplate';
|
|
Unfocus(document.getElementById('defaultUserRadios'));
|
|
document.getElementById('storeDefaults').onclick = (): void => storeDefaults(userIDs);
|
|
userDefaultsModal.show();
|
|
};
|
|
|
|
(<HTMLSelectElement>document.getElementById('defaultsSource')).addEventListener('change', function (): void {
|
|
const radios = document.getElementById('defaultUserRadios');
|
|
if (this.value == 'userTemplate') {
|
|
Unfocus(radios);
|
|
} else {
|
|
Focus(radios);
|
|
}
|
|
});
|
|
|
|
(<HTMLButtonElement>document.getElementById('newUserCreate')).onclick = function (): void {
|
|
const button = this as HTMLButtonElement;
|
|
const ogText = button.textContent;
|
|
button.innerHTML = `
|
|
<span class="spinner-border spinner-border-sm" role="status" aria-hidden="true"></span> Creating...
|
|
`;
|
|
const email: string = (<HTMLInputElement>document.getElementById('newUserEmail')).value;
|
|
var username: string = email;
|
|
if (document.getElementById('newUserName') != null) {
|
|
username = (<HTMLInputElement>document.getElementById('newUserName')).value;
|
|
}
|
|
const password: string = (<HTMLInputElement>document.getElementById('newUserPassword')).value;
|
|
if (!validateEmail(email) && email != "") {
|
|
return;
|
|
}
|
|
const send = {
|
|
'username': username,
|
|
'password': password,
|
|
'email': email
|
|
};
|
|
_post("/newUserAdmin", send, function (): void {
|
|
if (this.readyState == 4) {
|
|
rmAttr(button, 'btn-primary');
|
|
if (this.status == 200) {
|
|
addAttr(button, 'btn-success');
|
|
button.textContent = 'Success';
|
|
setTimeout((): void => {
|
|
rmAttr(button, 'btn-success');
|
|
addAttr(button, 'btn-primary');
|
|
button.textContent = ogText;
|
|
newUserModal.hide();
|
|
}, 1000);
|
|
populateUsers();
|
|
} else {
|
|
addAttr(button, 'btn-danger');
|
|
if ("error" in this.response) {
|
|
button.textContent = this.response["error"];
|
|
} else {
|
|
button.textContent = 'Failed';
|
|
}
|
|
setTimeout((): void => {
|
|
rmAttr(button, 'btn-danger');
|
|
addAttr(button, 'btn-primary');
|
|
button.textContent = ogText;
|
|
}, 2000);
|
|
populateUsers();
|
|
}
|
|
}
|
|
});
|
|
};
|
|
|
|
(<HTMLButtonElement>document.getElementById('accountsTabAddUser')).onclick = function (): void {
|
|
(<HTMLInputElement>document.getElementById('newUserEmail')).value = '';
|
|
(<HTMLInputElement>document.getElementById('newUserPassword')).value = '';
|
|
if (document.getElementById('newUserName') != null) {
|
|
(<HTMLInputElement>document.getElementById('newUserName')).value = '';
|
|
}
|
|
newUserModal.show();
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|