sync efactura-generator -> 0.9-beta-14

Mirror sincronizat cu repo canonic /workspace/efactura-generator.
CLAUDE.md: documentat workflow sync + exclude config.json din rsync deploy.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
Claude Agent
2026-05-05 13:02:16 +00:00
parent 881881658a
commit 85ccdae2cb
26 changed files with 12188 additions and 5872 deletions

View File

@@ -1,71 +1,71 @@
export class InvoiceFormatter {
constructor() {
this.locale = navigator.language;
this.currencyFormatter = new Intl.NumberFormat(this.locale, {
minimumFractionDigits: 2,
maximumFractionDigits: 2,
useGrouping: true
});
this.quantityFormatter = new Intl.NumberFormat(this.locale, {
minimumFractionDigits: 3,
maximumFractionDigits: 3,
useGrouping: true
});
this.numberFormatter = new Intl.NumberFormat(this.locale, {
minimumFractionDigits: 4,
maximumFractionDigits: 4,
useGrouping: true
});
}
formatCurrency(value) {
const numValue = parseFloat(value);
return isNaN(numValue) ? '0,00' : this.currencyFormatter.format(numValue);
}
formatQuantity(value) {
const numValue = parseFloat(value);
return isNaN(numValue) ? '0,000' : this.quantityFormatter.format(numValue);
}
formatNumber(value) {
const numValue = parseFloat(value);
return isNaN(numValue) ? '0,0000' : this.numberFormatter.format(numValue);
}
parseCurrency(value) {
if (typeof value !== 'string') {
value = value.toString();
}
// Remove all non-digit characters except decimal and minus
const normalized = value.replace(/[^\d\-.,]/g, '')
// Replace thousands separator
.replace(/[.,](?=.*[.,])/g, '')
// Last dot/comma is decimal separator
.replace(/[.,]/, '.');
return parseFloat(normalized) || 0;
}
parseQuantity(value) {
if (typeof value !== 'string') {
value = value.toString();
}
const normalized = value.replace(/[^\d\-.,]/g, '')
.replace(/[.,](?=.*[.,])/g, '')
.replace(/[.,]/, '.');
return parseFloat(normalized) || 0;
}
parseNumber(value) {
if (typeof value !== 'string') {
value = value.toString();
}
const normalized = value.replace(/[^\d\-.,]/g, '')
.replace(/[.,](?=.*[.,])/g, '')
.replace(/[.,]/, '.');
return parseFloat(normalized) || 0;
}
}
// js/formatter.js
//
// Compatibility-layer formatter folosit de print template + script.js
// pentru afișare. Internal delegate la js/numeric.js (PR-E E1+E3+E4).
//
// E2: locale hardcoded "ro-RO" (înlocuit `navigator.language`). Audiența
// țintă e RO; print PDF / display formular trebuie să fie consistent
// între browsere și OS-uri.
import { RO_LOCALE, parseStrict, parseStrictOr, format2, format3, format4 } from './numeric.js';
export class InvoiceFormatter {
constructor() {
this.locale = RO_LOCALE;
this.currencyFormatter = new Intl.NumberFormat(this.locale, {
minimumFractionDigits: 2,
maximumFractionDigits: 2,
useGrouping: true
});
this.quantityFormatter = new Intl.NumberFormat(this.locale, {
minimumFractionDigits: 3,
maximumFractionDigits: 3,
useGrouping: true
});
this.numberFormatter = new Intl.NumberFormat(this.locale, {
minimumFractionDigits: 4,
maximumFractionDigits: 4,
useGrouping: true
});
}
formatCurrency(value) {
const big = (value === '' || value === null || value === undefined)
? null
: parseStrict(value);
return big === null ? '0,00' : format2(big);
}
formatQuantity(value) {
const big = (value === '' || value === null || value === undefined)
? null
: parseStrict(value);
return big === null ? '0,000' : format3(big);
}
formatNumber(value) {
const big = (value === '' || value === null || value === undefined)
? null
: parseStrict(value);
return big === null ? '0,0000' : format4(big);
}
/**
* Strict-but-pragmatic parsing → number (pentru consumatorii vechi).
* Pentru cod nou, preferă `parseStrict` din numeric.js (returnează Big).
*/
parseCurrency(value) {
return Number(parseStrictOr(value, '0').toString());
}
parseQuantity(value) {
return Number(parseStrictOr(value, '0').toString());
}
parseNumber(value) {
return Number(parseStrictOr(value, '0').toString());
}
}