forked from MapComplete/MapComplete
		
	
		
			
				
	
	
		
			372 lines
		
	
	
		
			No EOL
		
	
	
		
			14 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			372 lines
		
	
	
		
			No EOL
		
	
	
		
			14 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| 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";
 | |
| import {Translation} from "../i18n/Translation";
 | |
| import Constants from "../../Models/Constants";
 | |
| import {SubstitutedTranslation} from "../SubstitutedTranslation";
 | |
| import {TagsFilter} from "../../Logic/Tags/TagsFilter";
 | |
| import {Tag} from "../../Logic/Tags/Tag";
 | |
| import {And} from "../../Logic/Tags/And";
 | |
| import {TagUtils} from "../../Logic/Tags/TagUtils";
 | |
| import BaseUIElement from "../BaseUIElement";
 | |
| import {DropDown} from "../Input/DropDown";
 | |
| import {Unit} from "../../Customizations/JSON/Denomination";
 | |
| import CombinedInputElement from "../Input/CombinedInputElement";
 | |
| 
 | |
| /**
 | |
|  * 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 {
 | |
|     private readonly _tags: UIEventSource<any>;
 | |
|     private _configuration: TagRenderingConfig;
 | |
| 
 | |
|     private _saveButton: BaseUIElement;
 | |
| 
 | |
|     private _inputElement: InputElement<TagsFilter>;
 | |
|     private _cancelButton: BaseUIElement;
 | |
|     private _appliedTags: BaseUIElement;
 | |
|     private readonly _applicableUnit: Unit;
 | |
|     private _question: BaseUIElement;
 | |
| 
 | |
|     constructor(tags: UIEventSource<any>,
 | |
|                 configuration: TagRenderingConfig,
 | |
|                 units: Unit[],
 | |
|                 afterSave?: () => void,
 | |
|                 cancelButton?: BaseUIElement
 | |
|     ) {
 | |
|         super(tags);
 | |
|         this._applicableUnit = units.filter(unit => unit.isApplicableToKey(configuration.freeform?.key))[0];
 | |
|         this._tags = tags;
 | |
|         this._configuration = configuration;
 | |
|         this._cancelButton = cancelButton;
 | |
|         this._question = new SubstitutedTranslation(this._configuration.question, tags)
 | |
|             .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;
 | |
|             console.log("Save button clicked, the tags are is", selection)
 | |
|             if (selection) {
 | |
|                 (State.state?.changes ?? new Changes())
 | |
|                     .addTag(tags.data.id, selection, tags);
 | |
|             }
 | |
| 
 | |
|             if (afterSave) {
 | |
|                 afterSave();
 | |
|             }
 | |
|         }
 | |
| 
 | |
| 
 | |
|         this._saveButton = new SaveButton(this._inputElement.GetValue(),
 | |
|             State.state?.osmConnection)
 | |
|             .onClick(save)
 | |
| 
 | |
| 
 | |
|         this._appliedTags = new VariableUiElement(
 | |
|             self._inputElement.GetValue().map(
 | |
|                 (tags: TagsFilter) => {
 | |
|                     const csCount = State.state?.osmConnection?.userDetails?.data?.csCount ?? 1000;
 | |
|                     if (csCount < Constants.userJourney.tagsVisibleAt) {
 | |
|                         return "";
 | |
|                     }
 | |
|                     if (tags === undefined) {
 | |
|                         return Translations.t.general.noTagsSelected.SetClass("subtle");
 | |
|                     }
 | |
|                     if (csCount < Constants.userJourney.tagsVisibleAndWikiLinked) {
 | |
|                         const tagsStr = tags.asHumanString(false, true, self._tags.data);
 | |
|                         return new FixedUiElement(tagsStr).SetClass("subtle");
 | |
|                     }
 | |
|                     return tags.asHumanString(true, true, self._tags.data);
 | |
|                 }
 | |
|             )
 | |
|         ).SetClass("block break-all")
 | |
| 
 | |
| 
 | |
|     }
 | |
| 
 | |
|     InnerRender() {
 | |
|         return new Combine([
 | |
|             this._question,
 | |
|             this._inputElement,
 | |
|             this._cancelButton,
 | |
|             this._saveButton,
 | |
|             this._appliedTags]
 | |
|         )
 | |
|             .SetClass("question")
 | |
|     }
 | |
| 
 | |
|     private GenerateInputElement(): InputElement<TagsFilter> {
 | |
|         const self = this;
 | |
|         let inputEls: InputElement<TagsFilter>[];
 | |
| 
 | |
|         const mappings = (this._configuration.mappings ?? [])
 | |
|             .filter(mapping => {
 | |
|                 if (mapping.hideInAnswer === true) {
 | |
|                     return false;
 | |
|                 }
 | |
|                 if (typeof (mapping.hideInAnswer) !== "boolean" && mapping.hideInAnswer.matchesProperties(this._tags.data)) {
 | |
|                     return false;
 | |
|                 }
 | |
|                 return true;
 | |
|             })
 | |
| 
 | |
| 
 | |
|         let allIfNots: TagsFilter[] = Utils.NoNull(this._configuration.mappings?.map(m => m.ifnot) ?? []);
 | |
|         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) {
 | |
|             return ff;
 | |
|         }
 | |
| 
 | |
|         if (ff) {
 | |
|             inputEls.push(ff);
 | |
|         }
 | |
| 
 | |
|         if (this._configuration.multiAnswer) {
 | |
|             return this.GenerateMultiAnswer(inputEls, ff, this._configuration.mappings.map(mp => mp.ifnot))
 | |
|         } else {
 | |
|             return new RadioButton(inputEls, false)
 | |
|         }
 | |
| 
 | |
|     }
 | |
| 
 | |
|     private GenerateMultiAnswer(elements: InputElement<TagsFilter>[], freeformField: InputElement<TagsFilter>, ifNotSelected: TagsFilter[]): InputElement<TagsFilter> {
 | |
|         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);
 | |
|                 const oppositeTags: TagsFilter[] = [];
 | |
|                 for (let i = 0; i < ifNotSelected.length; i++) {
 | |
|                     if (indices.indexOf(i) >= 0) {
 | |
|                         continue;
 | |
|                     }
 | |
|                     const notSelected = ifNotSelected[i];
 | |
|                     if (notSelected === undefined) {
 | |
|                         continue;
 | |
|                     }
 | |
|                     oppositeTags.push(notSelected);
 | |
|                 }
 | |
|                 tags.push(TagUtils.FlattenMultiAnswer(oppositeTags));
 | |
|                 const actualTags = TagUtils.FlattenMultiAnswer(tags);
 | |
|                 console.log("Converted ", indices.join(","), "into", actualTags.asHumanString(false, false, {}), "with elems", elements)
 | |
|                 return actualTags;
 | |
|             },
 | |
|             (tags: TagsFilter) => {
 | |
|                 // {key --> values[]}
 | |
|                 const presentTags = TagUtils.SplitKeys([tags]);
 | |
|                 const indices: number[] = []
 | |
|                 // 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] ?? [])]
 | |
|                 }
 | |
| 
 | |
|                 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);
 | |
|                                 }
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                 }
 | |
|                 if (freeformField) {
 | |
|                     if (freeformExtras.length > 0) {
 | |
|                         freeformField.GetValue().setData(new Tag(this._configuration.freeform.key, freeformExtras.join(";")));
 | |
|                         indices.push(elements.indexOf(freeformField))
 | |
|                     } else {
 | |
|                         freeformField.GetValue().setData(undefined);
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
| 
 | |
|                 return indices;
 | |
|             },
 | |
|             elements.map(el => el.GetValue())
 | |
|         );
 | |
| 
 | |
| 
 | |
|         freeformField?.GetValue()?.addCallbackAndRun(value => {
 | |
|             // The list of indices of the selected elements
 | |
|             const es = checkBoxes.GetValue();
 | |
|             const i = elements.length - 1;
 | |
|             // The actual index of the freeform-element
 | |
|             const index = es.data.indexOf(i);
 | |
|             if (value === undefined) {
 | |
|                 // No data is set in the freeform text field; so we delete the checkmark if it is selected
 | |
|                 if (index >= 0) {
 | |
|                     es.data.splice(index, 1);
 | |
|                     es.ping();
 | |
|                 }
 | |
|             } else if (index < 0) {
 | |
|                 // 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
 | |
|                 es.data.push(i);
 | |
|                 es.ping();
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         return inputEl;
 | |
|     }
 | |
| 
 | |
|     private GenerateMappingElement(mapping: {
 | |
|         if: TagsFilter,
 | |
|         then: Translation,
 | |
|         hideInAnswer: boolean | TagsFilter
 | |
|     }, ifNot?: TagsFilter[]): InputElement<TagsFilter> {
 | |
| 
 | |
|         let tagging = mapping.if;
 | |
|         if (ifNot.length > 0) {
 | |
|             tagging = new And([tagging, ...ifNot])
 | |
|         }
 | |
| 
 | |
|         return new FixedInputElement(
 | |
|             new SubstitutedTranslation(mapping.then, this._tags),
 | |
|             tagging,
 | |
|             (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;
 | |
|         }
 | |
| 
 | |
|         let input: InputElement<string> = ValidatedTextField.InputForType(this._configuration.freeform.type, {
 | |
|             isValid: (str) => (str.length <= 255),
 | |
|             country: () => this._tags.data._country,
 | |
|             location: [this._tags.data._lat, this._tags.data._lon],
 | |
|             mapBackgroundLayer: State.state.backgroundLayer
 | |
|         });
 | |
| 
 | |
|         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]);
 | |
| 
 | |
|         return new InputElementMap(
 | |
|             input, (a, b) => a === b || (a?.isEquivalent(b) ?? false),
 | |
|             pickString, toString
 | |
|         );
 | |
| 
 | |
|     }
 | |
| 
 | |
| } |