forked from MapComplete/MapComplete
		
	
		
			
				
	
	
		
			396 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			396 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import BaseUIElement from "../UI/BaseUIElement"
 | |
| import { Denomination } from "./Denomination"
 | |
| import UnitConfigJson from "./ThemeConfig/Json/UnitConfigJson"
 | |
| import unit from "../../assets/layers/unit/unit.json"
 | |
| import TagRenderingConfig from "./ThemeConfig/TagRenderingConfig"
 | |
| import Validators, { ValidatorType } from "../UI/InputElement/Validators"
 | |
| import { Validator } from "../UI/InputElement/Validator"
 | |
| 
 | |
| export class Unit {
 | |
|     private static allUnits = this.initUnits()
 | |
|     public readonly appliesToKeys: Set<string>
 | |
|     public readonly denominations: Denomination[]
 | |
|     public readonly denominationsSorted: Denomination[]
 | |
|     public readonly eraseInvalid: boolean
 | |
|     public readonly quantity: string
 | |
|     public readonly inverted: boolean
 | |
| 
 | |
|     constructor(
 | |
|         quantity: string,
 | |
|         appliesToKeys: string[],
 | |
|         applicableDenominations: Denomination[],
 | |
|         eraseInvalid: boolean,
 | |
|         validator: Validator,
 | |
|         inverted: boolean = false
 | |
|     ) {
 | |
|         this.quantity = quantity
 | |
|         if (
 | |
|             !inverted &&
 | |
|             ["string", "text", "key", "icon", "translation", "fediverse", "id"].indexOf(
 | |
|                 validator.name
 | |
|             ) >= 0
 | |
|         ) {
 | |
|             console.trace("Unit error")
 | |
|             throw (
 | |
|                 "Invalid unit configuration. The validator is of a forbidden type: " +
 | |
|                 validator.name +
 | |
|                 "; set a (number) type to your freeform key or set inverted. Hint: this unit is applied onto keys: " +
 | |
|                 appliesToKeys.join("; ")
 | |
|             )
 | |
|         }
 | |
|         this.inverted = inverted
 | |
|         this.appliesToKeys = new Set(appliesToKeys)
 | |
|         this.denominations = applicableDenominations
 | |
|         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) {
 | |
|             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)
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     static fromJson(
 | |
|         json:
 | |
|             | UnitConfigJson
 | |
|             | Record<
 | |
|                   string,
 | |
|                   string | { quantity: string; denominations: string[]; inverted?: boolean }
 | |
|               >,
 | |
|         tagRenderings: TagRenderingConfig[],
 | |
|         ctx: string
 | |
|     ): Unit[] {
 | |
|         const types: Record<string, ValidatorType> = {}
 | |
|         for (const tagRendering of tagRenderings) {
 | |
|             if (tagRendering.freeform?.type) {
 | |
|                 types[tagRendering.freeform.key] = tagRendering.freeform.type
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (!json.appliesToKey && !json.quantity) {
 | |
|             return this.loadFromLibrary(<any>json, types, ctx)
 | |
|         }
 | |
|         return this.parse(<UnitConfigJson>json, types, ctx)
 | |
|     }
 | |
| 
 | |
|     private static parseDenomination(
 | |
|         json: UnitConfigJson,
 | |
|         validator: Validator,
 | |
|         appliesToKey: string,
 | |
|         ctx: string
 | |
|     ): Unit {
 | |
|         const applicable = json.applicableUnits.map((u, i) =>
 | |
|             Denomination.fromJson(u, validator, `${ctx}.units[${i}]`)
 | |
|         )
 | |
| 
 | |
|         if (
 | |
|             json.defaultInput &&
 | |
|             !applicable.some((denom) => denom.canonical.trim() === json.defaultInput)
 | |
|         ) {
 | |
|             throw `${ctx}: no denomination has the specified default denomination. The default denomination is '${
 | |
|                 json.defaultInput
 | |
|             }', but the available denominations are ${applicable
 | |
|                 .map((denom) => denom.canonical)
 | |
|                 .join(", ")}`
 | |
|         }
 | |
| 
 | |
|         return new Unit(
 | |
|             json.quantity ?? "",
 | |
|             appliesToKey === undefined ? undefined : [appliesToKey],
 | |
|             applicable,
 | |
|             json.eraseInvalidValues ?? false,
 | |
|             validator
 | |
|         )
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      *
 | |
|      * // Should detect invalid defaultInput
 | |
|      * let threwError = false
 | |
|      * try{
 | |
|      *   Unit.parse({
 | |
|      *     appliesToKey: ["length"],
 | |
|      *     defaultInput: "xcm",
 | |
|      *     applicableUnits: [
 | |
|      *         {
 | |
|      *             canonicalDenomination: "m",
 | |
|      *             useIfNoUnitGiven: true,
 | |
|      *             human: "meter"
 | |
|      *         }
 | |
|      *     ]
 | |
|      *   },"test")
 | |
|      * }catch(e){
 | |
|      *     threwError = true
 | |
|      * }
 | |
|      * threwError // => true
 | |
|      *
 | |
|      * // Should work
 | |
|      * Unit.parse({
 | |
|      *     appliesToKey: ["length"],
 | |
|      *     defaultInput: "xcm",
 | |
|      *     applicableUnits: [
 | |
|      *         {
 | |
|      *             canonicalDenomination: "m",
 | |
|      *             useIfNoUnitGiven: true,
 | |
|      *             humen: "meter"
 | |
|      *         },
 | |
|      *         {
 | |
|      *             canonicalDenomination: "cm",
 | |
|      *             human: "centimeter"
 | |
|      *         }
 | |
|      *     ]
 | |
|      * }, "test")
 | |
|      */
 | |
|     private static parse(
 | |
|         json: UnitConfigJson,
 | |
|         types: Record<string, ValidatorType>,
 | |
|         ctx: string
 | |
|     ): Unit[] {
 | |
|         const appliesTo = json.appliesToKey
 | |
|         for (let i = 0; i < (appliesTo ?? []).length; i++) {
 | |
|             const 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 units: Unit[] = []
 | |
|         if (appliesTo === undefined) {
 | |
|             units.push(this.parseDenomination(json, Validators.get("float"), undefined, ctx))
 | |
|         }
 | |
|         for (const key of appliesTo ?? []) {
 | |
|             const validator = Validators.get(types[key] ?? "float")
 | |
|             units.push(this.parseDenomination(json, validator, undefined, ctx))
 | |
|         }
 | |
|         return units
 | |
|     }
 | |
| 
 | |
|     private static initUnits(): Map<string, Unit> {
 | |
|         const m = new Map<string, Unit>()
 | |
|         const units = (<UnitConfigJson[]>unit.units).flatMap((json, i) =>
 | |
|             this.parse(json, {}, "unit.json.units." + i)
 | |
|         )
 | |
| 
 | |
|         for (const unit of units) {
 | |
|             m.set(unit.quantity, unit)
 | |
|         }
 | |
|         return m
 | |
|     }
 | |
| 
 | |
|     private static getFromLibrary(name: string, ctx: string): Unit {
 | |
|         const loaded = this.allUnits.get(name)
 | |
|         if (loaded === undefined) {
 | |
|             throw (
 | |
|                 "No unit with quantity name " +
 | |
|                 name +
 | |
|                 " found (at " +
 | |
|                 ctx +
 | |
|                 "). Try one of: " +
 | |
|                 Array.from(this.allUnits.keys()).join(", ")
 | |
|             )
 | |
|         }
 | |
|         return loaded
 | |
|     }
 | |
| 
 | |
|     private static loadFromLibrary(
 | |
|         spec: Record<
 | |
|             string,
 | |
|             | string
 | |
|             | { quantity: string; denominations: string[]; canonical?: string; inverted?: boolean }
 | |
|         >,
 | |
|         types: Record<string, ValidatorType>,
 | |
|         ctx: string
 | |
|     ): Unit[] {
 | |
|         const units: Unit[] = []
 | |
|         for (const key in spec) {
 | |
|             const toLoad = spec[key]
 | |
|             const validator = Validators.get(types[key] ?? "float")
 | |
|             if (typeof toLoad === "string") {
 | |
|                 const loaded = this.getFromLibrary(toLoad, ctx)
 | |
|                 units.push(
 | |
|                     new Unit(
 | |
|                         loaded.quantity,
 | |
|                         [key],
 | |
|                         loaded.denominations,
 | |
|                         loaded.eraseInvalid,
 | |
|                         validator,
 | |
|                         toLoad["inverted"]
 | |
|                     )
 | |
|                 )
 | |
|                 continue
 | |
|             }
 | |
| 
 | |
|             const loaded = this.getFromLibrary(toLoad.quantity, ctx)
 | |
|             const quantity = toLoad.quantity
 | |
| 
 | |
|             const fetchDenom = (d: string): Denomination => {
 | |
|                 const found = loaded.denominations.find(
 | |
|                     (denom) => denom.canonical.toLowerCase() === d
 | |
|                 )
 | |
|                 if (!found) {
 | |
|                     throw (
 | |
|                         `Could not find a denomination \`${d}\`for quantity ${quantity} at ${ctx}. Perhaps you meant to use on of ` +
 | |
|                         loaded.denominations.map((d) => d.canonical).join(", ")
 | |
|                     )
 | |
|                 }
 | |
|                 return found
 | |
|             }
 | |
| 
 | |
|             if (!Array.isArray(toLoad.denominations)) {
 | |
|                 throw (
 | |
|                     "toLoad is not an array. Did you forget the [ and ] around the denominations at " +
 | |
|                     ctx +
 | |
|                     "?"
 | |
|                 )
 | |
|             }
 | |
|             const denoms = toLoad.denominations
 | |
|                 .map((d) => d.toLowerCase())
 | |
|                 .map((d) => fetchDenom(d))
 | |
|                 .map((d) => d.withValidator(validator))
 | |
| 
 | |
|             if (toLoad.canonical) {
 | |
|                 const canonical = fetchDenom(toLoad.canonical).withValidator(validator)
 | |
|                 denoms.unshift(canonical.withBlankCanonical())
 | |
|             }
 | |
|             units.push(
 | |
|                 new Unit(
 | |
|                     loaded.quantity,
 | |
|                     [key],
 | |
|                     denoms,
 | |
|                     loaded.eraseInvalid,
 | |
|                     validator,
 | |
|                     toLoad["inverted"]
 | |
|                 )
 | |
|             )
 | |
|         }
 | |
|         return units
 | |
|     }
 | |
| 
 | |
|     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, country: () => string): [string, Denomination] {
 | |
|         if (valueWithDenom === undefined) {
 | |
|             return undefined
 | |
|         }
 | |
|         const defaultDenom = this.getDefaultDenomination(country)
 | |
|         for (const denomination of this.denominationsSorted) {
 | |
|             const bare = denomination.StrippedValue(
 | |
|                 valueWithDenom,
 | |
|                 defaultDenom === denomination,
 | |
|                 this.inverted
 | |
|             )
 | |
|             if (bare !== null) {
 | |
|                 return [bare, denomination]
 | |
|             }
 | |
|         }
 | |
|         return [undefined, undefined]
 | |
|     }
 | |
| 
 | |
|     asHumanLongValue(value: string, country: () => string): BaseUIElement | string {
 | |
|         if (value === undefined) {
 | |
|             return undefined
 | |
|         }
 | |
|         const [stripped, denom] = this.findDenomination(value, country)
 | |
|         const human = denom?.human
 | |
|         if (this.inverted) {
 | |
|             return human.Subs({ quantity: stripped + "/" })
 | |
|         }
 | |
|         if (stripped === "1") {
 | |
|             return denom?.humanSingular ?? stripped
 | |
|         }
 | |
|         if (human === undefined) {
 | |
|             return stripped ?? value
 | |
|         }
 | |
| 
 | |
|         return human.Subs({ quantity: stripped })
 | |
|     }
 | |
| 
 | |
|     public toOsm(value: string, denomination: string) {
 | |
|         const denom = this.denominations.find((d) => d.canonical === denomination)
 | |
|         if (this.inverted) {
 | |
|             return value + "/" + denom._canonicalSingular
 | |
|         }
 | |
| 
 | |
|         const space = denom.addSpace ? " " : ""
 | |
|         if (denom.prefix) {
 | |
|             return denom.canonical + space + value
 | |
|         }
 | |
|         return value + space + denom.canonical
 | |
|     }
 | |
| 
 | |
|     public getDefaultDenomination(country: () => string): Denomination {
 | |
|         for (const denomination of this.denominations) {
 | |
|             if (denomination.useIfNoUnitGiven === true) {
 | |
|                 return denomination
 | |
|             }
 | |
|             if (
 | |
|                 denomination.useIfNoUnitGiven === undefined ||
 | |
|                 denomination.useIfNoUnitGiven === false
 | |
|             ) {
 | |
|                 continue
 | |
|             }
 | |
|             let countries: string | string[] = country() ?? []
 | |
|             if (typeof countries === "string") {
 | |
|                 countries = countries.split(",")
 | |
|             }
 | |
|             const denominationCountries: string[] = denomination.useIfNoUnitGiven
 | |
|             if (countries.some((country) => denominationCountries.indexOf(country) >= 0)) {
 | |
|                 return denomination
 | |
|             }
 | |
|         }
 | |
|         for (const denomination of this.denominations) {
 | |
|             if (denomination.canonical === "") {
 | |
|                 return denomination
 | |
|             }
 | |
|         }
 | |
|         return this.denominations[0]
 | |
|     }
 | |
| }
 |