MapComplete/UI/NewPoint/ConfirmLocationOfPoint.ts

209 lines
8.5 KiB
TypeScript
Raw Normal View History

2022-11-08 14:37:48 +01:00
import { UIEventSource } from "../../Logic/UIEventSource"
2022-09-08 21:40:48 +02:00
import BaseUIElement from "../BaseUIElement"
import LocationInput from "../Input/LocationInput"
2022-11-08 14:37:48 +01:00
import { BBox } from "../../Logic/BBox"
import { TagUtils } from "../../Logic/Tags/TagUtils"
import { SubtleButton } from "../Base/SubtleButton"
2022-09-08 21:40:48 +02:00
import Combine from "../Base/Combine"
import Translations from "../i18n/Translations"
import Svg from "../../Svg"
import Toggle from "../Input/Toggle"
import { PresetInfo } from "../BigComponents/SimpleAddUI"
2022-09-08 21:40:48 +02:00
import Title from "../Base/Title"
2022-11-08 14:37:48 +01:00
import { VariableUiElement } from "../Base/VariableUIElement"
import { Tag } from "../../Logic/Tags/Tag"
import { WayId } from "../../Models/OsmFeature"
2022-11-14 01:29:22 +01:00
import { Translation } from "../i18n/Translation"
2023-03-24 19:21:15 +01:00
import { Feature } from "geojson"
2023-03-28 05:13:48 +02:00
import { AvailableRasterLayers } from "../../Models/RasterLayers"
import { SpecialVisualizationState } from "../SpecialVisualization"
import ClippedFeatureSource from "../../Logic/FeatureSource/Sources/ClippedFeatureSource"
export default class ConfirmLocationOfPoint extends Combine {
constructor(
2023-03-28 05:13:48 +02:00
state: SpecialVisualizationState,
filterViewIsOpened: UIEventSource<boolean>,
preset: PresetInfo,
confirmText: BaseUIElement,
2022-09-08 21:40:48 +02:00
loc: { lon: number; lat: number },
confirm: (
tags: any[],
location: { lat: number; lon: number },
2022-09-21 02:21:20 +02:00
snapOntoWayId: WayId | undefined
2022-09-08 21:40:48 +02:00
) => void,
cancel: () => void,
closePopup: () => void,
options?: {
2022-11-14 01:29:22 +01:00
cancelIcon: BaseUIElement
cancelText?: string | Translation
}
) {
let preciseInput: LocationInput = undefined
if (preset.preciseInput !== undefined) {
// Create location input
// We uncouple the event source
2022-11-08 14:37:48 +01:00
const zloc = { ...loc, zoom: 19 }
2022-09-08 21:40:48 +02:00
const locationSrc = new UIEventSource(zloc)
2022-09-08 21:40:48 +02:00
let backgroundLayer = new UIEventSource(
2023-03-28 05:13:48 +02:00
state?.mapProperties.rasterLayer?.data ?? AvailableRasterLayers.osmCarto
2022-09-08 21:40:48 +02:00
)
if (preset.preciseInput.preferredBackground) {
const defaultBackground = AvailableRasterLayers.SelectBestLayerAccordingTo(
2022-09-08 21:40:48 +02:00
locationSrc,
new UIEventSource<string | string[]>(preset.preciseInput.preferredBackground)
)
// Note that we _break the link_ here, as the minimap will take care of the switching!
backgroundLayer.setData(defaultBackground.data)
}
let snapToFeatures: UIEventSource<Feature[]> = undefined
let mapBounds: UIEventSource<BBox> = undefined
if (preset.preciseInput.snapToLayers && preset.preciseInput.snapToLayers.length > 0) {
2023-03-24 19:21:15 +01:00
snapToFeatures = new UIEventSource<Feature[]>([])
mapBounds = new UIEventSource<BBox>(undefined)
}
2022-09-08 21:40:48 +02:00
const tags = TagUtils.KVtoProperties(preset.tags ?? [])
preciseInput = new LocationInput({
mapBackground: backgroundLayer,
centerLocation: locationSrc,
snapTo: snapToFeatures,
renderLayerForSnappedPoint: preset.layerToAddTo.layerDef,
snappedPointTags: tags,
maxSnapDistance: preset.preciseInput.maxSnapDistance,
2022-09-08 21:40:48 +02:00
bounds: mapBounds,
2022-10-27 01:50:41 +02:00
state: <any>state,
})
2022-01-25 21:55:51 +01:00
preciseInput.installBounds(preset.boundsFactor ?? 0.25, true)
2022-09-08 21:40:48 +02:00
preciseInput
.SetClass("rounded-xl overflow-hidden border border-gray")
.SetStyle("height: 18rem; max-height: 50vh")
if (preset.preciseInput.snapToLayers && preset.preciseInput.snapToLayers.length > 0) {
// We have to snap to certain layers.
// Lets fetch them
let loadedBbox: BBox = undefined
2022-09-08 21:40:48 +02:00
mapBounds?.addCallbackAndRunD((bbox) => {
if (loadedBbox !== undefined && bbox.isContainedIn(loadedBbox)) {
// All is already there
// return;
}
2022-09-08 21:40:48 +02:00
bbox = bbox.pad(
Math.max(preset.boundsFactor ?? 0.25, 2),
Math.max(preset.boundsFactor ?? 0.25, 2)
)
loadedBbox = bbox
2023-03-28 05:13:48 +02:00
const sources = preset.preciseInput.snapToLayers.map(
(layerId) =>
new ClippedFeatureSource(
state.perLayer.get(layerId),
bbox.asGeoJson({})
)
)
})
}
}
2022-09-08 21:40:48 +02:00
let confirmButton: BaseUIElement = new SubtleButton(
preset.icon(),
2022-11-08 14:37:48 +01:00
new Combine([confirmText]).SetClass("flex flex-col")
2022-09-08 21:40:48 +02:00
)
.SetClass("font-bold break-words")
.onClick(() => {
2022-09-08 21:40:48 +02:00
const globalFilterTagsToAdd: Tag[][] = state.globalFilters.data
.filter((gf) => gf.onNewPoint !== undefined)
.map((gf) => gf.onNewPoint.tags)
const globalTags: Tag[] = [].concat(...globalFilterTagsToAdd)
console.log("Global tags to add are: ", globalTags)
2022-09-08 21:40:48 +02:00
confirm(
[...preset.tags, ...globalTags],
preciseInput?.GetValue()?.data ?? loc,
preciseInput?.snappedOnto?.data?.properties?.id
)
})
if (preciseInput !== undefined) {
confirmButton = new Combine([preciseInput, confirmButton])
} else {
confirmButton = new Combine([confirmButton])
}
let openLayerOrConfirm = confirmButton
const disableFilter = new SubtleButton(
new Combine([
Svg.filter_ui().SetClass("absolute w-full"),
2022-09-08 21:40:48 +02:00
Svg.cross_bottom_right_svg().SetClass("absolute red-svg"),
]).SetClass("relative"),
2022-09-08 21:40:48 +02:00
new Combine([
Translations.t.general.add.disableFiltersExplanation.Clone(),
Translations.t.general.add.disableFilters.Clone().SetClass("text-xl"),
]).SetClass("flex flex-col")
).onClick(() => {
2022-09-08 21:40:48 +02:00
const appliedFilters = preset.layerToAddTo.appliedFilters
2022-01-08 04:22:50 +01:00
appliedFilters.data.forEach((_, k) => appliedFilters.data.set(k, undefined))
appliedFilters.ping()
cancel()
2022-01-08 04:22:50 +01:00
closePopup()
})
// We assume the number of global filters won't change during the run of the program
for (let i = 0; i < state.globalFilters.data.length; i++) {
2022-09-08 21:40:48 +02:00
const hasBeenCheckedOf = new UIEventSource(false)
const filterConfirmPanel = new VariableUiElement(
2022-09-08 21:40:48 +02:00
state.globalFilters.map((gfs) => {
const gf = gfs[i]
2022-11-08 14:37:48 +01:00
const confirm = gf.onNewPoint?.confirmAddNew?.Subs({ preset: preset.title })
2022-09-08 21:40:48 +02:00
return new Combine([
gf.onNewPoint?.safetyCheck,
new SubtleButton(Svg.confirm_svg(), confirm).onClick(() =>
hasBeenCheckedOf.setData(true)
),
])
})
)
openLayerOrConfirm = new Toggle(
2022-09-08 21:40:48 +02:00
openLayerOrConfirm,
filterConfirmPanel,
state.globalFilters.map(
(f) => hasBeenCheckedOf.data || f[i]?.onNewPoint === undefined,
[hasBeenCheckedOf]
)
)
}
2022-01-08 04:22:50 +01:00
// If at least one filter is active which _might_ hide a newly added item, this blocks the preset and requests the filter to be disabled
const disableFiltersOrConfirm = new Toggle(openLayerOrConfirm, disableFilter)
2022-09-08 21:40:48 +02:00
const cancelButton = new SubtleButton(
options?.cancelIcon ?? Svg.close_ui(),
options?.cancelText ?? Translations.t.general.cancel
).onClick(cancel)
2022-09-08 21:40:48 +02:00
let examples: BaseUIElement = undefined
if (preset.exampleImages !== undefined && preset.exampleImages.length > 0) {
examples = new Combine([new Title()])
}
super([
new Toggle(
Translations.t.general.testing.SetClass("alert"),
undefined,
state.featureSwitchIsTesting
),
disableFiltersOrConfirm,
cancelButton,
preset.description,
examples,
])
this.SetClass("flex flex-col")
}
2022-09-08 21:40:48 +02:00
}