surwiki/assets/script.js

227 lines
7.4 KiB
JavaScript
Raw Permalink Normal View History

2018-08-22 09:14:25 +03:00
"use strict";
function autosizeTextarea(textarea, shadow) {
shadow.style.width = textarea.clientWidth + "px";
shadow.value = textarea.value;
textarea.style.height = shadow.scrollHeight + "px";
}
function queryArgsFromForm(form) {
const items = [];
2018-09-24 09:43:36 +03:00
for (const {name, value, type, checked} of form.elements) {
if (!name) continue;
2018-09-24 09:43:36 +03:00
if (type === "radio" && !checked) continue;
items.push(encodeURIComponent(name) + '=' + encodeURIComponent(value));
}
return items.join('&');
}
function isEdited(form) {
2018-09-24 09:43:36 +03:00
for (const {name, value, defaultValue, checked, defaultChecked} of form.elements) {
if (name && ((value !== defaultValue) || (checked !== defaultChecked))) return true;
}
return false;
}
2017-12-02 13:01:24 +03:00
function instantiate(templateId) {
return document.getElementById(templateId).firstElementChild.cloneNode(true);
2017-12-01 18:52:39 +03:00
}
2017-12-02 13:01:24 +03:00
function popup(dialog) {
document.body.appendChild(dialog);
2017-12-02 13:01:24 +03:00
dialog.querySelector(".primary").focus();
return new Promise((resolve, reject) => {
2017-12-02 13:01:24 +03:00
function handler(ev) {
document.body.removeChild(dialog);
resolve(ev.target.getAttribute("data-value"));
}
const buttons = dialog.querySelectorAll('.btn-row>*');
for (let i = 0; i < buttons.length; ++i)
buttons[i].addEventListener("click", handler);
});
}
2017-12-02 13:01:24 +03:00
function loginDialog(loginUrl) {
const dialog = instantiate("login");
dialog.querySelector("a").setAttribute("href", loginUrl);
return popup(dialog);
}
2017-12-01 18:37:41 +03:00
2017-12-02 13:01:24 +03:00
function alertAsync(message) {
const dialog = instantiate("alert");
dialog.querySelector(".message").textContent = message;
return popup(dialog);
2017-12-01 18:37:41 +03:00
}
2017-12-02 13:06:20 +03:00
function confirmDiscard() {
return popup(instantiate("confirm-discard"));
2017-12-01 18:44:07 +03:00
}
let hasBeenOpen = false;
function openEditor() {
2018-09-24 09:43:36 +03:00
const bodyElement = document.querySelector("body");
const container = document.querySelector(".container");
const rendered = container.querySelector(".rendered");
const editor = container.querySelector(".editor");
const textarea = editor.querySelector('textarea[name="body"]');
const shadow = editor.querySelector('textarea.shadow-control');
2018-07-16 23:11:53 +03:00
const form = document.getElementById('article-editor');
const cancel = form.querySelector('.cancel');
const footer = document.querySelector("footer");
const lastUpdated = footer.querySelector(".last-updated");
textarea.style.height = rendered.clientHeight + "px";
container.classList.add('edit');
autosizeTextarea(textarea, shadow);
textarea.setSelectionRange(textarea.value.length, textarea.value.length);
textarea.focus();
if (hasBeenOpen) return;
hasBeenOpen = true;
textarea.addEventListener('input', () => autosizeTextarea(textarea, shadow));
window.addEventListener('resize', () => autosizeTextarea(textarea, shadow));
2018-08-22 09:14:25 +03:00
function doSave() {
2017-09-21 13:18:09 +03:00
const body = queryArgsFromForm(form);
textarea.disabled = true;
// TODO Disable other interaction as well: title editor, cancel and OK buttons
2017-09-21 13:18:09 +03:00
fetch(
form.getAttribute("action"),
{
method: 'PUT',
headers: {
"Content-Type": "application/x-www-form-urlencoded"
},
body: body,
credentials: "same-origin",
2017-09-21 13:18:09 +03:00
}
).then(response => {
// I don't know how to more precisely determine that we hit a login redirect:
const probablyLoginRedirect = response.redirected &&
(response.headers.get("content-type") !== "application/json");
if (probablyLoginRedirect) {
return loginDialog(response.url)
.then(() => {
textarea.disabled = false;
});
}
if (!response.ok) throw new Error("Unexpected status code (" + response.status + ")");
return response.json()
.then(result => {
// Update url-bar, page title and footer
window.history.replaceState(null, result.title, result.slug == "" ? "." : result.slug);
2018-09-24 09:43:36 +03:00
// TODO Cancel-link URL should be updated to new slug
document.querySelector("title").textContent = result.title;
lastUpdated.innerHTML = result.last_updated;
lastUpdated.classList.remove("missing");
// Update body:
rendered.innerHTML = result.rendered;
form.elements.title.value = result.title;
shadow.value = textarea.value = result.body;
2018-09-24 09:43:36 +03:00
form.querySelector(`.theme-picker--option[value=${JSON.stringify(result.theme)}]`).checked = true;
bodyElement.className = `theme-${result.theme}`;
// Update form:
form.elements.base_revision.value = result.revision;
for (const element of form.elements) {
element.defaultValue = element.value;
2018-09-24 09:43:36 +03:00
element.defaultChecked = element.checked;
}
if (!result.conflict) {
container.classList.remove('edit');
}
textarea.disabled = false;
autosizeTextarea(textarea, shadow);
if (result.conflict) {
2017-12-01 18:37:41 +03:00
return alertAsync("Your edit came into conflict with another change " +
"and has not been saved.\n" +
"Please resolve the merge conflict and save again.");
}
});
2017-09-21 13:18:09 +03:00
}).catch(err => {
textarea.disabled = false;
console.error(err);
2017-12-01 18:37:41 +03:00
return alertAsync(err.toString());
2017-09-21 13:18:09 +03:00
});
2018-08-22 09:14:25 +03:00
}
2018-08-22 09:14:25 +03:00
function doCancel() {
2017-12-02 13:06:20 +03:00
Promise.resolve(!isEdited(form) || confirmDiscard())
2017-12-01 18:44:07 +03:00
.then(doReset => {
if (doReset) {
container.classList.remove('edit');
form.reset();
}
});
2018-08-22 09:14:25 +03:00
}
form.addEventListener("submit", function (ev) {
ev.preventDefault();
ev.stopPropagation();
doSave();
});
cancel.addEventListener('click', function (ev) {
ev.preventDefault();
ev.stopPropagation();
doCancel();
});
window.addEventListener("beforeunload", function (ev) {
if (isEdited(form)) {
ev.preventDefault();
return ev.returnValue = "Discard changes?";
}
});
2018-08-22 09:14:25 +03:00
document.addEventListener("keypress", function (ev) {
const accel = ev.ctrlKey || ev.metaKey; // Imprecise, but works cross platform
if (ev.key === "Enter" && accel) {
const isEditing = container.classList.contains('edit');
if (!isEditing) return;
2018-08-22 09:14:25 +03:00
ev.stopPropagation();
ev.preventDefault();
doSave();
}
});
2018-09-24 09:43:36 +03:00
const themeOptions = form.querySelectorAll(".theme-picker--option");
for (let themeOption of themeOptions) {
themeOption.addEventListener("click", function (ev) {
bodyElement.className = `theme-${ev.target.value}`;
});
}
}
document
.getElementById("openEditor")
.addEventListener("click", function (ev) {
ev.preventDefault();
ev.stopPropagation();
openEditor();
})
if (document.querySelector(".container").classList.contains("edit")) {
openEditor();
}