Fix: opening the popup if defined in the hash
This commit is contained in:
parent
1c4cf78a03
commit
196d76d9dc
3 changed files with 174 additions and 75 deletions
|
@ -3,15 +3,20 @@ import {OsmObject} from "../Osm/OsmObject";
|
||||||
import Loc from "../../Models/Loc";
|
import Loc from "../../Models/Loc";
|
||||||
import {ElementStorage} from "../ElementStorage";
|
import {ElementStorage} from "../ElementStorage";
|
||||||
import FeaturePipeline from "../FeatureSource/FeaturePipeline";
|
import FeaturePipeline from "../FeatureSource/FeaturePipeline";
|
||||||
|
import {GeoOperations} from "../GeoOperations";
|
||||||
|
import LayoutConfig from "../../Models/ThemeConfig/LayoutConfig";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Makes sure the hash shows the selected element and vice-versa.
|
* Makes sure the hash shows the selected element and vice-versa.
|
||||||
*/
|
*/
|
||||||
export default class SelectedFeatureHandler {
|
export default class SelectedFeatureHandler {
|
||||||
private static readonly _no_trigger_on = new Set(["welcome", "copyright", "layers", "new", "", undefined])
|
private static readonly _no_trigger_on = new Set(["welcome", "copyright", "layers", "new", "", undefined])
|
||||||
hash: UIEventSource<string>;
|
private readonly hash: UIEventSource<string>;
|
||||||
private readonly state: {
|
private readonly state: {
|
||||||
selectedElement: UIEventSource<any>
|
selectedElement: UIEventSource<any>,
|
||||||
|
allElements: ElementStorage,
|
||||||
|
locationControl: UIEventSource<Loc>,
|
||||||
|
layoutToUse: LayoutConfig
|
||||||
}
|
}
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
|
@ -19,7 +24,9 @@ export default class SelectedFeatureHandler {
|
||||||
state: {
|
state: {
|
||||||
selectedElement: UIEventSource<any>,
|
selectedElement: UIEventSource<any>,
|
||||||
allElements: ElementStorage,
|
allElements: ElementStorage,
|
||||||
featurePipeline: FeaturePipeline
|
featurePipeline: FeaturePipeline,
|
||||||
|
locationControl: UIEventSource<Loc>,
|
||||||
|
layoutToUse: LayoutConfig
|
||||||
}
|
}
|
||||||
) {
|
) {
|
||||||
this.hash = hash;
|
this.hash = hash;
|
||||||
|
@ -27,30 +34,9 @@ export default class SelectedFeatureHandler {
|
||||||
|
|
||||||
|
|
||||||
// If the hash changes, set the selected element correctly
|
// If the hash changes, set the selected element correctly
|
||||||
function setSelectedElementFromHash(h){
|
|
||||||
if (h === undefined || h === "") {
|
|
||||||
// Hash has been cleared - we clear the selected element
|
|
||||||
state.selectedElement.setData(undefined);
|
|
||||||
}else{
|
|
||||||
// we search the element to select
|
|
||||||
const feature = state.allElements.ContainingFeatures.get(h)
|
|
||||||
if(feature === undefined){
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const currentlySeleced = state.selectedElement.data
|
|
||||||
if(currentlySeleced === undefined){
|
|
||||||
state.selectedElement.setData(feature)
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if(currentlySeleced.properties?.id === feature.properties.id){
|
|
||||||
// We already have the right feature
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
state.selectedElement.setData(feature)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
hash.addCallback(setSelectedElementFromHash)
|
const self = this;
|
||||||
|
hash.addCallback(() => self.setSelectedElementFromHash())
|
||||||
|
|
||||||
|
|
||||||
// IF the selected element changes, set the hash correctly
|
// IF the selected element changes, set the hash correctly
|
||||||
|
@ -67,40 +53,102 @@ export default class SelectedFeatureHandler {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
state.featurePipeline.newDataLoadedSignal.addCallbackAndRunD(_ => {
|
state.featurePipeline?.newDataLoadedSignal?.addCallbackAndRunD(_ => {
|
||||||
// New data was loaded. In initial startup, the hash might be set (via the URL) but might not be selected yet
|
// New data was loaded. In initial startup, the hash might be set (via the URL) but might not be selected yet
|
||||||
if(hash.data === undefined || SelectedFeatureHandler._no_trigger_on.has(hash.data)){
|
if (hash.data === undefined || SelectedFeatureHandler._no_trigger_on.has(hash.data)) {
|
||||||
// This is an invalid hash anyway
|
// This is an invalid hash anyway
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if(state.selectedElement.data !== undefined){
|
if (state.selectedElement.data !== undefined) {
|
||||||
// We already have something selected
|
// We already have something selected
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
setSelectedElementFromHash(hash.data)
|
self.setSelectedElementFromHash()
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
this.initialLoad()
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* On startup: check if the hash is loaded and eventually zoom to it
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
private initialLoad() {
|
||||||
|
const hash = this.hash.data
|
||||||
|
if (hash === undefined || hash === "" || hash.indexOf("-") >= 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (SelectedFeatureHandler._no_trigger_on.has(hash)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
OsmObject.DownloadObjectAsync(hash).then(obj => {
|
||||||
|
|
||||||
|
try {
|
||||||
|
|
||||||
|
console.log("Downloaded selected object from OSM-API for initial load: ", hash)
|
||||||
|
const geojson = obj.asGeoJson()
|
||||||
|
this.state.allElements.addOrGetElement(geojson)
|
||||||
|
this.state.selectedElement.setData(geojson)
|
||||||
|
this.zoomToSelectedFeature()
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e)
|
||||||
|
}
|
||||||
|
|
||||||
})
|
})
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// If a feature is selected via the hash, zoom there
|
private setSelectedElementFromHash() {
|
||||||
public zoomToSelectedFeature(location: UIEventSource<Loc>) {
|
const state = this.state
|
||||||
const hash = this.hash.data;
|
const h = this.hash.data
|
||||||
if (hash === undefined || SelectedFeatureHandler._no_trigger_on.has(hash)) {
|
if (h === undefined || h === "") {
|
||||||
return; // No valid feature selected
|
// Hash has been cleared - we clear the selected element
|
||||||
}
|
state.selectedElement.setData(undefined);
|
||||||
// We should have a valid osm-ID and zoom to it... But we wrap it in try-catch to be sure
|
} else {
|
||||||
try {
|
// we search the element to select
|
||||||
|
const feature = state.allElements.ContainingFeatures.get(h)
|
||||||
OsmObject.DownloadObject(hash).addCallbackAndRunD(element => {
|
if (feature === undefined) {
|
||||||
const centerpoint = element.centerpoint();
|
return;
|
||||||
console.log("Zooming to location for select point: ", centerpoint)
|
}
|
||||||
location.data.lat = centerpoint[0]
|
const currentlySeleced = state.selectedElement.data
|
||||||
location.data.lon = centerpoint[1]
|
if (currentlySeleced === undefined) {
|
||||||
location.ping();
|
state.selectedElement.setData(feature)
|
||||||
})
|
return;
|
||||||
} catch (e) {
|
}
|
||||||
console.error("Could not download OSM-object with id", hash, " - probably a weird hash")
|
if (currentlySeleced.properties?.id === feature.properties.id) {
|
||||||
|
// We already have the right feature
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
state.selectedElement.setData(feature)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If a feature is selected via the hash, zoom there
|
||||||
|
private zoomToSelectedFeature() {
|
||||||
|
|
||||||
|
const selected = this.state.selectedElement.data
|
||||||
|
if(selected === undefined){
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const centerpoint= GeoOperations.centerpointCoordinates(selected)
|
||||||
|
const location = this.state.locationControl;
|
||||||
|
location.data.lon = centerpoint[0]
|
||||||
|
location.data.lat = centerpoint[1]
|
||||||
|
|
||||||
|
const minZoom = Math.max(14, ...(this.state.layoutToUse?.layers?.map(l => l.minzoomVisible) ?? []))
|
||||||
|
if(location.data.zoom < minZoom ){
|
||||||
|
location.data.zoom = minZoom
|
||||||
|
}
|
||||||
|
|
||||||
|
location.ping();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -23,6 +23,10 @@ import ScrollableFullScreen from "./Base/ScrollableFullScreen";
|
||||||
import Translations from "./i18n/Translations";
|
import Translations from "./i18n/Translations";
|
||||||
import SimpleAddUI from "./BigComponents/SimpleAddUI";
|
import SimpleAddUI from "./BigComponents/SimpleAddUI";
|
||||||
import StrayClickHandler from "../Logic/Actors/StrayClickHandler";
|
import StrayClickHandler from "../Logic/Actors/StrayClickHandler";
|
||||||
|
import Lazy from "./Base/Lazy";
|
||||||
|
import ShowDataMultiLayer from "./ShowDataLayer/ShowDataMultiLayer";
|
||||||
|
import StaticFeatureSource from "../Logic/FeatureSource/Sources/StaticFeatureSource";
|
||||||
|
import FilteredLayer from "../Models/FilteredLayer";
|
||||||
|
|
||||||
export class DefaultGuiState {
|
export class DefaultGuiState {
|
||||||
public readonly welcomeMessageIsOpened;
|
public readonly welcomeMessageIsOpened;
|
||||||
|
@ -81,12 +85,22 @@ export default class DefaultGUI {
|
||||||
constructor(state: FeaturePipelineState, guiState: DefaultGuiState) {
|
constructor(state: FeaturePipelineState, guiState: DefaultGuiState) {
|
||||||
this.state = state;
|
this.state = state;
|
||||||
this._guiState = guiState;
|
this._guiState = guiState;
|
||||||
const self = this;
|
|
||||||
|
|
||||||
if (state.layoutToUse.customCss !== undefined) {
|
if (state.layoutToUse.customCss !== undefined) {
|
||||||
Utils.LoadCustomCss(state.layoutToUse.customCss);
|
Utils.LoadCustomCss(state.layoutToUse.customCss);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
this.SetupUIElements();
|
||||||
|
this.SetupMap()
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private SetupMap(){
|
||||||
|
const state = this.state;
|
||||||
|
const guiState = this._guiState;
|
||||||
|
|
||||||
// Attach the map
|
// Attach the map
|
||||||
state.mainMapObject.SetClass("w-full h-full")
|
state.mainMapObject.SetClass("w-full h-full")
|
||||||
.AttachTo("leafletDiv")
|
.AttachTo("leafletDiv")
|
||||||
|
@ -96,8 +110,28 @@ export default class DefaultGUI {
|
||||||
state
|
state
|
||||||
)
|
)
|
||||||
|
|
||||||
this.InitWelcomeMessage();
|
|
||||||
|
|
||||||
|
new ShowDataLayer({
|
||||||
|
leafletMap: state.leafletMap,
|
||||||
|
layerToShow: AllKnownLayers.sharedLayers.get("home_location"),
|
||||||
|
features: state.homeLocation,
|
||||||
|
enablePopups: false,
|
||||||
|
})
|
||||||
|
|
||||||
|
state.leafletMap.addCallbackAndRunD(_ => {
|
||||||
|
// Lets assume that all showDataLayers are initialized at this point
|
||||||
|
state.selectedElement.ping()
|
||||||
|
State.state.locationControl.ping();
|
||||||
|
return true;
|
||||||
|
})
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private SetupUIElements(){
|
||||||
|
const state = this.state;
|
||||||
|
const guiState = this._guiState;
|
||||||
|
|
||||||
|
const self =this
|
||||||
Toggle.If(state.featureSwitchUserbadge,
|
Toggle.If(state.featureSwitchUserbadge,
|
||||||
() => new UserBadge(state)
|
() => new UserBadge(state)
|
||||||
).AttachTo("userbadge")
|
).AttachTo("userbadge")
|
||||||
|
@ -119,36 +153,21 @@ export default class DefaultGUI {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
new Toggle(self.InitWelcomeMessage(),
|
new Toggle(new Lazy(() => self.InitWelcomeMessage()),
|
||||||
Toggle.If(state.featureSwitchIframePopoutEnabled, iframePopout),
|
Toggle.If(state.featureSwitchIframePopoutEnabled, iframePopout),
|
||||||
state.featureSwitchWelcomeMessage
|
state.featureSwitchWelcomeMessage
|
||||||
).AttachTo("messagesbox");
|
).AttachTo("messagesbox");
|
||||||
|
|
||||||
new LeftControls(state, guiState).AttachTo("bottom-left");
|
new LeftControls(state, guiState).AttachTo("bottom-left");
|
||||||
new RightControls(state).AttachTo("bottom-right");
|
new RightControls(state).AttachTo("bottom-right");
|
||||||
State.state.locationControl.ping();
|
|
||||||
new CenterMessageBox(state).AttachTo("centermessage");
|
new CenterMessageBox(state).AttachTo("centermessage");
|
||||||
document
|
document
|
||||||
.getElementById("centermessage")
|
.getElementById("centermessage")
|
||||||
.classList.add("pointer-events-none");
|
.classList.add("pointer-events-none");
|
||||||
|
|
||||||
|
|
||||||
new ShowDataLayer({
|
|
||||||
leafletMap: state.leafletMap,
|
|
||||||
layerToShow: AllKnownLayers.sharedLayers.get("home_location"),
|
|
||||||
features: state.homeLocation,
|
|
||||||
enablePopups: false,
|
|
||||||
})
|
|
||||||
|
|
||||||
state.leafletMap.addCallbackAndRunD(_ => {
|
|
||||||
// Lets assume that all showDataLayers are initialized at this point
|
|
||||||
state.selectedElement.ping()
|
|
||||||
return true;
|
|
||||||
})
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private InitWelcomeMessage() {
|
private InitWelcomeMessage() : BaseUIElement{
|
||||||
const isOpened = this._guiState.welcomeMessageIsOpened
|
const isOpened = this._guiState.welcomeMessageIsOpened
|
||||||
const fullOptions = new FullWelcomePaneWithTabs(isOpened, this._guiState.welcomeMessageOpenedTab, this.state);
|
const fullOptions = new FullWelcomePaneWithTabs(isOpened, this._guiState.welcomeMessageOpenedTab, this.state);
|
||||||
|
|
||||||
|
@ -180,7 +199,7 @@ export default class DefaultGUI {
|
||||||
|
|
||||||
public setupClickDialogOnMap(filterViewIsOpened: UIEventSource<boolean>, state: FeaturePipelineState) {
|
public setupClickDialogOnMap(filterViewIsOpened: UIEventSource<boolean>, state: FeaturePipelineState) {
|
||||||
|
|
||||||
function setup(){
|
function setup() {
|
||||||
let presetCount = 0;
|
let presetCount = 0;
|
||||||
for (const layer of state.layoutToUse.layers) {
|
for (const layer of state.layoutToUse.layers) {
|
||||||
for (const preset of layer.presets) {
|
for (const preset of layer.presets) {
|
||||||
|
|
|
@ -5,6 +5,10 @@ import SelectedElementTagsUpdater from "../Logic/Actors/SelectedElementTagsUpdat
|
||||||
import UserRelatedState from "../Logic/State/UserRelatedState";
|
import UserRelatedState from "../Logic/State/UserRelatedState";
|
||||||
import {Utils} from "../Utils";
|
import {Utils} from "../Utils";
|
||||||
import ScriptUtils from "../scripts/ScriptUtils";
|
import ScriptUtils from "../scripts/ScriptUtils";
|
||||||
|
import SelectedFeatureHandler from "../Logic/Actors/SelectedFeatureHandler";
|
||||||
|
import {UIEventSource} from "../Logic/UIEventSource";
|
||||||
|
import {ElementStorage} from "../Logic/ElementStorage";
|
||||||
|
import Loc from "../Models/Loc";
|
||||||
|
|
||||||
export default class ActorsSpec extends T {
|
export default class ActorsSpec extends T {
|
||||||
|
|
||||||
|
@ -100,7 +104,35 @@ export default class ActorsSpec extends T {
|
||||||
// The fixme should be removed
|
// The fixme should be removed
|
||||||
T.equals(undefined, feature.properties.fixme)
|
T.equals(undefined, feature.properties.fixme)
|
||||||
|
|
||||||
}]
|
}],
|
||||||
|
["Hash without selected element should download geojson from OSM-API", async () => {
|
||||||
|
const hash = new UIEventSource("node/5568693115")
|
||||||
|
const selected = new UIEventSource(undefined)
|
||||||
|
const loc = new UIEventSource<Loc>({
|
||||||
|
lat: 0,
|
||||||
|
lon: 0,
|
||||||
|
zoom: 0
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
loc.addCallback(_ => {
|
||||||
|
T.equals("node/5568693115", selected.data.properties.id)
|
||||||
|
T.equals(14, loc.data.zoom)
|
||||||
|
T.equals( 51.2179199, loc.data.lat)
|
||||||
|
})
|
||||||
|
|
||||||
|
new SelectedFeatureHandler(hash, {
|
||||||
|
selectedElement: selected,
|
||||||
|
allElements: new ElementStorage(),
|
||||||
|
featurePipeline: undefined,
|
||||||
|
locationControl: loc,
|
||||||
|
layoutToUse: undefined
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}]
|
||||||
|
|
||||||
|
|
||||||
]);
|
]);
|
||||||
|
|
Loading…
Reference in a new issue