import ImportFlow, { ImportFlowArguments } from "./ImportFlow" import { SpecialVisualizationState } from "../../SpecialVisualization" import { Feature, LineString, Polygon } from "geojson" import { Store, UIEventSource } from "../../../Logic/UIEventSource" import { Tag } from "../../../Logic/Tags/Tag" import { And } from "../../../Logic/Tags/And" import CreateWayWithPointReuseAction, { MergePointConfig, } from "../../../Logic/Osm/Actions/CreateWayWithPointReuseAction" import { TagUtils } from "../../../Logic/Tags/TagUtils" import { OsmCreateAction, PreviewableAction } from "../../../Logic/Osm/Actions/OsmChangeAction" import { FeatureSource, IndexedFeatureSource } from "../../../Logic/FeatureSource/FeatureSource" import CreateMultiPolygonWithPointReuseAction from "../../../Logic/Osm/Actions/CreateMultiPolygonWithPointReuseAction" import LayoutConfig from "../../../Models/ThemeConfig/LayoutConfig" import { Changes } from "../../../Logic/Osm/Changes" import FullNodeDatabaseSource from "../../../Logic/FeatureSource/TiledFeatureSource/FullNodeDatabaseSource" export interface WayImportFlowArguments extends ImportFlowArguments { max_snap_distance: string snap_onto_layers: string snap_to_layer_max_distance: string max_move_distance: string move_osm_point_if snap_to_point_if } export default class WayImportFlowState extends ImportFlow { public readonly originalFeature: Feature private readonly action: OsmCreateAction & { getPreview?(): Promise } constructor( state: SpecialVisualizationState, originalFeature: Feature, args: WayImportFlowArguments, tagsToApply: Store, originalFeatureTags: UIEventSource> ) { super(state, args, tagsToApply, originalFeatureTags) this.originalFeature = originalFeature const mergeConfigs = WayImportFlowState.GetMergeConfig(args) this.action = WayImportFlowState.CreateAction( originalFeature, args, state, tagsToApply, mergeConfigs ) } public static CreateAction( feature: Feature, args: WayImportFlowArguments, state: { layout: LayoutConfig changes: Changes indexedFeatures: IndexedFeatureSource fullNodeDatabase?: FullNodeDatabaseSource }, tagsToApply: Store, mergeConfigs: MergePointConfig[] ): OsmCreateAction & PreviewableAction & { newElementId?: string } { if (feature.geometry.type === "Polygon" && feature.geometry.coordinates.length > 1) { const coors = (feature.geometry).coordinates const outer = coors[0] const inner = [...coors] inner.splice(0, 1) return new CreateMultiPolygonWithPointReuseAction( tagsToApply.data, outer, inner, state, mergeConfigs, "import" ) } else if (feature.geometry.type === "Polygon") { const coors = feature.geometry.coordinates const outer = coors[0] return new CreateWayWithPointReuseAction(tagsToApply.data, outer, state, mergeConfigs) } else if (feature.geometry.type === "LineString") { const coors = feature.geometry.coordinates return new CreateWayWithPointReuseAction(tagsToApply.data, coors, state, mergeConfigs) } else { throw "Unsupported type" } } public static GetMergeConfig(args: WayImportFlowArguments): MergePointConfig[] { const nodesMustMatch = args.snap_to_point_if ?.split(";") ?.map((tag, i) => TagUtils.Tag(tag, "TagsSpec for import button " + i)) const mergeConfigs = [] if (nodesMustMatch !== undefined && nodesMustMatch.length > 0) { const mergeConfig: MergePointConfig = { mode: "reuse_osm_point", ifMatches: new And(nodesMustMatch), withinRangeOfM: Number(args.max_snap_distance), } mergeConfigs.push(mergeConfig) } const moveOsmPointIfTags = args["move_osm_point_if"] ?.split(";") ?.map((tag, i) => TagUtils.Tag(tag, "TagsSpec for import button " + i)) if (nodesMustMatch !== undefined && moveOsmPointIfTags.length > 0) { const moveDistance = Math.min(20, Number(args["max_move_distance"])) const mergeConfig: MergePointConfig = { mode: "move_osm_point", ifMatches: new And(moveOsmPointIfTags), withinRangeOfM: moveDistance, } mergeConfigs.push(mergeConfig) } return mergeConfigs } // noinspection JSUnusedGlobalSymbols public async onConfirm() { const originalFeatureTags = this._originalFeatureTags originalFeatureTags.data["_imported"] = "yes" originalFeatureTags.ping() // will set isImported as per its definition const action = this.action await this.state.changes.applyAction(action) const newId = action.newElementId ?? action.mainObjectId this.state.selectedLayer.setData(this.targetLayer.layerDef) this.state.selectedElement.setData(this.state.indexedFeatures.featuresById.data.get(newId)) } public GetPreview(): undefined | Promise { if (!this.action?.getPreview) { return undefined } return this.action.getPreview() } }