forked from MapComplete/MapComplete
204 lines
No EOL
7.1 KiB
TypeScript
204 lines
No EOL
7.1 KiB
TypeScript
import {Translation} from "../../UI/i18n/Translation";
|
|
import UnitConfigJson from "./UnitConfigJson";
|
|
import Translations from "../../UI/i18n/Translations";
|
|
import BaseUIElement from "../../UI/BaseUIElement";
|
|
import Combine from "../../UI/Base/Combine";
|
|
|
|
export class Unit {
|
|
public readonly appliesToKeys: Set<string>;
|
|
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<string>();
|
|
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<string>()
|
|
function addPostfixesOf(str){
|
|
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)
|
|
denomination.alternativeDenominations.forEach(addPostfixesOf)
|
|
}
|
|
this.possiblePostFixes = Array.from(possiblePostFixes)
|
|
this.possiblePostFixes.sort((a, b) => b.length - a .length)
|
|
}
|
|
|
|
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 = denom.human
|
|
|
|
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;
|
|
}
|
|
}
|
|
|
|
export class Denomination {
|
|
public readonly canonical: string;
|
|
readonly default: boolean;
|
|
readonly prefix: boolean;
|
|
private readonly _human: Translation;
|
|
public readonly alternativeDenominations: string [];
|
|
|
|
constructor(json: UnitConfigJson, context: string) {
|
|
context = `${context}.unit(${json.canonicalDenomination})`
|
|
this.canonical = json.canonicalDenomination.trim()
|
|
if (this.canonical === undefined) {
|
|
throw `${context}: this unit has no decent canonical value defined`
|
|
}
|
|
|
|
json.alternativeDenomination.forEach((v, i) => {
|
|
if (((v?.trim() ?? "") === "")) {
|
|
throw `${context}.alternativeDenomination.${i}: invalid alternative denomination: undefined, null or only whitespace`
|
|
}
|
|
})
|
|
|
|
this.alternativeDenominations = json.alternativeDenomination?.map(v => v.trim()) ?? []
|
|
|
|
this.default = json.default ?? false;
|
|
|
|
this._human = Translations.T(json.human, context + "human")
|
|
|
|
this.prefix = json.prefix ?? false;
|
|
|
|
}
|
|
|
|
get human(): Translation {
|
|
return this._human.Clone()
|
|
}
|
|
|
|
public canonicalValue(value: string, actAsDefault?: boolean) {
|
|
if (value === undefined) {
|
|
return undefined;
|
|
}
|
|
const stripped = this.StrippedValue(value, actAsDefault)
|
|
if (stripped === null) {
|
|
return null;
|
|
}
|
|
return stripped + " " + this.canonical.trim()
|
|
}
|
|
|
|
/**
|
|
* Returns the core value (without unit) if:
|
|
* - the value ends with the canonical or an alternative value (or begins with if prefix is set)
|
|
* - the value is a Number (without unit) and default is set
|
|
*
|
|
* Returns null if it doesn't match this unit
|
|
*/
|
|
public StrippedValue(value: string, actAsDefault?: boolean): string {
|
|
|
|
if (value === undefined) {
|
|
return undefined;
|
|
}
|
|
|
|
value = value.toLowerCase()
|
|
if (this.prefix) {
|
|
if (value.startsWith(this.canonical) && this.canonical !== "") {
|
|
return value.substring(this.canonical.length).trim();
|
|
}
|
|
for (const alternativeValue of this.alternativeDenominations) {
|
|
if (value.startsWith(alternativeValue)) {
|
|
return value.substring(alternativeValue.length).trim();
|
|
}
|
|
}
|
|
} else {
|
|
if (value.endsWith(this.canonical.toLowerCase()) && this.canonical !== "") {
|
|
return value.substring(0, value.length - this.canonical.length).trim();
|
|
}
|
|
for (const alternativeValue of this.alternativeDenominations) {
|
|
if (value.endsWith(alternativeValue.toLowerCase())) {
|
|
return value.substring(0, value.length - alternativeValue.length).trim();
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
if (this.default || actAsDefault) {
|
|
const parsed = Number(value.trim())
|
|
if (!isNaN(parsed)) {
|
|
return value.trim();
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
|
|
} |