forked from MapComplete/MapComplete
		
	Fixed the personal layer
This commit is contained in:
		
							parent
							
								
									bf7e6376c0
								
							
						
					
					
						commit
						106d9927aa
					
				
					 3 changed files with 110 additions and 118 deletions
				
			
		|  | @ -12,7 +12,6 @@ import Combine from "../../UI/Base/Combine"; | ||||||
| import {VariableUiElement} from "../../UI/Base/VariableUIElement"; | import {VariableUiElement} from "../../UI/Base/VariableUIElement"; | ||||||
| import {UIEventSource} from "../../Logic/UIEventSource"; | import {UIEventSource} from "../../Logic/UIEventSource"; | ||||||
| import {FixedUiElement} from "../../UI/Base/FixedUiElement"; | import {FixedUiElement} from "../../UI/Base/FixedUiElement"; | ||||||
| import {SubstitutedTranslation} from "../../UI/SubstitutedTranslation"; |  | ||||||
| import SourceConfig from "./SourceConfig"; | import SourceConfig from "./SourceConfig"; | ||||||
| import {TagsFilter} from "../../Logic/Tags/TagsFilter"; | import {TagsFilter} from "../../Logic/Tags/TagsFilter"; | ||||||
| import {Tag} from "../../Logic/Tags/Tag"; | import {Tag} from "../../Logic/Tags/Tag"; | ||||||
|  | @ -290,7 +289,7 @@ export default class LayerConfig { | ||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public GenerateLeafletStyle(tags: UIEventSource<any>, clickable: boolean): |     public GenerateLeafletStyle(tags: UIEventSource<any>, clickable: boolean, widthHeight= "100%"): | ||||||
|         { |         { | ||||||
|             icon: |             icon: | ||||||
|                 { |                 { | ||||||
|  |  | ||||||
|  | @ -92,9 +92,16 @@ export class UIEventSource<T> { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public map<J>(f: ((T) => J), |     /** | ||||||
|  |      * Monoidal map: | ||||||
|  |      * Given a function 'f', will construct a new UIEventSource where the contents will always be "f(this.data)' | ||||||
|  |      * @param f: The transforming function | ||||||
|  |      * @param extraSources: also trigger the update if one of these sources change | ||||||
|  |      * @param g: a 'backfunction to let the sync run in two directions. (data of the new UIEVEntSource, currentData) => newData | ||||||
|  |      */ | ||||||
|  |     public map<J>(f: ((t: T) => J), | ||||||
|                   extraSources: UIEventSource<any>[] = [], |                   extraSources: UIEventSource<any>[] = [], | ||||||
|                   g: ((J) => T) = undefined): UIEventSource<J> { |                   g: ((j:J, t:T) => T) = undefined): UIEventSource<J> { | ||||||
|         const self = this; |         const self = this; | ||||||
| 
 | 
 | ||||||
|         const newSource = new UIEventSource<J>( |         const newSource = new UIEventSource<J>( | ||||||
|  | @ -113,7 +120,7 @@ export class UIEventSource<T> { | ||||||
| 
 | 
 | ||||||
|         if (g !== undefined) { |         if (g !== undefined) { | ||||||
|             newSource.addCallback((latest) => { |             newSource.addCallback((latest) => { | ||||||
|                 self.setData(g(latest)); |                 self.setData(g(latest, self.data)); | ||||||
|             }) |             }) | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,6 +1,3 @@ | ||||||
| import {UIEventSource} from "../../Logic/UIEventSource"; |  | ||||||
| import {UIElement} from "../UIElement"; |  | ||||||
| import LayoutConfig from "../../Customizations/JSON/LayoutConfig"; |  | ||||||
| import {AllKnownLayouts} from "../../Customizations/AllKnownLayouts"; | import {AllKnownLayouts} from "../../Customizations/AllKnownLayouts"; | ||||||
| import Svg from "../../Svg"; | import Svg from "../../Svg"; | ||||||
| import State from "../../State"; | import State from "../../State"; | ||||||
|  | @ -8,130 +5,119 @@ import Combine from "../Base/Combine"; | ||||||
| import Toggle from "../Input/Toggle"; | import Toggle from "../Input/Toggle"; | ||||||
| import {SubtleButton} from "../Base/SubtleButton"; | import {SubtleButton} from "../Base/SubtleButton"; | ||||||
| import Translations from "../i18n/Translations"; | import Translations from "../i18n/Translations"; | ||||||
| import * as personal from "../../assets/themes/personalLayout/personalLayout.json" |  | ||||||
| import Locale from "../i18n/Locale"; |  | ||||||
| import BaseUIElement from "../BaseUIElement"; | import BaseUIElement from "../BaseUIElement"; | ||||||
|  | import {VariableUiElement} from "../Base/VariableUIElement"; | ||||||
|  | import LayerConfig from "../../Customizations/JSON/LayerConfig"; | ||||||
|  | import Img from "../Base/Img"; | ||||||
|  | import {UIEventSource} from "../../Logic/UIEventSource"; | ||||||
| 
 | 
 | ||||||
| export default class PersonalLayersPanel extends UIElement { | export default class PersonalLayersPanel extends VariableUiElement { | ||||||
|     private checkboxes: BaseUIElement[] = []; |  | ||||||
| 
 | 
 | ||||||
|     constructor() { |     constructor() { | ||||||
|         super(State.state.favouriteLayers); |         super( | ||||||
|         this.ListenTo(State.state.osmConnection.userDetails); |             State.state.installedThemes.map(installedThemes => { | ||||||
|         this.ListenTo(Locale.language); |                 const t = Translations.t.favourite; | ||||||
|         this.UpdateView([]); |  | ||||||
|         const self = this; |  | ||||||
|         State.state.installedThemes.addCallback(extraThemes => { |  | ||||||
|             self.UpdateView(extraThemes.map(layout => layout.layout)); |  | ||||||
|         }) |  | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|  |                 // Lets get all the layers
 | ||||||
|  |                 const allThemes = AllKnownLayouts.layoutsList.concat(installedThemes.map(layout => layout.layout)) | ||||||
|  |                     .filter(theme => !theme.hideFromOverview) | ||||||
| 
 | 
 | ||||||
|     private UpdateView(extraThemes: LayoutConfig[]) { |                 const allLayers = [] | ||||||
|         this.checkboxes = []; |                 { | ||||||
|         const favs = State.state.favouriteLayers.data ?? []; |                     const seenLayers = new Set<string>() | ||||||
|         const controls = new Map<string, UIEventSource<boolean>>(); |                     for (const layers of allThemes.map(theme => theme.layers)) { | ||||||
|         const allLayouts = AllKnownLayouts.layoutsList.concat(extraThemes); |                         for (const layer of layers) { | ||||||
|         for (const layout of allLayouts) { |                             if (seenLayers.has(layer.id)) { | ||||||
|             if (layout.id === personal.id) { |                                 continue | ||||||
|                 continue; |                             } | ||||||
|             } |                             seenLayers.add(layer.id) | ||||||
|              |                             allLayers.push(layer) | ||||||
|             if(layout.hideFromOverview){ |  | ||||||
|                 continue; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             const header = |  | ||||||
|                 new Combine([ |  | ||||||
|                     `<img style="max-width: 3em;max-height: 3em; float: left; padding: 0.1em; margin-right: 0.3em;" src='${layout.icon}'>`, |  | ||||||
|                     "<b>", |  | ||||||
|                     layout.title, |  | ||||||
|                     "</b><br/>", |  | ||||||
|                     layout.shortDescription ?? "" |  | ||||||
|                 ]).SetClass("block p1 overflow-auto rounded") |  | ||||||
|                     .SetStyle("background: #eee;") |  | ||||||
|             this.checkboxes.push(header); |  | ||||||
| 
 |  | ||||||
|             for (const layer of layout.layers) { |  | ||||||
|                 if(layer === undefined){ |  | ||||||
|                     console.warn("Undefined layer for ",layout.id) |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (typeof layer === "string") { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 let icon :BaseUIElement = layer.GenerateLeafletStyle(new UIEventSource<any>({id:"node/-1"}), false).icon.html |  | ||||||
|                     ?? Svg.checkmark_svg(); |  | ||||||
|                 let iconUnset =new Combine([icon]); |  | ||||||
|                 icon.SetClass("single-layer-selection-toggle") |  | ||||||
|                 iconUnset.SetClass("single-layer-selection-toggle") |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|                 let name = layer.name ?? layer.id; |  | ||||||
|                 if (name === undefined) { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 const content = new Combine([ |  | ||||||
|                     "<b>",  |  | ||||||
|                     name,  |  | ||||||
|                     "</b> ", |  | ||||||
|                     layer.description !== undefined ? new Combine(["<br/>", layer.description]) : "", |  | ||||||
|                 ]) |  | ||||||
|                  |  | ||||||
|                  |  | ||||||
|                 const cb = new Toggle( |  | ||||||
|                     new SubtleButton( |  | ||||||
|                         icon,  |  | ||||||
|                         content), |  | ||||||
|                     new SubtleButton( |  | ||||||
|                         iconUnset.SetStyle("opacity:0.1"), |  | ||||||
|                         new Combine(["<del>", |  | ||||||
|                             content, |  | ||||||
|                             "</del>" |  | ||||||
|                         ])), |  | ||||||
|                     controls[layer.id] ?? (favs.indexOf(layer.id) >= 0) |  | ||||||
|                 ).ToggleOnClick(); |  | ||||||
|                 cb.SetClass("custom-layer-checkbox"); |  | ||||||
|                 controls[layer.id] = cb.isEnabled; |  | ||||||
| 
 |  | ||||||
|                 cb.isEnabled.addCallback((isEnabled) => { |  | ||||||
|                     const favs = State.state.favouriteLayers; |  | ||||||
|                     if (isEnabled) { |  | ||||||
|                         if(favs.data.indexOf(layer.id)>= 0){ |  | ||||||
|                             return; // Already added
 |  | ||||||
|                         } |                         } | ||||||
|                         favs.data.push(layer.id); |  | ||||||
|                     } else { |  | ||||||
|                         favs.data.splice(favs.data.indexOf(layer.id), 1); |  | ||||||
|                     } |                     } | ||||||
|                     favs.ping(); |                 } | ||||||
|                 }) |  | ||||||
| 
 | 
 | ||||||
|                 this.checkboxes.push(cb); |                 // Time to create a panel based on them!
 | ||||||
|  |                 const panel: BaseUIElement = new Combine(allLayers.map(PersonalLayersPanel.CreateLayerToggle)); | ||||||
| 
 | 
 | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         State.state.favouriteLayers.addCallback((layers) => { |  | ||||||
|             for (const layerId of layers) { |  | ||||||
|                 controls[layerId]?.setData(true); |  | ||||||
|             } |  | ||||||
|         }); |  | ||||||
| 
 | 
 | ||||||
|  |                 return new Toggle( | ||||||
|  |                     new Combine([ | ||||||
|  |                         t.panelIntro.Clone(), | ||||||
|  |                         panel | ||||||
|  |                     ]).SetClass("flex flex-col"), | ||||||
|  |                     new SubtleButton( | ||||||
|  |                         Svg.osm_logo_ui(), | ||||||
|  |                         t.loginNeeded.Clone().SetClass("text-center") | ||||||
|  |                     ).onClick(() => State.state.osmConnection.AttemptLogin()), | ||||||
|  |                     State.state.osmConnection.isLoggedIn | ||||||
|  |                 ) | ||||||
|  |             }) | ||||||
|  |         ) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     InnerRender(): BaseUIElement { |     /*** | ||||||
|         const t = Translations.t.favourite; |      * Creates a toggle for the given layer, which'll update State.state.favouriteLayers right away | ||||||
|  |      * @param layer | ||||||
|  |      * @constructor | ||||||
|  |      * @private | ||||||
|  |      */ | ||||||
|  |     private static CreateLayerToggle(layer: LayerConfig): Toggle { | ||||||
|  |         const iconUrl = layer.icon.GetRenderValue({id: "node/-1"}).txt | ||||||
|  |         let icon :BaseUIElement =new Combine([ layer.GenerateLeafletStyle( | ||||||
|  |             new UIEventSource<any>({id: "node/-1"}), | ||||||
|  |             false, | ||||||
|  |             "2em" | ||||||
|  |         ).icon.html]).SetClass("relative") | ||||||
|  |         let iconUnset =new Combine([ layer.GenerateLeafletStyle( | ||||||
|  |             new UIEventSource<any>({id: "node/-1"}), | ||||||
|  |             false, | ||||||
|  |             "2em" | ||||||
|  |         ).icon.html]).SetClass("relative") | ||||||
|  | 
 | ||||||
|  |         iconUnset.SetStyle("opacity:0.1") | ||||||
|  | 
 | ||||||
|  |         let name = layer.name ; | ||||||
|  |         if (name === undefined) { | ||||||
|  |             return undefined; | ||||||
|  |         } | ||||||
|  |         const content = new Combine([ | ||||||
|  |             Translations.WT(name).Clone().SetClass("font-bold"), | ||||||
|  |             Translations.WT(layer.description)?.Clone() | ||||||
|  |         ]).SetClass("flex flex-col") | ||||||
|  | 
 | ||||||
|  |         const contentUnselected = new Combine([ | ||||||
|  |             Translations.WT(name).Clone().SetClass("font-bold"), | ||||||
|  |             Translations.WT(layer.description)?.Clone() | ||||||
|  |         ]).SetClass("flex flex-col line-through") | ||||||
|  | 
 | ||||||
|         return new Toggle( |         return new Toggle( | ||||||
|             new Combine([ |             new SubtleButton( | ||||||
|                 t.panelIntro, |                 icon, | ||||||
|                 ...this.checkboxes |                 content ), | ||||||
|             ]), |             new SubtleButton( | ||||||
|             t.loginNeeded, |                 iconUnset, | ||||||
|             State.state.osmConnection.isLoggedIn |                 contentUnselected | ||||||
|              |             ), | ||||||
|         ) |             State.state.favouriteLayers.map(favLayers => { | ||||||
|  |                 return favLayers.indexOf(layer.id) >= 0 | ||||||
|  |             }, [], (selected, current) => { | ||||||
|  |                 if (!selected && current.indexOf(layer.id) <= 0) { | ||||||
|  |                     // Not selected and not contained: nothing to change: we return current as is
 | ||||||
|  |                     return current; | ||||||
|  |                 } | ||||||
|  |                 if (selected && current.indexOf(layer.id) >= 0) { | ||||||
|  |                     // Selected and contained: this is fine!
 | ||||||
|  |                     return current; | ||||||
|  |                 } | ||||||
|  |                 const clone = [...current] | ||||||
|  |                 if (selected) { | ||||||
|  |                     clone.push(layer.id) | ||||||
|  |                 } else { | ||||||
|  |                     clone.splice(clone.indexOf(layer.id), 1) | ||||||
|  |                 } | ||||||
|  |                 return clone | ||||||
|  |             }) | ||||||
|  |         ).ToggleOnClick(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue