forked from MapComplete/MapComplete
		
	Fix rendering of multianswers, other small bug fixes
This commit is contained in:
		
							parent
							
								
									46254434db
								
							
						
					
					
						commit
						ad08a55517
					
				
					 13 changed files with 68 additions and 62 deletions
				
			
		
							
								
								
									
										1
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							|  | @ -11,3 +11,4 @@ assets/generated/* | ||||||
| .parcel-cache | .parcel-cache | ||||||
| Docs/Tools/stats.*.json | Docs/Tools/stats.*.json | ||||||
| Docs/Tools/stats.csv | Docs/Tools/stats.csv | ||||||
|  | 
 | ||||||
|  |  | ||||||
|  | @ -1,4 +1,4 @@ | ||||||
| import {And, TagsFilter} from "../../Logic/Tags"; | import {And, TagsFilter, TagUtils} from "../../Logic/Tags"; | ||||||
| import {TagRenderingConfigJson} from "./TagRenderingConfigJson"; | import {TagRenderingConfigJson} from "./TagRenderingConfigJson"; | ||||||
| import Translations from "../../UI/i18n/Translations"; | import Translations from "../../UI/i18n/Translations"; | ||||||
| import {FromJSON} from "./FromJSON"; | import {FromJSON} from "./FromJSON"; | ||||||
|  | @ -152,6 +152,40 @@ export default class TagRenderingConfig { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Returns true if it is known or not shown, false if the question should be asked | ||||||
|  |      * @constructor | ||||||
|  |      */ | ||||||
|  |     public IsKnown(tags: any): boolean { | ||||||
|  |         if (this.condition && | ||||||
|  |             !this.condition.matchesProperties(tags)) { | ||||||
|  |             // Filtered away by the condition
 | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |         if(this.multiAnswer){ | ||||||
|  |             for (const m of this.mappings) { | ||||||
|  |                 if(TagUtils.MatchesMultiAnswer(m.if, tags)){ | ||||||
|  |                     return true; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             const free = this.freeform?.key | ||||||
|  |             if(free !== undefined){ | ||||||
|  |                 return tags[free] !== undefined | ||||||
|  |             } | ||||||
|  |             return false | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (this.GetRenderValue(tags) !== undefined) { | ||||||
|  |             // This value is known and can be rendered
 | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |      | ||||||
|     /** |     /** | ||||||
|      * Gets the correct rendering value (or undefined if not known) |      * Gets the correct rendering value (or undefined if not known) | ||||||
|      * @constructor |      * @constructor | ||||||
|  |  | ||||||
|  | @ -257,7 +257,7 @@ export class InitUiElements { | ||||||
|                 isOpened.setData(false); |                 isOpened.setData(false); | ||||||
|             } |             } | ||||||
|         }) |         }) | ||||||
|         isOpened.setData(true) |         isOpened.setData(Hash.hash.data === undefined) | ||||||
|          |          | ||||||
|   |   | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -1,5 +1,4 @@ | ||||||
| import {UIEventSource} from "../UIEventSource"; | import {UIEventSource} from "../UIEventSource"; | ||||||
| import {UIElement} from "../../UI/UIElement"; |  | ||||||
| import FeatureSource from "../FeatureSource/FeatureSource"; | import FeatureSource from "../FeatureSource/FeatureSource"; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  | @ -18,9 +17,7 @@ export default class SelectedFeatureHandler { | ||||||
|         this._featureSource = featureSource; |         this._featureSource = featureSource; | ||||||
|         const self = this; |         const self = this; | ||||||
|         hash.addCallback(h => { |         hash.addCallback(h => { | ||||||
|             console.log("SelectedFeatureHandler: hash is now ", h) |  | ||||||
|             if (h === undefined || h === "") { |             if (h === undefined || h === "") { | ||||||
|                 console.log("Deselecting...") |  | ||||||
|                 selectedFeature.setData(undefined); |                 selectedFeature.setData(undefined); | ||||||
|             }else{ |             }else{ | ||||||
|                 self.selectFeature(); |                 self.selectFeature(); | ||||||
|  | @ -30,7 +27,10 @@ export default class SelectedFeatureHandler { | ||||||
|         featureSource.features.addCallback(_ => self.selectFeature()); |         featureSource.features.addCallback(_ => self.selectFeature()); | ||||||
| 
 | 
 | ||||||
|         selectedFeature.addCallback(feature => { |         selectedFeature.addCallback(feature => { | ||||||
|             hash.setData(feature?.properties?.id ?? ""); |             const h = feature?.properties?.id; | ||||||
|  |             if(h !== undefined){ | ||||||
|  |                 hash.setData(h) | ||||||
|  |             } | ||||||
|         }) |         }) | ||||||
| 
 | 
 | ||||||
|         this.selectFeature(); |         this.selectFeature(); | ||||||
|  | @ -51,7 +51,6 @@ export default class SelectedFeatureHandler { | ||||||
|         if(hash === undefined || hash === "" || hash === "#"){ |         if(hash === undefined || hash === "" || hash === "#"){ | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         console.log("Selecting a feature from the hash...") |  | ||||||
|         for (const feature of features) { |         for (const feature of features) { | ||||||
|             const id = feature.feature?.properties?.id; |             const id = feature.feature?.properties?.id; | ||||||
|             if(id === hash){ |             if(id === hash){ | ||||||
|  |  | ||||||
|  | @ -125,7 +125,7 @@ export default class MetaTagging { | ||||||
|                             tags["_isOpen:oldvalue"] = tags.opening_hours |                             tags["_isOpen:oldvalue"] = tags.opening_hours | ||||||
|                             window.setTimeout( |                             window.setTimeout( | ||||||
|                                 () => { |                                 () => { | ||||||
|                                     console.log("Updating the _isOpen tag for ", tags.id); |                                     console.log("Updating the _isOpen tag for ", tags.id, ", it's timer expired after", timeout); | ||||||
|                                     updateTags(); |                                     updateTags(); | ||||||
|                                 }, |                                 }, | ||||||
|                                 timeout |                                 timeout | ||||||
|  |  | ||||||
|  | @ -43,16 +43,14 @@ export default class Hash { | ||||||
| 
 | 
 | ||||||
|         window.onhashchange = () => { |         window.onhashchange = () => { | ||||||
|             let newValue = window.location.hash.substr(1); |             let newValue = window.location.hash.substr(1); | ||||||
|             console.log("The hash is now:", newValue) |  | ||||||
|             if (newValue === "") { |             if (newValue === "") { | ||||||
|                 newValue = undefined; |                 newValue = undefined; | ||||||
|             } |             } | ||||||
|             hash.setData(newValue) |             hash.setData(newValue) | ||||||
|         } |         } | ||||||
|          |          | ||||||
|         window.addEventListener('popstate', e => { |         window.addEventListener('popstate', _ => { | ||||||
|             let newValue = window.location.hash.substr(1); |             let newValue = window.location.hash.substr(1); | ||||||
|             console.log("Popstate: the hash is now:", newValue) |  | ||||||
|             if (newValue === "") { |             if (newValue === "") { | ||||||
|                 newValue = undefined; |                 newValue = undefined; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|  | @ -2,7 +2,7 @@ import { Utils } from "../Utils"; | ||||||
| 
 | 
 | ||||||
| export default class Constants { | export default class Constants { | ||||||
|      |      | ||||||
|     public static vNumber = "0.5.8"; |     public static vNumber = "0.5.10"; | ||||||
| 
 | 
 | ||||||
|     // The user journey states thresholds when a new feature gets unlocked
 |     // The user journey states thresholds when a new feature gets unlocked
 | ||||||
|     public static userJourney = { |     public static userJourney = { | ||||||
|  |  | ||||||
|  | @ -83,7 +83,7 @@ export default class ScrollableFullScreen extends UIElement { | ||||||
|     private static clear() { |     private static clear() { | ||||||
|         ScrollableFullScreen.empty.AttachTo("fullscreen") |         ScrollableFullScreen.empty.AttachTo("fullscreen") | ||||||
|         const fs = document.getElementById("fullscreen"); |         const fs = document.getElementById("fullscreen"); | ||||||
|         ScrollableFullScreen._currentlyOpen.isShown.setData(false); |         ScrollableFullScreen._currentlyOpen?.isShown?.setData(false); | ||||||
|         fs.classList.add("hidden") |         fs.classList.add("hidden") | ||||||
|         Hash.hash.setData(undefined); |         Hash.hash.setData(undefined); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -53,13 +53,8 @@ export default class EditableTagRendering extends UIElement { | ||||||
|         if (this._editMode.data) { |         if (this._editMode.data) { | ||||||
|             return this._question.Render(); |             return this._question.Render(); | ||||||
|         } |         } | ||||||
|         if (this._configuration.multiAnswer) { |         if(!this._configuration.IsKnown(this._tags.data)){ | ||||||
|             const atLeastOneMatch = this._configuration.mappings.some(mp =>TagUtils.MatchesMultiAnswer(mp.if, this._tags.data)); |             return "" | ||||||
|             if (!atLeastOneMatch) { |  | ||||||
|                 return ""; |  | ||||||
|             } |  | ||||||
|         } else if (this._configuration.GetRenderValue(this._tags.data) === undefined) { |  | ||||||
|             return ""; |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         return new Combine([this._answer, |         return new Combine([this._answer, | ||||||
|  |  | ||||||
|  | @ -56,24 +56,19 @@ export default class FeatureInfoBox extends ScrollableFullScreen { | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         let questionBoxIsUsed = false; |         let questionBoxIsUsed = false; | ||||||
|         const renderings = layerConfig.tagRenderings.map(tr => { |         const renderings = layerConfig.tagRenderings.map((tr,i) => { | ||||||
|             if (tr.question === null) { |             if (tr.question === null) { | ||||||
|                 // This is the question box!
 |                 // This is the question box!
 | ||||||
|                 questionBoxIsUsed = true; |                 questionBoxIsUsed = true; | ||||||
|                 return questionBox; |                 return questionBox; | ||||||
|             } |             } | ||||||
|             return new EditableTagRendering(tags, tr); |             return  new EditableTagRendering(tags, tr); | ||||||
|         }); |         }); | ||||||
|         if (!questionBoxIsUsed) { |         if (!questionBoxIsUsed) { | ||||||
|             renderings.push(questionBox); |             renderings.push(questionBox); | ||||||
|         } |         } | ||||||
|         const tail = new Combine([]).SetClass("only-on-mobile"); |  | ||||||
| 
 | 
 | ||||||
|         return new Combine([ |         return new Combine(renderings).SetClass("block") | ||||||
|                 ...renderings, |  | ||||||
|                 tail.SetClass("featureinfobox-tail") |  | ||||||
|             ] |  | ||||||
|         ).SetClass("block") |  | ||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -46,37 +46,11 @@ export default class QuestionBox extends UIElement { | ||||||
|             }) |             }) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |  | ||||||
|      * Returns true if it is known or not shown, false if the question should be asked |  | ||||||
|      * @constructor |  | ||||||
|      */ |  | ||||||
|     IsKnown(tagRendering: TagRenderingConfig): boolean { |  | ||||||
|         if (tagRendering.condition && |  | ||||||
|             !tagRendering.condition.matchesProperties(this._tags.data)) { |  | ||||||
|             // Filtered away by the condition
 |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
|         if(tagRendering.multiAnswer){ |  | ||||||
|             for (const m of tagRendering.mappings) { |  | ||||||
|                 if(TagUtils.MatchesMultiAnswer(m.if, this._tags.data)){ |  | ||||||
|                     return true; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (tagRendering.GetRenderValue(this._tags.data) !== undefined) { |  | ||||||
|             // This value is known and can be rendered
 |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     InnerRender(): string { |     InnerRender(): string { | ||||||
|         for (let i = 0; i < this._tagRenderingQuestions.length; i++) { |         for (let i = 0; i < this._tagRenderingQuestions.length; i++) { | ||||||
|             let tagRendering = this._tagRenderings[i]; |             let tagRendering = this._tagRenderings[i]; | ||||||
| 
 | 
 | ||||||
|             if(this.IsKnown(tagRendering)){ |             if(tagRendering.IsKnown(this._tags.data)){ | ||||||
|                 continue; |                 continue; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -5,6 +5,7 @@ import {Utils} from "../../Utils"; | ||||||
| import Combine from "../Base/Combine"; | import Combine from "../Base/Combine"; | ||||||
| import {TagUtils} from "../../Logic/Tags"; | import {TagUtils} from "../../Logic/Tags"; | ||||||
| import {SubstitutedTranslation} from "../SubstitutedTranslation"; | import {SubstitutedTranslation} from "../SubstitutedTranslation"; | ||||||
|  | import {Translation} from "../i18n/Translation"; | ||||||
| 
 | 
 | ||||||
| /*** | /*** | ||||||
|  * Displays the correct value for a known tagrendering |  * Displays the correct value for a known tagrendering | ||||||
|  | @ -43,7 +44,7 @@ export default class TagRenderingAnswer extends UIElement { | ||||||
| 
 | 
 | ||||||
|         // The render value doesn't work well with multi-answers (checkboxes), so we have to check for them manually
 |         // The render value doesn't work well with multi-answers (checkboxes), so we have to check for them manually
 | ||||||
|         if (this._configuration.multiAnswer) { |         if (this._configuration.multiAnswer) { | ||||||
|             const applicableThens = Utils.NoNull(this._configuration.mappings.map(mapping => { |             const applicableThens: Translation[] = Utils.NoNull(this._configuration.mappings.map(mapping => { | ||||||
|                 if (mapping.if === undefined) { |                 if (mapping.if === undefined) { | ||||||
|                     return mapping.then; |                     return mapping.then; | ||||||
|                 } |                 } | ||||||
|  | @ -52,12 +53,20 @@ export default class TagRenderingAnswer extends UIElement { | ||||||
|                 } |                 } | ||||||
|                 return undefined; |                 return undefined; | ||||||
|             })) |             })) | ||||||
|             if (applicableThens.length >= 0) { | 
 | ||||||
|                 if (applicableThens.length === 1) { |             if (this._configuration.freeform !== undefined && tags[this._configuration.freeform.key] !== undefined) { | ||||||
|                     this._content = applicableThens[0]; |                 applicableThens.push(this._configuration.render) | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             const self = this | ||||||
|  |             const valuesToRender: UIElement[] = applicableThens.map(tr => SubstitutedTranslation.construct(tr, self._tags)) | ||||||
|  | 
 | ||||||
|  |             if (valuesToRender.length >= 0) { | ||||||
|  |                 if (valuesToRender.length === 1) { | ||||||
|  |                     this._content = valuesToRender[0]; | ||||||
|                 } else { |                 } else { | ||||||
|                     this._content = new Combine(["<ul>", |                     this._content = new Combine(["<ul>", | ||||||
|                         ...applicableThens.map(tr => new Combine(["<li>", tr, "</li>"])) |                         ...valuesToRender.map(tr => new Combine(["<li>", tr, "</li>"])) | ||||||
|                         , |                         , | ||||||
|                         "</ul>" |                         "</ul>" | ||||||
|                     ]) |                     ]) | ||||||
|  | @ -66,13 +75,13 @@ export default class TagRenderingAnswer extends UIElement { | ||||||
|                 return this._content.SetClass(this._contentClass).SetStyle(this._contentStyle).Render(); |                 return this._content.SetClass(this._contentClass).SetStyle(this._contentStyle).Render(); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          | 
 | ||||||
|         const tr = this._configuration.GetRenderValue(tags); |         const tr = this._configuration.GetRenderValue(tags); | ||||||
|         if (tr !== undefined) { |         if (tr !== undefined) { | ||||||
|             this._content = SubstitutedTranslation.construct(tr, this._tags); |             this._content = SubstitutedTranslation.construct(tr, this._tags); | ||||||
|             return this._content.SetClass(this._contentClass).SetStyle(this._contentStyle).Render(); |             return this._content.SetClass(this._contentClass).SetStyle(this._contentStyle).Render(); | ||||||
|         } |         } | ||||||
|         | 
 | ||||||
|         return ""; |         return ""; | ||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -19,6 +19,7 @@ | ||||||
|     "build": "rm -rf dist/ && npm run generate && parcel build --public-url ./ *.html assets/** assets/**/** assets/**/**/** vendor/* vendor/*/*", |     "build": "rm -rf dist/ && npm run generate && parcel build --public-url ./ *.html assets/** assets/**/** assets/**/**/** vendor/* vendor/*/*", | ||||||
|     "prepare-deploy": "npm run test && npm run generate:editor-layer-index && npm run generate:layouts && npm run generate && npm run build && rm -rf .cache", |     "prepare-deploy": "npm run test && npm run generate:editor-layer-index && npm run generate:layouts && npm run generate && npm run build && rm -rf .cache", | ||||||
|     "deploy:staging": "npm run prepare-deploy && rm -rf /home/pietervdvn/git/pietervdvn.github.io/Staging/* && cp -r dist/* /home/pietervdvn/git/pietervdvn.github.io/Staging/ && cd /home/pietervdvn/git/pietervdvn.github.io/ && git add * && git commit -m 'New MapComplete Version' && git push && cd - && npm run clean", |     "deploy:staging": "npm run prepare-deploy && rm -rf /home/pietervdvn/git/pietervdvn.github.io/Staging/* && cp -r dist/* /home/pietervdvn/git/pietervdvn.github.io/Staging/ && cd /home/pietervdvn/git/pietervdvn.github.io/ && git add * && git commit -m 'New MapComplete Version' && git push && cd - && npm run clean", | ||||||
|  |     "deploy:pietervdvn": "npm run prepare-deploy && rm -rf /home/pietervdvn/git/pietervdvn.github.io/MapComplete/* && cp -r dist/* /home/pietervdvn/git/pietervdvn.github.io/MapComplete/ && cd /home/pietervdvn/git/pietervdvn.github.io/ && git add * && git commit -m 'New MapComplete Version' && git push && cd - && npm run clean", | ||||||
|     "deploy:production": "rm -rf ./assets/generated && npm run prepare-deploy && npm run optimize-images && rm -rf /home/pietervdvn/git/mapcomplete.github.io/* && cp -r dist/* /home/pietervdvn/git/mapcomplete.github.io/ && cd /home/pietervdvn/git/mapcomplete.github.io/ && echo \"mapcomplete.osm.be\" > CNAME  && git add * && git commit -m 'New MapComplete Version' && git push && cd - && npm run clean", |     "deploy:production": "rm -rf ./assets/generated && npm run prepare-deploy && npm run optimize-images && rm -rf /home/pietervdvn/git/mapcomplete.github.io/* && cp -r dist/* /home/pietervdvn/git/mapcomplete.github.io/ && cd /home/pietervdvn/git/mapcomplete.github.io/ && echo \"mapcomplete.osm.be\" > CNAME  && git add * && git commit -m 'New MapComplete Version' && git push && cd - && npm run clean", | ||||||
|     "clean": "rm -rf .cache/ && (find *.html | grep -v \"\\(index\\|land\\|test\\|preferences\\|customGenerator\\).html\" | xargs rm) && rm *.webmanifest" |     "clean": "rm -rf .cache/ && (find *.html | grep -v \"\\(index\\|land\\|test\\|preferences\\|customGenerator\\).html\" | xargs rm) && rm *.webmanifest" | ||||||
|   }, |   }, | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue