forked from MapComplete/MapComplete
		
	Add share tab
This commit is contained in:
		
							parent
							
								
									dcdfa8159b
								
							
						
					
					
						commit
						7b91d2574f
					
				
					 17 changed files with 573 additions and 104 deletions
				
			
		|  | @ -103,22 +103,13 @@ export class WelcomeMessage extends UIElement { | |||
|     } | ||||
| 
 | ||||
|     InnerRender(): string { | ||||
|         return "<span id='welcomeMessage'>" + | ||||
|         return "<span>" + | ||||
|             this.description.Render() + | ||||
|             (this.userDetails.data.loggedIn ? this.welcomeBack : this.plzLogIn).Render() + | ||||
|             this.tail.Render() + | ||||
|             "<br/>" + | ||||
|             this.languagePicker.Render() + | ||||
|             "</span>" | ||||
| 
 | ||||
|             ; | ||||
|         /* | ||||
|         return new VariableUiElement( | ||||
|             this.userDetails.map((userdetails) => { | ||||
|             }), | ||||
|             function () { | ||||
|                 | ||||
|             }).ListenTo(Locale.language);*/ | ||||
|             "</span>"; | ||||
|     } | ||||
| 
 | ||||
|     protected InnerUpdate(htmlElement: HTMLElement) { | ||||
|  |  | |||
							
								
								
									
										74
									
								
								InitUiElements.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								InitUiElements.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,74 @@ | |||
| import {Layout, WelcomeMessage} from "./Customizations/Layout"; | ||||
| import Locale from "./UI/i18n/Locale"; | ||||
| import Translations from "./UI/i18n/Translations"; | ||||
| import {TabbedComponent} from "./UI/Base/TabbedComponent"; | ||||
| import {ShareScreen} from "./UI/ShareScreen"; | ||||
| import {FixedUiElement} from "./UI/Base/FixedUiElement"; | ||||
| import {CheckBox} from "./UI/Input/CheckBox"; | ||||
| import Combine from "./UI/Base/Combine"; | ||||
| import {OsmConnection} from "./Logic/OsmConnection"; | ||||
| import {Basemap} from "./Logic/Basemap"; | ||||
| import {UIEventSource} from "./UI/UIEventSource"; | ||||
| import {UIElement} from "./UI/UIElement"; | ||||
| 
 | ||||
| export class InitUiElements { | ||||
| 
 | ||||
| 
 | ||||
|     static OnlyIf(featureSwitch: UIEventSource<string>, callback: () => void) { | ||||
|         featureSwitch.addCallback(() => { | ||||
| 
 | ||||
|             if (featureSwitch.data === "false") { | ||||
|                 return; | ||||
|             } | ||||
|             callback(); | ||||
|         }); | ||||
| 
 | ||||
|         if (featureSwitch.data !== "false") { | ||||
|             callback(); | ||||
|         } | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     static InitWelcomeMessage(layoutToUse: Layout, osmConnection: OsmConnection, bm: Basemap, | ||||
|                               fullScreenMessage: UIEventSource<UIElement>) { | ||||
| 
 | ||||
|         const welcome = new WelcomeMessage(layoutToUse, | ||||
|             Locale.CreateLanguagePicker(layoutToUse, Translations.t.general.pickLanguage), | ||||
|             osmConnection); | ||||
| 
 | ||||
|         const fullOptions = new TabbedComponent([ | ||||
|             {header: `<img src='${layoutToUse.icon}'>`, content: welcome}, | ||||
|             {header: `<img src='${'./assets/osm-logo.svg'}'>`, content: Translations.t.general.openStreetMapIntro}, | ||||
|             {header: `<img src='${'./assets/share.svg'}'>`, content: new ShareScreen(layoutToUse, bm.Location)} | ||||
|         ]) | ||||
| 
 | ||||
|         const help = new FixedUiElement(`<div class='collapse-button-img'><img src='assets/help.svg'  alt='help'></div>`); | ||||
|         const close = new FixedUiElement(`<div class='collapse-button-img'><img src='assets/close.svg'  alt='close'></div>`); | ||||
|         new CheckBox( | ||||
|             new Combine([ | ||||
|                 "<span class='collapse-button'>", close, "</span>", | ||||
|                 "<span id='welcomeMessage'>", fullOptions.onClick(() => { | ||||
|                 }), "</span>"]), | ||||
|             new Combine(["<span class='open-button'>", help, "</span>"]) | ||||
|             , true | ||||
|         ).AttachTo("messagesbox"); | ||||
| 
 | ||||
| 
 | ||||
|         const welcome2 = new WelcomeMessage(layoutToUse, Locale.CreateLanguagePicker(layoutToUse, Translations.t.general.pickLanguage), osmConnection) | ||||
| 
 | ||||
|         const fullOptions2 = new TabbedComponent([ | ||||
|             {header: `<img src='${layoutToUse.icon}'>`, content: welcome2}, | ||||
|             {header: `<img src='${'./assets/osm-logo.svg'}'>`, content: Translations.t.general.openStreetMapIntro}, | ||||
|             {header: `<img src='${'./assets/share.svg'}'>`, content: new ShareScreen(layoutToUse, bm.Location)} | ||||
|         ]) | ||||
| 
 | ||||
| 
 | ||||
|         fullScreenMessage.setData(fullOptions2) | ||||
|         new FixedUiElement(`<div class='collapse-button-img' class="shadow"><img src='assets/help.svg'  alt='help'></div>`).onClick(() => { | ||||
|             fullScreenMessage.setData(fullOptions2) | ||||
|         }).AttachTo("help-button-mobile"); | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -7,10 +7,10 @@ export class QueryParameters { | |||
| 
 | ||||
|     private static order: string [] = ["layout","test","z","lat","lon"]; | ||||
|     private static knownSources = QueryParameters.init(); | ||||
|     private static defaults = {} | ||||
|      | ||||
|     private static addOrder(key){ | ||||
|         if(this.order.indexOf(key) < 0){ | ||||
|             console.log("Adding order", key) | ||||
|             this.order.push(key) | ||||
|         } | ||||
|     } | ||||
|  | @ -41,18 +41,22 @@ export class QueryParameters { | |||
|             if (QueryParameters.knownSources[key] === undefined || QueryParameters.knownSources[key].data === undefined) { | ||||
|                 continue; | ||||
|             } | ||||
|             if (QueryParameters.knownSources[key].data == QueryParameters.defaults[key]) { | ||||
|                 continue; | ||||
|             } | ||||
|             parts.push(encodeURIComponent(key) + "=" + encodeURIComponent(QueryParameters.knownSources[key].data)) | ||||
|         } | ||||
|         history.replaceState(null, "", "?" + parts.join("&")); | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     public static GetQueryParameter(key: string): UIEventSource<string> { | ||||
|     public static GetQueryParameter(key: string, deflt: string): UIEventSource<string> { | ||||
|         QueryParameters.defaults[key] = deflt; | ||||
|         if (QueryParameters.knownSources[key] !== undefined) { | ||||
|             return QueryParameters.knownSources[key]; | ||||
|         } | ||||
|         QueryParameters.addOrder(key); | ||||
|         const source = new UIEventSource<string>(undefined); | ||||
|         const source = new UIEventSource<string>(deflt); | ||||
|         QueryParameters.knownSources[key] = source; | ||||
|         source.addCallback(() => QueryParameters.Serialize()) | ||||
|         return source; | ||||
|  |  | |||
							
								
								
									
										42
									
								
								UI/Base/TabbedComponent.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								UI/Base/TabbedComponent.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,42 @@ | |||
| import {UIElement} from "../UIElement"; | ||||
| import {UIEventSource} from "../UIEventSource"; | ||||
| import Translations from "../i18n/Translations"; | ||||
| 
 | ||||
| 
 | ||||
| export class TabbedComponent extends UIElement { | ||||
| 
 | ||||
|     private headers: UIElement[] = []; | ||||
|     private content: UIElement[] = []; | ||||
| 
 | ||||
|     constructor(elements: { header: UIElement | string, content: UIElement | string }[]) { | ||||
|         super(new UIEventSource<number>(0)); | ||||
|         const self = this; | ||||
|         for (let i = 0; i < elements.length; i++) { | ||||
|             let element = elements[i]; | ||||
|             this.headers.push(Translations.W(element.header).onClick(() => self._source.setData(i))); | ||||
|             this.content.push(Translations.W(element.content)); | ||||
|         } | ||||
| 
 | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     InnerRender(): string { | ||||
|         let html = ""; | ||||
| 
 | ||||
|         let headerBar = ""; | ||||
|         for (let i = 0; i < this.headers.length; i++) { | ||||
|             let header = this.headers[i]; | ||||
| 
 | ||||
|             headerBar += `<div class=\'tab-single-header ${i == this._source.data ? 'tab-active' : 'tab-non-active'}\'>` + | ||||
|                 header.Render() + "</div>" | ||||
|         } | ||||
| 
 | ||||
| 
 | ||||
|         headerBar = "<div class='tabs-header-bar'>" + headerBar + "</div>" | ||||
| 
 | ||||
|         const content = this.content[this._source.data].Render(); | ||||
| 
 | ||||
|         return headerBar + "<div class='tab-content'>" + content + "</div>"; | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -10,7 +10,7 @@ export class VerticalCombine extends UIElement { | |||
|         this._className = className; | ||||
|     } | ||||
| 
 | ||||
|     protected InnerRender(): string { | ||||
|     InnerRender(): string { | ||||
|         let html = ""; | ||||
|         for (const element of this._elements) { | ||||
|             if (!element.IsEmpty()) { | ||||
|  |  | |||
|  | @ -1,4 +1,6 @@ | |||
| export class Img { | ||||
|     static readonly checkmark = `<svg width="26" height="18" viewBox="0 0 26 18" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M3 7.28571L10.8261 15L23 3" stroke="black" stroke-width="4" stroke-linejoin="round"/></svg>`; | ||||
|     static readonly no_checkmark = `<svg width="26" height="18" viewBox="0 0 26 18" fill="none" xmlns="http://www.w3.org/2000/svg"></svg>`; | ||||
| 
 | ||||
|     static osmAbstractLogo: string = | ||||
|         "<svg class='osm-logo' xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" width=\"24px\" version=\"1.1\" viewBox=\"0 0 66 64\">" + | ||||
|  | @ -16,6 +18,7 @@ export class Img { | |||
|         <path d="M26.3988 13.7412C26.2956 13.9661 26.1026 14.081 25.8927 14.1924C21.8198 16.3577 17.749 18.5258 13.6815 20.7013C13.492 20.8025 13.3602 20.7902 13.1795 20.6938C9.09638 18.5114 5.01059 16.3359 0.924798 14.1582C0.399637 13.8786 0.307921 13.2646 0.735251 12.838C0.829005 12.7443 0.947217 12.6705 1.06407 12.6055C1.56545 12.3279 2.07635 12.0654 2.57297 11.7789C2.74214 11.6812 2.86579 11.6921 3.03291 11.7817C6.27492 13.5155 9.52303 15.2378 12.761 16.9792C13.2352 17.2343 13.6394 17.2322 14.1129 16.9772C17.3509 15.2358 20.5996 13.5142 23.8416 11.7796C24.0095 11.69 24.1338 11.6818 24.3016 11.7789C24.7384 12.0339 25.1821 12.2794 25.6352 12.5037C25.9701 12.6691 26.2426 12.8831 26.3995 13.2304C26.3988 13.4014 26.3988 13.5716 26.3988 13.7412Z" fill="#003B8B"/> | ||||
|         </svg>  ` | ||||
| 
 | ||||
|      | ||||
|     static openFilterButton: string = `<svg width="22" height="22" viewBox="0 0 22 22" fill="none" xmlns="http://www.w3.org/2000/svg">
 | ||||
| <path d="M20 2L2 20M20 20L2 2" stroke="#003B8B" stroke-width="4"/> | ||||
| </svg>  ` | ||||
|  |  | |||
|  | @ -6,26 +6,26 @@ import instantiate = WebAssembly.instantiate; | |||
| 
 | ||||
| 
 | ||||
| export class CheckBox extends UIElement{ | ||||
|     private readonly _data: UIEventSource<boolean>; | ||||
|     public readonly isEnabled: UIEventSource<boolean>; | ||||
|     private readonly _showEnabled: string|UIElement; | ||||
|     private readonly _showDisabled: string|UIElement; | ||||
| 
 | ||||
|     constructor(showEnabled: string | UIElement, showDisabled: string | UIElement, data: UIEventSource<boolean> | boolean = false) { | ||||
|         super(undefined); | ||||
|         this._data = | ||||
|         this.isEnabled = | ||||
|             data instanceof UIEventSource ? data : new UIEventSource(data ?? false); | ||||
|         this.ListenTo(this._data); | ||||
|         this.ListenTo(this.isEnabled); | ||||
|         this._showEnabled = showEnabled; | ||||
|         this._showDisabled = showDisabled; | ||||
|         const self = this; | ||||
|         this.onClick(() => { | ||||
|             self._data.setData(!self._data.data); | ||||
|             self.isEnabled.setData(!self.isEnabled.data); | ||||
|         }) | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     InnerRender(): string { | ||||
|         if (this._data.data) { | ||||
|         if (this.isEnabled.data) { | ||||
|             return Translations.W(this._showEnabled).Render(); | ||||
|         } else { | ||||
|             return Translations.W(this._showDisabled).Render(); | ||||
|  |  | |||
|  | @ -3,6 +3,7 @@ import { FilteredLayer } from "../Logic/FilteredLayer"; | |||
| import { CheckBox } from "./Input/CheckBox"; | ||||
| import Combine from "./Base/Combine"; | ||||
| import {Utils} from "../Utils"; | ||||
| import {Img} from "./Img"; | ||||
| 
 | ||||
| export class LayerSelection extends UIElement{ | ||||
| 
 | ||||
|  | @ -25,9 +26,7 @@ export class LayerSelection extends UIElement{ | |||
|           this._checkboxes.push(new CheckBox( | ||||
|               new Combine([checkbox, icon, name]), | ||||
|               new Combine([ | ||||
|                   `<svg width="26" height="18" viewBox="0 0 26 18" fill="none" xmlns="http://www.w3.org/2000/svg">
 | ||||
|             <path d="M3 7.28571L10.8261 15L23 3" stroke="#ffffff" stroke-width="4" stroke-linejoin="round"/> | ||||
|             </svg>`,
 | ||||
|                   Img.checkmark, | ||||
|                   icon, | ||||
|                   layer.layerDef.name]), | ||||
|               layer.isDisplayed)); | ||||
|  |  | |||
							
								
								
									
										159
									
								
								UI/ShareScreen.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										159
									
								
								UI/ShareScreen.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,159 @@ | |||
| import {UIElement} from "./UIElement"; | ||||
| import {Layout} from "../Customizations/Layout"; | ||||
| import Translations from "./i18n/Translations"; | ||||
| import {FixedUiElement} from "./Base/FixedUiElement"; | ||||
| import Combine from "./Base/Combine"; | ||||
| import {VariableUiElement} from "./Base/VariableUIElement"; | ||||
| import {UIEventSource} from "./UIEventSource"; | ||||
| import {CheckBox} from "./Input/CheckBox"; | ||||
| import {VerticalCombine} from "./Base/VerticalCombine"; | ||||
| import {QueryParameters} from "../Logic/QueryParameters"; | ||||
| import {Img} from "./Img"; | ||||
| 
 | ||||
| export class ShareScreen extends UIElement { | ||||
| 
 | ||||
|     private _shareButton: UIElement; | ||||
| 
 | ||||
|     private _options: UIElement; | ||||
|     private _iframeCode: UIElement; | ||||
|     private _link: UIElement; | ||||
|     private _linkStatus: UIElement; | ||||
| 
 | ||||
|     constructor(layout: Layout, currentLocation: UIEventSource<{ zoom: number, lat: number, lon: number }>) { | ||||
|         super(undefined) | ||||
|         const tr = Translations.t.general.sharescreen; | ||||
| 
 | ||||
|         const optionCheckboxes: UIElement[] = [] | ||||
|         const optionParts: (UIEventSource<string>)[] = []; | ||||
| 
 | ||||
|         const includeLocation = new CheckBox( | ||||
|             new Combine([Img.checkmark, "Include current location"]), | ||||
|             new Combine([Img.no_checkmark, "Include current location"]), | ||||
|             true | ||||
|         ) | ||||
|         optionCheckboxes.push(includeLocation); | ||||
|         optionParts.push(includeLocation.isEnabled.map((includeL) => { | ||||
|             if (includeL) { | ||||
|                 return `z=${currentLocation.data.zoom}&lat=${currentLocation.data.lat}&lon=${currentLocation.data.lon}` | ||||
|             } else { | ||||
|                 return null; | ||||
|             } | ||||
|         }, [currentLocation])); | ||||
| 
 | ||||
| 
 | ||||
|         const switches = [{urlName: "fs-userbadge", human: "Enable the login-button"}, | ||||
|             {urlName: "fs-search", human: "Enable search bar"}, | ||||
|             {urlName: "fs-welcome-message", human: "Enable the welcome message"}, | ||||
|             {urlName: "fs-layers", human: "Enable layer control"}, | ||||
|             {urlName: "fs-add-new", human: "Enable the 'add new POI' button"} | ||||
|         ] | ||||
| 
 | ||||
| 
 | ||||
|         for (const swtch of switches) { | ||||
| 
 | ||||
|             const checkbox = new CheckBox( | ||||
|                 new Combine([Img.checkmark, swtch.human]), | ||||
|                 new Combine([Img.no_checkmark, swtch.human]), | ||||
|                 true | ||||
|             ); | ||||
|             optionCheckboxes.push(checkbox); | ||||
|             optionParts.push(checkbox.isEnabled.map((isEn) => { | ||||
|                 if (isEn) { | ||||
|                     return null; | ||||
|                 } else { | ||||
|                     return `${swtch.urlName}=false` | ||||
|                 } | ||||
|             })) | ||||
| 
 | ||||
| 
 | ||||
|         } | ||||
| 
 | ||||
| 
 | ||||
|         this._options = new VerticalCombine(optionCheckboxes) | ||||
|         const url = currentLocation.map(() => { | ||||
| 
 | ||||
|             let literalText = "https://pietervdvn.github.io/MapComplete/" + layout.name + ".html" | ||||
| 
 | ||||
|             const parts = []; | ||||
|             for (const part of optionParts) { | ||||
|                 if (part.data === null) { | ||||
|                     continue; | ||||
|                 } | ||||
|                 parts.push(part.data); | ||||
|             } | ||||
| 
 | ||||
|             if (parts.length === 0) { | ||||
|                 return literalText; | ||||
|             } | ||||
| 
 | ||||
|             return literalText + "?" + parts.join("&"); | ||||
|         }, optionParts); | ||||
|         this._iframeCode = new VariableUiElement( | ||||
|             url.map((url) => { | ||||
|                 return `<span class='literal-code iframe-code-block'>
 | ||||
|                         <iframe src="${url}" title="${layout.name} with MapComplete"></iframe>  | ||||
|                     </span>` | ||||
|             }) | ||||
|         ); | ||||
| 
 | ||||
| 
 | ||||
|         this._link = new VariableUiElement( | ||||
|             url.map((url) => { | ||||
|                 return `<input type="text" value=" ${url}" id="code-link--copyable" style="width:90%"readonly>` | ||||
|             }) | ||||
|         ); | ||||
| 
 | ||||
| 
 | ||||
|         const status = new UIEventSource(" "); | ||||
|         this._linkStatus = new VariableUiElement(status); | ||||
|         const self = this; | ||||
|         this._link.onClick(async () => { | ||||
| 
 | ||||
|             const shareData = { | ||||
|                 title: Translations.W(layout.name).InnerRender(), | ||||
|                 text: Translations.W(layout.description).InnerRender(), | ||||
|                 url: self._link.data, | ||||
|             } | ||||
| 
 | ||||
|             function rejected() { | ||||
|                 status.setData("Copying to clipboard...") | ||||
|                 var copyText = document.getElementById("code-link--copyable"); | ||||
| 
 | ||||
|                 // @ts-ignore
 | ||||
|                 copyText.select(); | ||||
|                 // @ts-ignore
 | ||||
|                 copyText.setSelectionRange(0, 99999); /*For mobile devices*/ | ||||
| 
 | ||||
|                 document.execCommand("copy"); | ||||
|                 status.setData("Copied to clipboard") | ||||
|             } | ||||
| 
 | ||||
|             try { | ||||
|                 navigator.share(shareData) | ||||
|                     .then(() => { | ||||
|                         status.setData("Thanks for sharing!") | ||||
|                     }, rejected) | ||||
|                     .catch(rejected) | ||||
|             } catch (err) { | ||||
|                 rejected(); | ||||
|             } | ||||
| 
 | ||||
|         }); | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     InnerRender(): string { | ||||
| 
 | ||||
|         const tr = Translations.t.general.sharescreen; | ||||
| 
 | ||||
|         return new VerticalCombine([ | ||||
|             tr.intro, | ||||
|             this._link, | ||||
|             this._linkStatus, | ||||
|             tr.embedIntro, | ||||
|             this._options, | ||||
|             this._iframeCode | ||||
|         ]).Render() | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -136,6 +136,8 @@ export abstract class UIElement extends UIEventSource<string>{ | |||
|     public IsEmpty(): boolean { | ||||
|         return this.InnerRender() === ""; | ||||
|     } | ||||
|      | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|  |  | |||
|  | @ -528,6 +528,33 @@ export default class Translations { | |||
|                     nl: "Het telefoonnummer van {category} is <a href='tel:{phone}' target='_blank'>{phone}</a>" | ||||
|                 }) | ||||
| 
 | ||||
|             }, | ||||
|             openStreetMapIntro: new T({ | ||||
|                 en: "<h2>An Open Map</h2>" + | ||||
|                     "<p></p>Wouldn't it be cool if there was a single map, which everyone could freely use and edit?" + | ||||
|                     "A single place to store all geo-information? Then, all those websites with different, small and incompatible maps (which are always outdated) wouldn't be needed anymore.</p>" + | ||||
|                     "<p><b><a href='https://OpenStreetMap.org' target='_blank'>OpenStreetMap</a></b> is this map. The map data can be used for free (with <a href='https://osm.org/copyright' target='_blank'>attribution and publication of changes to that data</a>)." + | ||||
|                     " On top of that, everyone can freely add new data and fix errors. This website uses OpenStreetMap as well. All the data is from there, and your answers and corrections are added there as well.</p>" + | ||||
|                     "<p>A ton of people and application already use OpenStreetMap:  <a href='https://maps.me/' traget='_blank'>Maps.me</a>, <a href='https://osmAnd.net' traget='_blank'>OsmAnd</a>, but also the maps at Facebook, Intsagram, Apple-maps and Bing-maps are (partly) powered by OpenStreetMap." + | ||||
|                     "If you change something here, it'll be reflected in those applications too - after their next update!</p>", | ||||
|                 nl: "<h2>Een open kaart</h2>" + | ||||
|                     "<p>Zou het niet fantastisch zijn als er een open kaart zou zijn, die door iedereen aangepast én gebruikt kon worden? Waar iedereen zijn interesses aan zou kunnen toevoegen?" + | ||||
|                     "Dan zouden er geen duizend-en-één verschillende kleine kaartjes, websites, ... meer nodig zijn</p>" + | ||||
|                     "<p><b><a href='https://OpenStreetMap.org' target='_blank'>OpenStreetMap</a></b> is deze open kaart. Je mag de kaartdata gratis gebruiken (mits <a href='https://osm.org/copyright' target='_blank'>bronvermelding en herpublicatie van aanpassingen</a>). Daarenboven mag je de kaart ook gratis aanpassen als je een account maakt." + | ||||
|                     "Ook deze website is gebaseerd op OpenStreetMap. Als je hier een vraag beantwoord, gaat het antwoord daar ook naartoe</p>" + | ||||
|                     "<p>Tenslotte zijn er reeds vele gebruikers van OpenStreetMap. Denk maar <a href='https://maps.me/' traget='_blank'>Maps.me</a>, <a href='https://osmAnd.net' traget='_blank'>OsmAnd</a>, verschillende gespecialiseerde routeplanners, de achtergrondkaarten op Facebook, Instagram, ... Zelfs Apple Maps en Bing-Maps gebruiken OpenStreetMap in hun kaarten!</p>" + | ||||
|                     "<p></p>Kortom, als je hier een antwoord geeft of een fout aanpast, zal dat na een tijdje ook in al dié applicaties te zien zijn.</p>" | ||||
|             }), | ||||
|              | ||||
|             sharescreen: { | ||||
|                 intro:new T({ | ||||
|                     en: "<h3>Share this map</h3> Share this map by copying the link below and sending it to friends and family:" | ||||
|                 }), | ||||
|                 embedIntro: new T({ | ||||
|                     en: "<h3>Embed on your website</h3>Please, embed this map into your website. We encourage you to do it - you don't even have to ask permission. It is free, and always will be. The more people using this, the more valuable it becomes." | ||||
|                 }) | ||||
| 
 | ||||
| 
 | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  |  | |||
							
								
								
									
										100
									
								
								assets/share.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								assets/share.svg
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,100 @@ | |||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> | ||||
| 
 | ||||
| <svg | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    width="100" | ||||
|    height="100" | ||||
|    viewBox="0 0 26.458333 26.458334" | ||||
|    version="1.1" | ||||
|    id="svg8" | ||||
|    inkscape:version="0.92.4 (5da689c313, 2019-01-14)" | ||||
|    sodipodi:docname="share.svg"> | ||||
|   <defs | ||||
|      id="defs2" /> | ||||
|   <sodipodi:namedview | ||||
|      id="base" | ||||
|      pagecolor="#ffffff" | ||||
|      bordercolor="#666666" | ||||
|      borderopacity="1.0" | ||||
|      inkscape:pageopacity="0.0" | ||||
|      inkscape:pageshadow="2" | ||||
|      inkscape:zoom="4" | ||||
|      inkscape:cx="-15.237738" | ||||
|      inkscape:cy="36.323203" | ||||
|      inkscape:document-units="px" | ||||
|      inkscape:current-layer="layer1" | ||||
|      showgrid="false" | ||||
|      units="px" | ||||
|      showguides="true" | ||||
|      inkscape:guide-bbox="true" | ||||
|      inkscape:window-width="1920" | ||||
|      inkscape:window-height="1001" | ||||
|      inkscape:window-x="0" | ||||
|      inkscape:window-y="1050" | ||||
|      inkscape:window-maximized="1"> | ||||
|     <sodipodi:guide | ||||
|        position="13.229167,23.859748" | ||||
|        orientation="1,0" | ||||
|        id="guide815" | ||||
|        inkscape:locked="false" /> | ||||
|     <sodipodi:guide | ||||
|        position="14.944824,13.229167" | ||||
|        orientation="0,1" | ||||
|        id="guide817" | ||||
|        inkscape:locked="false" /> | ||||
|     <sodipodi:guide | ||||
|        position="19.182291,3.4395834" | ||||
|        orientation="1,0" | ||||
|        id="guide852" | ||||
|        inkscape:locked="false" /> | ||||
|   </sodipodi:namedview> | ||||
|   <metadata | ||||
|      id="metadata5"> | ||||
|     <rdf:RDF> | ||||
|       <cc:Work | ||||
|          rdf:about=""> | ||||
|         <dc:format>image/svg+xml</dc:format> | ||||
|         <dc:type | ||||
|            rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|         <dc:title /> | ||||
|       </cc:Work> | ||||
|     </rdf:RDF> | ||||
|   </metadata> | ||||
|   <g | ||||
|      inkscape:label="Layer 1" | ||||
|      inkscape:groupmode="layer" | ||||
|      id="layer1" | ||||
|      transform="translate(0,-270.54165)"> | ||||
|     <path | ||||
|        style="fill:none;stroke:#000000;stroke-width:2.43863511;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" | ||||
|        d="m 19.212364,278.17517 -11.9689358,5.52059 11.9388628,5.50669" | ||||
|        id="path819" | ||||
|        inkscape:connector-curvature="0" | ||||
|        sodipodi:nodetypes="ccc" /> | ||||
|     <circle | ||||
|        style="fill:#000000;fill-opacity:1;stroke:none;stroke-width:0.53329796;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:0.97014926" | ||||
|        id="path820" | ||||
|        cx="7.2434282" | ||||
|        cy="283.69574" | ||||
|        r="3.9119694" /> | ||||
|     <circle | ||||
|        style="fill:#000000;fill-opacity:1;stroke:none;stroke-width:0.53329796;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:0.97014926" | ||||
|        id="path820-3" | ||||
|        cx="19.48818" | ||||
|        cy="289.22873" | ||||
|        r="3.9119689" /> | ||||
|     <circle | ||||
|        style="fill:#000000;fill-opacity:1;stroke:none;stroke-width:0.53329796;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:0.97014926" | ||||
|        id="path820-3-6" | ||||
|        cx="19.48818" | ||||
|        cy="277.56281" | ||||
|        r="3.9119689" /> | ||||
|   </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 3.3 KiB | 
							
								
								
									
										110
									
								
								index.css
									
										
									
									
									
								
							
							
						
						
									
										110
									
								
								index.css
									
										
									
									
									
								
							|  | @ -21,8 +21,6 @@ form { | |||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| #leafletDiv { | ||||
|     height: 100%; | ||||
| } | ||||
|  | @ -343,18 +341,14 @@ form { | |||
| 
 | ||||
| #welcomeMessage { | ||||
|     display: inline-block; | ||||
|     background-color: white; | ||||
|     padding: 1em; | ||||
|     margin-left: 3.5em; | ||||
|     padding-left: 1em; | ||||
|     padding-bottom: 2em; | ||||
|     border-radius: 2em; | ||||
|     border-top-left-radius: 0; | ||||
|     border-bottom-left-radius: 0; | ||||
|     max-width: calc(max(35vw, 30em)); | ||||
|     max-width: 30em; | ||||
|     width: 35vw; | ||||
|     max-height: calc(100vh - 15em); | ||||
|     overflow-y: auto; | ||||
|     box-shadow: 0 0 10px #00000066; | ||||
| } | ||||
| 
 | ||||
| #messagesbox { | ||||
|  | @ -647,32 +641,15 @@ form { | |||
|     height: auto; | ||||
| } | ||||
| 
 | ||||
| #bottomRight { | ||||
| #top-right { | ||||
| 
 | ||||
|     display: block ruby; | ||||
|     position: absolute; | ||||
| 
 | ||||
|     margin: auto; | ||||
|     right: 1%; | ||||
|     bottom: 1.5em; | ||||
|     height: auto; | ||||
|     min-height: 1em; | ||||
|     width: auto; | ||||
| 
 | ||||
|     font-size: large; | ||||
| 
 | ||||
|     padding: 2em; | ||||
|     border-radius: 2em; | ||||
|     display: block; | ||||
|     right: 0.5em; | ||||
|     top: 0.5em; | ||||
|     z-index: 5000; | ||||
| 
 | ||||
|     opacity: 1; | ||||
|     background-color: white; | ||||
| 
 | ||||
|     transition: all 500ms linear; | ||||
| 
 | ||||
|     text-align: center; | ||||
|     horiz-align: center; | ||||
|     font-weight: bold; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
|  | @ -844,7 +821,6 @@ form { | |||
| 
 | ||||
| 
 | ||||
| .license-picker { | ||||
|     background-color: orange; | ||||
|     float: left; | ||||
| } | ||||
| 
 | ||||
|  | @ -1130,3 +1106,77 @@ form { | |||
|     padding-right: 0.3em; | ||||
|     border-radius: 1.5em; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /******** TabbedElement ****/ | ||||
| 
 | ||||
| .tabs-header-bar { | ||||
|     padding-left: 1em; | ||||
|     padding-top: 10px; /* For the shadow */ | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|     flex-wrap: nowrap; | ||||
|     justify-content: flex-start; | ||||
|     align-items: start; | ||||
|     background-color: white; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| .tab-single-header img { | ||||
|     height: 3em; | ||||
|     width: 3em; | ||||
|     padding: 0.5em; | ||||
| } | ||||
| 
 | ||||
| .tab-content { | ||||
|     padding: 1em; | ||||
|     z-index: 5002; | ||||
|     background-color: white; | ||||
|     position: relative; | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| .tab-single-header { | ||||
|     border-top-left-radius: 1em; | ||||
|     border-top-right-radius: 1em; | ||||
|     box-shadow: 0 0 10px black; | ||||
|     z-index: 5000; | ||||
|     border-bottom: 1px solid white; | ||||
|     padding-bottom: 1px; | ||||
| } | ||||
| 
 | ||||
| .tab-active { | ||||
|     background-color: white; | ||||
|     z-index: 5001; | ||||
| } | ||||
| 
 | ||||
| .tab-non-active { | ||||
|     background-color: #e5f5ff; | ||||
|     opacity: 0.5; | ||||
|     border-bottom: 1px solid lightgray; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /****** ShareScreen *****/ | ||||
| 
 | ||||
| .literal-code { | ||||
|     display: inline-block; | ||||
|     background-color: lightgray; | ||||
|     padding: 0.5em; | ||||
| } | ||||
| 
 | ||||
| .iframe-code-block { | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| .iframe-escape { | ||||
|     background-color: white; | ||||
|     border-radius: 2em; | ||||
|     display: block; | ||||
| } | ||||
| 
 | ||||
| .iframe-escape img{ | ||||
|     padding: 1em; | ||||
|     width: 2em; | ||||
|     height: 2em; | ||||
| } | ||||
|  | @ -28,8 +28,6 @@ | |||
| <div id="topleft-tools"> | ||||
|     <div id="userbadge-and-search"> | ||||
|         <div id="userbadge" class="shadow"> | ||||
|             Loading... If this message persists, check if javascript is enabled and if no extension (uMatrix) is | ||||
|             blocking it. | ||||
|         </div> | ||||
|         <div id="searchbox" class="shadow"></div> | ||||
|     </div> | ||||
|  | @ -42,8 +40,8 @@ | |||
|     <div id="filter__selection"></div> | ||||
| </div> | ||||
| 
 | ||||
| <div id="centermessage">Loading...</div> | ||||
| <div id="bottomRight" style="display: none">ADD</div> | ||||
| <div id="centermessage">Loading MapComplete, hang on...</div> | ||||
| <div id="top-right"></div> | ||||
| 
 | ||||
| <div id="geolocate-button"></div> | ||||
| <div id="leafletDiv"></div> | ||||
|  |  | |||
							
								
								
									
										96
									
								
								index.ts
									
										
									
									
									
								
							
							
						
						
									
										96
									
								
								index.ts
									
										
									
									
									
								
							|  | @ -32,6 +32,9 @@ import {QueryParameters} from "./Logic/QueryParameters"; | |||
| import {Utils} from "./Utils"; | ||||
| import {LocalStorageSource} from "./Logic/LocalStorageSource"; | ||||
| import {Button} from "./UI/Base/Button"; | ||||
| import {TabbedComponent} from "./UI/Base/TabbedComponent"; | ||||
| import {ShareScreen} from "./UI/ShareScreen"; | ||||
| import {InitUiElements} from "./InitUiElements"; | ||||
| 
 | ||||
| 
 | ||||
| // --------------------- Special actions based on the parameters -----------------
 | ||||
|  | @ -44,7 +47,7 @@ if (location.href.startsWith("http://buurtnatuur.be")) { | |||
| 
 | ||||
| if (location.hostname === "localhost" || location.hostname === "127.0.0.1") { | ||||
|     // Set to true if testing and changes should NOT be saved
 | ||||
|     const testing = QueryParameters.GetQueryParameter("test"); | ||||
|     const testing = QueryParameters.GetQueryParameter("test", "true"); | ||||
|     testing.setData(testing.data ?? "true") | ||||
|     // If you have a testfile somewhere, enable this to spoof overpass
 | ||||
|     // This should be hosted independantly, e.g. with `cd assets; webfsd -p 8080` + a CORS plugin to disable cors rules
 | ||||
|  | @ -54,7 +57,7 @@ if (location.hostname === "localhost" || location.hostname === "127.0.0.1") { | |||
| 
 | ||||
| // ----------------- SELECT THE RIGHT QUESTSET -----------------
 | ||||
| 
 | ||||
| let defaultLayout = "all" | ||||
| let defaultLayout = "bookcases" | ||||
| 
 | ||||
| const path = window.location.pathname.split("/").slice(-1)[0]; | ||||
| if (path !== "index.html") { | ||||
|  | @ -76,7 +79,7 @@ for (const k in AllKnownLayouts.allSets) { | |||
|     } | ||||
| } | ||||
| 
 | ||||
| defaultLayout = QueryParameters.GetQueryParameter("layout").data ?? defaultLayout; | ||||
| defaultLayout = QueryParameters.GetQueryParameter("layout", defaultLayout).data; | ||||
| 
 | ||||
| const layoutToUse: Layout = AllKnownLayouts.allSets[defaultLayout] ?? AllKnownLayouts["all"]; | ||||
| console.log("Using layout: ", layoutToUse.name); | ||||
|  | @ -102,13 +105,21 @@ const fullScreenMessage = new UIEventSource<UIElement>(undefined); | |||
| // The latest element that was selected - used to generate the right UI at the right place
 | ||||
| const selectedElement = new UIEventSource<{ feature: any }>(undefined); | ||||
| 
 | ||||
| const zoom = QueryParameters.GetQueryParameter("z") | ||||
| const zoom = QueryParameters.GetQueryParameter("z", "" + layoutToUse.startzoom) | ||||
|     .syncWith(LocalStorageSource.Get("zoom")); | ||||
| const lat = QueryParameters.GetQueryParameter("lat") | ||||
| const lat = QueryParameters.GetQueryParameter("lat", "" + layoutToUse.startLat) | ||||
|     .syncWith(LocalStorageSource.Get("lat")); | ||||
| const lon = QueryParameters.GetQueryParameter("lon") | ||||
| const lon = QueryParameters.GetQueryParameter("lon", "" + layoutToUse.startLon) | ||||
|     .syncWith(LocalStorageSource.Get("lon")); | ||||
| 
 | ||||
| const featureSwitchUserbadge = QueryParameters.GetQueryParameter("fs-userbadge", "true"); | ||||
| const featureSwitchSearch = QueryParameters.GetQueryParameter("fs-search", "true"); | ||||
| const featureSwitchWelcomeMessage = QueryParameters.GetQueryParameter("fs-welcome-message", "true"); | ||||
| const featureSwitchLayers = QueryParameters.GetQueryParameter("fs-layers", "true"); | ||||
| const featureSwitchEmbedded = QueryParameters.GetQueryParameter("fs-embedded", "true"); | ||||
| const featureSwitchAddNew = QueryParameters.GetQueryParameter("fs-add-new", "true"); | ||||
| const featureSwitchIframe = QueryParameters.GetQueryParameter("fs-iframe", "false"); | ||||
| 
 | ||||
| 
 | ||||
| const locationControl = new UIEventSource<{ lat: number, lon: number, zoom: number }>({ | ||||
|     zoom: Utils.asFloat(zoom.data) ?? layoutToUse.startzoom, | ||||
|  | @ -126,7 +137,7 @@ locationControl.addCallback((latlonz) => { | |||
| // ----------------- Prepare the important objects -----------------
 | ||||
| 
 | ||||
| const osmConnection = new OsmConnection( | ||||
|     QueryParameters.GetQueryParameter("test").data === "true" | ||||
|     QueryParameters.GetQueryParameter("test", "false").data === "true" | ||||
| ); | ||||
| 
 | ||||
| 
 | ||||
|  | @ -169,6 +180,7 @@ const bm = new Basemap("leafletDiv", locationControl, new VariableUiElement( | |||
| )); | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| // ------------- Setup the layers -------------------------------
 | ||||
| const addButtons: { | ||||
|     name: UIElement, | ||||
|  | @ -228,7 +240,9 @@ if (flayers.length > 1) { | |||
|     layerControl = new Combine([layerSelection, backgroundMapPicker]); | ||||
| } | ||||
| 
 | ||||
| new CheckBox(layerControl, closedFilterButton).AttachTo("filter__selection"); | ||||
| InitUiElements.OnlyIf(featureSwitchLayers, () => { | ||||
|     new CheckBox(layerControl, closedFilterButton).AttachTo("filter__selection"); | ||||
| }); | ||||
| 
 | ||||
| 
 | ||||
| // ------------------ Setup various other UI elements ------------
 | ||||
|  | @ -240,16 +254,19 @@ Locale.language.addCallback(e => { | |||
| }) | ||||
| 
 | ||||
| 
 | ||||
| new StrayClickHandler(bm, selectedElement, fullScreenMessage, () => { | ||||
|     return new SimpleAddUI(bm.Location, | ||||
|         bm.LastClickLocation, | ||||
|         changes, | ||||
|         selectedElement, | ||||
|         layerUpdater.runningQuery, | ||||
|         osmConnection.userDetails, | ||||
|         addButtons); | ||||
|     } | ||||
| ); | ||||
| InitUiElements.OnlyIf(featureSwitchAddNew, () => { | ||||
|     new StrayClickHandler(bm, selectedElement, fullScreenMessage, () => { | ||||
|             return new SimpleAddUI(bm.Location, | ||||
|                 bm.LastClickLocation, | ||||
|                 changes, | ||||
|                 selectedElement, | ||||
|                 layerUpdater.runningQuery, | ||||
|                 osmConnection.userDetails, | ||||
|                 addButtons); | ||||
|         } | ||||
|     ); | ||||
| }); | ||||
| 
 | ||||
| 
 | ||||
| /** | ||||
|  * Show the questions and information for the selected element | ||||
|  | @ -283,40 +300,31 @@ selectedElement.addCallback((feature) => { | |||
| 
 | ||||
| const pendingChanges = new PendingChanges(changes, secondsTillChangesAreSaved,); | ||||
| 
 | ||||
| new UserBadge(osmConnection.userDetails, | ||||
|     pendingChanges, | ||||
|     Locale.CreateLanguagePicker(layoutToUse), | ||||
|     bm) | ||||
|     .AttachTo('userbadge'); | ||||
| InitUiElements.OnlyIf(featureSwitchUserbadge, () => { | ||||
| 
 | ||||
| new SearchAndGo(bm).AttachTo("searchbox"); | ||||
| 
 | ||||
| const welcome = new WelcomeMessage(layoutToUse,  | ||||
|     Locale.CreateLanguagePicker(layoutToUse, Translations.t.general.pickLanguage), | ||||
|     osmConnection).onClick(() => { | ||||
|     new UserBadge(osmConnection.userDetails, | ||||
|         pendingChanges, | ||||
|         Locale.CreateLanguagePicker(layoutToUse), | ||||
|         bm) | ||||
|         .AttachTo('userbadge'); | ||||
| }); | ||||
| 
 | ||||
| const help = new FixedUiElement(`<div class='collapse-button-img'><img src='assets/help.svg'  alt='help'></div>`); | ||||
| const close = new FixedUiElement(`<div class='collapse-button-img'><img src='assets/close.svg'  alt='close'></div>`); | ||||
| new CheckBox( | ||||
|     new Combine([ | ||||
|         new Combine(["<span class='collapse-button'>", close, "</span>"]), | ||||
|         welcome]), | ||||
|     new Combine(["<span class='open-button'>", help, "</span>"]) | ||||
|     , true | ||||
| ).AttachTo("messagesbox"); | ||||
| 
 | ||||
| InitUiElements.OnlyIf((featureSwitchSearch), () => { | ||||
|     new SearchAndGo(bm).AttachTo("searchbox"); | ||||
| }); | ||||
| 
 | ||||
| new FullScreenMessageBoxHandler(fullScreenMessage, () => { | ||||
|     selectedElement.setData(undefined) | ||||
| }).update(); | ||||
| 
 | ||||
| const welcome2 = new WelcomeMessage(layoutToUse, Locale.CreateLanguagePicker(layoutToUse, Translations.t.general.pickLanguage), osmConnection) | ||||
| fullScreenMessage.setData(welcome2) | ||||
| new FixedUiElement(`<div class='collapse-button-img' class="shadow"><img src='assets/help.svg'  alt='help'></div>`).onClick(() => { | ||||
|     fullScreenMessage.setData(welcome2) | ||||
| }) | ||||
|     .AttachTo("help-button-mobile") | ||||
| InitUiElements.OnlyIf(featureSwitchWelcomeMessage, () => { | ||||
|     InitUiElements.InitWelcomeMessage(layoutToUse, osmConnection, bm, fullScreenMessage) | ||||
| }); | ||||
| 
 | ||||
| if (window != window.top || featureSwitchIframe.data !== "false") { | ||||
|     new FixedUiElement(`<a href='${window.location}' target='_blank'><span class='iframe-escape'><img src='assets/pencil.svg'></span></a>`).AttachTo("top-right") | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| new CenterMessageBox( | ||||
|     minZoom, | ||||
|  |  | |||
|  | @ -5,9 +5,7 @@ | |||
|     <link href="index.css" rel="stylesheet"/>    | ||||
| </head> | ||||
| <body> | ||||
| <span class="image-delete-container"> | ||||
| <div id="maindiv">'maindiv' not attached</div> | ||||
| </span> | ||||
| <div id="extradiv">'extradiv' not attached</div> | ||||
| <script src="./test.ts"></script> | ||||
| </body> | ||||
|  |  | |||
							
								
								
									
										14
									
								
								test.ts
									
										
									
									
									
								
							
							
						
						
									
										14
									
								
								test.ts
									
										
									
									
									
								
							|  | @ -0,0 +1,14 @@ | |||
| import {TabbedComponent} from "./UI/Base/TabbedComponent"; | ||||
| import {FixedUiElement} from "./UI/Base/FixedUiElement"; | ||||
| import {Bookcases} from "./Customizations/Layouts/Bookcases"; | ||||
| import {ShareScreen} from "./UI/ShareScreen"; | ||||
| import {UIEventSource} from "./UI/UIEventSource"; | ||||
| 
 | ||||
| 
 | ||||
| const layout = new Bookcases(); | ||||
| 
 | ||||
| new ShareScreen(layout, new UIEventSource<{zoom: number, lat: number, lon: number}>({ | ||||
|     zoom: 16, | ||||
|     lat: 51.5, | ||||
|     lon:3.2 | ||||
| })).AttachTo("maindiv") | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue