forked from MapComplete/MapComplete
Refactoring fullscreenhandling
This commit is contained in:
parent
e1a4c75391
commit
00f610c589
23 changed files with 346 additions and 245 deletions
|
@ -12,7 +12,6 @@ import CenterMessageBox from "./UI/CenterMessageBox";
|
||||||
import {AllKnownLayouts} from "./Customizations/AllKnownLayouts";
|
import {AllKnownLayouts} from "./Customizations/AllKnownLayouts";
|
||||||
import UserBadge from "./UI/BigComponents/UserBadge";
|
import UserBadge from "./UI/BigComponents/UserBadge";
|
||||||
import SearchAndGo from "./UI/BigComponents/SearchAndGo";
|
import SearchAndGo from "./UI/BigComponents/SearchAndGo";
|
||||||
import FullScreenMessageBox from "./UI/FullScreenMessageBoxHandler";
|
|
||||||
import GeoLocationHandler from "./Logic/Actors/GeoLocationHandler";
|
import GeoLocationHandler from "./Logic/Actors/GeoLocationHandler";
|
||||||
import {LocalStorageSource} from "./Logic/Web/LocalStorageSource";
|
import {LocalStorageSource} from "./Logic/Web/LocalStorageSource";
|
||||||
import {Utils} from "./Utils";
|
import {Utils} from "./Utils";
|
||||||
|
@ -33,7 +32,6 @@ import FeatureSwitched from "./UI/Base/FeatureSwitched";
|
||||||
import LayerConfig from "./Customizations/JSON/LayerConfig";
|
import LayerConfig from "./Customizations/JSON/LayerConfig";
|
||||||
import ShowDataLayer from "./UI/ShowDataLayer";
|
import ShowDataLayer from "./UI/ShowDataLayer";
|
||||||
import Hash from "./Logic/Web/Hash";
|
import Hash from "./Logic/Web/Hash";
|
||||||
import HistoryHandling from "./Logic/Actors/HistoryHandling";
|
|
||||||
import FeaturePipeline from "./Logic/FeatureSource/FeaturePipeline";
|
import FeaturePipeline from "./Logic/FeatureSource/FeaturePipeline";
|
||||||
|
|
||||||
export class InitUiElements {
|
export class InitUiElements {
|
||||||
|
@ -118,8 +116,6 @@ export class InitUiElements {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
new HistoryHandling(Hash.hash, State.state.fullScreenMessage);
|
|
||||||
|
|
||||||
InitUiElements.OnlyIf(State.state.featureSwitchUserbadge, () => {
|
InitUiElements.OnlyIf(State.state.featureSwitchUserbadge, () => {
|
||||||
new UserBadge().AttachTo('userbadge');
|
new UserBadge().AttachTo('userbadge');
|
||||||
});
|
});
|
||||||
|
@ -129,9 +125,6 @@ export class InitUiElements {
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
new FullScreenMessageBox().AttachTo("messagesboxmobile");
|
|
||||||
|
|
||||||
|
|
||||||
InitUiElements.OnlyIf(State.state.featureSwitchWelcomeMessage, () => {
|
InitUiElements.OnlyIf(State.state.featureSwitchWelcomeMessage, () => {
|
||||||
InitUiElements.InitWelcomeMessage()
|
InitUiElements.InitWelcomeMessage()
|
||||||
});
|
});
|
||||||
|
@ -214,18 +207,18 @@ export class InitUiElements {
|
||||||
private static InitWelcomeMessage() {
|
private static InitWelcomeMessage() {
|
||||||
|
|
||||||
const isOpened = new UIEventSource<boolean>(true);
|
const isOpened = new UIEventSource<boolean>(true);
|
||||||
const fullOptions = new FullWelcomePaneWithTabs(() => isOpened.setData(false));
|
const fullOptions = new FullWelcomePaneWithTabs(() => {
|
||||||
|
console.log("Closing the welcome message...")
|
||||||
|
isOpened.setData(false);
|
||||||
|
});
|
||||||
|
|
||||||
// ?-Button on Desktop, opens panel with close-X.
|
// ?-Button on Desktop, opens panel with close-X.
|
||||||
const help = Svg.help_svg().SetClass("open-welcome-button block");
|
const help = Svg.help_svg().SetClass("open-welcome-button block");
|
||||||
const close = Svg.close_svg().SetClass("close-welcome-button");
|
|
||||||
const checkbox = new CheckBox(
|
const checkbox = new CheckBox(
|
||||||
new Combine([
|
|
||||||
close,
|
|
||||||
fullOptions
|
fullOptions
|
||||||
.SetClass("welcomeMessage")
|
.SetClass("welcomeMessage")
|
||||||
.onClick(() => {/*Catch the click*/
|
.onClick(() => {/*Catch the click*/
|
||||||
})]),
|
}),
|
||||||
help
|
help
|
||||||
, isOpened
|
, isOpened
|
||||||
).AttachTo("messagesbox");
|
).AttachTo("messagesbox");
|
||||||
|
@ -252,15 +245,8 @@ export class InitUiElements {
|
||||||
const layerControlPanel = new LayerControlPanel(
|
const layerControlPanel = new LayerControlPanel(
|
||||||
() => State.state.layerControlIsOpened.setData(false))
|
() => State.state.layerControlIsOpened.setData(false))
|
||||||
.SetClass("block p-1 rounded-full");
|
.SetClass("block p-1 rounded-full");
|
||||||
const closeButton = Svg.close_svg()
|
const checkbox = new CheckBox(
|
||||||
.SetClass("layer-selection-toggle")
|
layerControlPanel,
|
||||||
.SetStyle(" background: var(--subtle-detail-color);")
|
|
||||||
const checkbox = new CheckBox(
|
|
||||||
new Combine([
|
|
||||||
closeButton,
|
|
||||||
layerControlPanel])
|
|
||||||
.SetClass("flex flex-row")
|
|
||||||
,
|
|
||||||
Svg.layers_svg().SetClass("layer-selection-toggle"),
|
Svg.layers_svg().SetClass("layer-selection-toggle"),
|
||||||
State.state.layerControlIsOpened
|
State.state.layerControlIsOpened
|
||||||
).AttachTo("layer-selection");
|
).AttachTo("layer-selection");
|
||||||
|
@ -388,7 +374,6 @@ export class InitUiElements {
|
||||||
State.state.selectedElement,
|
State.state.selectedElement,
|
||||||
State.state.filteredLayers,
|
State.state.filteredLayers,
|
||||||
State.state.leafletMap,
|
State.state.leafletMap,
|
||||||
State.state.fullScreenMessage,
|
|
||||||
() => {
|
() => {
|
||||||
return new SimpleAddUI(
|
return new SimpleAddUI(
|
||||||
() => State.state.LastClickLocation.setData(undefined)
|
() => State.state.LastClickLocation.setData(undefined)
|
||||||
|
|
|
@ -17,7 +17,6 @@ export default class StrayClickHandler {
|
||||||
selectedElement: UIEventSource<string>,
|
selectedElement: UIEventSource<string>,
|
||||||
filteredLayers: UIEventSource<{ readonly isDisplayed: UIEventSource<boolean> }[]>,
|
filteredLayers: UIEventSource<{ readonly isDisplayed: UIEventSource<boolean> }[]>,
|
||||||
leafletMap: UIEventSource<L.Map>,
|
leafletMap: UIEventSource<L.Map>,
|
||||||
fullscreenMessage: UIEventSource<{content: UIElement, hashText: string}>,
|
|
||||||
uiToShow: (() => UIElement)) {
|
uiToShow: (() => UIElement)) {
|
||||||
this._uiToShow = uiToShow;
|
this._uiToShow = uiToShow;
|
||||||
const self = this;
|
const self = this;
|
||||||
|
|
|
@ -3,31 +3,70 @@ import LayoutConfig from "../../Customizations/JSON/LayoutConfig";
|
||||||
import Translations from "../../UI/i18n/Translations";
|
import Translations from "../../UI/i18n/Translations";
|
||||||
import Locale from "../../UI/i18n/Locale";
|
import Locale from "../../UI/i18n/Locale";
|
||||||
import {UIElement} from "../../UI/UIElement";
|
import {UIElement} from "../../UI/UIElement";
|
||||||
|
import TagRenderingAnswer from "../../UI/Popup/TagRenderingAnswer";
|
||||||
|
import {ElementStorage} from "../ElementStorage";
|
||||||
|
import Combine from "../../UI/Base/Combine";
|
||||||
|
|
||||||
export default class TitleHandler{
|
class TitleElement extends UIElement {
|
||||||
constructor(layoutToUse: UIEventSource<LayoutConfig>, fullScreenMessage: UIEventSource<{ content: UIElement, hashText: string, titleText?: UIElement }>) {
|
private readonly _layoutToUse: UIEventSource<LayoutConfig>;
|
||||||
|
private readonly _selectedFeature: UIEventSource<any>;
|
||||||
|
private readonly _allElementsStorage: ElementStorage;
|
||||||
|
|
||||||
|
constructor(layoutToUse: UIEventSource<LayoutConfig>,
|
||||||
|
selectedFeature: UIEventSource<any>,
|
||||||
|
allElementsStorage : ElementStorage) {
|
||||||
|
super(layoutToUse);
|
||||||
|
this._layoutToUse = layoutToUse;
|
||||||
|
this._selectedFeature = selectedFeature;
|
||||||
|
this._allElementsStorage = allElementsStorage;
|
||||||
|
this.ListenTo(Locale.language);
|
||||||
|
this.dumbMode = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
InnerRender(): string {
|
||||||
|
|
||||||
|
const defaultTitle = Translations.WT(this._layoutToUse.data?.title)?.txt ?? "MapComplete"
|
||||||
|
const feature = this._selectedFeature.data;
|
||||||
|
|
||||||
|
if(feature === undefined){
|
||||||
|
return defaultTitle;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
layoutToUse.map((layoutToUse) => {
|
const layout = this._layoutToUse.data;
|
||||||
return Translations.WT(layoutToUse?.title)?.txt ?? "MapComplete"
|
const properties = this._selectedFeature.data.properties;
|
||||||
}, [Locale.language]
|
for (const layer of layout.layers) {
|
||||||
).addCallbackAndRun((title) => {
|
if(layer.title === undefined){
|
||||||
document.title = title
|
continue;
|
||||||
});
|
}
|
||||||
|
if (layer.overpassTags.matchesProperties(properties)) {
|
||||||
|
|
||||||
fullScreenMessage.addCallbackAndRun(selected => {
|
const title = new TagRenderingAnswer(
|
||||||
const title = Translations.WT(layoutToUse.data?.title)?.txt ?? "MapComplete"
|
this._allElementsStorage.getEventSourceFor(feature),
|
||||||
if(selected?.titleText?.data === undefined){
|
layer.title
|
||||||
document.title = title
|
)
|
||||||
}else{
|
return new Combine([defaultTitle," | ", title]).Render();
|
||||||
selected.titleText.Update();
|
|
||||||
var d = document.createElement('div');
|
|
||||||
d.innerHTML = selected.titleText.InnerRender();
|
|
||||||
const poi = (d.textContent || d.innerText)
|
|
||||||
document.title = title + " | " + poi;
|
|
||||||
}
|
}
|
||||||
})
|
}
|
||||||
|
return defaultTitle;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
export default class TitleHandler {
|
||||||
|
constructor(layoutToUse: UIEventSource<LayoutConfig>,
|
||||||
|
selectedFeature: UIEventSource<any>,
|
||||||
|
allElementsStorage : ElementStorage) {
|
||||||
|
|
||||||
|
new TitleElement(layoutToUse, selectedFeature, allElementsStorage)
|
||||||
|
.addCallbackAndRun(contents => {
|
||||||
|
|
||||||
|
const d = document.createElement('div');
|
||||||
|
d.innerHTML = contents;
|
||||||
|
// We pass everything into a div to strip out images etc...
|
||||||
|
document.title = (d.textContent || d.innerText);
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -74,7 +74,7 @@ export default class MetaTagging {
|
||||||
const tagsSource = State.state.allElements.getEventSourceFor(feature);
|
const tagsSource = State.state.allElements.getEventSourceFor(feature);
|
||||||
tagsSource.ping();
|
tagsSource.ping();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error(e)
|
console.warn(e)
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,7 @@ import {Utils} from "../../Utils";
|
||||||
|
|
||||||
export class OsmPreferences {
|
export class OsmPreferences {
|
||||||
|
|
||||||
public preferences = new UIEventSource<any>({});
|
public preferences = new UIEventSource<any>({}, "all-osm-preferences");
|
||||||
public preferenceSources: any = {}
|
public preferenceSources: any = {}
|
||||||
private auth: any;
|
private auth: any;
|
||||||
private userDetails: UIEventSource<UserDetails>;
|
private userDetails: UIEventSource<UserDetails>;
|
||||||
|
@ -29,7 +29,7 @@ export class OsmPreferences {
|
||||||
return this.longPreferences[prefix + key];
|
return this.longPreferences[prefix + key];
|
||||||
}
|
}
|
||||||
|
|
||||||
const source = new UIEventSource<string>(undefined);
|
const source = new UIEventSource<string>(undefined, "long-osm-preference:"+prefix+key);
|
||||||
this.longPreferences[prefix + key] = source;
|
this.longPreferences[prefix + key] = source;
|
||||||
|
|
||||||
const allStartWith = prefix + key + "-combined";
|
const allStartWith = prefix + key + "-combined";
|
||||||
|
@ -106,7 +106,7 @@ export class OsmPreferences {
|
||||||
if (this.userDetails.data.loggedIn && this.preferences.data[key] === undefined) {
|
if (this.userDetails.data.loggedIn && this.preferences.data[key] === undefined) {
|
||||||
this.UpdatePreferences();
|
this.UpdatePreferences();
|
||||||
}
|
}
|
||||||
const pref = new UIEventSource<string>(this.preferences.data[key]);
|
const pref = new UIEventSource<string>(this.preferences.data[key],"osm-preference:"+key);
|
||||||
pref.addCallback((v) => {
|
pref.addCallback((v) => {
|
||||||
this.SetPreference(key, v);
|
this.SetPreference(key, v);
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,14 +1,64 @@
|
||||||
export class UIEventSource<T>{
|
export class UIEventSource<T> {
|
||||||
|
|
||||||
public data: T;
|
public data: T;
|
||||||
|
private readonly tag: string;
|
||||||
private _callbacks = [];
|
private _callbacks = [];
|
||||||
|
|
||||||
|
private static allSources : UIEventSource<any>[] = UIEventSource.PrepPerf();
|
||||||
|
|
||||||
|
static PrepPerf(){
|
||||||
|
// @ts-ignore
|
||||||
|
window.mcperf = () => {
|
||||||
|
console.log(UIEventSource.allSources.length, "uieventsources created");
|
||||||
|
const copy = [...UIEventSource.allSources];
|
||||||
|
copy.sort((a,b) => b._callbacks.length - a._callbacks.length);
|
||||||
|
console.log("Topten is:")
|
||||||
|
for (let i = 0; i < 10; i++) {
|
||||||
|
console.log(copy[i].tag, copy[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
constructor(data: T) {
|
constructor(data: T, tag: string = "") {
|
||||||
|
this.tag = tag;
|
||||||
this.data = data;
|
this.data = data;
|
||||||
|
UIEventSource.allSources.push(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static flatten<X>(source: UIEventSource<UIEventSource<X>>, possibleSources: UIEventSource<any>[]): UIEventSource<X> {
|
||||||
|
const sink = new UIEventSource<X>(source.data?.data);
|
||||||
|
|
||||||
|
source.addCallback((latestData) => {
|
||||||
|
sink.setData(latestData?.data);
|
||||||
|
});
|
||||||
|
|
||||||
|
for (const possibleSource of possibleSources) {
|
||||||
|
possibleSource?.addCallback(() => {
|
||||||
|
sink.setData(source.data?.data);
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return sink;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Chronic(millis: number, asLong: () => boolean = undefined): UIEventSource<Date> {
|
||||||
|
const source = new UIEventSource<Date>(undefined);
|
||||||
|
|
||||||
|
function run() {
|
||||||
|
source.setData(new Date());
|
||||||
|
if (asLong === undefined || asLong()) {
|
||||||
|
window.setTimeout(run, millis);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
run();
|
||||||
|
return source;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public addCallback(callback: ((latestData: T) => void)): UIEventSource<T> {
|
public addCallback(callback: ((latestData: T) => void)): UIEventSource<T> {
|
||||||
if(callback === console.log){
|
if (callback === console.log) {
|
||||||
// This ^^^ actually works!
|
// This ^^^ actually works!
|
||||||
throw "Don't add console.log directly as a callback - you'll won't be able to find it afterwards. Wrap it in a lambda instead."
|
throw "Don't add console.log directly as a callback - you'll won't be able to find it afterwards. Wrap it in a lambda instead."
|
||||||
}
|
}
|
||||||
|
@ -36,25 +86,9 @@ export class UIEventSource<T>{
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static flatten<X>(source: UIEventSource<UIEventSource<X>>, possibleSources: UIEventSource<any>[]): UIEventSource<X> {
|
|
||||||
const sink = new UIEventSource<X>(source.data?.data);
|
|
||||||
|
|
||||||
source.addCallback((latestData) => {
|
|
||||||
sink.setData(latestData?.data);
|
|
||||||
});
|
|
||||||
|
|
||||||
for (const possibleSource of possibleSources) {
|
|
||||||
possibleSource?.addCallback(() => {
|
|
||||||
sink.setData(source.data?.data);
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
return sink;
|
|
||||||
}
|
|
||||||
|
|
||||||
public map<J>(f: ((T) => J),
|
public map<J>(f: ((T) => J),
|
||||||
extraSources: UIEventSource<any>[] = [],
|
extraSources: UIEventSource<any>[] = [],
|
||||||
g: ((J) => T) = undefined ): UIEventSource<J> {
|
g: ((J) => T) = undefined): UIEventSource<J> {
|
||||||
const self = this;
|
const self = this;
|
||||||
|
|
||||||
const newSource = new UIEventSource<J>(
|
const newSource = new UIEventSource<J>(
|
||||||
|
@ -70,7 +104,7 @@ export class UIEventSource<T>{
|
||||||
extraSource?.addCallback(update);
|
extraSource?.addCallback(update);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(g !== undefined) {
|
if (g !== undefined) {
|
||||||
newSource.addCallback((latest) => {
|
newSource.addCallback((latest) => {
|
||||||
self.setData(g(latest));
|
self.setData(g(latest));
|
||||||
})
|
})
|
||||||
|
@ -79,7 +113,6 @@ export class UIEventSource<T>{
|
||||||
return newSource;
|
return newSource;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public syncWith(otherSource: UIEventSource<T>, reverseOverride = false): UIEventSource<T> {
|
public syncWith(otherSource: UIEventSource<T>, reverseOverride = false): UIEventSource<T> {
|
||||||
this.addCallback((latest) => otherSource.setData(latest));
|
this.addCallback((latest) => otherSource.setData(latest));
|
||||||
const self = this;
|
const self = this;
|
||||||
|
@ -94,7 +127,7 @@ export class UIEventSource<T>{
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
public stabilized(millisToStabilize) : UIEventSource<T>{
|
public stabilized(millisToStabilize): UIEventSource<T> {
|
||||||
|
|
||||||
const newSource = new UIEventSource<T>(this.data);
|
const newSource = new UIEventSource<T>(this.data);
|
||||||
|
|
||||||
|
@ -103,7 +136,7 @@ export class UIEventSource<T>{
|
||||||
currentCallback++;
|
currentCallback++;
|
||||||
const thisCallback = currentCallback;
|
const thisCallback = currentCallback;
|
||||||
window.setTimeout(() => {
|
window.setTimeout(() => {
|
||||||
if(thisCallback === currentCallback){
|
if (thisCallback === currentCallback) {
|
||||||
newSource.setData(latestData);
|
newSource.setData(latestData);
|
||||||
}
|
}
|
||||||
}, millisToStabilize)
|
}, millisToStabilize)
|
||||||
|
@ -112,19 +145,4 @@ export class UIEventSource<T>{
|
||||||
return newSource;
|
return newSource;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Chronic(millis: number, asLong: () => boolean = undefined): UIEventSource<Date> {
|
|
||||||
const source = new UIEventSource<Date>(undefined);
|
|
||||||
|
|
||||||
function run() {
|
|
||||||
source.setData(new Date());
|
|
||||||
if (asLong === undefined || asLong()) {
|
|
||||||
window.setTimeout(run, millis);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
run();
|
|
||||||
return source;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
|
@ -5,7 +5,7 @@ export class LocalStorageSource {
|
||||||
static Get(key: string, defaultValue: string = undefined): UIEventSource<string> {
|
static Get(key: string, defaultValue: string = undefined): UIEventSource<string> {
|
||||||
try {
|
try {
|
||||||
const saved = localStorage.getItem(key);
|
const saved = localStorage.getItem(key);
|
||||||
const source = new UIEventSource<string>(saved ?? defaultValue);
|
const source = new UIEventSource<string>(saved ?? defaultValue, "localstorage:"+key);
|
||||||
|
|
||||||
source.addCallback((data) => {
|
source.addCallback((data) => {
|
||||||
localStorage.setItem(key, data);
|
localStorage.setItem(key, data);
|
||||||
|
|
6
State.ts
6
State.ts
|
@ -72,10 +72,6 @@ export default class State {
|
||||||
* The message that should be shown at the center of the screen
|
* The message that should be shown at the center of the screen
|
||||||
*/
|
*/
|
||||||
public readonly centerMessage = new UIEventSource<string>("");
|
public readonly centerMessage = new UIEventSource<string>("");
|
||||||
/**
|
|
||||||
This message is shown full screen on mobile devices
|
|
||||||
*/
|
|
||||||
public readonly fullScreenMessage = new UIEventSource<{ content: UIElement, hashText: string, titleText?: UIElement }>(undefined)
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
The latest element that was selected - used to generate the right UI at the right place
|
The latest element that was selected - used to generate the right UI at the right place
|
||||||
|
@ -244,7 +240,7 @@ export default class State {
|
||||||
}
|
}
|
||||||
}).ping()
|
}).ping()
|
||||||
|
|
||||||
new TitleHandler(this.layoutToUse, this.fullScreenMessage);
|
new TitleHandler(this.layoutToUse, this.selectedElement, this.allElements);
|
||||||
|
|
||||||
|
|
||||||
this.allElements = new ElementStorage();
|
this.allElements = new ElementStorage();
|
||||||
|
|
|
@ -1,22 +1,25 @@
|
||||||
import {UIElement} from "../UIElement";
|
import {UIElement} from "../UIElement";
|
||||||
|
|
||||||
export default class LazyElement extends UIElement {
|
export default class LazyElement<T extends UIElement> extends UIElement {
|
||||||
|
|
||||||
|
|
||||||
private _content: UIElement = undefined;
|
public Activate: (onElement?: (element: T) => void) => void;
|
||||||
|
private _content: T = undefined;
|
||||||
|
private readonly _loadingContent: string;
|
||||||
|
|
||||||
public Activate: () => void;
|
constructor(content: (() => T), loadingContent = "Rendering...") {
|
||||||
private _loadingContent: string;
|
|
||||||
|
|
||||||
constructor(content: (() => UIElement), loadingContent = "Rendering...") {
|
|
||||||
super();
|
super();
|
||||||
this._loadingContent = loadingContent;
|
this._loadingContent = loadingContent;
|
||||||
this.dumbMode = false;
|
this.dumbMode = false;
|
||||||
const self = this;
|
const self = this;
|
||||||
this.Activate = () => {
|
this.Activate = (onElement?: (element: T) => void) => {
|
||||||
|
console.log("ACTIVATED")
|
||||||
if (this._content === undefined) {
|
if (this._content === undefined) {
|
||||||
self._content = content();
|
self._content = content();
|
||||||
}
|
}
|
||||||
|
if (onElement) {
|
||||||
|
onElement(self._content)
|
||||||
|
}
|
||||||
self.Update();
|
self.Update();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -27,5 +30,7 @@ export default class LazyElement extends UIElement {
|
||||||
}
|
}
|
||||||
return this._content.InnerRender();
|
return this._content.InnerRender();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
|
@ -1,6 +1,5 @@
|
||||||
import {UIElement} from "../UIElement";
|
import {UIElement} from "../UIElement";
|
||||||
import Svg from "../../Svg";
|
import Svg from "../../Svg";
|
||||||
import State from "../../State";
|
|
||||||
import Combine from "./Combine";
|
import Combine from "./Combine";
|
||||||
import Ornament from "./Ornament";
|
import Ornament from "./Ornament";
|
||||||
|
|
||||||
|
@ -8,63 +7,150 @@ import Ornament from "./Ornament";
|
||||||
* Wraps some contents into a panel that scrolls the content _under_ the title
|
* Wraps some contents into a panel that scrolls the content _under_ the title
|
||||||
*/
|
*/
|
||||||
export default class ScrollableFullScreen extends UIElement {
|
export default class ScrollableFullScreen extends UIElement {
|
||||||
|
private static _isInited = false;
|
||||||
|
private title: UIElement;
|
||||||
|
private content: UIElement;
|
||||||
private _component: UIElement;
|
private _component: UIElement;
|
||||||
private elementsToRestore: Set<HTMLElement> = new Set<HTMLElement>();
|
|
||||||
|
|
||||||
constructor(title: UIElement, content: UIElement, onClose: (() => void)) {
|
constructor(title: UIElement, content: UIElement, onClose: (() => void)) {
|
||||||
super();
|
super();
|
||||||
|
this.content = content;
|
||||||
|
this.title = title;
|
||||||
|
if (!ScrollableFullScreen._isInited) {
|
||||||
|
ScrollableFullScreen._isInited = ScrollableFullScreen.PreparePatchesForFullscreen();
|
||||||
|
}
|
||||||
|
if (onClose === undefined) {
|
||||||
|
console.error("ScrollableFullScreen initialized without onClose!")
|
||||||
|
}
|
||||||
this.dumbMode = false;
|
this.dumbMode = false;
|
||||||
const returnToTheMap = Svg.back_svg().onClick(() => {
|
const returnToTheMap =
|
||||||
console.log("Clicked back!");
|
new Combine([
|
||||||
this.RestoreLeaflet();
|
Svg.back_svg().SetClass("block sm:hidden"),
|
||||||
if (onClose() !== undefined) {
|
Svg.close_svg().SetClass("hidden sm:block")
|
||||||
console.error("WARNING: onClose is not defined")
|
])
|
||||||
onClose();
|
.onClick(() => {
|
||||||
}
|
console.log("Clicked back!");
|
||||||
}).SetClass("block sm:hidden mb-2 bg-blue-50 rounded-full w-12 h-12 p-1.5")
|
ScrollableFullScreen.RestoreLeaflet();
|
||||||
|
if (onClose !== undefined) {
|
||||||
|
onClose();
|
||||||
|
} else {
|
||||||
|
console.error("WARNING: onClose is not defined")
|
||||||
|
}
|
||||||
|
}).SetClass("mb-2 bg-blue-50 rounded-full w-12 h-12 p-1.5")
|
||||||
|
|
||||||
title.SetClass("block w-full")
|
title.SetClass("block w-full text-2xl font-bold p-2 pl-4")
|
||||||
const ornament = new Combine([new Ornament().SetStyle("height:5em;")])
|
const ornament = new Combine([new Ornament().SetStyle("height:5em;")])
|
||||||
.SetClass("block sm:hidden h-5")
|
.SetClass("block sm:hidden h-5")
|
||||||
|
|
||||||
|
|
||||||
this._component =
|
this._component =
|
||||||
new Combine([
|
new Combine([
|
||||||
new Combine([
|
new Combine([
|
||||||
new Combine([returnToTheMap, title])
|
new Combine([returnToTheMap, title])
|
||||||
.AddClass("border-b-2 border-black shadow sm:shadow-none bg-white p-2 pb-0 sm:p-0 flex overflow-x-hidden flex-shrink-0 max-h-20vh"),
|
.SetClass("border-b-2 border-black shadow sm:shadow-none bg-white p-2 pb-0 sm:p-0 flex overflow-x-hidden flex-shrink-0 max-h-20vh"),
|
||||||
new Combine(["<span>", content, "</span>", ornament])
|
new Combine(["<span>", content, "</span>", ornament])
|
||||||
.SetClass("block p-2 sm:pt-4 w-full h-screen landscape:h-screen sm:h-full sm:w-full overflow-y-auto overflow-x-hidden"),
|
.SetClass("block p-2 sm:pt-4 w-full h-screen landscape:h-screen sm:h-full sm:w-full overflow-y-auto overflow-x-hidden"),
|
||||||
// We add an ornament which takes around 5em. This is in order to make sure the Web UI doesn't hide
|
// We add an ornament which takes around 5em. This is in order to make sure the Web UI doesn't hide
|
||||||
]).SetClass("block flex flex-col relative bg-white")
|
]).SetClass("block flex flex-col relative bg-white")
|
||||||
]).SetClass("fixed top-0 left-0 right-0 h-screen w-screen sm:max-h-65vh sm:w-auto");
|
]).SetClass("fixed top-0 left-0 right-0 h-screen w-screen sm:max-h-65vh sm:w-auto sm:relative");
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static HideClutter(htmlElement: HTMLElement) {
|
||||||
|
const whiteList = new Set<Element>();
|
||||||
|
do {
|
||||||
|
if(htmlElement === null){
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (htmlElement.classList.contains("clutter")) {
|
||||||
|
// Don't hide the parent element
|
||||||
|
whiteList.add(htmlElement)
|
||||||
|
}
|
||||||
|
htmlElement = htmlElement.parentElement;
|
||||||
|
} while (htmlElement != null)
|
||||||
|
|
||||||
|
const clutter = document.getElementsByClassName("clutter");
|
||||||
|
for (let i = 0; i < clutter.length; ++i) {
|
||||||
|
if (whiteList.has(clutter[i])) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const classlist = clutter[i].classList;
|
||||||
|
if (classlist.contains("clutter-hidden")) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
classlist.add("clutter-hidden");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adds the 'clutter' class (which merely acts as a tag) onto some elements, e.g. the leaflet attributions
|
||||||
|
* @constructor
|
||||||
|
*/
|
||||||
|
private static PreparePatchesForFullscreen(): boolean {
|
||||||
|
const toHide = document.getElementsByClassName("leaflet-control-container");
|
||||||
|
for (let i = 0; i < toHide.length; ++i) {
|
||||||
|
toHide[i].classList.add("clutter");
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static PatchLeaflet(htmlElement) {
|
||||||
|
if(htmlElement === null){
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
do {
|
||||||
|
// A leaflet workaround: in order for fullscreen to work, we need to get the parent element which does a transform3d and remove/read the transform
|
||||||
|
if (htmlElement.style.transform !== "") {
|
||||||
|
if (!htmlElement.classList.contains("no-transform")) {
|
||||||
|
htmlElement.classList.add("no-transform");
|
||||||
|
htmlElement.classList.add("scrollable-fullscreen-no-transform")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
htmlElement = htmlElement.parentElement;
|
||||||
|
} while (htmlElement != null)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private static RestoreLeaflet() {
|
||||||
|
console.log("Restoring")
|
||||||
|
const noTransf = document.getElementsByClassName("scrollable-fullscreen-no-transform");
|
||||||
|
for (let i = 0; i < noTransf.length; ++i) {
|
||||||
|
noTransf[i].classList.remove("no-transform");
|
||||||
|
noTransf[i].classList.remove("scrollable-fullscreen-no-transform");
|
||||||
|
}
|
||||||
|
let clutter = document.getElementsByClassName("clutter-hidden");
|
||||||
|
|
||||||
|
do {
|
||||||
|
for (let i = 0; i < clutter.length; ++i) {
|
||||||
|
clutter[i].classList.remove("clutter-hidden");
|
||||||
|
}
|
||||||
|
clutter = document.getElementsByClassName("clutter-hidden");
|
||||||
|
} while (clutter.length > 0)
|
||||||
|
}
|
||||||
|
|
||||||
InnerRender(): string {
|
InnerRender(): string {
|
||||||
return this._component.Render();
|
return this._component.Render();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected InnerUpdate(htmlElement: HTMLElement) {
|
Update() {
|
||||||
|
console.log("Updating the scrollableFullScreen")
|
||||||
do {
|
super.Update();
|
||||||
// A leaflet workaround: in order for fullscreen to work, we need to get the parent element which does a transform3d and remove/read the transform
|
this._component.Update();
|
||||||
if (htmlElement.style.transform !== "") {
|
|
||||||
this.elementsToRestore.add(htmlElement);
|
|
||||||
htmlElement.classList.add("no-transform")
|
|
||||||
}
|
|
||||||
htmlElement = htmlElement.parentElement;
|
|
||||||
} while (htmlElement != null)
|
|
||||||
|
|
||||||
super.InnerUpdate(htmlElement);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private RestoreLeaflet() {
|
public PrepFullscreen(htmlElement = undefined) {
|
||||||
this.elementsToRestore.forEach(
|
htmlElement = htmlElement ?? document.getElementById(this.id);
|
||||||
el => el.classList.remove("no-transform")
|
ScrollableFullScreen.PatchLeaflet(htmlElement);
|
||||||
);
|
ScrollableFullScreen.HideClutter(htmlElement);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected InnerUpdate(htmlElement: HTMLElement) {
|
||||||
|
this.PrepFullscreen(htmlElement)
|
||||||
|
super.InnerUpdate(htmlElement);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
|
@ -24,9 +24,9 @@ export class SubtleButton extends UIElement{
|
||||||
} else {
|
} else {
|
||||||
img = imageUrl;
|
img = imageUrl;
|
||||||
}
|
}
|
||||||
img.AddClass("block flex items-center justify-center h-11 w-11 flex-shrink0")
|
img.SetClass("block flex items-center justify-center h-11 w-11 flex-shrink0")
|
||||||
this.image = new Combine([img])
|
this.image = new Combine([img])
|
||||||
.AddClass("flex-shrink-0");
|
.SetClass("flex-shrink-0");
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -53,7 +53,7 @@ export class SubtleButton extends UIElement{
|
||||||
return new Combine([
|
return new Combine([
|
||||||
this.image,
|
this.image,
|
||||||
this.message,
|
this.message,
|
||||||
]).AddClass("block flex p-3 my-2 bg-blue-100 rounded-lg hover:shadow-xl hover:bg-blue-200")
|
]).SetClass("block flex p-3 my-2 bg-blue-100 rounded-lg hover:shadow-xl hover:bg-blue-200")
|
||||||
.Render();
|
.Render();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -7,22 +7,22 @@ export default class IndexText extends Combine {
|
||||||
constructor() {
|
constructor() {
|
||||||
super([
|
super([
|
||||||
new FixedUiElement(`<img class="w-12 h-12 sm:h-24 sm:w-24" src="./assets/svg/logo.svg" alt="MapComplete Logo">`)
|
new FixedUiElement(`<img class="w-12 h-12 sm:h-24 sm:w-24" src="./assets/svg/logo.svg" alt="MapComplete Logo">`)
|
||||||
.AddClass("flex-none m-3"),
|
.SetClass("flex-none m-3"),
|
||||||
|
|
||||||
new Combine([
|
new Combine([
|
||||||
Translations.t.index.title
|
Translations.t.index.title
|
||||||
.AddClass("text-2xl tracking-tight font-extrabold text-gray-900 sm:text-5xl md:text-6xl block text-gray-800 xl:inline"),
|
.SetClass("text-2xl tracking-tight font-extrabold text-gray-900 sm:text-5xl md:text-6xl block text-gray-800 xl:inline"),
|
||||||
|
|
||||||
Translations.t.index.intro.AddClass(
|
Translations.t.index.intro.SetClass(
|
||||||
"mt-3 text-base font-semibold text-gray-500 sm:mt-5 sm:text-lg sm:max-w-xl sm:mx-auto md:mt-5 md:text-xl lg:mx-0"),
|
"mt-3 text-base font-semibold text-gray-500 sm:mt-5 sm:text-lg sm:max-w-xl sm:mx-auto md:mt-5 md:text-xl lg:mx-0"),
|
||||||
|
|
||||||
Translations.t.index.pickTheme.AddClass("mt-3 text-base text-green-600 sm:mt-5 sm:text-lg sm:max-w-xl sm:mx-auto md:mt-5 md:text-xl lg:mx-0")
|
Translations.t.index.pickTheme.SetClass("mt-3 text-base text-green-600 sm:mt-5 sm:text-lg sm:max-w-xl sm:mx-auto md:mt-5 md:text-xl lg:mx-0")
|
||||||
|
|
||||||
]).AddClass("flex flex-col sm:text-center lg:text-left m-1 mt-2 md:m-2 md:mt-4")
|
]).SetClass("flex flex-col sm:text-center lg:text-left m-1 mt-2 md:m-2 md:mt-4")
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
|
||||||
this.AddClass("flex flex-row");
|
this.SetClass("flex flex-row");
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -22,7 +22,7 @@ export default class SearchAndGo extends UIElement {
|
||||||
);
|
);
|
||||||
|
|
||||||
private _foundEntries = new UIEventSource([]);
|
private _foundEntries = new UIEventSource([]);
|
||||||
private _goButton = Svg.search_ui().AddClass('w-8 h-8 full-rounded border-black float-right');
|
private _goButton = Svg.search_ui().SetClass('w-8 h-8 full-rounded border-black float-right');
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
super(undefined);
|
super(undefined);
|
||||||
|
|
|
@ -1,32 +0,0 @@
|
||||||
import {UIElement} from "./UIElement";
|
|
||||||
import State from "../State";
|
|
||||||
import Combine from "./Base/Combine";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Handles the full screen popup on mobile
|
|
||||||
*/
|
|
||||||
export default class FullScreenMessageBox extends UIElement {
|
|
||||||
|
|
||||||
private _content: UIElement;
|
|
||||||
|
|
||||||
constructor() {
|
|
||||||
super(State.state.fullScreenMessage);
|
|
||||||
this.HideOnEmpty(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
InnerRender(): string {
|
|
||||||
if (State.state.fullScreenMessage.data === undefined) {
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
this._content = State.state.fullScreenMessage.data.content;
|
|
||||||
return new Combine([this._content])
|
|
||||||
.SetClass("block max-h-screen h-screen overflow-x-hidden overflow-y-auto bg-white p-0").Render();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected InnerUpdate(htmlElement: HTMLElement) {
|
|
||||||
super.InnerUpdate(htmlElement);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
|
@ -9,29 +9,41 @@ import State from "../../State";
|
||||||
import TagRenderingConfig from "../../Customizations/JSON/TagRenderingConfig";
|
import TagRenderingConfig from "../../Customizations/JSON/TagRenderingConfig";
|
||||||
import ScrollableFullScreen from "../Base/ScrollableFullScreen";
|
import ScrollableFullScreen from "../Base/ScrollableFullScreen";
|
||||||
|
|
||||||
export default class FeatureInfoBox extends UIElement {
|
export default class FeatureInfoBox extends ScrollableFullScreen {
|
||||||
private _component: UIElement;
|
|
||||||
|
|
||||||
public title: UIElement ;
|
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
tags: UIEventSource<any>,
|
tags: UIEventSource<any>,
|
||||||
layerConfig: LayerConfig,
|
layerConfig: LayerConfig,
|
||||||
onClose: () => {}
|
onClose: () => void
|
||||||
) {
|
) {
|
||||||
super();
|
super(
|
||||||
|
FeatureInfoBox.GenerateTitleBar(tags, layerConfig),
|
||||||
|
FeatureInfoBox.GenerateContent(tags, layerConfig),
|
||||||
|
onClose
|
||||||
|
);
|
||||||
if (layerConfig === undefined) {
|
if (layerConfig === undefined) {
|
||||||
throw "Undefined layerconfig"
|
throw "Undefined layerconfig"
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static GenerateTitleBar( tags: UIEventSource<any>,
|
||||||
|
layerConfig: LayerConfig): UIElement{
|
||||||
const title = new TagRenderingAnswer(tags, layerConfig.title ?? new TagRenderingConfig("POI", undefined))
|
const title = new TagRenderingAnswer(tags, layerConfig.title ?? new TagRenderingConfig("POI", undefined))
|
||||||
.AddClass("text-2xl break-words font-bold p-2");
|
.SetClass("text-2xl break-words font-bold p-2");
|
||||||
this.title = title;
|
|
||||||
const titleIcons = new Combine(
|
const titleIcons = new Combine(
|
||||||
layerConfig.titleIcons.map(icon => new TagRenderingAnswer(tags, icon)
|
layerConfig.titleIcons.map(icon => new TagRenderingAnswer(tags, icon)
|
||||||
.AddClass("block w-8 h-8 align-baseline box-content p-0.5")))
|
.SetClass("block w-8 h-8 align-baseline box-content p-0.5")))
|
||||||
.AddClass("flex flex-row flex-wrap pt-1 items-center mr-2");
|
.SetClass("flex flex-row flex-wrap pt-1 items-center mr-2");
|
||||||
|
|
||||||
|
return new Combine([
|
||||||
|
new Combine([title, titleIcons]).SetClass("flex flex-grow justify-between")
|
||||||
|
])
|
||||||
|
}
|
||||||
|
|
||||||
|
private static GenerateContent(tags: UIEventSource<any>,
|
||||||
|
layerConfig: LayerConfig): UIElement{
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
let questionBox: UIElement = undefined;
|
let questionBox: UIElement = undefined;
|
||||||
if (State.state.featureSwitchUserbadge.data) {
|
if (State.state.featureSwitchUserbadge.data) {
|
||||||
|
@ -53,20 +65,12 @@ export default class FeatureInfoBox extends UIElement {
|
||||||
}
|
}
|
||||||
const tail = new Combine([]).SetClass("only-on-mobile");
|
const tail = new Combine([]).SetClass("only-on-mobile");
|
||||||
|
|
||||||
const content = new Combine([
|
return new Combine([
|
||||||
...renderings,
|
...renderings,
|
||||||
tail.SetClass("featureinfobox-tail")
|
tail.SetClass("featureinfobox-tail")
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
const titleBar = new Combine([
|
|
||||||
new Combine([title, titleIcons]).SetClass("flex flex-grow justify-between")
|
|
||||||
])
|
|
||||||
|
|
||||||
this._component = new ScrollableFullScreen(titleBar, content, onClose)
|
|
||||||
}
|
|
||||||
|
|
||||||
InnerRender(): string {
|
|
||||||
return this._component.Render();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,7 +21,7 @@ export default class TagRenderingAnswer extends UIElement {
|
||||||
if (configuration === undefined) {
|
if (configuration === undefined) {
|
||||||
throw "Trying to generate a tagRenderingAnswer without configuration..."
|
throw "Trying to generate a tagRenderingAnswer without configuration..."
|
||||||
}
|
}
|
||||||
this.AddClass("flex items-center flex-row text-lg")
|
this.SetClass("flex items-center flex-row text-lg")
|
||||||
}
|
}
|
||||||
|
|
||||||
InnerRender(): string {
|
InnerRender(): string {
|
||||||
|
|
|
@ -88,7 +88,7 @@ export default class TagRenderingQuestion extends UIElement {
|
||||||
return tags.asHumanString(true, true);
|
return tags.asHumanString(true, true);
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
).AddClass("block")
|
).SetClass("block")
|
||||||
}
|
}
|
||||||
|
|
||||||
private GenerateInputElement(): InputElement<TagsFilter> {
|
private GenerateInputElement(): InputElement<TagsFilter> {
|
||||||
|
|
|
@ -44,10 +44,10 @@ export default class ShowDataLayer {
|
||||||
|
|
||||||
let geoLayer = self.CreateGeojsonLayer(feats)
|
let geoLayer = self.CreateGeojsonLayer(feats)
|
||||||
if (layoutToUse.clustering.minNeededElements <= features.data.length) {
|
if (layoutToUse.clustering.minNeededElements <= features.data.length) {
|
||||||
const cl = window["L"]; // This is a dirty workaround, the clustering plugin binds to the L of the window, not of the namespace or something
|
const cl = window["L"]; // This is a dirty workaround, the clustering plugin binds to the L of the window, not of the namespace or something
|
||||||
const cluster = cl.markerClusterGroup({ disableClusteringAtZoom: layoutToUse.clustering.maxZoom });
|
const cluster = cl.markerClusterGroup({disableClusteringAtZoom: layoutToUse.clustering.maxZoom});
|
||||||
cluster.addLayer(geoLayer);
|
cluster.addLayer(geoLayer);
|
||||||
geoLayer = cluster;
|
geoLayer = cluster;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (oldGeoLayer) {
|
if (oldGeoLayer) {
|
||||||
|
@ -60,22 +60,22 @@ export default class ShowDataLayer {
|
||||||
features.addCallbackAndRun(() => update());
|
features.addCallbackAndRun(() => update());
|
||||||
leafletMap.addCallback(() => update());
|
leafletMap.addCallback(() => update());
|
||||||
State.state.selectedElement.addCallback(feature => {
|
State.state.selectedElement.addCallback(feature => {
|
||||||
if(feature === undefined){
|
if (feature === undefined) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const id = feature.properties.id+feature.geometry.type+feature._matching_layer_id;
|
const id = feature.properties.id + feature.geometry.type + feature._matching_layer_id;
|
||||||
const action = self._onSelectedTrigger[id];
|
const action = self._onSelectedTrigger[id];
|
||||||
if(action){
|
if (action) {
|
||||||
action();
|
action();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
Hash.hash.addCallbackAndRun(id => {
|
Hash.hash.addCallbackAndRun(id => {
|
||||||
// This is a bit of an edge case: if the hash becomes an id to search, we have to show the corresponding popup
|
// This is a bit of an edge case: if the hash becomes an id to search, we have to show the corresponding popup
|
||||||
if(State.state.selectedElement !== undefined){
|
if (State.state.selectedElement !== undefined) {
|
||||||
return; // Something is already selected, we don't have to apply this fix
|
return; // Something is already selected, we don't have to apply this fix
|
||||||
}
|
}
|
||||||
const action = self._onSelectedTrigger[id];
|
const action = self._onSelectedTrigger[id];
|
||||||
if(action){
|
if (action) {
|
||||||
action();
|
action();
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
@ -126,41 +126,48 @@ export default class ShowDataLayer {
|
||||||
|
|
||||||
|
|
||||||
const tags = State.state.allElements.getEventSourceFor(feature);
|
const tags = State.state.allElements.getEventSourceFor(feature);
|
||||||
const uiElement: LazyElement = new LazyElement(() => new FeatureInfoBox(tags, layer, () => popup.closePopup()),
|
const uiElement: LazyElement<FeatureInfoBox> = new LazyElement(() => new FeatureInfoBox(tags, layer, () => {
|
||||||
|
console.log("Closing the popup!")
|
||||||
|
State.state.selectedElement.setData(undefined);
|
||||||
|
popup.remove();
|
||||||
|
|
||||||
|
}),
|
||||||
"<div style='height: 90vh'>Rendering</div>");
|
"<div style='height: 90vh'>Rendering</div>");
|
||||||
popup.setContent(uiElement.Render());
|
popup.setContent(uiElement.Render());
|
||||||
popup.on('remove', () => {
|
popup.on('remove', () => {
|
||||||
if(!popup.isOpen()){
|
if (!popup.isOpen()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
State.state.selectedElement.setData(undefined);
|
State.state.selectedElement.setData(undefined);
|
||||||
});
|
});
|
||||||
leafletLayer.bindPopup(popup);
|
leafletLayer.bindPopup(popup);
|
||||||
// We first render the UIelement (which'll still need an update later on...)
|
// We first render the UIelement (which'll still need an update later on...)
|
||||||
// But at least it'll be visible already
|
// But at least it'll be visible already
|
||||||
|
|
||||||
|
|
||||||
leafletLayer.on("click", (e) => {
|
leafletLayer.on("click", () => {
|
||||||
// We set the element as selected...
|
// We set the element as selected...
|
||||||
uiElement.Activate();
|
|
||||||
|
uiElement.Activate(e => e.PrepFullscreen());
|
||||||
State.state.selectedElement.setData(feature);
|
State.state.selectedElement.setData(feature);
|
||||||
});
|
});
|
||||||
|
|
||||||
const id = feature.properties.id+feature.geometry.type+feature._matching_layer_id;
|
const id = feature.properties.id + feature.geometry.type + feature._matching_layer_id;
|
||||||
this._onSelectedTrigger[id]
|
this._onSelectedTrigger[id]
|
||||||
= () => {
|
= () => {
|
||||||
if(popup.isOpen()){
|
if (popup.isOpen()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
leafletLayer.openPopup();
|
leafletLayer.openPopup();
|
||||||
uiElement.Activate();
|
uiElement.Activate(e => e.PrepFullscreen());
|
||||||
State.state.selectedElement.setData(feature);
|
State.state.selectedElement.setData(feature);
|
||||||
}
|
}
|
||||||
this._onSelectedTrigger[feature.properties.id.replace("/","_")] = this._onSelectedTrigger[id];
|
this._onSelectedTrigger[feature.properties.id.replace("/", "_")] = this._onSelectedTrigger[id];
|
||||||
if (feature.properties.id.replace(/\//g, "_") === Hash.hash.data) {
|
if (feature.properties.id.replace(/\//g, "_") === Hash.hash.data && State.state.selectedElement.data === undefined) {
|
||||||
// This element is in the URL, so this is a share link
|
// This element is in the URL, so this is a share link
|
||||||
// We open the relevant popup straight away
|
// We open the relevant popup straight away
|
||||||
uiElement.Activate();
|
console.log("Opening the popup due to sharelink")
|
||||||
|
uiElement.Activate(e => e.PrepFullscreen());
|
||||||
popup.setContent(uiElement.Render());
|
popup.setContent(uiElement.Render());
|
||||||
|
|
||||||
const center = GeoOperations.centerpoint(feature).geometry.coordinates;
|
const center = GeoOperations.centerpoint(feature).geometry.coordinates;
|
||||||
|
|
|
@ -149,16 +149,12 @@ export abstract class UIElement extends UIEventSource<string> {
|
||||||
return this.InnerRender() === "";
|
return this.InnerRender() === "";
|
||||||
}
|
}
|
||||||
|
|
||||||
public SetClass(clss: string): UIElement {
|
|
||||||
return this.AddClass(clss);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds all the relevant classes, space seperated
|
* Adds all the relevant classes, space seperated
|
||||||
* @param clss
|
* @param clss
|
||||||
* @constructor
|
* @constructor
|
||||||
*/
|
*/
|
||||||
public AddClass(clss: string) {
|
public SetClass(clss: string) {
|
||||||
this.dumbMode = false;
|
this.dumbMode = false;
|
||||||
const all = clss.split(" ");
|
const all = clss.split(" ");
|
||||||
let recordedChange = false;
|
let recordedChange = false;
|
||||||
|
|
|
@ -78,11 +78,4 @@
|
||||||
max-width: 100%;
|
max-width: 100%;
|
||||||
max-height: 30vh;
|
max-height: 30vh;
|
||||||
border-radius: 1em;
|
border-radius: 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
.hidden {
|
|
||||||
/* This is used by the slideshow, to hide non-active slides*/
|
|
||||||
display: none !important;
|
|
||||||
|
|
||||||
}
|
|
16
index.css
16
index.css
|
@ -32,15 +32,21 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@media only screen and (max-width: 640px), only screen and (max-height: 640px) {
|
@media only screen and (max-width: 640px) {
|
||||||
.no-transform {
|
.no-transform {
|
||||||
/*This is a workaround to let popup contents escape the popup on mobile*/
|
/*This is a workaround to let popup contents escape the popup on mobile - see scrollableFullScreen.ts*/
|
||||||
transform: none !important;
|
transform: none !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.clutter-hidden {
|
||||||
|
/*This is a workaround to let popup contents escape the popup on mobile - see scrollableFullScreen.ts*/
|
||||||
|
visibility: hidden !important;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
:root {
|
:root {
|
||||||
--subtle-detail-color: #e5f5ff;
|
--subtle-detail-color: #e5f5ff;
|
||||||
--subtle-detail-color-contrast: black;
|
--subtle-detail-color-contrast: black;
|
||||||
|
@ -57,6 +63,11 @@
|
||||||
--return-to-the-map-height: 2em;
|
--return-to-the-map-height: 2em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.hide-when-fullscreen-is-shown {
|
||||||
|
/*Clutter is actually a class indicating that the element should be hidden when a scrollableFullScreen is opened
|
||||||
|
It doesn't actually define any rules*/
|
||||||
|
}
|
||||||
|
|
||||||
html, body {
|
html, body {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
min-height: 100vh;
|
min-height: 100vh;
|
||||||
|
@ -333,7 +344,6 @@ a {
|
||||||
|
|
||||||
.welcomeMessage {
|
.welcomeMessage {
|
||||||
display: block;
|
display: block;
|
||||||
margin-left: 4em;
|
|
||||||
max-width: calc(100vw - 5em);
|
max-width: calc(100vw - 5em);
|
||||||
width: 40em;
|
width: 40em;
|
||||||
max-height: calc(100vh - 15em);
|
max-height: calc(100vh - 15em);
|
||||||
|
|
10
index.html
10
index.html
|
@ -44,11 +44,7 @@
|
||||||
<!-- DECORATION 0 END -->
|
<!-- DECORATION 0 END -->
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="only-on-mobile">
|
<div id="topleft-tools" class="z-index-above-map clutter">
|
||||||
<div id="messagesboxmobile"></div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div id="topleft-tools" class="z-index-above-map">
|
|
||||||
<div id="userbadge-and-search" class="p-3">
|
<div id="userbadge-and-search" class="p-3">
|
||||||
<div id="userbadge" class="shadow rounded-3xl overflow-hidden"></div>
|
<div id="userbadge" class="shadow rounded-3xl overflow-hidden"></div>
|
||||||
<div id="searchbox" class="shadow rounded-3xl overflow-hidden"></div>
|
<div id="searchbox" class="shadow rounded-3xl overflow-hidden"></div>
|
||||||
|
@ -57,13 +53,13 @@
|
||||||
<div id="help-button-mobile"></div>
|
<div id="help-button-mobile"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="layer-selection" class="absolute bottom-3 left-3 rounded-3xl overflow-hidden"></div>
|
<div id="layer-selection" class="absolute bottom-3 left-3 rounded-3xl overflow-hidden clutter"></div>
|
||||||
|
|
||||||
<div id="centermessage" class="absolute rounded-3xl h-24 left-24 right-24 top-56 bg-white p-3 pt-5 sm:pt-8 text-xl font-bold text-center">
|
<div id="centermessage" class="absolute rounded-3xl h-24 left-24 right-24 top-56 bg-white p-3 pt-5 sm:pt-8 text-xl font-bold text-center">
|
||||||
Loading MapComplete, hang on...
|
Loading MapComplete, hang on...
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="geolocate-button"></div>
|
<div id="geolocate-button" class="clutter"></div>
|
||||||
<div id="leafletDiv"></div>
|
<div id="leafletDiv"></div>
|
||||||
|
|
||||||
<script src="./index.ts"></script>
|
<script src="./index.ts"></script>
|
||||||
|
|
3
index.ts
3
index.ts
|
@ -119,11 +119,10 @@ if (layoutFromBase64.startsWith("wiki:")) {
|
||||||
} else {
|
} else {
|
||||||
// We fall through: no theme loaded: just show a few buttons
|
// We fall through: no theme loaded: just show a few buttons
|
||||||
State.state = new State(undefined);
|
State.state = new State(undefined);
|
||||||
document.getElementById("messagesboxmobile").remove();
|
|
||||||
new Combine([new MoreScreen(true)
|
new Combine([new MoreScreen(true)
|
||||||
.SetStyle("pointer-events: all;"),
|
.SetStyle("pointer-events: all;"),
|
||||||
Translations.t.general.openStreetMapIntro
|
Translations.t.general.openStreetMapIntro
|
||||||
]).AddClass("block m-5 lg:w-3/4 lg:ml-40")
|
]).SetClass("block m-5 lg:w-3/4 lg:ml-40")
|
||||||
.AttachTo("topleft-tools");
|
.AttachTo("topleft-tools");
|
||||||
}
|
}
|
||||||
window.addEventListener('contextmenu', function (e) { // Not compatible with IE < 9
|
window.addEventListener('contextmenu', function (e) { // Not compatible with IE < 9
|
||||||
|
|
Loading…
Reference in a new issue