| 
									
										
										
										
											2021-08-07 23:11:34 +02:00
										 |  |  | import BaseUIElement from "../UI/BaseUIElement"; | 
					
						
							|  |  |  | import {FixedUiElement} from "../UI/Base/FixedUiElement"; | 
					
						
							|  |  |  | import Combine from "../UI/Base/Combine"; | 
					
						
							|  |  |  | import {Denomination} from "./Denomination"; | 
					
						
							| 
									
										
										
										
											2021-09-13 01:21:47 +02:00
										 |  |  | import UnitConfigJson from "./ThemeConfig/Json/UnitConfigJson"; | 
					
						
							| 
									
										
										
										
											2021-08-07 23:11:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 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) { | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |             if (str === undefined) { | 
					
						
							|  |  |  |                 return | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-08-07 23:11:34 +02:00
										 |  |  |             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) | 
					
						
							| 
									
										
										
										
											2021-09-13 02:38:20 +02:00
										 |  |  |             addPostfixesOf(denomination._canonicalSingular) | 
					
						
							| 
									
										
										
										
											2021-08-07 23:11:34 +02:00
										 |  |  |             denomination.alternativeDenominations.forEach(addPostfixesOf) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         this.possiblePostFixes = Array.from(possiblePostFixes) | 
					
						
							|  |  |  |         this.possiblePostFixes.sort((a, b) => b.length - a.length) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     static fromJson(json: UnitConfigJson, ctx: string) { | 
					
						
							| 
									
										
										
										
											2021-09-13 01:21:47 +02:00
										 |  |  |         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) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-07 23:11:34 +02:00
										 |  |  |     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) | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |         const human = stripped === "1" ? denom?.humanSingular : denom?.human | 
					
						
							| 
									
										
										
										
											2021-08-07 23:11:34 +02:00
										 |  |  |         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; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |