/* The Mia! Accounting Flask Project * journal-entry-form.js: The JavaScript for the journal entry form */ /* Copyright (c) 2023 imacat. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* Author: imacat@mail.imacat.idv.tw (imacat) * First written: 2023/2/25 */ "use strict"; document.addEventListener("DOMContentLoaded", () => { JournalEntryForm.initialize(); }); /** * The journal entry form * */ class JournalEntryForm { /** * The form element * @type {HTMLFormElement} */ #element; /** * The template to add a new line item * @type {string} */ lineItemTemplate; /** * The date * @type {HTMLInputElement} */ #date; /** * The error message of the date * @type {HTMLDivElement} */ #dateError; /** * The control of the currencies * @type {HTMLDivElement} */ #currencyControl; /** * The error message of the currencies * @type {HTMLDivElement} */ #currencyError; /** * The currency list * @type {HTMLDivElement} */ #currencyList; /** * The currency sub-forms * @type {CurrencySubForm[]} */ #currencies; /** * The button to add a new currency * @type {HTMLButtonElement} */ #addCurrencyButton; /** * The note * @type {HTMLTextAreaElement} */ #note; /** * The error message of the note * @type {HTMLDivElement} */ #noteError; /** * The line item editor * @type {JournalEntryLineItemEditor} */ lineItemEditor; /** * Constructs the journal entry form. * */ constructor() { this.#element = document.getElementById("accounting-form"); this.lineItemTemplate = this.#element.dataset.lineItemTemplate; this.#date = document.getElementById("accounting-date"); this.#dateError = document.getElementById("accounting-date-error"); this.#currencyControl = document.getElementById("accounting-currencies"); this.#currencyError = document.getElementById("accounting-currencies-error"); this.#currencyList = document.getElementById("accounting-currency-list"); this.#currencies = Array.from(document.getElementsByClassName("accounting-currency")) .map((element) => new CurrencySubForm(this, element)); this.#addCurrencyButton = document.getElementById("accounting-add-currency"); this.#note = document.getElementById("accounting-note"); this.#noteError = document.getElementById("accounting-note-error"); this.lineItemEditor = new JournalEntryLineItemEditor(this); this.#addCurrencyButton.onclick = () => { const newIndex = 1 + (this.#currencies.length === 0? 0: Math.max(...this.#currencies.map((currency) => currency.index))); const html = this.#element.dataset.currencyTemplate .replaceAll("CURRENCY_INDEX", escapeHtml(String(newIndex))); this.#currencyList.insertAdjacentHTML("beforeend", html); const element = document.getElementById("accounting-currency-" + String(newIndex)); this.#currencies.push(new CurrencySubForm(this, element)); this.#resetDeleteCurrencyButtons(); this.#initializeDragAndDropReordering(); }; this.#resetDeleteCurrencyButtons(); this.#initializeDragAndDropReordering(); this.#date.onchange = () => this.#validateDate(); this.#note.onchange = () => this.#validateNote(); this.#element.onsubmit = () => { return this.#validate(); }; } /** * Deletes a currency sub-form. * * @param currency {CurrencySubForm} the currency sub-form to delete */ deleteCurrency(currency) { const index = this.#currencies.indexOf(currency); this.#currencies.splice(index, 1); this.#resetDeleteCurrencyButtons(); } /** * Resets the buttons to delete the currency sub-forms * */ #resetDeleteCurrencyButtons() { if (this.#currencies.length === 1) { this.#currencies[0].deleteButton.classList.add("d-none"); } else { for (const currency of this.#currencies) { let isAnyLineItemMatched = false; for (const lineItem of currency.getLineItems()) { if (lineItem.isMatched) { isAnyLineItemMatched = true; break; } } if (isAnyLineItemMatched) { currency.deleteButton.classList.add("d-none"); } else { currency.deleteButton.classList.remove("d-none"); } } } } /** * Initializes the drag and drop reordering on the currency sub-forms. * */ #initializeDragAndDropReordering() { initializeDragAndDropReordering(this.#currencyList, () => { const currencyId = Array.from(this.#currencyList.children).map((currency) => currency.id); this.#currencies.sort((a, b) => currencyId.indexOf(a.element.id) - currencyId.indexOf(b.element.id)); for (let i = 0; i < this.#currencies.length; i++) { this.#currencies[i].no.value = String(i + 1); } }); } /** * Returns all the line items in the form. * * @param debitCredit {string|null} Either "debit" or "credit", or null for both * @return {LineItemSubForm[]} all the line item sub-forms */ getLineItems(debitCredit = null) { const lineItems = []; for (const currency of this.#currencies) { lineItems.push(...currency.getLineItems(debitCredit)); } return lineItems; } /** * Returns the account codes used in the form. * * @param debitCredit {string} either "debit" or "credit" * @return {string[]} the account codes used in the form */ getAccountCodesUsed(debitCredit) { return this.getLineItems(debitCredit).map((lineItem) => lineItem.getAccountCode()) .filter((code) => code !== null); } /** * Returns the date. * * @return {string} the date */ getDate() { return this.#date.value; } /** * Updates the minimal date. * */ updateMinDate() { let lastOriginalLineItemDate = null; for (const lineItem of this.getLineItems()) { const date = lineItem.getOriginalLineItemDate(); if (date !== null) { if (lastOriginalLineItemDate === null || lastOriginalLineItemDate < date) { lastOriginalLineItemDate = date; } } } this.#date.min = lastOriginalLineItemDate === null? "": lastOriginalLineItemDate; this.#validateDate(); } /** * Validates the form. * * @returns {boolean} true if valid, or false otherwise */ #validate() { let isValid = true; isValid = this.#validateDate() && isValid; isValid = this.#validateCurrencies() && isValid; isValid = this.#validateNote() && isValid; return isValid; } /** * Validates the date. * * @returns {boolean} true if valid, or false otherwise */ #validateDate() { this.#date.value = this.#date.value.trim(); this.#date.classList.remove("is-invalid"); if (this.#date.value === "") { this.#date.classList.add("is-invalid"); this.#dateError.innerText = A_("Please fill in the date."); return false; } if (this.#date.value < this.#date.min) { this.#date.classList.add("is-invalid"); this.#dateError.innerText = A_("The date cannot be earlier than the original line items."); return false; } this.#date.classList.remove("is-invalid"); this.#dateError.innerText = ""; return true; } /** * Validates the currencies. * * @returns {boolean} true if valid, or false otherwise */ #validateCurrencies() { let isValid = true; isValid = this.#validateCurrenciesReal() && isValid; for (const currency of this.#currencies) { isValid = currency.validate() && isValid; } return isValid; } /** * Validates the currency sub-forms, the validator itself. * * @returns {boolean} true if valid, or false otherwise */ #validateCurrenciesReal() { if (this.#currencies.length === 0) { this.#currencyControl.classList.add("is-invalid"); this.#currencyError.innerText = A_("Please add some currencies."); return false; } this.#currencyControl.classList.remove("is-invalid"); this.#currencyError.innerText = ""; return true; } /** * Validates the note. * * @returns {boolean} true if valid, or false otherwise */ #validateNote() { this.#note.value = this.#note.value .replace(/^\s*\n/, "") .trimEnd(); this.#note.classList.remove("is-invalid"); this.#noteError.innerText = ""; return true; } /** * The journal entry form * @type {JournalEntryForm} */ static #form; /** * Initializes the journal entry form. * */ static initialize() { this.#form = new JournalEntryForm() } } /** * The currency sub-form. * */ class CurrencySubForm { /** * The element * @type {HTMLDivElement} */ element; /** * The journal entry form * @type {JournalEntryForm} */ form; /** * The currency index * @type {number} */ index; /** * The prefix of the HTML ID and class * @type {string} */ #prefix; /** * The control * @type {HTMLDivElement} */ #control; /** * The error message * @type {HTMLDivElement} */ #error; /** * The number * @type {HTMLInputElement} */ no; /** * The currency code * @type {HTMLInputElement} */ #code; /** * The currency code selector * @type {HTMLSelectElement} */ #codeSelect; /** * The button to delete the currency * @type {HTMLButtonElement} */ deleteButton; /** * The debit sub-form * @type {DebitCreditSubForm|null} */ #debit; /** * The credit sub-form * @type {DebitCreditSubForm|null} */ #credit; /** * Constructs a currency sub-form * * @param form {JournalEntryForm} the journal entry form * @param element {HTMLDivElement} the currency sub-form element */ constructor(form, element) { this.element = element; this.form = form; this.index = parseInt(this.element.dataset.index); this.#prefix = "accounting-currency-" + String(this.index); this.#control = document.getElementById(this.#prefix + "-control"); this.#error = document.getElementById(this.#prefix + "-error"); this.no = document.getElementById(this.#prefix + "-no"); this.#code = document.getElementById(this.#prefix + "-code"); this.#codeSelect = document.getElementById(this.#prefix + "-code-select"); this.deleteButton = document.getElementById(this.#prefix + "-delete"); const debitElement = document.getElementById(this.#prefix + "-debit"); this.#debit = debitElement === null? null: new DebitCreditSubForm(this, debitElement, "debit"); const creditElement = document.getElementById(this.#prefix + "-credit"); this.#credit = creditElement == null? null: new DebitCreditSubForm(this, creditElement, "credit"); this.#codeSelect.onchange = () => this.#code.value = this.#codeSelect.value; this.deleteButton.onclick = () => { this.element.parentElement.removeChild(this.element); this.form.deleteCurrency(this); }; } /** * Returns the currency code. * * @return {string} the currency code */ getCurrencyCode() { return this.#code.value; } /** * Returns all the line items in the form. * * @param debitCredit {string|null} either "debit" or "credit", or null for both * @return {LineItemSubForm[]} all the line item sub-forms */ getLineItems(debitCredit = null) { const lineItems = [] for (const debitCreditSubForm of [this.#debit, this.#credit]) { if (debitCreditSubForm !== null ) { if (debitCredit === null || debitCreditSubForm.debitCredit === debitCredit) { lineItems.push(...debitCreditSubForm.lineItems); } } } return lineItems; } /** * Updates whether to enable the currency code selector * */ updateCodeSelectorStatus() { let isEnabled = true; for (const lineItem of this.getLineItems()) { if (lineItem.getOriginalLineItemId() !== null) { isEnabled = false; break; } } this.#codeSelect.disabled = !isEnabled; } /** * Validates the form. * * @returns {boolean} true if valid, or false otherwise */ validate() { let isValid = true; if (this.#debit !== null) { isValid = this.#debit.validate() && isValid; } if (this.#credit !== null) { isValid = this.#credit.validate() && isValid; } isValid = this.validateBalance() && isValid; return isValid; } /** * Validates the valance. * * @returns {boolean} true if valid, or false otherwise */ validateBalance() { if (this.#debit !== null && this.#credit !== null) { if (!this.#debit.getTotal().equals(this.#credit.getTotal())) { this.#control.classList.add("is-invalid"); this.#error.innerText = A_("The totals of the debit and credit amounts do not match."); return false; } } this.#control.classList.remove("is-invalid"); this.#error.innerText = ""; return true; } } /** * The debit or credit sub-form * */ class DebitCreditSubForm { /** * The currency sub-form * @type {CurrencySubForm} */ currency; /** * The element * @type {HTMLDivElement} */ #element; /** * The currencyIndex * @type {number} */ #currencyIndex; /** * Either "debit" or "credit" * @type {string} */ debitCredit; /** * The prefix of the HTML ID and class * @type {string} */ #prefix; /** * The error message * @type {HTMLDivElement} */ #error; /** * The line item list * @type {HTMLUListElement} */ #lineItemList; /** * The line item sub-forms * @type {LineItemSubForm[]} */ lineItems; /** * The total * @type {HTMLSpanElement} */ #total; /** * The button to add a new line item * @type {HTMLButtonElement} */ #addLineItemButton; /** * Constructs a debit or credit sub-form * * @param currency {CurrencySubForm} the currency sub-form * @param element {HTMLDivElement} the element * @param debitCredit {string} either "debit" or "credit" */ constructor(currency, element, debitCredit) { this.currency = currency; this.#element = element; this.#currencyIndex = currency.index; this.debitCredit = debitCredit; this.#prefix = "accounting-currency-" + String(this.#currencyIndex) + "-" + debitCredit; this.#error = document.getElementById(this.#prefix + "-error"); this.#lineItemList = document.getElementById(this.#prefix + "-list"); // noinspection JSValidateTypes this.lineItems = Array.from(document.getElementsByClassName(this.#prefix)).map((element) => new LineItemSubForm(this, element)); this.#total = document.getElementById(this.#prefix + "-total"); this.#addLineItemButton = document.getElementById(this.#prefix + "-add-line-item"); this.#addLineItemButton.onclick = () => this.currency.form.lineItemEditor.onAddNew(this); this.#resetDeleteLineItemButtons(); this.#initializeDragAndDropReordering(); } /** * Adds a new line item sub-form * * @returns {LineItemSubForm} the newly-added line item sub-form */ addLineItem() { const newIndex = 1 + (this.lineItems.length === 0? 0: Math.max(...this.lineItems.map((lineItem) => lineItem.lineItemIndex))); const html = this.currency.form.lineItemTemplate .replaceAll("CURRENCY_INDEX", escapeHtml(String(this.#currencyIndex))) .replaceAll("DEBIT_CREDIT", escapeHtml(this.debitCredit)) .replaceAll("LINE_ITEM_INDEX", escapeHtml(String(newIndex))); this.#lineItemList.insertAdjacentHTML("beforeend", html); const lineItem = new LineItemSubForm(this, document.getElementById(this.#prefix + "-" + String(newIndex))); this.lineItems.push(lineItem); this.#resetDeleteLineItemButtons(); this.#initializeDragAndDropReordering(); this.validate(); return lineItem; } /** * Deletes a line item sub-form * * @param lineItem {LineItemSubForm} */ deleteLineItem(lineItem) { const index = this.lineItems.indexOf(lineItem); this.lineItems.splice(index, 1); this.updateTotal(); this.currency.updateCodeSelectorStatus(); this.currency.form.updateMinDate(); this.#resetDeleteLineItemButtons(); } /** * Resets the buttons to delete the line item sub-forms * */ #resetDeleteLineItemButtons() { if (this.lineItems.length === 1) { this.lineItems[0].deleteButton.classList.add("d-none"); } else { for (const lineItem of this.lineItems) { if (lineItem.isMatched) { lineItem.deleteButton.classList.add("d-none"); } else { lineItem.deleteButton.classList.remove("d-none"); } } } } /** * Returns the total amount. * * @return {Decimal} the total amount */ getTotal() { let total = new Decimal("0"); for (const lineItem of this.lineItems) { const amount = lineItem.getAmount(); if (amount !== null) { total = total.plus(amount); } } return total; } /** * Updates the total * */ updateTotal() { this.#total.innerText = formatDecimal(this.getTotal()); this.currency.validateBalance(); } /** * Initializes the drag and drop reordering on the currency sub-forms. * */ #initializeDragAndDropReordering() { initializeDragAndDropReordering(this.#lineItemList, () => { const lineItemId = Array.from(this.#lineItemList.children).map((lineItem) => lineItem.id); this.lineItems.sort((a, b) => lineItemId.indexOf(a.element.id) - lineItemId.indexOf(b.element.id)); for (let i = 0; i < this.lineItems.length; i++) { this.lineItems[i].no.value = String(i + 1); } }); } /** * Validates the form. * * @returns {boolean} true if valid, or false otherwise */ validate() { let isValid = true; isValid = this.#validateReal() && isValid; for (const lineItem of this.lineItems) { isValid = lineItem.validate() && isValid; } return isValid; } /** * Validates the form, the validator itself. * * @returns {boolean} true if valid, or false otherwise */ #validateReal() { if (this.lineItems.length === 0) { this.#element.classList.add("is-invalid"); this.#error.innerText = A_("Please add some line items."); return false; } this.#element.classList.remove("is-invalid"); this.#error.innerText = ""; return true; } } /** * The line item sub-form. * */ class LineItemSubForm { /** * The debit or credit sub-form * @type {DebitCreditSubForm} */ debitCreditSubForm; /** * The element * @type {HTMLLIElement} */ element; /** * Either "debit" or "credit" * @type {string} */ debitCredit; /** * The line item index * @type {number} */ lineItemIndex; /** * Whether this is an original line item with offsets * @type {boolean} */ isMatched; /** * The prefix of the HTML ID and class * @type {string} */ #prefix; /** * The control * @type {HTMLDivElement} */ #control; /** * The error message * @type {HTMLDivElement} */ #error; /** * The number * @type {HTMLInputElement} */ no; /** * The account code * @type {HTMLInputElement} */ #accountCode; /** * The text display of the account * @type {HTMLDivElement} */ #accountText; /** * The description * @type {HTMLInputElement} */ #description; /** * The text display of the description * @type {HTMLDivElement} */ #descriptionText; /** * The ID of the original line item * @type {HTMLInputElement} */ #originalLineItemId; /** * The text of the original line item * @type {HTMLDivElement} */ #originalLineItemText; /** * The offset items * @type {HTMLInputElement} */ #offsets; /** * The amount * @type {HTMLInputElement} */ #amount; /** * The text display of the amount * @type {HTMLSpanElement} */ #amountText; /** * The button to delete line item * @type {HTMLButtonElement} */ deleteButton; /** * Constructs the line item sub-form. * * @param debitCredit {DebitCreditSubForm} the debit or credit sub-form * @param element {HTMLLIElement} the element */ constructor(debitCredit, element) { this.debitCreditSubForm = debitCredit; this.element = element; this.debitCredit = element.dataset.debitCredit; this.lineItemIndex = parseInt(element.dataset.lineItemIndex); this.isMatched = element.classList.contains("accounting-matched-line-item"); this.#prefix = "accounting-currency-" + element.dataset.currencyIndex + "-" + this.debitCredit + "-" + this.lineItemIndex; this.#control = document.getElementById(this.#prefix + "-control"); this.#error = document.getElementById(this.#prefix + "-error"); this.no = document.getElementById(this.#prefix + "-no"); this.#accountCode = document.getElementById(this.#prefix + "-account-code"); this.#accountText = document.getElementById(this.#prefix + "-account-text"); this.#description = document.getElementById(this.#prefix + "-description"); this.#descriptionText = document.getElementById(this.#prefix + "-description-text"); this.#originalLineItemId = document.getElementById(this.#prefix + "-original-line-item-id"); this.#originalLineItemText = document.getElementById(this.#prefix + "-original-line-item-text"); this.#offsets = document.getElementById(this.#prefix + "-offsets"); this.#amount = document.getElementById(this.#prefix + "-amount"); this.#amountText = document.getElementById(this.#prefix + "-amount-text"); this.deleteButton = document.getElementById(this.#prefix + "-delete"); this.#control.onclick = () => this.debitCreditSubForm.currency.form.lineItemEditor.onEdit(this); this.deleteButton.onclick = () => { this.element.parentElement.removeChild(this.element); this.debitCreditSubForm.deleteLineItem(this); }; } /** * Returns whether the line item needs offset. * * @return {boolean} true if the line item needs offset, or false otherwise */ isNeedOffset() { return "isNeedOffset" in this.element.dataset; } /** * Returns the ID of the original line item. * * @return {string|null} the ID of the original line item */ getOriginalLineItemId() { return this.#originalLineItemId.value === ""? null: this.#originalLineItemId.value; } /** * Returns the date of the original line item. * * @return {string|null} the date of the original line item */ getOriginalLineItemDate() { return this.#originalLineItemId.dataset.date === ""? null: this.#originalLineItemId.dataset.date; } /** * Returns the text of the original line item. * * @return {string|null} the text of the original line item */ getOriginalLineItemText() { return this.#originalLineItemId.dataset.text === ""? null: this.#originalLineItemId.dataset.text; } /** * Returns the description. * * @return {string|null} the description */ getDescription() { return this.#description.value === ""? null: this.#description.value; } /** * Returns the account code. * * @return {string|null} the account code */ getAccountCode() { return this.#accountCode.value === ""? null: this.#accountCode.value; } /** * Returns the account text. * * @return {string|null} the account text */ getAccountText() { return this.#accountCode.dataset.text === ""? null: this.#accountCode.dataset.text; } /** * Returns the amount. * * @return {Decimal|null} the amount */ getAmount() { return this.#amount.value === ""? null: new Decimal(this.#amount.value); } /** * Returns the minimal amount. * * @return {Decimal|null} the minimal amount */ getAmountMin() { return this.#amount.dataset.min === ""? null: new Decimal(this.#amount.dataset.min); } /** * Validates the form. * * @returns {boolean} true if valid, or false otherwise */ validate() { if (this.#accountCode.value === "") { this.#control.classList.add("is-invalid"); this.#error.innerText = A_("Please select the account."); return false; } if (this.#amount.value === "") { this.#control.classList.add("is-invalid"); this.#error.innerText = A_("Please fill in the amount."); return false; } this.#control.classList.remove("is-invalid"); this.#error.innerText = ""; return true; } /** * Stores the data into the line item sub-form. * * @param editor {JournalEntryLineItemEditor} the line item editor */ save(editor) { if (editor.isNeedOffset) { this.#offsets.classList.remove("d-none"); } else { this.#offsets.classList.add("d-none"); } this.#originalLineItemId.value = editor.originalLineItemId === null? "": editor.originalLineItemId; this.#originalLineItemId.dataset.date = editor.originalLineItemDate === null? "": editor.originalLineItemDate; this.#originalLineItemId.dataset.text = editor.originalLineItemText === null? "": editor.originalLineItemText; if (editor.originalLineItemText === null) { this.#originalLineItemText.classList.add("d-none"); this.#originalLineItemText.innerText = ""; } else { this.#originalLineItemText.classList.remove("d-none"); this.#originalLineItemText.innerText = A_("Offset %(item)s", {item: editor.originalLineItemText}); } this.#accountCode.value = editor.accountCode === null? "": editor.accountCode; this.#accountCode.dataset.text = editor.accountText === null? "": editor.accountText; this.#accountText.innerText = editor.accountText === null? "": editor.accountText; this.#description.value = editor.description === null? "": editor.description; this.#descriptionText.innerText = editor.description === null? "": editor.description; this.#amount.value = editor.amount; this.#amountText.innerText = formatDecimal(new Decimal(editor.amount)); this.validate(); this.debitCreditSubForm.updateTotal(); this.debitCreditSubForm.currency.updateCodeSelectorStatus(); this.debitCreditSubForm.currency.form.updateMinDate(); } } /** * Escapes the HTML special characters and returns. * * @param s {string} the original string * @returns {string} the string with HTML special character escaped * @private */ function escapeHtml(s) { return String(s) .replaceAll("&", "&") .replaceAll("<", "<") .replaceAll(">", ">") .replaceAll("\"", """); } /** * Formats a Decimal number. * * @param number {Decimal} the Decimal number * @returns {string} the formatted Decimal number */ function formatDecimal(number) { if (number.equals(new Decimal("0"))) { return "-"; } const frac = number.modulo(1); const whole = Number(number.minus(frac)).toLocaleString(); return whole + String(frac).substring(1); }