| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  | import {UIElement} from "../UIElement"; | 
					
						
							|  |  |  | import {UIEventSource} from "../../Logic/UIEventSource"; | 
					
						
							|  |  |  | import Combine from "../Base/Combine"; | 
					
						
							|  |  |  | import TagRenderingConfig from "../../Customizations/JSON/TagRenderingConfig"; | 
					
						
							|  |  |  | import {InputElement} from "../Input/InputElement"; | 
					
						
							|  |  |  | import ValidatedTextField from "../Input/ValidatedTextField"; | 
					
						
							|  |  |  | import {FixedInputElement} from "../Input/FixedInputElement"; | 
					
						
							|  |  |  | import {RadioButton} from "../Input/RadioButton"; | 
					
						
							|  |  |  | import {Utils} from "../../Utils"; | 
					
						
							|  |  |  | import CheckBoxes from "../Input/Checkboxes"; | 
					
						
							|  |  |  | import InputElementMap from "../Input/InputElementMap"; | 
					
						
							|  |  |  | import {SaveButton} from "./SaveButton"; | 
					
						
							|  |  |  | import State from "../../State"; | 
					
						
							|  |  |  | import {Changes} from "../../Logic/Osm/Changes"; | 
					
						
							|  |  |  | import {VariableUiElement} from "../Base/VariableUIElement"; | 
					
						
							|  |  |  | import Translations from "../i18n/Translations"; | 
					
						
							|  |  |  | import {FixedUiElement} from "../Base/FixedUiElement"; | 
					
						
							| 
									
										
										
										
											2020-11-06 01:58:26 +01:00
										 |  |  | import {Translation} from "../i18n/Translation"; | 
					
						
							| 
									
										
										
										
											2021-01-02 19:09:49 +01:00
										 |  |  | import Constants from "../../Models/Constants"; | 
					
						
							| 
									
										
										
										
											2021-02-05 16:32:37 +01:00
										 |  |  | import {SubstitutedTranslation} from "../SubstitutedTranslation"; | 
					
						
							| 
									
										
										
										
											2021-03-29 00:41:53 +02:00
										 |  |  | import {TagsFilter} from "../../Logic/Tags/TagsFilter"; | 
					
						
							|  |  |  | import {Tag} from "../../Logic/Tags/Tag"; | 
					
						
							|  |  |  | import {And} from "../../Logic/Tags/And"; | 
					
						
							|  |  |  | import {TagUtils} from "../../Logic/Tags/TagUtils"; | 
					
						
							| 
									
										
										
										
											2021-06-12 02:58:32 +02:00
										 |  |  | import BaseUIElement from "../BaseUIElement"; | 
					
						
							| 
									
										
										
										
											2021-06-18 01:50:03 +02:00
										 |  |  | import {DropDown} from "../Input/DropDown"; | 
					
						
							| 
									
										
										
										
											2021-06-22 03:16:45 +02:00
										 |  |  | import {Unit} from "../../Customizations/JSON/Denomination"; | 
					
						
							|  |  |  | import CombinedInputElement from "../Input/CombinedInputElement"; | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Shows the question element. | 
					
						
							|  |  |  |  * Note that the value _migh_ already be known, e.g. when selected or when changing the value | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export default class TagRenderingQuestion extends UIElement { | 
					
						
							| 
									
										
										
										
											2021-01-04 04:06:21 +01:00
										 |  |  |     private readonly _tags: UIEventSource<any>; | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |     private _configuration: TagRenderingConfig; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-14 02:39:23 +02:00
										 |  |  |     private _saveButton: BaseUIElement; | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     private _inputElement: InputElement<TagsFilter>; | 
					
						
							| 
									
										
										
										
											2021-06-14 02:39:23 +02:00
										 |  |  |     private _cancelButton: BaseUIElement; | 
					
						
							| 
									
										
										
										
											2021-06-12 02:58:32 +02:00
										 |  |  |     private _appliedTags: BaseUIElement; | 
					
						
							| 
									
										
										
										
											2021-06-22 03:16:45 +02:00
										 |  |  |     private readonly _applicableUnit: Unit; | 
					
						
							| 
									
										
										
										
											2021-06-14 02:39:23 +02:00
										 |  |  |     private _question: BaseUIElement; | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     constructor(tags: UIEventSource<any>, | 
					
						
							|  |  |  |                 configuration: TagRenderingConfig, | 
					
						
							| 
									
										
										
										
											2021-06-22 03:16:45 +02:00
										 |  |  |                 units: Unit[], | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |                 afterSave?: () => void, | 
					
						
							| 
									
										
										
										
											2021-06-14 02:39:23 +02:00
										 |  |  |                 cancelButton?: BaseUIElement | 
					
						
							| 
									
										
										
										
											2021-03-12 13:48:49 +01:00
										 |  |  |     ) { | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |         super(tags); | 
					
						
							| 
									
										
										
										
											2021-06-22 03:16:45 +02:00
										 |  |  |         this._applicableUnit = units.filter(unit => unit.isApplicableToKey(configuration.freeform?.key))[0]; | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |         this._tags = tags; | 
					
						
							|  |  |  |         this._configuration = configuration; | 
					
						
							|  |  |  |         this._cancelButton = cancelButton; | 
					
						
							| 
									
										
										
										
											2021-06-14 02:39:23 +02:00
										 |  |  |         this._question = new SubstitutedTranslation(this._configuration.question, tags) | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |             .SetClass("question-text"); | 
					
						
							|  |  |  |         if (configuration === undefined) { | 
					
						
							|  |  |  |             throw "A question is needed for a question visualization" | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this._inputElement = this.GenerateInputElement() | 
					
						
							|  |  |  |         const self = this; | 
					
						
							|  |  |  |         const save = () => { | 
					
						
							|  |  |  |             const selection = self._inputElement.GetValue().data; | 
					
						
							| 
									
										
										
										
											2021-03-29 02:53:06 +02:00
										 |  |  |             console.log("Save button clicked, the tags are is", selection) | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |             if (selection) { | 
					
						
							|  |  |  |                 (State.state?.changes ?? new Changes()) | 
					
						
							|  |  |  |                     .addTag(tags.data.id, selection, tags); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (afterSave) { | 
					
						
							|  |  |  |                 afterSave(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-18 01:50:03 +02:00
										 |  |  |         this._saveButton = new SaveButton(this._inputElement.GetValue(), | 
					
						
							| 
									
										
										
										
											2021-06-16 21:23:03 +02:00
										 |  |  |             State.state?.osmConnection) | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |             .onClick(save) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this._appliedTags = new VariableUiElement( | 
					
						
							|  |  |  |             self._inputElement.GetValue().map( | 
					
						
							|  |  |  |                 (tags: TagsFilter) => { | 
					
						
							|  |  |  |                     const csCount = State.state?.osmConnection?.userDetails?.data?.csCount ?? 1000; | 
					
						
							| 
									
										
										
										
											2021-01-02 19:09:49 +01:00
										 |  |  |                     if (csCount < Constants.userJourney.tagsVisibleAt) { | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |                         return ""; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     if (tags === undefined) { | 
					
						
							| 
									
										
										
										
											2021-06-12 02:58:32 +02:00
										 |  |  |                         return Translations.t.general.noTagsSelected.SetClass("subtle"); | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |                     } | 
					
						
							| 
									
										
										
										
											2021-01-02 19:09:49 +01:00
										 |  |  |                     if (csCount < Constants.userJourney.tagsVisibleAndWikiLinked) { | 
					
						
							| 
									
										
										
										
											2021-03-29 02:31:08 +02:00
										 |  |  |                         const tagsStr = tags.asHumanString(false, true, self._tags.data); | 
					
						
							| 
									
										
										
										
											2021-06-12 02:58:32 +02:00
										 |  |  |                         return new FixedUiElement(tagsStr).SetClass("subtle"); | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |                     } | 
					
						
							| 
									
										
										
										
											2021-03-29 02:31:08 +02:00
										 |  |  |                     return tags.asHumanString(true, true, self._tags.data); | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |                 } | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2021-06-16 21:23:03 +02:00
										 |  |  |         ).SetClass("block break-all") | 
					
						
							| 
									
										
										
										
											2021-06-18 01:50:03 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-10 01:36:20 +02:00
										 |  |  |     InnerRender() { | 
					
						
							| 
									
										
										
										
											2021-03-31 15:50:29 +02:00
										 |  |  |         return new Combine([ | 
					
						
							|  |  |  |             this._question, | 
					
						
							|  |  |  |             this._inputElement, | 
					
						
							|  |  |  |             this._cancelButton, | 
					
						
							|  |  |  |             this._saveButton, | 
					
						
							|  |  |  |             this._appliedTags] | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |             .SetClass("question") | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private GenerateInputElement(): InputElement<TagsFilter> { | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |         const self = this; | 
					
						
							| 
									
										
										
										
											2021-06-18 01:50:03 +02:00
										 |  |  |         let inputEls: InputElement<TagsFilter>[]; | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-22 03:16:45 +02:00
										 |  |  |         const mappings = (this._configuration.mappings ?? []) | 
					
						
							|  |  |  |             .filter(mapping => { | 
					
						
							|  |  |  |                 if (mapping.hideInAnswer === true) { | 
					
						
							| 
									
										
										
										
											2021-06-18 01:50:03 +02:00
										 |  |  |                     return false; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (typeof (mapping.hideInAnswer) !== "boolean" && mapping.hideInAnswer.matchesProperties(this._tags.data)) { | 
					
						
							|  |  |  |                     return false; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |             }) | 
					
						
							| 
									
										
										
										
											2021-06-22 03:16:45 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let allIfNots: TagsFilter[] = Utils.NoNull(this._configuration.mappings?.map(m => m.ifnot) ?? []); | 
					
						
							| 
									
										
										
										
											2021-06-18 01:50:03 +02:00
										 |  |  |         const ff = this.GenerateFreeform(); | 
					
						
							|  |  |  |         const hasImages = mappings.filter(mapping => mapping.then.ExtractImages().length > 0).length > 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (mappings.length < 8 || this._configuration.multiAnswer || hasImages) { | 
					
						
							|  |  |  |             inputEls = (mappings ?? []).map(mapping => self.GenerateMappingElement(mapping, allIfNots)); | 
					
						
							|  |  |  |             inputEls = Utils.NoNull(inputEls); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             const dropdown: InputElement<TagsFilter> = new DropDown("", | 
					
						
							|  |  |  |                 mappings.map(mapping => { | 
					
						
							|  |  |  |                     return { | 
					
						
							|  |  |  |                         value: new And([mapping.if, ...allIfNots]), | 
					
						
							|  |  |  |                         shown: Translations.WT(mapping.then).Clone() | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 }) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (ff == undefined) { | 
					
						
							|  |  |  |                 return dropdown; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 inputEls = [dropdown] | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (inputEls.length == 0) { | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |             return ff; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-31 15:50:29 +02:00
										 |  |  |         if (ff) { | 
					
						
							| 
									
										
										
										
											2021-06-18 01:50:03 +02:00
										 |  |  |             inputEls.push(ff); | 
					
						
							| 
									
										
										
										
											2021-02-20 01:45:51 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |         if (this._configuration.multiAnswer) { | 
					
						
							| 
									
										
										
										
											2021-06-18 01:50:03 +02:00
										 |  |  |             return this.GenerateMultiAnswer(inputEls, ff, this._configuration.mappings.map(mp => mp.ifnot)) | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2021-06-18 01:50:03 +02:00
										 |  |  |             return new RadioButton(inputEls, false) | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-20 16:48:42 +01:00
										 |  |  |     private GenerateMultiAnswer(elements: InputElement<TagsFilter>[], freeformField: InputElement<TagsFilter>, ifNotSelected: TagsFilter[]): InputElement<TagsFilter> { | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |         const checkBoxes = new CheckBoxes(elements); | 
					
						
							|  |  |  |         const inputEl = new InputElementMap<number[], TagsFilter>( | 
					
						
							|  |  |  |             checkBoxes, | 
					
						
							|  |  |  |             (t0, t1) => { | 
					
						
							|  |  |  |                 return t0?.isEquivalent(t1) ?? false | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             (indices) => { | 
					
						
							|  |  |  |                 if (indices.length === 0) { | 
					
						
							|  |  |  |                     return undefined; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 const tags: TagsFilter[] = indices.map(i => elements[i].GetValue().data); | 
					
						
							| 
									
										
										
										
											2021-02-20 16:48:42 +01:00
										 |  |  |                 const oppositeTags: TagsFilter[] = []; | 
					
						
							|  |  |  |                 for (let i = 0; i < ifNotSelected.length; i++) { | 
					
						
							| 
									
										
										
										
											2021-03-31 15:50:29 +02:00
										 |  |  |                     if (indices.indexOf(i) >= 0) { | 
					
						
							| 
									
										
										
										
											2021-02-20 16:48:42 +01:00
										 |  |  |                         continue; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     const notSelected = ifNotSelected[i]; | 
					
						
							| 
									
										
										
										
											2021-03-31 15:50:29 +02:00
										 |  |  |                     if (notSelected === undefined) { | 
					
						
							| 
									
										
										
										
											2021-02-20 16:48:42 +01:00
										 |  |  |                         continue; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     oppositeTags.push(notSelected); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 tags.push(TagUtils.FlattenMultiAnswer(oppositeTags)); | 
					
						
							| 
									
										
										
										
											2021-06-16 21:23:03 +02:00
										 |  |  |                 const actualTags = TagUtils.FlattenMultiAnswer(tags); | 
					
						
							|  |  |  |                 console.log("Converted ", indices.join(","), "into", actualTags.asHumanString(false, false, {}), "with elems", elements) | 
					
						
							|  |  |  |                 return actualTags; | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |             }, | 
					
						
							|  |  |  |             (tags: TagsFilter) => { | 
					
						
							| 
									
										
										
										
											2020-10-27 14:13:37 +01:00
										 |  |  |                 // {key --> values[]}
 | 
					
						
							|  |  |  |                 const presentTags = TagUtils.SplitKeys([tags]); | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |                 const indices: number[] = [] | 
					
						
							| 
									
										
										
										
											2020-10-27 14:13:37 +01:00
										 |  |  |                 // We also collect the values that have to be added to the freeform field
 | 
					
						
							|  |  |  |                 let freeformExtras: string[] = [] | 
					
						
							|  |  |  |                 if (this._configuration.freeform?.key) { | 
					
						
							|  |  |  |                     freeformExtras = [...(presentTags[this._configuration.freeform.key] ?? [])] | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-27 14:13:37 +01:00
										 |  |  |                 for (let j = 0; j < elements.length; j++) { | 
					
						
							|  |  |  |                     const inputElement = elements[j]; | 
					
						
							|  |  |  |                     if (inputElement === freeformField) { | 
					
						
							|  |  |  |                         continue; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     const val = inputElement.GetValue(); | 
					
						
							|  |  |  |                     const neededTags = TagUtils.SplitKeys([val.data]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     // if every 'neededKeys'-value is present in presentKeys, we have a match and enable the index
 | 
					
						
							|  |  |  |                     if (TagUtils.AllKeysAreContained(presentTags, neededTags)) { | 
					
						
							|  |  |  |                         indices.push(j); | 
					
						
							|  |  |  |                         if (freeformExtras.length > 0) { | 
					
						
							|  |  |  |                             const freeformsToRemove: string[] = (neededTags[this._configuration.freeform.key] ?? []); | 
					
						
							|  |  |  |                             for (const toRm of freeformsToRemove) { | 
					
						
							|  |  |  |                                 const i = freeformExtras.indexOf(toRm); | 
					
						
							|  |  |  |                                 if (i >= 0) { | 
					
						
							|  |  |  |                                     freeformExtras.splice(i, 1); | 
					
						
							|  |  |  |                                 } | 
					
						
							|  |  |  |                             } | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2020-10-27 14:13:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (freeformField) { | 
					
						
							|  |  |  |                     if (freeformExtras.length > 0) { | 
					
						
							|  |  |  |                         freeformField.GetValue().setData(new Tag(this._configuration.freeform.key, freeformExtras.join(";"))); | 
					
						
							| 
									
										
										
										
											2020-10-28 11:19:47 +01:00
										 |  |  |                         indices.push(elements.indexOf(freeformField)) | 
					
						
							| 
									
										
										
										
											2020-10-27 14:13:37 +01:00
										 |  |  |                     } else { | 
					
						
							|  |  |  |                         freeformField.GetValue().setData(undefined); | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-10-27 14:13:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |                 return indices; | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             elements.map(el => el.GetValue()) | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         freeformField?.GetValue()?.addCallbackAndRun(value => { | 
					
						
							| 
									
										
										
										
											2021-02-21 00:15:03 +01:00
										 |  |  |             // The list of indices of the selected elements
 | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |             const es = checkBoxes.GetValue(); | 
					
						
							|  |  |  |             const i = elements.length - 1; | 
					
						
							| 
									
										
										
										
											2021-02-21 00:15:03 +01:00
										 |  |  |             // The actual index of the freeform-element
 | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |             const index = es.data.indexOf(i); | 
					
						
							|  |  |  |             if (value === undefined) { | 
					
						
							| 
									
										
										
										
											2021-02-21 00:15:03 +01:00
										 |  |  |                 // No data is set in the freeform text field; so we delete the checkmark if it is selected
 | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |                 if (index >= 0) { | 
					
						
							|  |  |  |                     es.data.splice(index, 1); | 
					
						
							|  |  |  |                     es.ping(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } else if (index < 0) { | 
					
						
							| 
									
										
										
										
											2021-02-21 00:15:03 +01:00
										 |  |  |                 // There is data defined in the checkmark, but the checkmark isn't checked, so we check it
 | 
					
						
							|  |  |  |                 // This is of course because the data changed
 | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |                 es.data.push(i); | 
					
						
							|  |  |  |                 es.ping(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return inputEl; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private GenerateMappingElement(mapping: { | 
					
						
							|  |  |  |         if: TagsFilter, | 
					
						
							|  |  |  |         then: Translation, | 
					
						
							| 
									
										
										
										
											2020-12-07 03:02:50 +01:00
										 |  |  |         hideInAnswer: boolean | TagsFilter | 
					
						
							| 
									
										
										
										
											2021-06-18 01:50:03 +02:00
										 |  |  |     }, ifNot?: TagsFilter[]): InputElement<TagsFilter> { | 
					
						
							| 
									
										
										
										
											2021-06-22 03:16:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-18 01:50:03 +02:00
										 |  |  |         let tagging = mapping.if; | 
					
						
							|  |  |  |         if (ifNot.length > 0) { | 
					
						
							|  |  |  |             tagging = new And([tagging, ...ifNot]) | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-18 01:50:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |         return new FixedInputElement( | 
					
						
							| 
									
										
										
										
											2021-06-14 02:39:23 +02:00
										 |  |  |             new SubstitutedTranslation(mapping.then, this._tags), | 
					
						
							| 
									
										
										
										
											2021-06-18 01:50:03 +02:00
										 |  |  |             tagging, | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |             (t0, t1) => t1.isEquivalent(t0)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private GenerateFreeform(): InputElement<TagsFilter> { | 
					
						
							|  |  |  |         const freeform = this._configuration.freeform; | 
					
						
							|  |  |  |         if (freeform === undefined) { | 
					
						
							|  |  |  |             return undefined; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const pickString = | 
					
						
							|  |  |  |             (string: any) => { | 
					
						
							|  |  |  |                 if (string === "" || string === undefined) { | 
					
						
							|  |  |  |                     return undefined; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 const tag = new Tag(freeform.key, string); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (freeform.addExtraTags === undefined) { | 
					
						
							|  |  |  |                     return tag; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 return new And([ | 
					
						
							|  |  |  |                         tag, | 
					
						
							|  |  |  |                         ...freeform.addExtraTags | 
					
						
							|  |  |  |                     ] | 
					
						
							|  |  |  |                 ); | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const toString = (tag) => { | 
					
						
							|  |  |  |             if (tag instanceof And) { | 
					
						
							|  |  |  |                 for (const subtag of tag.and) { | 
					
						
							|  |  |  |                     if (subtag instanceof Tag && subtag.key === freeform.key) { | 
					
						
							|  |  |  |                         return subtag.value; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 return undefined; | 
					
						
							|  |  |  |             } else if (tag instanceof Tag) { | 
					
						
							|  |  |  |                 return tag.value | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return undefined; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-22 03:16:45 +02:00
										 |  |  |         let input: InputElement<string> = ValidatedTextField.InputForType(this._configuration.freeform.type, { | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |             isValid: (str) => (str.length <= 255), | 
					
						
							| 
									
										
										
										
											2020-12-05 03:22:17 +01:00
										 |  |  |             country: () => this._tags.data._country, | 
					
						
							| 
									
										
										
										
											2021-06-23 02:15:28 +02:00
										 |  |  |             location: [this._tags.data._lat, this._tags.data._lon], | 
					
						
							|  |  |  |             mapBackgroundLayer: State.state.backgroundLayer | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-22 03:16:45 +02:00
										 |  |  |         if (this._applicableUnit) { | 
					
						
							|  |  |  |             // We need to apply a unit.
 | 
					
						
							|  |  |  |             // This implies:
 | 
					
						
							|  |  |  |             // We have to create a dropdown with applicable denominations, and fuse those values
 | 
					
						
							|  |  |  |             const unit = this._applicableUnit | 
					
						
							|  |  |  |             const unitDropDown = new DropDown("", | 
					
						
							|  |  |  |                 unit.denominations.map(denom => { | 
					
						
							|  |  |  |                     return { | 
					
						
							|  |  |  |                         shown: denom.human, | 
					
						
							|  |  |  |                         value: denom | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 }) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             unitDropDown.GetValue().setData(this._applicableUnit.defaultDenom) | 
					
						
							|  |  |  |             unitDropDown.SetStyle("width: min-content") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             input = new CombinedInputElement( | 
					
						
							|  |  |  |                 input, | 
					
						
							|  |  |  |                 unitDropDown, | 
					
						
							|  |  |  |                 (text, denom) => denom?.canonicalValue(text, true) ?? text, | 
					
						
							|  |  |  |                 (valueWithDenom: string) => unit.findDenomination(valueWithDenom) | 
					
						
							|  |  |  |             ).SetClass("flex") | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         input.GetValue().setData(this._tags.data[this._configuration.freeform.key]); | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return new InputElementMap( | 
					
						
							| 
									
										
										
										
											2021-06-22 03:16:45 +02:00
										 |  |  |             input, (a, b) => a === b || (a?.isEquivalent(b) ?? false), | 
					
						
							| 
									
										
										
										
											2020-10-27 01:01:34 +01:00
										 |  |  |             pickString, toString | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |