forked from MapComplete/MapComplete
162 lines
5.9 KiB
TypeScript
162 lines
5.9 KiB
TypeScript
import Combine from "../Base/Combine"
|
|
import { Store, UIEventSource } from "../../Logic/UIEventSource"
|
|
import { BBox } from "../../Logic/BBox"
|
|
import UserRelatedState from "../../Logic/State/UserRelatedState"
|
|
import Translations from "../i18n/Translations"
|
|
import { AllKnownLayouts } from "../../Customizations/AllKnownLayouts"
|
|
import { DropDown } from "../Input/DropDown"
|
|
import { Utils } from "../../Utils"
|
|
import LayerConfig from "../../Models/ThemeConfig/LayerConfig"
|
|
import Loc from "../../Models/Loc"
|
|
import StaticFeatureSource from "../../Logic/FeatureSource/Sources/StaticFeatureSource"
|
|
import Toggle from "../Input/Toggle"
|
|
import { VariableUiElement } from "../Base/VariableUIElement"
|
|
import { FlowStep } from "./FlowStep"
|
|
import Title from "../Base/Title"
|
|
import CheckBoxes from "../Input/Checkboxes"
|
|
import { Feature, Point } from "geojson"
|
|
import DivContainer from "../Base/DivContainer"
|
|
import { AvailableRasterLayers, RasterLayerPolygon } from "../../Models/RasterLayers"
|
|
import { MapLibreAdaptor } from "../Map/MapLibreAdaptor"
|
|
import ShowDataLayer from "../Map/ShowDataLayer"
|
|
|
|
/**
|
|
* Shows the data to import on a map, asks for the correct layer to be selected
|
|
*/
|
|
export class MapPreview
|
|
extends Combine
|
|
implements FlowStep<{ bbox: BBox; layer: LayerConfig; features: Feature<Point>[] }>
|
|
{
|
|
public readonly IsValid: Store<boolean>
|
|
public readonly Value: Store<{ bbox: BBox; layer: LayerConfig; features: any[] }>
|
|
|
|
constructor(state: UserRelatedState, geojson: { features: Feature[] }) {
|
|
const t = Translations.t.importHelper.mapPreview
|
|
|
|
const propertyKeys = new Set<string>()
|
|
for (const f of geojson.features) {
|
|
Object.keys(f.properties).forEach((key) => propertyKeys.add(key))
|
|
}
|
|
|
|
const availableLayers = AllKnownLayouts.AllPublicLayers().filter(
|
|
(l) => l.name !== undefined && l.source !== undefined
|
|
)
|
|
const layerPicker = new DropDown(
|
|
t.selectLayer,
|
|
[{ shown: t.selectLayer, value: undefined }].concat(
|
|
availableLayers.map((l) => ({
|
|
shown: l.name,
|
|
value: l,
|
|
}))
|
|
)
|
|
)
|
|
|
|
let autodetected = new UIEventSource(false)
|
|
for (const layer of availableLayers) {
|
|
const mismatched = geojson.features.some(
|
|
(f) => !layer.source.osmTags.matchesProperties(f.properties)
|
|
)
|
|
if (!mismatched) {
|
|
console.log("Autodected layer", layer.id)
|
|
layerPicker.GetValue().setData(layer)
|
|
layerPicker.GetValue().addCallback((_) => autodetected.setData(false))
|
|
autodetected.setData(true)
|
|
break
|
|
}
|
|
}
|
|
|
|
const withId = geojson.features.map((f, i) => {
|
|
const copy = Utils.Clone(f)
|
|
copy.properties.id = "to-import/" + i
|
|
return copy
|
|
})
|
|
|
|
// Create a store which has only features matching the selected layer
|
|
const matching: Store<Feature[]> = layerPicker.GetValue().map((layer: LayerConfig) => {
|
|
if (layer === undefined) {
|
|
console.log("No matching layer found")
|
|
return []
|
|
}
|
|
const matching: Feature[] = []
|
|
|
|
for (const feature of withId) {
|
|
if (layer.source.osmTags.matchesProperties(feature.properties)) {
|
|
matching.push(feature)
|
|
}
|
|
}
|
|
console.log("Matching features: ", matching)
|
|
|
|
return matching
|
|
})
|
|
const background = new UIEventSource<RasterLayerPolygon>(AvailableRasterLayers.osmCarto)
|
|
const location = new UIEventSource<Loc>({ lat: 0, lon: 0, zoom: 1 })
|
|
const currentBounds = new UIEventSource<BBox>(undefined)
|
|
const { ui, mapproperties, map } = MapLibreAdaptor.construct()
|
|
|
|
ui.SetClass("w-full").SetStyle("height: 500px")
|
|
|
|
layerPicker.GetValue().addCallbackAndRunD((layerToShow) => {
|
|
new ShowDataLayer(map, {
|
|
layer: layerToShow,
|
|
zoomToFeatures: true,
|
|
features: new StaticFeatureSource(matching),
|
|
})
|
|
})
|
|
|
|
const bbox = matching.map((feats) =>
|
|
BBox.bboxAroundAll(
|
|
feats.map((f) => new BBox([(<Feature<Point>>f).geometry.coordinates]))
|
|
)
|
|
)
|
|
|
|
const mismatchIndicator = new VariableUiElement(
|
|
matching.map((matching) => {
|
|
if (matching === undefined) {
|
|
return undefined
|
|
}
|
|
const diff = geojson.features.length - matching.length
|
|
if (diff === 0) {
|
|
return undefined
|
|
}
|
|
const obligatory = layerPicker
|
|
.GetValue()
|
|
.data?.source?.osmTags?.asHumanString(false, false, {})
|
|
return t.mismatch.Subs({ count: diff, tags: obligatory }).SetClass("alert")
|
|
})
|
|
)
|
|
|
|
const confirm = new CheckBoxes([t.confirm])
|
|
super([
|
|
new Title(t.title, 1),
|
|
layerPicker,
|
|
new Toggle(t.autodetected.SetClass("thanks"), undefined, autodetected),
|
|
mismatchIndicator,
|
|
ui,
|
|
new DivContainer("fullscreen"),
|
|
confirm,
|
|
])
|
|
|
|
this.Value = bbox.map(
|
|
(bbox) => ({
|
|
bbox,
|
|
features: matching.data,
|
|
layer: layerPicker.GetValue().data,
|
|
}),
|
|
[layerPicker.GetValue(), matching]
|
|
)
|
|
|
|
this.IsValid = matching.map(
|
|
(matching) => {
|
|
if (matching === undefined) {
|
|
return false
|
|
}
|
|
if (confirm.GetValue().data.length !== 1) {
|
|
return false
|
|
}
|
|
const diff = geojson.features.length - matching.length
|
|
return diff === 0
|
|
},
|
|
[confirm.GetValue()]
|
|
)
|
|
}
|
|
}
|