2024-11-30 09:55:59 +00:00
|
|
|
function initErrorDialog() {
|
|
|
|
const dialog = document.querySelector("dialog#error");
|
|
|
|
const message = document.querySelector("textarea#error-message");
|
|
|
|
const close = document.querySelector("button#error-close");
|
|
|
|
|
|
|
|
close.addEventListener("click", () => dialog.close());
|
|
|
|
|
|
|
|
return (err) => {
|
|
|
|
message.value = err.toString();
|
|
|
|
dialog.showModal();
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function initEnterPassword(errorDialog) {
|
2024-11-10 09:46:40 +00:00
|
|
|
const form = document.querySelector("form#enter-password");
|
|
|
|
const fieldset = document.querySelector("fieldset#enter-password");
|
|
|
|
const submit = document.querySelector("button#enter-password");
|
|
|
|
|
|
|
|
const dialog = document.querySelector("dialog#link");
|
|
|
|
const link = document.querySelector("input#link");
|
|
|
|
const copy = document.querySelector("button#link-copy");
|
|
|
|
const close = document.querySelector("button#link-close");
|
|
|
|
|
|
|
|
form.addEventListener("submit", async (ev) => {
|
|
|
|
const data = new FormData(ev.target);
|
|
|
|
ev.preventDefault();
|
|
|
|
|
|
|
|
try {
|
|
|
|
fieldset.disabled = true;
|
|
|
|
submit.ariaBusy = "true";
|
|
|
|
|
|
|
|
const password = await encryptPassword(data.get("password"));
|
|
|
|
|
|
|
|
const id = await uploadPassword(
|
|
|
|
password.password,
|
|
|
|
parseInt(data.get("expires-in")),
|
|
|
|
);
|
|
|
|
|
|
|
|
const url = new URL(window.location.toString());
|
|
|
|
url.hash = [id, password.key, password.iv].join(":");
|
|
|
|
|
|
|
|
link.value = url.toString();
|
|
|
|
dialog.showModal();
|
2024-11-30 09:55:59 +00:00
|
|
|
} catch (error) {
|
|
|
|
errorDialog(error);
|
2024-11-10 09:46:40 +00:00
|
|
|
} finally {
|
|
|
|
fieldset.disabled = false;
|
|
|
|
submit.ariaBusy = "false";
|
|
|
|
ev.target.reset();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
dialog.addEventListener("close", (ev) => {
|
|
|
|
this.link.value = "";
|
|
|
|
});
|
|
|
|
|
|
|
|
copy.addEventListener("click", (ev) => {
|
|
|
|
link.select();
|
|
|
|
link.setSelectionRange(0, 99999);
|
|
|
|
navigator.clipboard.writeText(link.value);
|
|
|
|
});
|
|
|
|
|
|
|
|
close.addEventListener("click", (ev) => {
|
|
|
|
dialog.close();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-11-30 09:55:59 +00:00
|
|
|
async function confirmViewPassword(errorDialog) {
|
2024-11-10 09:46:40 +00:00
|
|
|
const dialog = document.querySelector("dialog#confirm");
|
|
|
|
const close = document.querySelector("button#confirm-close");
|
|
|
|
const ok = document.querySelector("button#confirm-ok");
|
|
|
|
const notFoundDialog = document.querySelector("dialog#not-found");
|
|
|
|
const notFoundClose = document.querySelector("button#not-found-close");
|
|
|
|
|
|
|
|
let hash = window.location.hash;
|
|
|
|
hash = hash.substring(1);
|
|
|
|
if (hash.trim() == "") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const [id, key, iv] = hash.split(":");
|
|
|
|
const has = await hasPassword(id);
|
|
|
|
if (!has) {
|
|
|
|
notFoundClose.addEventListener("click", () => notFoundDialog.close());
|
|
|
|
notFoundDialog.addEventListener("close", () => (window.location.hash = ""));
|
|
|
|
|
|
|
|
notFoundDialog.showModal();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
dialog.addEventListener("close", () => (window.location.hash = ""));
|
|
|
|
|
|
|
|
close.addEventListener("click", () => {
|
|
|
|
dialog.close();
|
|
|
|
});
|
|
|
|
|
|
|
|
ok.addEventListener("click", async () => {
|
|
|
|
try {
|
|
|
|
close.disabled = true;
|
|
|
|
ok.disabled = true;
|
|
|
|
ok.ariaBusy = "true";
|
|
|
|
|
|
|
|
const encryptedPassword = await getPassword(id);
|
|
|
|
const password = await decryptPassword(encryptedPassword, key, iv);
|
|
|
|
|
|
|
|
viewPassword(password);
|
2024-11-30 09:55:59 +00:00
|
|
|
} catch (error) {
|
|
|
|
errorDialog(error);
|
2024-11-10 09:46:40 +00:00
|
|
|
} finally {
|
|
|
|
close.disabled = false;
|
|
|
|
ok.disabled = false;
|
|
|
|
ok.ariaBusy = "false";
|
|
|
|
dialog.close();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
dialog.showModal();
|
|
|
|
}
|
|
|
|
|
|
|
|
async function viewPassword(password) {
|
|
|
|
const dialog = document.querySelector("dialog#view");
|
|
|
|
const viewPassword = document.querySelector("textarea#view-password");
|
|
|
|
const close = document.querySelector("button#view-close");
|
|
|
|
|
|
|
|
close.addEventListener("click", () => dialog.close());
|
|
|
|
dialog.addEventListener("close", () => (window.location.hash = ""), 0);
|
|
|
|
|
|
|
|
viewPassword.value = password;
|
|
|
|
dialog.showModal();
|
|
|
|
}
|
|
|
|
|
|
|
|
window.addEventListener("load", () => {
|
2024-11-30 09:55:59 +00:00
|
|
|
const errorDialog = initErrorDialog();
|
|
|
|
initEnterPassword(errorDialog);
|
|
|
|
confirmViewPassword(errorDialog);
|
2024-11-10 09:46:40 +00:00
|
|
|
});
|