209 lines
7.1 KiB
TypeScript
209 lines
7.1 KiB
TypeScript
import { Translation, TypedTranslation } from "../UI/i18n/Translation"
|
|
import { DenominationConfigJson } from "./ThemeConfig/Json/UnitConfigJson"
|
|
import Translations from "../UI/i18n/Translations"
|
|
|
|
/**
|
|
* A 'denomination' is one way to write a certain quantity.
|
|
* For example, 'meter', 'kilometer', 'mile' and 'foot' are all possible ways to quantify 'length'
|
|
*/
|
|
export class Denomination {
|
|
public readonly canonical: string
|
|
public readonly _canonicalSingular: string
|
|
public readonly useIfNoUnitGiven: boolean | string[]
|
|
public readonly prefix: boolean
|
|
public readonly addSpace: boolean
|
|
public readonly alternativeDenominations: string[]
|
|
public readonly human: TypedTranslation<{ quantity: string }>
|
|
public readonly humanSingular?: Translation
|
|
|
|
private constructor(
|
|
canonical: string,
|
|
_canonicalSingular: string,
|
|
useIfNoUnitGiven: boolean | string[],
|
|
prefix: boolean,
|
|
addSpace: boolean,
|
|
alternativeDenominations: string[],
|
|
_human: TypedTranslation<{ quantity: string }>,
|
|
_humanSingular?: Translation
|
|
) {
|
|
this.canonical = canonical
|
|
this._canonicalSingular = _canonicalSingular
|
|
this.useIfNoUnitGiven = useIfNoUnitGiven
|
|
this.prefix = prefix
|
|
this.addSpace = addSpace
|
|
this.alternativeDenominations = alternativeDenominations
|
|
this.human = _human
|
|
this.humanSingular = _humanSingular
|
|
}
|
|
|
|
public static fromJson(json: DenominationConfigJson, context: string) {
|
|
context = `${context}.unit(${json.canonicalDenomination})`
|
|
const canonical = json.canonicalDenomination.trim()
|
|
if (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`
|
|
}
|
|
})
|
|
|
|
if (json["default" /* @code-quality: ignore*/] !== undefined) {
|
|
throw `${context} uses the old 'default'-key. Use "useIfNoUnitGiven" or "useAsDefaultInput" instead`
|
|
}
|
|
|
|
const humanTexts = Translations.T(json.human, context + "human")
|
|
humanTexts.OnEveryLanguage((text, language) => {
|
|
if (text.indexOf("{quantity}") < 0) {
|
|
throw `In denomination: a human text should contain {quantity} (at ${context}.human.${language})`
|
|
}
|
|
return text
|
|
})
|
|
return new Denomination(
|
|
canonical,
|
|
json.canonicalDenominationSingular?.trim(),
|
|
json.useIfNoUnitGiven,
|
|
json.prefix ?? false,
|
|
json.addSpace ?? false,
|
|
json.alternativeDenomination?.map((v) => v.trim()) ?? [],
|
|
humanTexts,
|
|
Translations.T(json.humanSingular, context + "humanSingular")
|
|
)
|
|
}
|
|
|
|
public clone() {
|
|
return new Denomination(
|
|
this.canonical,
|
|
this._canonicalSingular,
|
|
this.useIfNoUnitGiven,
|
|
this.prefix,
|
|
this.addSpace,
|
|
this.alternativeDenominations,
|
|
this.human,
|
|
this.humanSingular
|
|
)
|
|
}
|
|
|
|
public withBlankCanonical() {
|
|
return new Denomination(
|
|
"",
|
|
this._canonicalSingular,
|
|
this.useIfNoUnitGiven,
|
|
this.prefix,
|
|
this.addSpace,
|
|
[this.canonical, ...this.alternativeDenominations],
|
|
this.human,
|
|
this.humanSingular
|
|
)
|
|
}
|
|
|
|
/**
|
|
* Create the canonical, human representation of the given value
|
|
* @param value the value from OSM
|
|
* @param actAsDefault if set and the value can be parsed as number, will be parsed and trimmed
|
|
*
|
|
* const unit = Denomination.fromJson({
|
|
* canonicalDenomination: "m",
|
|
* alternativeDenomination: ["meter"],
|
|
* human: {
|
|
* en: "{quantity} meter"
|
|
* }
|
|
* }, "test")
|
|
* unit.canonicalValue("42m", true) // =>"42 m"
|
|
* unit.canonicalValue("42", true) // =>"42 m"
|
|
* unit.canonicalValue("42 m", true) // =>"42 m"
|
|
* unit.canonicalValue("42 meter", true) // =>"42 m"
|
|
* unit.canonicalValue("42m", true) // =>"42 m"
|
|
* unit.canonicalValue("42", true) // =>"42 m"
|
|
*
|
|
* // Should be trimmed if canonical is empty
|
|
* const unit = Denomination.fromJson({
|
|
* canonicalDenomination: "",
|
|
* alternativeDenomination: ["meter","m"],
|
|
* human: {
|
|
* en: "{quantity} meter"
|
|
* }
|
|
* }, "test")
|
|
* unit.canonicalValue("42m", true) // =>"42"
|
|
* unit.canonicalValue("42", true) // =>"42"
|
|
* unit.canonicalValue("42 m", true) // =>"42"
|
|
* unit.canonicalValue("42 meter", true) // =>"42"
|
|
*
|
|
*
|
|
*/
|
|
public canonicalValue(value: string, actAsDefault: boolean): string {
|
|
if (value === undefined) {
|
|
return undefined
|
|
}
|
|
const stripped = this.StrippedValue(value, actAsDefault)
|
|
if (stripped === null) {
|
|
return null
|
|
}
|
|
if (stripped === "1" && this._canonicalSingular !== undefined) {
|
|
return ("1 " + this._canonicalSingular).trim()
|
|
}
|
|
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()
|
|
const self = this
|
|
|
|
function startsWith(key) {
|
|
if (self.prefix) {
|
|
return value.startsWith(key)
|
|
} else {
|
|
return value.endsWith(key)
|
|
}
|
|
}
|
|
|
|
function substr(key) {
|
|
if (self.prefix) {
|
|
return value.substr(key.length).trim()
|
|
} else {
|
|
return value.substring(0, value.length - key.length).trim()
|
|
}
|
|
}
|
|
|
|
if (this.canonical !== "" && startsWith(this.canonical.toLowerCase())) {
|
|
return substr(this.canonical)
|
|
}
|
|
|
|
if (
|
|
this._canonicalSingular !== undefined &&
|
|
this._canonicalSingular !== "" &&
|
|
startsWith(this._canonicalSingular)
|
|
) {
|
|
return substr(this._canonicalSingular)
|
|
}
|
|
|
|
for (const alternativeValue of this.alternativeDenominations) {
|
|
if (startsWith(alternativeValue)) {
|
|
return substr(alternativeValue)
|
|
}
|
|
}
|
|
|
|
if (!actAsDefault) {
|
|
return null
|
|
}
|
|
|
|
const parsed = Number(value.trim())
|
|
if (!isNaN(parsed)) {
|
|
return value.trim()
|
|
}
|
|
|
|
return null
|
|
}
|
|
}
|