MapComplete/src/Models/ThemeViewState.ts

1143 lines
45 KiB
TypeScript
Raw Normal View History

import ThemeConfig from "./ThemeConfig/ThemeConfig"
2023-10-16 14:27:05 +02:00
import { SpecialVisualizationState } from "../UI/SpecialVisualization"
import { Changes } from "../Logic/Osm/Changes"
import { Store, UIEventSource } from "../Logic/UIEventSource"
import { FeatureSource, IndexedFeatureSource, WritableFeatureSource } from "../Logic/FeatureSource/FeatureSource"
2023-10-16 14:27:05 +02:00
import { OsmConnection } from "../Logic/Osm/OsmConnection"
import { ExportableMap, MapProperties } from "./MapProperties"
import LayerState from "../Logic/State/LayerState"
import { Feature, Point, Polygon } from "geojson"
import FullNodeDatabaseSource from "../Logic/FeatureSource/TiledFeatureSource/FullNodeDatabaseSource"
import { Map as MlMap } from "maplibre-gl"
import InitialMapPositioning from "../Logic/Actors/InitialMapPositioning"
import { MapLibreAdaptor } from "../UI/Map/MapLibreAdaptor"
import { GeoLocationState } from "../Logic/State/GeoLocationState"
import FeatureSwitchState from "../Logic/State/FeatureSwitchState"
import { QueryParameters } from "../Logic/Web/QueryParameters"
import UserRelatedState from "../Logic/State/UserRelatedState"
import LayerConfig from "./ThemeConfig/LayerConfig"
import GeoLocationHandler from "../Logic/Actors/GeoLocationHandler"
import { AvailableRasterLayers, RasterLayerPolygon, RasterLayerUtils } from "./RasterLayers"
import ThemeSource from "../Logic/FeatureSource/Sources/ThemeSource"
2023-10-16 14:27:05 +02:00
import StaticFeatureSource from "../Logic/FeatureSource/Sources/StaticFeatureSource"
import FeaturePropertiesStore from "../Logic/FeatureSource/Actors/FeaturePropertiesStore"
import PerLayerFeatureSourceSplitter from "../Logic/FeatureSource/PerLayerFeatureSourceSplitter"
import FilteringFeatureSource from "../Logic/FeatureSource/Sources/FilteringFeatureSource"
import ShowDataLayer from "../UI/Map/ShowDataLayer"
import TitleHandler from "../Logic/Actors/TitleHandler"
import ChangeToElementsActor from "../Logic/Actors/ChangeToElementsActor"
import PendingChangesUploader from "../Logic/Actors/PendingChangesUploader"
import SelectedElementTagsUpdater from "../Logic/Actors/SelectedElementTagsUpdater"
import { BBox } from "../Logic/BBox"
import Constants from "./Constants"
import Hotkeys from "../UI/Base/Hotkeys"
import Translations from "../UI/i18n/Translations"
import { GeoIndexedStoreForLayer } from "../Logic/FeatureSource/Actors/GeoIndexedStore"
import { LastClickFeatureSource } from "../Logic/FeatureSource/Sources/LastClickFeatureSource"
import { MenuState } from "./MenuState"
import MetaTagging from "../Logic/MetaTagging"
import ChangeGeometryApplicator from "../Logic/FeatureSource/Sources/ChangeGeometryApplicator"
import { NewGeometryFromChangesFeatureSource } from "../Logic/FeatureSource/Sources/NewGeometryFromChangesFeatureSource"
import OsmObjectDownloader from "../Logic/Osm/OsmObjectDownloader"
import ShowOverlayRasterLayer from "../UI/Map/ShowOverlayRasterLayer"
import { Utils } from "../Utils"
import { EliCategory } from "./RasterLayerProperties"
import BackgroundLayerResetter from "../Logic/Actors/BackgroundLayerResetter"
import SaveFeatureSourceToLocalStorage from "../Logic/FeatureSource/Actors/SaveFeatureSourceToLocalStorage"
import BBoxFeatureSource from "../Logic/FeatureSource/Sources/TouchesBboxFeatureSource"
import ThemeViewStateHashActor from "../Logic/Web/ThemeViewStateHashActor"
import NoElementsInViewDetector, { FeatureViewState } from "../Logic/Actors/NoElementsInViewDetector"
2023-10-16 14:27:05 +02:00
import FilteredLayer from "./FilteredLayer"
import { PreferredRasterLayerSelector } from "../Logic/Actors/PreferredRasterLayerSelector"
import { ImageUploadManager } from "../Logic/ImageProviders/ImageUploadManager"
import NearbyFeatureSource from "../Logic/FeatureSource/Sources/NearbyFeatureSource"
2023-11-22 19:39:19 +01:00
import FavouritesFeatureSource from "../Logic/FeatureSource/Sources/FavouritesFeatureSource"
import { ProvidedImage } from "../Logic/ImageProviders/ImageProvider"
2023-12-18 01:30:02 +01:00
import { GeolocationControlState } from "../UI/BigComponents/GeolocationControl"
import Zoomcontrol from "../UI/Zoomcontrol"
import {
SummaryTileSource,
SummaryTileSourceRewriter
} from "../Logic/FeatureSource/TiledFeatureSource/SummaryTileSource"
2024-02-15 17:39:59 +01:00
import summaryLayer from "../assets/generated/layers/summary.json"
import last_click_layerconfig from "../assets/generated/layers/last_click.json"
2024-02-15 17:39:59 +01:00
import { LayerConfigJson } from "./ThemeConfig/Json/LayerConfigJson"
2024-05-08 21:46:33 +02:00
import Hash from "../Logic/Web/Hash"
import { GeoOperations } from "../Logic/GeoOperations"
2024-07-19 11:37:20 +02:00
import { CombinedFetcher } from "../Logic/Web/NearbyImagesSearch"
2024-09-11 01:46:55 +02:00
import { GeocodeResult, GeocodingUtils } from "../Logic/Search/GeocodingProvider"
2024-08-30 02:18:29 +02:00
import SearchState from "../Logic/State/SearchState"
import { ShowDataLayerOptions } from "../UI/Map/ShowDataLayerOptions"
import { PanoramaxUploader } from "../Logic/ImageProviders/Panoramax"
import { Tag } from "../Logic/Tags/Tag"
2023-03-28 05:13:48 +02:00
/**
*
* The themeviewState contains all the state needed for the themeViewGUI.
*
* This is pretty much the 'brain' or the HQ of MapComplete
*
* It ties up all the needed elements and starts some actors.
*/
export default class ThemeViewState implements SpecialVisualizationState {
readonly theme: ThemeConfig
2023-10-16 14:27:05 +02:00
readonly map: UIEventSource<MlMap>
readonly changes: Changes
readonly featureSwitches: FeatureSwitchState
readonly featureSwitchIsTesting: Store<boolean>
readonly featureSwitchUserbadge: Store<boolean>
readonly featureProperties: FeaturePropertiesStore
readonly osmConnection: OsmConnection
readonly selectedElement: UIEventSource<Feature>
readonly mapProperties: MapLibreAdaptor & MapProperties & ExportableMap
2023-10-16 14:27:05 +02:00
readonly osmObjectDownloader: OsmObjectDownloader
readonly dataIsLoading: Store<boolean>
/**
* Indicates if there is _some_ data in view, even if it is not shown due to the filters
*/
readonly hasDataInView: Store<FeatureViewState>
readonly guistate: MenuState
readonly fullNodeDatabase?: FullNodeDatabaseSource
readonly historicalUserLocations: WritableFeatureSource<Feature<Point>>
readonly indexedFeatures: IndexedFeatureSource & ThemeSource
2023-10-16 14:27:05 +02:00
readonly currentView: FeatureSource<Feature<Polygon>>
readonly featuresInView: FeatureSource
2023-11-22 19:39:19 +01:00
readonly favourites: FavouritesFeatureSource
/**
* Contains a few (<10) >features that are near the center of the map.
*/
2023-11-22 19:39:19 +01:00
readonly closestFeatures: NearbyFeatureSource
2023-10-16 14:27:05 +02:00
readonly newFeatures: WritableFeatureSource
readonly layerState: LayerState
readonly perLayer: ReadonlyMap<string, GeoIndexedStoreForLayer>
readonly perLayerFiltered: ReadonlyMap<string, FilteringFeatureSource>
readonly availableLayers: { store: Store<RasterLayerPolygon[]> }
2023-10-16 14:27:05 +02:00
readonly userRelatedState: UserRelatedState
readonly geolocation: GeoLocationHandler
2023-12-18 01:30:02 +01:00
readonly geolocationControl: GeolocationControlState
2023-10-16 14:27:05 +02:00
readonly imageUploadManager: ImageUploadManager
readonly previewedImage = new UIEventSource<ProvidedImage>(undefined)
2023-10-16 14:27:05 +02:00
readonly addNewPoint: UIEventSource<boolean> = new UIEventSource<boolean>(false)
2023-12-21 17:36:43 +01:00
/**
* When using arrow keys to move, the accessibility mode is activated, which has a small rectangle set.
* This is the 'viewport' which 'closestFeatures' uses to filter wilt
*/
readonly visualFeedbackViewportBounds: UIEventSource<BBox> = new UIEventSource<BBox>(undefined)
2023-10-16 14:27:05 +02:00
readonly lastClickObject: LastClickFeatureSource
readonly overlayLayerStates: ReadonlyMap<
string,
{ readonly isDisplayed: UIEventSource<boolean> }
>
/**
* All 'level'-tags that are available with the current features
*/
readonly floors: Store<string[]>
/**
* If true, the user interface will toggle some extra aids for people using screenreaders and keyboard navigation
* Triggered by navigating the map with arrows or by pressing 'space' or 'enter'
*/
public readonly visualFeedback: UIEventSource<boolean> = new UIEventSource<boolean>(false)
public readonly toCacheSavers: ReadonlyMap<string, SaveFeatureSourceToLocalStorage>
2023-10-16 14:27:05 +02:00
2024-08-15 01:51:33 +02:00
public readonly nearbyImageSearcher: CombinedFetcher
/**
* Geocoded images that should be shown on the main map; probably only the currently hovered image
*/
public readonly geocodedImages: UIEventSource<Feature[]> = new UIEventSource<Feature[]>([])
2024-07-19 11:37:20 +02:00
2024-08-30 02:18:29 +02:00
public readonly searchState: SearchState
2024-10-01 23:46:04 +02:00
/**
* Used to check in the download panel if used
*/
public readonly featureSummary: SummaryTileSourceRewriter
2024-07-19 11:37:20 +02:00
constructor(layout: ThemeConfig, mvtAvailableLayers: Set<string>) {
2023-10-16 14:27:05 +02:00
Utils.initDomPurify()
this.theme = layout
2023-10-16 14:27:05 +02:00
this.featureSwitches = new FeatureSwitchState(layout)
this.guistate = new MenuState(
this.featureSwitches.featureSwitchWelcomeMessage.data,
2024-10-19 14:44:55 +02:00
layout.id
2023-10-16 14:27:05 +02:00
)
this.map = new UIEventSource<MlMap>(undefined)
2024-06-17 19:27:21 +02:00
const geolocationState = new GeoLocationState()
2023-10-16 14:27:05 +02:00
this.osmConnection = new OsmConnection({
dryRun: this.featureSwitches.featureSwitchIsTesting,
fakeUser: this.featureSwitches.featureSwitchFakeUser.data,
oauth_token: QueryParameters.GetQueryParameter(
"oauth_token",
undefined,
2024-10-19 14:44:55 +02:00
"Used to complete the login"
)
2023-10-16 14:27:05 +02:00
})
const initial = new InitialMapPositioning(layout, geolocationState, this.osmConnection)
this.mapProperties = new MapLibreAdaptor(this.map, initial, { correctClick: 20 })
this.featureSwitchIsTesting = this.featureSwitches.featureSwitchIsTesting
this.featureSwitchUserbadge = this.featureSwitches.featureSwitchEnableLogin
2023-10-16 14:27:05 +02:00
this.userRelatedState = new UserRelatedState(
this.osmConnection,
layout,
this.featureSwitches,
2024-10-19 14:44:55 +02:00
this.mapProperties
2023-10-16 14:27:05 +02:00
)
this.userRelatedState.fixateNorth.addCallbackAndRunD((fixated) => {
this.mapProperties.allowRotating.setData(fixated !== "yes")
2023-10-06 03:34:26 +02:00
})
2023-10-16 14:27:05 +02:00
this.selectedElement = new UIEventSource<Feature | undefined>(undefined, "Selected element")
this.geolocation = new GeoLocationHandler(
geolocationState,
this.selectedElement,
this.mapProperties,
2024-10-19 14:44:55 +02:00
this.userRelatedState.gpsLocationHistoryRetentionTime
2023-10-16 14:27:05 +02:00
)
2023-12-18 01:30:02 +01:00
this.geolocationControl = new GeolocationControlState(this.geolocation, this.mapProperties)
2023-10-16 14:27:05 +02:00
this.availableLayers = AvailableRasterLayers.layersAvailableAt(
this.mapProperties.location,
2024-10-19 14:44:55 +02:00
this.osmConnection.isLoggedIn
)
2023-10-16 14:27:05 +02:00
2024-07-09 13:42:08 +02:00
this.layerState = new LayerState(
this.osmConnection,
layout.layers,
layout.id,
2024-10-19 14:44:55 +02:00
this.featureSwitches.featureSwitchLayerDefault
2024-07-09 13:42:08 +02:00
)
{
2023-10-16 14:27:05 +02:00
const overlayLayerStates = new Map<string, { isDisplayed: UIEventSource<boolean> }>()
for (const rasterInfo of this.theme.tileLayerSources) {
2023-10-16 14:27:05 +02:00
const isDisplayed = QueryParameters.GetBooleanQueryParameter(
"overlay-" + rasterInfo.id,
rasterInfo.defaultState ?? true,
2024-10-19 14:44:55 +02:00
"Whether or not overlay layer " + rasterInfo.id + " is shown"
2023-10-16 14:27:05 +02:00
)
const state = { isDisplayed }
overlayLayerStates.set(rasterInfo.id, state)
new ShowOverlayRasterLayer(rasterInfo, this.map, this.mapProperties, state)
}
this.overlayLayerStates = overlayLayerStates
}
2023-10-16 14:27:05 +02:00
{
2024-06-27 02:29:53 +02:00
/* Set up the layout source
* A bit tricky, as this is heavily intertwined with the 'changes'-element, which generates a stream of new and changed features too
2023-10-16 14:27:05 +02:00
*/
if (this.theme.layers.some((l) => l._needsFullNodeDatabase)) {
2023-10-16 14:27:05 +02:00
this.fullNodeDatabase = new FullNodeDatabaseSource()
}
const layoutSource = new ThemeSource(
2023-10-16 14:27:05 +02:00
layout.layers,
this.featureSwitches,
this.mapProperties,
this.osmConnection.Backend(),
(id) => this.layerState.filteredLayers.get(id).isDisplayed,
2024-02-19 15:38:46 +01:00
mvtAvailableLayers,
2024-10-19 14:44:55 +02:00
this.fullNodeDatabase
2023-10-16 14:27:05 +02:00
)
let currentViewIndex = 0
const empty = []
this.currentView = new StaticFeatureSource(
this.mapProperties.bounds.map((bbox) => {
if (!bbox) {
return empty
}
currentViewIndex++
return <Feature[]>[
bbox.asGeoJson({
zoom: this.mapProperties.zoom.data,
...this.mapProperties.location.data,
id: "current_view_" + currentViewIndex,
}),
2023-10-16 14:27:05 +02:00
]
2024-10-19 14:44:55 +02:00
})
2023-10-16 14:27:05 +02:00
)
this.featuresInView = new BBoxFeatureSource(layoutSource, this.mapProperties.bounds)
2023-10-16 14:27:05 +02:00
this.dataIsLoading = layoutSource.isLoading
2023-11-22 19:39:19 +01:00
this.indexedFeatures = layoutSource
this.featureProperties = new FeaturePropertiesStore(layoutSource)
2023-10-16 14:27:05 +02:00
this.changes = new Changes(
this,
layout?.isLeftRightSensitive() ?? false,
2024-10-19 14:44:55 +02:00
(e, extraMsg) => this.reportError(e, extraMsg)
2023-10-16 14:27:05 +02:00
)
this.historicalUserLocations = this.geolocation.historicalUserLocations
this.newFeatures = new NewGeometryFromChangesFeatureSource(
this.changes,
2023-11-22 19:39:19 +01:00
layoutSource,
2024-10-19 14:44:55 +02:00
this.featureProperties
2023-10-16 14:27:05 +02:00
)
layoutSource.addSource(this.newFeatures)
const perLayer = new PerLayerFeatureSourceSplitter(
Array.from(this.layerState.filteredLayers.values()).filter(
2024-10-19 14:44:55 +02:00
(l) => l.layerDef?.source !== null
2023-10-16 14:27:05 +02:00
),
new ChangeGeometryApplicator(this.indexedFeatures, this.changes),
{
constructStore: (features, layer) =>
new GeoIndexedStoreForLayer(features, layer),
handleLeftovers: (features) => {
console.warn(
"Got ",
features.length,
"leftover features, such as",
2024-10-19 14:44:55 +02:00
features[0].properties
2023-10-16 14:27:05 +02:00
)
},
2024-10-19 14:44:55 +02:00
}
2023-10-16 14:27:05 +02:00
)
this.perLayer = perLayer.perLayer
2023-10-06 03:34:26 +02:00
}
2023-10-16 14:27:05 +02:00
this.floors = this.featuresInView.features.stabilized(500).map((features) => {
if (!features) {
return []
}
const floors = new Set<string>()
for (const feature of features) {
const level = feature.properties["_level"]
2023-10-16 14:27:05 +02:00
if (level) {
const levels = level.split(";")
for (const l of levels) {
floors.add(l)
}
} else {
floors.add("0") // '0' is the default and is thus _always_ present
}
}
const sorted = Array.from(floors)
// Sort alphabetically first, to deal with floor "A", "B" and "C"
sorted.sort()
sorted.sort((a, b) => {
// We use the laxer 'parseInt' to deal with floor '1A'
const na = parseInt(a)
const nb = parseInt(b)
if (isNaN(na) || isNaN(nb)) {
return 0
}
return na - nb
})
sorted.reverse(/* new list, no side-effects */)
return sorted
})
2024-06-20 04:21:29 +02:00
this.lastClickObject = new LastClickFeatureSource(
this.theme,
this.mapProperties.lastClickLocation,
2024-10-19 14:44:55 +02:00
this.userRelatedState.addNewFeatureMode
2024-06-20 04:21:29 +02:00
)
2023-10-16 14:27:05 +02:00
this.osmObjectDownloader = new OsmObjectDownloader(
this.osmConnection.Backend(),
2024-10-19 14:44:55 +02:00
this.changes
2023-10-16 14:27:05 +02:00
)
this.perLayerFiltered = this.showNormalDataOn(this.map)
this.closestFeatures = new NearbyFeatureSource(
this.mapProperties.location,
this.perLayerFiltered,
2023-12-21 17:36:43 +01:00
{
currentZoom: this.mapProperties.zoom,
layerState: this.layerState,
2024-10-19 14:44:55 +02:00
bounds: this.visualFeedbackViewportBounds.map(
(bounds) => bounds ?? this.mapProperties.bounds?.data,
[this.mapProperties.bounds]
),
}
)
2024-10-01 23:46:04 +02:00
this.featureSummary = this.setupSummaryLayer()
2023-10-16 14:27:05 +02:00
this.hasDataInView = new NoElementsInViewDetector(this).hasFeatureInView
this.imageUploadManager = new ImageUploadManager(
layout,
2024-10-24 00:56:24 +02:00
new PanoramaxUploader(
Constants.panoramax.url,
Constants.panoramax.token,
this.featureSwitchIsTesting.map((t) =>
t ? Constants.panoramax.testsequence : Constants.panoramax.sequence
)
),
2023-10-16 14:27:05 +02:00
this.featureProperties,
this.osmConnection,
this.changes,
this.geolocation.geolocationState.currentGPSLocation,
this.indexedFeatures,
2024-10-19 14:44:55 +02:00
this.reportError
2023-10-16 14:27:05 +02:00
)
this.favourites = new FavouritesFeatureSource(this)
2024-07-19 11:37:20 +02:00
const longAgo = new Date()
2024-07-21 10:52:51 +02:00
longAgo.setTime(new Date().getTime() - 5 * 365 * 24 * 60 * 60 * 1000)
2024-07-19 11:37:20 +02:00
this.nearbyImageSearcher = new CombinedFetcher(50, longAgo, this.indexedFeatures)
2023-10-16 14:27:05 +02:00
this.toCacheSavers = layout.enableCache ? this.initSaveToLocalStorage() : undefined
2024-08-15 01:51:33 +02:00
2024-08-30 02:18:29 +02:00
this.searchState = new SearchState(this)
2024-08-15 01:51:33 +02:00
2023-10-16 14:27:05 +02:00
this.initActors()
this.drawSpecialLayers()
this.initHotkeys()
this.miscSetup()
this.focusOnMap()
2023-10-16 14:27:05 +02:00
if (!Utils.runningFromConsole) {
console.log("State setup completed", this)
}
2023-03-28 05:13:48 +02:00
}
2023-10-16 14:27:05 +02:00
/* By focussing on the map, the keyboard panning and zoom with '+' and '+' works */
public focusOnMap() {
if (this.map.data) {
this.map.data.getCanvas().focus()
return
}
this.map.addCallbackAndRunD((map) => {
map.on("load", () => {
map.getCanvas().focus()
})
return true
})
}
public initSaveToLocalStorage() {
const toLocalStorage = new Map<string, SaveFeatureSourceToLocalStorage>()
this.perLayer.forEach((fs, layerId) => {
if (fs.layer.layerDef.source.geojsonSource !== undefined) {
return // We don't cache external data layers
}
const storage = new SaveFeatureSourceToLocalStorage(
this.osmConnection.Backend(),
fs.layer.layerDef.id,
ThemeSource.fromCacheZoomLevel,
fs,
this.featureProperties,
2024-10-19 14:44:55 +02:00
fs.layer.layerDef.maxAgeOfCache
)
toLocalStorage.set(layerId, storage)
})
return toLocalStorage
}
2023-10-16 14:27:05 +02:00
public showNormalDataOn(map: Store<MlMap>): ReadonlyMap<string, FilteringFeatureSource> {
const filteringFeatureSource = new Map<string, FilteringFeatureSource>()
this.perLayer.forEach((fs, layerName) => {
const doShowLayer = this.mapProperties.zoom.map(
(z) => {
2024-11-07 11:19:15 +01:00
if (
(fs.layer.isDisplayed?.data ?? true) &&
z >= (fs.layer.layerDef?.minzoom ?? 0)
) {
return true
}
2024-11-07 11:19:15 +01:00
if (this.layerState.globalFilters.data.some((f) => f.forceShowOnMatch)) {
return true
}
return false
},
[fs.layer.isDisplayed, this.layerState.globalFilters]
2023-10-16 14:27:05 +02:00
)
if (!doShowLayer.data && this.featureSwitches.featureSwitchFilter.data === false) {
/* This layer is hidden and there is no way to enable it (filterview is disabled or this layer doesn't show up in the filter view as the name is not defined)
*
* This means that we don't have to filter it, nor do we have to display it
*
* Note: it is tempting to also permanently disable the layer if it is not visible _and_ the layer name is hidden.
* However, this is _not_ correct: the layer might be hidden because zoom is not enough. Zooming in more _will_ reveal the layer!
* */
return
}
const filtered = new FilteringFeatureSource(
fs.layer,
fs,
(id) => this.featureProperties.getStore(id),
this.layerState.globalFilters,
undefined,
this.mapProperties.zoom,
this.selectedElement
2023-10-16 14:27:05 +02:00
)
filteringFeatureSource.set(layerName, filtered)
new ShowDataLayer(map, {
layer: fs.layer.layerDef,
features: filtered,
doShowLayer,
2023-11-14 17:35:12 +01:00
metaTags: this.userRelatedState.preferencesAsTags,
2023-10-16 14:27:05 +02:00
selectedElement: this.selectedElement,
fetchStore: (id) => this.featureProperties.getStore(id),
2023-10-16 14:27:05 +02:00
})
})
return filteringFeatureSource
}
2023-10-16 14:27:05 +02:00
public openNewDialog() {
this.selectedElement.setData(undefined)
const { lon, lat } = this.mapProperties.location.data
const feature = this.lastClickObject.createFeature(lon, lat)
this.featureProperties.trackFeature(feature)
this.selectedElement.setData(feature)
}
public showCurrentLocationOn(map: Store<MlMap>): ShowDataLayer {
const id = "gps_location"
const flayerGps = this.layerState.filteredLayers.get(id)
if (flayerGps === undefined) {
return
}
const features = this.geolocation.currentUserLocation
return new ShowDataLayer(map, {
features,
doShowLayer: flayerGps.isDisplayed,
layer: flayerGps.layerDef,
metaTags: this.userRelatedState.preferencesAsTags,
selectedElement: this.selectedElement,
})
}
2023-10-16 14:27:05 +02:00
/**
* Various small methods that need to be called
*/
private miscSetup() {
this.userRelatedState.a11y.addCallbackAndRunD((a11y) => {
if (a11y === "always") {
this.visualFeedback.setData(true)
} else if (a11y === "never") {
this.visualFeedback.setData(false)
}
})
this.mapProperties.onKeyNavigationEvent((keyEvent) => {
if (this.userRelatedState.a11y.data === "never") {
return
}
if (["north", "east", "south", "west"].indexOf(keyEvent.key) >= 0) {
this.visualFeedback.setData(true)
return true // Our job is done, unregister
}
})
this.userRelatedState.markLayoutAsVisited(this.theme)
2023-10-16 14:27:05 +02:00
this.selectedElement.addCallback((selected) => {
if (selected === undefined) {
Zoomcontrol.resetzoom()
}
})
if (this.theme.customCss !== undefined && window.location.pathname.indexOf("theme") >= 0) {
Utils.LoadCustomCss(this.theme.customCss)
2023-03-28 05:13:48 +02:00
}
2024-05-08 21:46:33 +02:00
2024-06-16 16:06:26 +02:00
Hash.hash.addCallbackAndRunD((hash) => {
if (hash === "current_view" || hash.match(/current_view_[0-9]+/)) {
2024-05-08 21:46:33 +02:00
this.selectCurrentView()
}
})
2023-10-16 14:27:05 +02:00
}
2024-11-28 12:00:23 +01:00
private setSelectedElement(feature: Feature) {
const current = this.selectedElement.data
2024-11-28 12:00:23 +01:00
if (
current?.properties?.id !== undefined &&
current.properties.id === feature.properties.id
) {
console.log("Not setting selected, same id", current, feature)
return // already set
}
this.selectedElement.setData(feature)
}
/**
* Selects the feature that is 'i' closest to the map center
*/
private selectClosestAtCenter(i: number = 0) {
console.log("Selecting closest", i)
if (this.userRelatedState.a11y.data !== "never") {
this.visualFeedback.setData(true)
}
2024-10-02 00:18:57 +02:00
2023-12-21 17:36:43 +01:00
const toSelect = this.closestFeatures.features?.data?.[i]
if (!toSelect) {
2023-12-21 17:36:43 +01:00
window.requestAnimationFrame(() => {
const toSelect = this.closestFeatures.features?.data?.[i]
if (!toSelect) {
return
}
this.setSelectedElement(toSelect)
2023-12-21 17:36:43 +01:00
})
return
}
this.setSelectedElement(toSelect)
}
2023-10-16 14:27:05 +02:00
private initHotkeys() {
const docs = Translations.t.hotkeyDocumentation
Hotkeys.RegisterHotkey({ nomod: "Escape", onUp: true }, docs.closeSidebar, () => {
if (this.previewedImage.data !== undefined) {
this.previewedImage.setData(undefined)
return
2023-10-16 14:27:05 +02:00
}
if (this.selectedElement.data) {
this.selectedElement.setData(undefined)
return
}
if (this.searchState.showSearchDrawer.data) {
this.searchState.showSearchDrawer.set(false)
return
}
if (this.guistate.closeAll()) {
return
}
Zoomcontrol.resetzoom()
this.focusOnMap()
})
2023-10-16 14:27:05 +02:00
Hotkeys.RegisterHotkey({ nomod: "f" }, docs.selectFavourites, () => {
this.guistate.pageStates.favourites.set(true)
})
Hotkeys.RegisterHotkey(
{
nomod: " ",
onUp: true,
},
docs.selectItem,
() => {
if (this.selectedElement.data !== undefined) {
return false
}
2024-09-02 12:48:15 +02:00
if (this.guistate.isSomethingOpen() || this.previewedImage.data !== undefined) {
return
}
2024-10-19 14:44:55 +02:00
if (
document.activeElement.tagName === "button" ||
document.activeElement.tagName === "input"
) {
2024-08-22 22:50:37 +02:00
return
}
this.selectClosestAtCenter(0)
2024-10-19 14:44:55 +02:00
}
)
for (let i = 1; i < 9; i++) {
2023-12-20 21:56:16 +01:00
let doc = docs.selectItemI.Subs({ i })
if (i === 1) {
doc = docs.selectItem
} else if (i === 2) {
doc = docs.selectItem2
} else if (i === 3) {
doc = docs.selectItem3
}
Hotkeys.RegisterHotkey(
{
nomod: "" + i,
onUp: true,
},
2023-12-20 21:56:16 +01:00
doc,
2024-10-19 14:44:55 +02:00
() => this.selectClosestAtCenter(i - 1)
)
}
2024-10-19 14:44:55 +02:00
Hotkeys.RegisterHotkey(
{ ctrl: "F" },
Translations.t.hotkeyDocumentation.selectSearch,
() => {
this.searchState.feedback.set(undefined)
this.searchState.searchIsFocused.set(true)
}
)
2023-10-16 14:27:05 +02:00
this.featureSwitches.featureSwitchBackgroundSelection.addCallbackAndRun((enable) => {
if (!enable) {
return
}
Hotkeys.RegisterHotkey(
{
nomod: "b",
2023-10-16 14:27:05 +02:00
},
docs.openLayersPanel,
2023-12-09 15:28:11 +01:00
() => {
if (this.featureSwitches.featureSwitchBackgroundSelection.data) {
this.guistate.pageStates.background.setData(true)
2023-12-09 15:28:11 +01:00
}
2024-10-19 14:44:55 +02:00
}
2023-12-09 15:28:11 +01:00
)
Hotkeys.RegisterHotkey(
{
nomod: "s",
2023-12-09 15:28:11 +01:00
},
Translations.t.hotkeyDocumentation.openFilterPanel,
2023-10-16 14:27:05 +02:00
() => {
if (this.featureSwitches.featureSwitchFilter.data) {
this.guistate.openFilterView()
}
2024-10-19 14:44:55 +02:00
}
2023-10-16 14:27:05 +02:00
)
const setLayerCategory = (category: EliCategory, skipLayers: number = 0) => {
const timeOfCall = new Date()
2024-08-22 03:01:21 +02:00
this.availableLayers.store.addCallbackAndRunD((available) => {
2024-08-14 13:53:56 +02:00
const now = new Date()
const timeDiff = (now.getTime() - timeOfCall.getTime()) / 1000
if (timeDiff > 3) {
return true // unregister
}
const current = this.mapProperties.rasterLayer
const best = RasterLayerUtils.SelectBestLayerAccordingTo(
available,
category,
current.data,
2024-10-19 14:44:55 +02:00
skipLayers
2024-08-14 13:53:56 +02:00
)
if (!best) {
return
}
console.log("Best layer for category", category, "is", best?.properties?.id)
2024-08-14 13:53:56 +02:00
current.setData(best)
})
}
2023-10-16 14:27:05 +02:00
Hotkeys.RegisterHotkey(
{ nomod: "O" },
Translations.t.hotkeyDocumentation.selectOsmbasedmap,
2024-10-19 14:44:55 +02:00
() => setLayerCategory("osmbasedmap")
2023-10-16 14:27:05 +02:00
)
Hotkeys.RegisterHotkey(
{ nomod: "M" },
Translations.t.hotkeyDocumentation.selectMap,
2024-10-19 14:44:55 +02:00
() => setLayerCategory("map")
2023-10-16 14:27:05 +02:00
)
Hotkeys.RegisterHotkey(
{ nomod: "P" },
Translations.t.hotkeyDocumentation.selectAerial,
2024-10-19 14:44:55 +02:00
() => setLayerCategory("photo")
2023-10-16 14:27:05 +02:00
)
Hotkeys.RegisterHotkey(
{ shift: "O" },
Translations.t.hotkeyDocumentation.selectOsmbasedmap,
2024-10-19 14:44:55 +02:00
() => setLayerCategory("osmbasedmap", 2)
)
Hotkeys.RegisterHotkey(
{ shift: "M" },
Translations.t.hotkeyDocumentation.selectMap,
2024-10-19 14:44:55 +02:00
() => setLayerCategory("map", 2)
)
Hotkeys.RegisterHotkey(
{ shift: "P" },
Translations.t.hotkeyDocumentation.selectAerial,
2024-10-19 14:44:55 +02:00
() => setLayerCategory("photo", 2)
2023-10-16 14:27:05 +02:00
)
2023-12-18 01:30:02 +01:00
Hotkeys.RegisterHotkey(
{ nomod: "L" },
Translations.t.hotkeyDocumentation.geolocate,
() => {
this.geolocationControl.handleClick()
2024-10-19 14:44:55 +02:00
}
2023-12-18 01:30:02 +01:00
)
2023-10-16 14:27:05 +02:00
return true
})
Hotkeys.RegisterHotkey(
{
shift: "T",
},
Translations.t.hotkeyDocumentation.translationMode,
() => {
2024-08-26 17:23:04 +02:00
const tm = this.userRelatedState.translationMode
2024-09-02 12:48:15 +02:00
if (tm.data === "false") {
2024-08-26 17:23:04 +02:00
tm.setData("true")
} else {
tm.setData("false")
}
2024-10-19 14:44:55 +02:00
}
)
2023-10-16 14:27:05 +02:00
}
private setupSummaryLayer(): SummaryTileSourceRewriter | undefined {
/**
* MaxZoom for the summary layer
*/
2024-10-19 14:44:55 +02:00
const normalLayers = this.theme.layers.filter((l) => l.isNormal())
const maxzoom = Math.min(...normalLayers.map((l) => l.minzoom))
const layers = this.theme.layers.filter(
2024-02-15 17:39:59 +01:00
(l) =>
(<string[]><unknown>Constants.priviliged_layers).indexOf(l.id) < 0 &&
l.source.geojsonSource === undefined &&
2024-10-19 14:44:55 +02:00
l.doCount
2024-02-15 17:39:59 +01:00
)
if (!Constants.SummaryServer || layers.length === 0) {
return undefined
}
const summaryTileSource = new SummaryTileSource(
2024-06-12 15:03:10 +02:00
Constants.SummaryServer,
2024-02-15 17:39:59 +01:00
layers.map((l) => l.id),
this.mapProperties.zoom.map((z) => Math.max(Math.floor(z), 0)),
2024-02-20 02:56:23 +01:00
this.mapProperties,
{
isActive: this.mapProperties.zoom.map((z) => z < maxzoom),
2024-10-19 14:44:55 +02:00
}
2024-02-15 17:39:59 +01:00
)
return new SummaryTileSourceRewriter(summaryTileSource, this.layerState.filteredLayers)
2024-02-15 17:39:59 +01:00
}
/**
2023-10-16 14:27:05 +02:00
* Add the special layers to the map
*/
2023-10-16 14:27:05 +02:00
private drawSpecialLayers() {
type AddedByDefaultTypes = (typeof Constants.added_by_default)[number]
const empty = []
/**
* A listing which maps the layerId onto the featureSource
*/
2024-06-20 04:21:29 +02:00
const specialLayers: Record<AddedByDefaultTypes | "current_view", FeatureSource> = {
2023-10-16 14:27:05 +02:00
home_location: this.userRelatedState.homeLocation,
gps_location: this.geolocation.currentUserLocation,
gps_location_history: this.geolocation.historicalUserLocations,
gps_track: this.geolocation.historicalUserLocationsTrack,
geocoded_image: new StaticFeatureSource(this.geocodedImages),
2023-10-16 14:27:05 +02:00
selected_element: new StaticFeatureSource(
2024-10-19 14:44:55 +02:00
this.selectedElement.map((f) => (f === undefined ? empty : [f]))
2023-10-16 14:27:05 +02:00
),
range: new StaticFeatureSource(
this.mapProperties.maxbounds.map((bbox) =>
2024-10-19 14:44:55 +02:00
bbox === undefined ? empty : <Feature[]>[bbox.asGeoJson({ id: "range" })]
)
2023-10-16 14:27:05 +02:00
),
current_view: this.currentView,
2023-11-22 19:39:19 +01:00
favourite: this.favourites,
2024-10-01 23:46:04 +02:00
summary: this.featureSummary,
2024-06-20 04:21:29 +02:00
last_click: this.lastClickObject,
search: this.searchState.locationResults,
}
2023-11-22 19:39:19 +01:00
this.closestFeatures.registerSource(specialLayers.favourite, "favourite")
if (this.theme?.lockLocation) {
const bbox = new BBox(<[[number, number], [number, number]]>this.theme.lockLocation)
2023-10-16 14:27:05 +02:00
this.mapProperties.maxbounds.setData(bbox)
ShowDataLayer.showRange(
this.map,
new StaticFeatureSource([bbox.asGeoJson({ id: "range" })]),
2024-10-19 14:44:55 +02:00
this.featureSwitches.featureSwitchIsTesting
2023-10-16 14:27:05 +02:00
)
}
const currentViewLayer = this.theme.layers.find((l) => l.id === "current_view")
2023-10-16 14:27:05 +02:00
if (currentViewLayer?.tagRenderings?.length > 0) {
const params = MetaTagging.createExtraFuncParams(this)
this.featureProperties.trackFeatureSource(specialLayers.current_view)
specialLayers.current_view.features.addCallbackAndRunD((features) => {
MetaTagging.addMetatags(
features,
params,
currentViewLayer,
this.theme,
2023-10-16 14:27:05 +02:00
this.osmObjectDownloader,
2024-10-19 14:44:55 +02:00
this.featureProperties
2023-10-16 14:27:05 +02:00
)
})
}
const rangeFLayer: FilteredLayer = this.layerState.filteredLayers.get("range")
const rangeIsDisplayed = rangeFLayer?.isDisplayed
if (
2023-12-06 12:25:47 +01:00
rangeFLayer &&
2023-10-16 14:27:05 +02:00
!QueryParameters.wasInitialized(FilteredLayer.queryParameterKey(rangeFLayer.layerDef))
) {
rangeIsDisplayed?.syncWith(this.featureSwitches.featureSwitchIsTesting, true)
}
2024-02-15 17:39:59 +01:00
// enumerate all 'normal' layers and match them with the appropriate 'special' layer - if applicable
2023-10-16 14:27:05 +02:00
this.layerState.filteredLayers.forEach((flayer) => {
const id = flayer.layerDef.id
const features: FeatureSource = specialLayers[id]
if (!features?.features) {
2023-10-16 14:27:05 +02:00
return
}
if (id === "summary" || id === "last_click") {
2024-02-15 17:39:59 +01:00
return
2023-11-22 19:39:19 +01:00
}
2023-03-28 05:13:48 +02:00
2023-10-16 14:27:05 +02:00
this.featureProperties.trackFeatureSource(features)
const options: ShowDataLayerOptions & { layer: LayerConfig } = {
2023-10-16 14:27:05 +02:00
features,
doShowLayer: flayer.isDisplayed,
layer: flayer.layerDef,
2023-11-14 17:35:12 +01:00
metaTags: this.userRelatedState.preferencesAsTags,
selectedElement: this.selectedElement,
}
if (flayer.layerDef.id === "search") {
options.onClick = (feature) => {
this.searchState.clickedOnMap(feature)
}
delete options.selectedElement
}
new ShowDataLayer(this.map, options)
2023-10-16 14:27:05 +02:00
})
// last click
{
const lastClickLayerConfig = new LayerConfig(
<LayerConfigJson>last_click_layerconfig,
2024-10-19 14:44:55 +02:00
"last_click"
)
const lastClickFiltered =
lastClickLayerConfig.isShown === undefined
? specialLayers.last_click
: specialLayers.last_click.features.mapD((fs) =>
2024-10-19 14:44:55 +02:00
fs.filter((f) => {
const matches = lastClickLayerConfig.isShown.matchesProperties(
f.properties
)
console.debug("LastClick ", f, "matches", matches)
return matches
})
)
// show last click = new point/note marker
new ShowDataLayer(this.map, {
features: new StaticFeatureSource(lastClickFiltered),
layer: lastClickLayerConfig,
onClick: (feature) => {
if (this.mapProperties.zoom.data >= Constants.minZoomLevelToAddNewPoint) {
this.selectedElement.setData(feature)
return
}
this.map.data.flyTo({
zoom: Constants.minZoomLevelToAddNewPoint,
center: GeoOperations.centerpointCoordinates(feature),
})
},
})
}
2024-09-24 18:08:01 +02:00
if (specialLayers.summary) {
new ShowDataLayer(this.map, {
features: specialLayers.summary,
layer: new LayerConfig(<LayerConfigJson>summaryLayer, "summaryLayer"),
// doShowLayer: this.mapProperties.zoom.map((z) => z < maxzoom),
selectedElement: this.selectedElement,
})
}
2023-03-28 05:13:48 +02:00
}
2023-10-16 14:27:05 +02:00
/**
* Setup various services for which no reference are needed
*/
private initActors() {
if (!this.theme.official) {
// Add custom themes to the "visited custom themes"
const th = this.theme
this.userRelatedState.addUnofficialTheme({
id: th.id,
icon: th.icon,
title: th.title.translations,
2024-10-19 14:44:55 +02:00
shortDescription: th.shortDescription.translations,
layers: th.layers.filter((l) => l.isNormal()).map((l) => l.id),
})
}
2023-10-16 14:27:05 +02:00
this.selectedElement.addCallback((selected) => {
if (selected === undefined) {
this.focusOnMap()
this.geocodedImages.set([])
2024-07-21 10:52:51 +02:00
} else {
this.lastClickObject.clear()
2023-10-16 14:27:05 +02:00
}
})
Object.values(this.guistate.pageStates).forEach((toggle) => {
toggle.addCallbackD((isOpened) => {
if (!isOpened) {
if (!this.guistate.isSomethingOpen()) {
this.focusOnMap()
}
}
})
})
// Add the selected element to the recently visited history
2024-10-19 14:44:55 +02:00
this.selectedElement.addCallbackD((selected) => {
const [osm_type, osm_id] = selected.properties.id.split("/")
const [lon, lat] = GeoOperations.centerpointCoordinates(selected)
const layer = this.theme.getMatchingLayer(selected.properties)
const nameOptions = [
selected?.properties?.name,
2024-10-19 14:44:55 +02:00
selected?.properties?.alt_name,
selected?.properties?.local_name,
layer?.title.GetRenderValue(selected?.properties ?? {}).txt,
selected.properties.display_name,
selected.properties.id,
]
const r = <GeocodeResult>{
feature: selected,
2024-10-19 14:44:55 +02:00
display_name: nameOptions.find((opt) => opt !== undefined),
osm_id,
osm_type,
lon,
lat,
}
this.userRelatedState.recentlyVisitedSearch.add(r)
})
2024-11-28 12:00:23 +01:00
this.mapProperties.lastClickLocation.addCallbackD((lastClick) => {
if (lastClick.mode !== "left" || !lastClick.nearestFeature) {
return
}
const f = lastClick.nearestFeature
this.setSelectedElement(f)
})
2024-10-19 14:44:55 +02:00
this.userRelatedState.showScale.addCallbackAndRun((showScale) => {
this.mapProperties.showScale.set(showScale)
})
2024-11-07 11:19:15 +01:00
this.layerState.filteredLayers
.get("favourite")
?.isDisplayed?.addCallbackAndRunD((favouritesShown) => {
const oldGlobal = this.layerState.globalFilters.data
const key = "show-favourite"
if (favouritesShown) {
this.layerState.globalFilters.set([
...oldGlobal,
{
forceShowOnMatch: true,
id: key,
osmTags: new Tag("_favourite", "yes"),
state: 0,
onNewPoint: undefined,
},
])
} else {
this.layerState.globalFilters.set(oldGlobal.filter((gl) => gl.id !== key))
}
})
2023-10-16 14:27:05 +02:00
new ThemeViewStateHashActor(this)
new MetaTagging(this)
2024-09-27 03:19:31 +02:00
new TitleHandler(this.selectedElement, this)
2023-10-16 14:27:05 +02:00
new ChangeToElementsActor(this.changes, this.featureProperties)
new PendingChangesUploader(this.changes, this.selectedElement, this.imageUploadManager)
2023-10-16 14:27:05 +02:00
new SelectedElementTagsUpdater(this)
new BackgroundLayerResetter(this.mapProperties.rasterLayer, this.availableLayers)
new PreferredRasterLayerSelector(
this.mapProperties.rasterLayer,
this.availableLayers,
this.featureSwitches.backgroundLayerId,
2024-10-19 14:44:55 +02:00
this.userRelatedState.preferredBackgroundLayer
2023-10-16 14:27:05 +02:00
)
}
2024-05-08 21:46:33 +02:00
2024-06-16 16:06:26 +02:00
public selectCurrentView() {
2024-05-08 21:46:33 +02:00
this.guistate.closeAll()
this.selectedElement.setData(this.currentView.features?.data?.[0])
}
2024-08-23 02:16:24 +02:00
/**
* Searches the appropriate layer - will first try if a special layer matches; if not, a normal layer will be used by delegating to the theme
2024-08-23 02:16:24 +02:00
*/
2024-11-25 23:44:26 +01:00
public getMatchingLayer(properties: Record<string, string>): LayerConfig | undefined {
2024-08-23 02:16:24 +02:00
const id = properties.id
if (id.startsWith("summary_")) {
// We don't select 'summary'-objects
return undefined
}
if (id === "settings") {
return UserRelatedState.usersettingsConfig
}
if (id.startsWith(LastClickFeatureSource.newPointElementId)) {
return this.theme.layers.find((l) => l.id === "last_click")
2024-08-23 02:16:24 +02:00
}
if (id.startsWith("search_result")) {
return GeocodingUtils.searchLayer
}
if (id === "location_track") {
return this.theme.layers.find((l) => l.id === "gps_track")
2024-08-23 02:16:24 +02:00
}
return this.theme.getMatchingLayer(properties)
2024-08-23 02:16:24 +02:00
}
public async reportError(message: string | Error | XMLHttpRequest, extramessage: string = "") {
if (Utils.runningFromConsole) {
2024-09-10 14:44:25 +02:00
console.error("Got (in themeViewSTate.reportError):", message, extramessage)
return
}
2024-10-13 13:19:54 +02:00
const isTesting = this.featureSwitchIsTesting?.data
2024-07-21 10:52:51 +02:00
console.log(
isTesting
? ">>> _Not_ reporting error to report server as testmode is on"
: ">>> Reporting error to",
Constants.ErrorReportServer,
2024-10-19 14:44:55 +02:00
message
2024-07-21 10:52:51 +02:00
)
if (isTesting) {
return
}
if ("" + message === "[object XMLHttpRequest]") {
const req = <XMLHttpRequest>message
let body = ""
try {
body = req.responseText
} catch (e) {
// pass
}
2024-09-02 12:48:15 +02:00
message =
"XMLHttpRequest with status code " +
req.status +
", " +
req.statusText +
", received: " +
body
}
if (extramessage) {
2024-10-10 23:02:12 +02:00
message += " (" + extramessage + ")"
}
const stacktrace: string = new Error().stack
try {
await fetch(Constants.ErrorReportServer, {
method: "POST",
body: JSON.stringify({
stacktrace,
message: "" + message,
theme: this.theme.id,
version: Constants.vNumber,
language: this.userRelatedState.language.data,
username: this.osmConnection.userDetails.data?.name,
userid: this.osmConnection.userDetails.data?.uid,
pendingChanges: this.changes.pendingChanges.data,
previousChanges: this.changes.allChanges.data,
changeRewrites: Utils.MapToObj(this.changes._changesetHandler._remappings),
}),
})
} catch (e) {
console.error("Could not upload an error report")
}
}
2023-03-28 05:13:48 +02:00
}