import BaseUIElement from "../UI/BaseUIElement"; import {FixedUiElement} from "../UI/Base/FixedUiElement"; import Combine from "../UI/Base/Combine"; import {Denomination} from "./Denomination"; import UnitConfigJson from "./ThemeConfig/Json/UnitConfigJson"; export class Unit { public readonly appliesToKeys: Set; public readonly denominations: Denomination[]; public readonly denominationsSorted: Denomination[]; public readonly defaultDenom: Denomination; public readonly eraseInvalid: boolean; private readonly possiblePostFixes: string[] = [] constructor(appliesToKeys: string[], applicableUnits: Denomination[], eraseInvalid: boolean) { this.appliesToKeys = new Set(appliesToKeys); this.denominations = applicableUnits; this.defaultDenom = applicableUnits.filter(denom => denom.default)[0] this.eraseInvalid = eraseInvalid const seenUnitExtensions = new Set(); for (const denomination of this.denominations) { if (seenUnitExtensions.has(denomination.canonical)) { throw "This canonical unit is already defined in another denomination: " + denomination.canonical } const duplicate = denomination.alternativeDenominations.filter(denom => seenUnitExtensions.has(denom)) if (duplicate.length > 0) { throw "A denomination is used multiple times: " + duplicate.join(", ") } seenUnitExtensions.add(denomination.canonical) denomination.alternativeDenominations.forEach(d => seenUnitExtensions.add(d)) } this.denominationsSorted = [...this.denominations] this.denominationsSorted.sort((a, b) => b.canonical.length - a.canonical.length) const possiblePostFixes = new Set() function addPostfixesOf(str) { if (str === undefined) { return } str = str.toLowerCase() for (let i = 0; i < str.length + 1; i++) { const substr = str.substring(0, i) possiblePostFixes.add(substr) } } for (const denomination of this.denominations) { addPostfixesOf(denomination.canonical) addPostfixesOf(denomination._canonicalSingular) denomination.alternativeDenominations.forEach(addPostfixesOf) } this.possiblePostFixes = Array.from(possiblePostFixes) this.possiblePostFixes.sort((a, b) => b.length - a.length) } static fromJson(json: UnitConfigJson, ctx: string) { const appliesTo = json.appliesToKey for (let i = 0; i < appliesTo.length; i++) { let key = appliesTo[i]; if (key.trim() !== key) { throw `${ctx}.appliesToKey[${i}] is invalid: it starts or ends with whitespace` } } if ((json.applicableUnits ?? []).length === 0) { throw `${ctx}: define at least one applicable unit` } // Some keys do have unit handling const defaultSet = json.applicableUnits.filter(u => u.default === true) // No default is defined - we pick the first as default if (defaultSet.length === 0) { json.applicableUnits[0].default = true } // Check that there are not multiple defaults if (defaultSet.length > 1) { throw `Multiple units are set as default: they have canonical values of ${defaultSet.map(u => u.canonicalDenomination).join(", ")}` } const applicable = json.applicableUnits.map((u, i) => new Denomination(u, `${ctx}.units[${i}]`)) return new Unit(appliesTo, applicable, json.eraseInvalidValues ?? false) } isApplicableToKey(key: string | undefined): boolean { if (key === undefined) { return false; } return this.appliesToKeys.has(key); } /** * Finds which denomination is applicable and gives the stripped value back */ findDenomination(valueWithDenom: string): [string, Denomination] { if (valueWithDenom === undefined) { return undefined; } for (const denomination of this.denominationsSorted) { const bare = denomination.StrippedValue(valueWithDenom) if (bare !== null) { return [bare, denomination] } } return [undefined, undefined] } asHumanLongValue(value: string): BaseUIElement { if (value === undefined) { return undefined; } const [stripped, denom] = this.findDenomination(value) const human = stripped === "1" ? denom?.humanSingular : denom?.human if (human === undefined) { return new FixedUiElement(stripped ?? value); } const elems = denom.prefix ? [human, stripped] : [stripped, human]; return new Combine(elems) } /** * Returns the value without any (sub)parts of any denomination - usefull as preprocessing step for validating inputs. * E.g. * if 'megawatt' is a possible denomination, then '5 Meg' will be rewritten to '5' (which can then be validated as a valid pnat) * * Returns the original string if nothign matches */ stripUnitParts(str: string) { if (str === undefined) { return undefined; } for (const denominationPart of this.possiblePostFixes) { if (str.endsWith(denominationPart)) { return str.substring(0, str.length - denominationPart.length).trim() } } return str; } }