/** * content for invoices */ import { DeesElement, property, html, customElement, type TemplateResult, css, render, domtools, } from "@design.estate/dees-element"; import * as plugins from "../plugins.js"; import { dedocumentSharedStyle } from "../style.js"; import type { TranslationKey } from "ts_shared/translation.js"; declare global { interface HTMLElementTagNameMap { "dedocument-contentinvoice": DeContentInvoice; } } @customElement("dedocument-contentinvoice") export class DeContentInvoice extends DeesElement { public static demo = () => html`
`; @property({ type: Object, reflect: true, }) public letterData: plugins.tsclass.finance.TInvoice; @property({ type: Object, reflect: true, }) public documentSettings: plugins.shared.interfaces.IDocumentSettings; constructor() { super(); domtools.DomTools.setupDomTools(); } public static styles = [ domtools.elementBasic.staticStyles, dedocumentSharedStyle, css` .trimmedContent { display: none; } .grid { display: grid; grid-template-columns: 40px auto 50px 50px 100px 50px 100px; } .topLine { margin-top: 5px; background: #e7e7e7; font-weight: bold; } .lineItem { font-size: 12px; padding: 5px; border-right: 1px dashed #ccc; white-space: nowrap; } .lineItem:last-child { border-right: none; } .value.rightAlign, .lineItem.rightAlign { text-align: right; } .invoiceLine { background: #ffffff00; border-bottom: 1px dotted #ccc; } .invoiceLine.highlighted { transition: background 0.2s; background: #ffc18f; border: 1px solid #ff9d4d; box-sizing: content-box; } .sums { margin-top: 5px; font-size: 12px; padding-left: 50%; } .sums .sumline { margin-top: 3px; display: grid; grid-template-columns: auto 100px; } .sums .sumline .label { padding: 2px 5px; border-right: 1px solid #ccc; text-align: right; font-weight: bold; } .sums .sumline .value { padding: 2px 5px; } .sums .sumline .value--total { font-weight: bold; } .divider { margin-top: 8px; border-top: 1px dotted #ccc; } .taxNote { font-size: 12px; padding: 4px; background: #eeeeeb; text-align: center; } .infoBox { margin-top: 22px; line-height: 1.4em; } .infoBox .label { padding-bottom: 2px; font-weight: bold; } `, ]; public render(): TemplateResult { return html`
`; } protected formatPrice( value: number, currency = "EUR", lang = "de-DE" ): string { return new Intl.NumberFormat(lang, { style: "currency", currency, }).format(value); } public getTotalNet = (): number => { let totalNet = 0; if (!this.letterData) { return totalNet; } for (const item of this.letterData.items) { totalNet += item.unitNetPrice * item.unitQuantity; } return totalNet; }; public getTotalGross = (): number => { let totalVat = 0; if (!this.letterData) { return totalVat; } for (const taxgroup of this.getVatGroups()) { totalVat += taxgroup.vatAmountSum; } return this.getTotalNet() + totalVat; }; public getVatGroups = () => { const vatGroups: { vatPercentage: number; items: plugins.tsclass.finance.TInvoice["items"]; vatAmountSum: number; }[] = []; if (!this.letterData) { return vatGroups; } const taxAmounts: number[] = []; for (const item of this.letterData.items) { taxAmounts.includes(item.vatPercentage) ? null : taxAmounts.push(item.vatPercentage); } for (const taxAmount of taxAmounts) { const matchingItems = this.letterData.items.filter( (itemArg) => itemArg.vatPercentage === taxAmount ); let sum = 0; for (const matchingItem of matchingItems) { sum += matchingItem.unitNetPrice * matchingItem.unitQuantity * (taxAmount / 100); } vatGroups.push({ items: matchingItems, vatPercentage: taxAmount, vatAmountSum: Math.round(sum * 100) / 100, }); } return vatGroups.sort((a, b) => b.vatPercentage - a.vatPercentage); }; public async getContentNodes() { await this.elementDomReady; return { currentContent: this.shadowRoot.querySelector( ".currentContent" ) as HTMLElement, trimmedContent: this.shadowRoot.querySelector( ".trimmedContent" ) as HTMLElement, repeatedContent: this.shadowRoot.querySelector( ".repeatedContent" ) as HTMLElement, }; } public async getContentLength() { await this.elementDomReady; return (await this.getContentNodes()).currentContent.children.length; } public async trimEndByOne() { await this.elementDomReady; this.shadowRoot .querySelector(".trimmedContent") .append( (await this.getContentNodes()).currentContent.children.item( (await this.getContentNodes()).currentContent.children.length - 1 ) ); } public async trimStartToOffset(contentOffsetArg: number) { await this.elementDomReady; const beginningLength = (await this.getContentNodes()).currentContent .children.length; while ( (await this.getContentNodes()).currentContent.children.length !== beginningLength - contentOffsetArg ) { (await this.getContentNodes()).trimmedContent.append( (await this.getContentNodes()).currentContent.children.item(0) ); } if ( (await this.getContentNodes()).currentContent.children .item(0) .classList.contains("needsDataHeader") ) { const trimmedContent = (await this.getContentNodes()).trimmedContent; let startPoint = trimmedContent.children.length; while (startPoint > 0) { const element = trimmedContent.children.item(startPoint - 1); if (element.classList.contains("dataHeader")) { (await this.getContentNodes()).repeatedContent.append(element); break; } startPoint--; } } } public translateKey(key: TranslationKey): string { return plugins.shared.translation.translate( this.documentSettings.languageCode, key ); } public async firstUpdated( _changedProperties: Map ) { super.firstUpdated(_changedProperties); this.attachInvoiceDom(); } private renderPaymentTerms(): TemplateResult { return html`
${this.translateKey("invoice@@payment.terms")}
${this.translateKey("invoice@@payment.terms.direct")} ${new Intl.DateTimeFormat(this.documentSettings.languageCode, { dateStyle: this.documentSettings.dateStyle, }).format( new Date(this.letterData.date).setDate( new Date(this.letterData.date).getDate() + this.letterData?.dueInDays ) )}
`; } private renderPaymentInfo(): TemplateResult { const bic = this.letterData?.from.sepaConnection.bic; const name = this.letterData?.from.name; const iban = this.letterData?.from.sepaConnection.iban; const currency = this.letterData?.currency; const totalGross = this.getTotalGross(); const reference = this.letterData?.id; return html`
${this.translateKey("invoice@@payment.qr")}
${this.translateKey("invoice@@payment.qr.description")}
`; } private renderReferencedContract(): TemplateResult { return null; // return this.documentSettings.enableInvoiceContractRefSection && // this.invoiceData?.content?.contractData?.contractDate // ? html` //
//
// ${this.translateKey("invoice@@referencedContract")} //
// ${this.translateKey("invoice@@referencedContract.text")} // ${new Intl.DateTimeFormat(this.documentSettings.languageCode, { // dateStyle: this.documentSettings.dateStyle, // }).format( // new Date(this.invoiceData?.content.contractData.contractDate) // )}. //
// ` // : null; } public async attachInvoiceDom() { const contentNodes = await this.getContentNodes(); render( html`
${this.translateKey("invoice@@introStatement")}
${this.translateKey("invoice@@item.position")}
${this.translateKey("invoice@@description")}
${this.translateKey("invoice@@quantity")}
${this.translateKey("invoice@@unit.type")}
${this.translateKey("invoice@@price.unit.net")}
${this.translateKey("invoice@@vat.short")}
${this.translateKey("invoice@@price.total.net")}
${this.letterData?.items?.map( (invoiceItem, index) => html`
${index + 1}
${invoiceItem.name}
${invoiceItem.unitQuantity}
${invoiceItem.unitType}
${this.formatPrice(invoiceItem.unitNetPrice)}
${invoiceItem.vatPercentage}%
${this.formatPrice( invoiceItem.unitQuantity * invoiceItem.unitNetPrice )}
` )}
${this.translateKey("invoice@@sum.total.net")}
${this.formatPrice(this.getTotalNet())}
${this.getVatGroups().map((vatGroupArg) => { let itemNumbers = vatGroupArg.items .map((item) => this.letterData.items.indexOf(item) + 1) .join(", "); return html`
${this.translateKey("vat.short")} ${vatGroupArg.vatPercentage}% ${this.documentSettings.vatGroupPositions ? html`
(${this.translateKey("invoice@@vat.position")}: ${itemNumbers}) ` : html``}
${this.formatPrice(vatGroupArg.vatAmountSum)}
`; })}
${this.translateKey("invoice@@totalGross")}
${this.formatPrice(this.getTotalGross())}
${this.letterData?.reverseCharge ? html`
${this.translateKey("invoice@@vat.reverseCharge.note")}
` : ``} ${this.renderReferencedContract()} ${this.renderPaymentTerms()} ${this.renderPaymentInfo()} `, contentNodes.currentContent ); } }