mirror of
https://git.anonymousland.org/anonymousland/synapse.git
synced 2024-12-22 16:14:59 -05:00
7a0dcea3e5
* fix validation and don't use built-in validation UI Co-authored-by: Bruno Windels <brunow@element.io>
117 lines
3.6 KiB
JavaScript
117 lines
3.6 KiB
JavaScript
const usernameField = document.getElementById("field-username");
|
|
const usernameOutput = document.getElementById("field-username-output");
|
|
const form = document.getElementById("form");
|
|
|
|
// needed to validate on change event when no input was changed
|
|
let needsValidation = true;
|
|
let isValid = false;
|
|
|
|
function throttle(fn, wait) {
|
|
let timeout;
|
|
const throttleFn = function() {
|
|
const args = Array.from(arguments);
|
|
if (timeout) {
|
|
clearTimeout(timeout);
|
|
}
|
|
timeout = setTimeout(fn.bind.apply(fn, [null].concat(args)), wait);
|
|
};
|
|
throttleFn.cancelQueued = function() {
|
|
clearTimeout(timeout);
|
|
};
|
|
return throttleFn;
|
|
}
|
|
|
|
function checkUsernameAvailable(username) {
|
|
let check_uri = 'check?username=' + encodeURIComponent(username);
|
|
return fetch(check_uri, {
|
|
// include the cookie
|
|
"credentials": "same-origin",
|
|
}).then(function(response) {
|
|
if(!response.ok) {
|
|
// for non-200 responses, raise the body of the response as an exception
|
|
return response.text().then((text) => { throw new Error(text); });
|
|
} else {
|
|
return response.json();
|
|
}
|
|
}).then(function(json) {
|
|
if(json.error) {
|
|
return {message: json.error};
|
|
} else if(json.available) {
|
|
return {available: true};
|
|
} else {
|
|
return {message: username + " is not available, please choose another."};
|
|
}
|
|
});
|
|
}
|
|
|
|
const allowedUsernameCharacters = new RegExp("^[a-z0-9\\.\\_\\-\\/\\=]+$");
|
|
const allowedCharactersString = "lowercase letters, digits, ., _, -, /, =";
|
|
|
|
function reportError(error) {
|
|
throttledCheckUsernameAvailable.cancelQueued();
|
|
usernameOutput.innerText = error;
|
|
usernameOutput.classList.add("error");
|
|
usernameField.parentElement.classList.add("invalid");
|
|
usernameField.focus();
|
|
}
|
|
|
|
function validateUsername(username) {
|
|
isValid = false;
|
|
needsValidation = false;
|
|
usernameOutput.innerText = "";
|
|
usernameField.parentElement.classList.remove("invalid");
|
|
usernameOutput.classList.remove("error");
|
|
if (!username) {
|
|
return reportError("Please provide a username");
|
|
}
|
|
if (username.length > 255) {
|
|
return reportError("Too long, please choose something shorter");
|
|
}
|
|
if (!allowedUsernameCharacters.test(username)) {
|
|
return reportError("Invalid username, please only use " + allowedCharactersString);
|
|
}
|
|
usernameOutput.innerText = "Checking if username is available …";
|
|
throttledCheckUsernameAvailable(username);
|
|
}
|
|
|
|
const throttledCheckUsernameAvailable = throttle(function(username) {
|
|
const handleError = function(err) {
|
|
// don't prevent form submission on error
|
|
usernameOutput.innerText = "";
|
|
isValid = true;
|
|
};
|
|
try {
|
|
checkUsernameAvailable(username).then(function(result) {
|
|
if (!result.available) {
|
|
reportError(result.message);
|
|
} else {
|
|
isValid = true;
|
|
usernameOutput.innerText = "";
|
|
}
|
|
}, handleError);
|
|
} catch (err) {
|
|
handleError(err);
|
|
}
|
|
}, 500);
|
|
|
|
form.addEventListener("submit", function(evt) {
|
|
if (needsValidation) {
|
|
validateUsername(usernameField.value);
|
|
evt.preventDefault();
|
|
return;
|
|
}
|
|
if (!isValid) {
|
|
evt.preventDefault();
|
|
usernameField.focus();
|
|
return;
|
|
}
|
|
});
|
|
usernameField.addEventListener("input", function(evt) {
|
|
validateUsername(usernameField.value);
|
|
});
|
|
usernameField.addEventListener("change", function(evt) {
|
|
if (needsValidation) {
|
|
validateUsername(usernameField.value);
|
|
}
|
|
});
|