mirror of
https://github.com/mauriceboe/TREK.git
synced 2026-06-27 01:01:47 +00:00
801bf0539f
Code-audit clean-ups: share one normCurrency between the router and the templates, lift the duplicated nearest-day resolver into formatters.resolveDayId, drop two needless as-unknown-as casts at the fillBookingWideFields call sites, restore routeExtraction's doc comment, and give the broker template readable names. Plus recognise ¥/JPY and fall back to a standalone symbol amount, so a Klook-style voucher whose price sits far from any label still yields a cost.
158 lines
6.6 KiB
TypeScript
158 lines
6.6 KiB
TypeScript
import type { AssignmentsMap, Day } from '../types'
|
|
|
|
// Collapses verbose Nominatim display_name strings (e.g. "Place, 1, Road, Neighbourhood,
|
|
// City, County, State, Country, Postcode, Country") into "Place, Postcode, Country".
|
|
// Clean short names (≤3 parts) pass through untouched.
|
|
export function formatLocationName(raw: string | null | undefined): string {
|
|
if (!raw) return ''
|
|
const parts = raw.split(',').map(p => p.trim()).filter(Boolean)
|
|
if (parts.length <= 3) return raw.trim()
|
|
|
|
// Dedup preserving insertion order
|
|
const seen = new Set<string>()
|
|
const unique: string[] = []
|
|
for (const p of parts) {
|
|
if (!seen.has(p.toLowerCase())) { seen.add(p.toLowerCase()); unique.push(p) }
|
|
}
|
|
if (unique.length <= 3) return unique.join(', ')
|
|
|
|
const name = unique[0]
|
|
const last = unique[unique.length - 1]
|
|
const secondLast = unique.length >= 2 ? unique[unique.length - 2] : null
|
|
|
|
// Detect postcode at tail: short alphanumeric with at least one digit, ≤10 chars
|
|
const postalRe = /^[A-Z0-9][A-Z0-9\s\-]{1,8}$/i
|
|
const isLastPostal = postalRe.test(last) && /\d/.test(last) && last.length <= 10
|
|
const postcode = isLastPostal ? last : null
|
|
const country = isLastPostal ? secondLast : last
|
|
|
|
const result: string[] = [name]
|
|
if (postcode && postcode !== name) result.push(postcode)
|
|
if (country && country !== name && country !== postcode) result.push(country)
|
|
|
|
return result.join(', ')
|
|
}
|
|
|
|
const ZERO_DECIMAL_CURRENCIES = new Set(['JPY', 'KRW', 'VND', 'CLP', 'ISK', 'HUF'])
|
|
|
|
export function currencyDecimals(currency: string): number {
|
|
return ZERO_DECIMAL_CURRENCIES.has(currency.toUpperCase()) ? 0 : 2
|
|
}
|
|
|
|
// Each currency formats in its own home convention (symbol position, grouping and
|
|
// decimal separators) regardless of the app language — so EUR is always "1.234,56 €"
|
|
// and USD always "$1,234.56". Intl derives all of that from the locale, so we map
|
|
// each supported currency to a representative locale (Latin-digit variants for the
|
|
// Arabic/Bengali ones to avoid non-Latin numerals).
|
|
const CURRENCY_LOCALE: Record<string, string> = {
|
|
EUR: 'de-DE', USD: 'en-US', GBP: 'en-GB', JPY: 'ja-JP', CHF: 'de-CH',
|
|
CZK: 'cs-CZ', PLN: 'pl-PL', SEK: 'sv-SE', NOK: 'nb-NO', DKK: 'da-DK',
|
|
TRY: 'tr-TR', THB: 'th-TH', AUD: 'en-AU', CAD: 'en-CA', NZD: 'en-NZ',
|
|
BRL: 'pt-BR', MXN: 'es-MX', INR: 'en-IN', IDR: 'id-ID', MYR: 'ms-MY',
|
|
PHP: 'en-PH', SGD: 'en-SG', KRW: 'ko-KR', CNY: 'zh-CN', HKD: 'en-HK',
|
|
TWD: 'zh-TW', ZAR: 'en-ZA', AED: 'en-AE', SAR: 'en-SA', ILS: 'he-IL',
|
|
EGP: 'en-EG', MAD: 'fr-MA', HUF: 'hu-HU', RON: 'ro-RO', BGN: 'bg-BG',
|
|
HRK: 'hr-HR', ISK: 'is-IS', RUB: 'ru-RU', UAH: 'uk-UA', BDT: 'en-BD',
|
|
LKR: 'en-LK', VND: 'vi-VN', CLP: 'es-CL', COP: 'es-CO', PEN: 'es-PE',
|
|
ARS: 'es-AR',
|
|
}
|
|
|
|
export function currencyLocale(currency: string): string {
|
|
return CURRENCY_LOCALE[(currency || '').toUpperCase()] || 'en-US'
|
|
}
|
|
|
|
/**
|
|
* Locale- and currency-correct money formatting via Intl: the symbol position,
|
|
* thousands/decimal separators and decimal count all follow the user's locale
|
|
* and the currency itself (e.g. de-DE EUR → "1.234,56 €", en-US USD → "$1,234.56",
|
|
* ja-JP JPY → "¥1,235"). Falls back to a "<number> CODE" suffix for unknown codes.
|
|
*/
|
|
export function formatMoney(
|
|
value: number,
|
|
currency: string,
|
|
locale: string,
|
|
opts?: { decimals?: number },
|
|
): string {
|
|
const cur = (currency || 'EUR').toUpperCase()
|
|
const decimals = opts?.decimals ?? currencyDecimals(cur)
|
|
// Format in the currency's home convention, not the app language, so the symbol
|
|
// position and separators are always correct for that currency. `locale` stays
|
|
// as a last-resort fallback for the error path.
|
|
const fmtLocale = currencyLocale(cur)
|
|
try {
|
|
return new Intl.NumberFormat(fmtLocale, {
|
|
style: 'currency',
|
|
currency: cur,
|
|
minimumFractionDigits: decimals,
|
|
maximumFractionDigits: decimals,
|
|
}).format(value || 0)
|
|
} catch {
|
|
return `${(value || 0).toLocaleString(locale || fmtLocale, { minimumFractionDigits: decimals, maximumFractionDigits: decimals })} ${cur}`
|
|
}
|
|
}
|
|
|
|
export function formatDate(dateStr: string | null | undefined, locale: string, timeZone?: string): string | null {
|
|
if (!dateStr) return null
|
|
const opts: Intl.DateTimeFormatOptions = {
|
|
weekday: 'short', day: 'numeric', month: 'short',
|
|
timeZone: timeZone || 'UTC',
|
|
}
|
|
return new Date(dateStr + 'T00:00:00Z').toLocaleDateString(locale, opts)
|
|
}
|
|
|
|
export function formatTime(timeStr: string | null | undefined, locale: string, timeFormat: string): string {
|
|
if (!timeStr) return ''
|
|
try {
|
|
const parts = timeStr.split(':')
|
|
const h = Number(parts[0]) || 0
|
|
const m = Number(parts[1]) || 0
|
|
if (isNaN(h)) return timeStr
|
|
if (timeFormat === '12h') {
|
|
const period = h >= 12 ? 'PM' : 'AM'
|
|
const h12 = h === 0 ? 12 : h > 12 ? h - 12 : h
|
|
return `${h12}:${String(m).padStart(2, '0')} ${period}`
|
|
}
|
|
const str = `${String(h).padStart(2, '0')}:${String(m).padStart(2, '0')}`
|
|
return locale?.startsWith('de') ? `${str} Uhr` : str
|
|
} catch { return timeStr }
|
|
}
|
|
|
|
export function splitReservationDateTime(value?: string | null): { date: string | null; time: string | null } {
|
|
if (!value) return { date: null, time: null }
|
|
const isoDate = /^\d{4}-\d{2}-\d{2}$/
|
|
if (value.includes('T')) {
|
|
const [d, t] = value.split('T')
|
|
return { date: isoDate.test(d) ? d : null, time: t ? t.slice(0, 5) : null }
|
|
}
|
|
if (isoDate.test(value)) return { date: value, time: null }
|
|
if (/^\d{1,2}:\d{2}/.test(value)) return { date: null, time: value.slice(0, 5) }
|
|
return { date: null, time: null }
|
|
}
|
|
|
|
/**
|
|
* Resolve a date (YYYY-MM-DD or an ISO timestamp) to a trip day id: exact match, else the
|
|
* nearest day so an out-of-range booking still lands on one. Returns '' when there is no
|
|
* usable date or the trip has no days — callers read that as "no day selected".
|
|
*/
|
|
export function resolveDayId(days: Day[], value: string | null | undefined): Day['id'] | '' {
|
|
const date = value ? String(value).slice(0, 10) : ''
|
|
if (!/^\d{4}-\d{2}-\d{2}$/.test(date) || days.length === 0) return ''
|
|
const exact = days.find(d => d.date === date)
|
|
if (exact) return exact.id
|
|
const target = new Date(date).getTime()
|
|
let best: Day['id'] | '' = ''
|
|
let bestDiff = Infinity
|
|
for (const d of days) {
|
|
if (!d.date) continue
|
|
const diff = Math.abs(new Date(d.date).getTime() - target)
|
|
if (diff < bestDiff) { bestDiff = diff; best = d.id }
|
|
}
|
|
return best
|
|
}
|
|
|
|
export function dayTotalCost(dayId: number, assignments: AssignmentsMap, currency: string): string | null {
|
|
const da = assignments[String(dayId)] || []
|
|
const total = da.reduce((s, a) => s + (parseFloat(String(a.place?.price ?? '')) || 0), 0)
|
|
return total > 0 ? `${total.toFixed(0)} ${currency}` : null
|
|
}
|