MapComplete/UI/Popup/SplitRoadWizard.ts

127 lines
5.4 KiB
TypeScript
Raw Normal View History

2021-07-13 11:26:50 +02:00
import Toggle from "../Input/Toggle";
import Svg from "../../Svg";
import {UIEventSource} from "../../Logic/UIEventSource";
import {SubtleButton} from "../Base/SubtleButton";
import Minimap from "../Base/Minimap";
import State from "../../State";
2021-07-13 16:11:57 +02:00
import ShowDataLayer from "../ShowDataLayer";
import {GeoOperations} from "../../Logic/GeoOperations";
import {LeafletMouseEvent} from "leaflet";
import Combine from "../Base/Combine";
import {Button} from "../Base/Button";
import Translations from "../i18n/Translations";
import LayoutConfig from "../../Customizations/JSON/LayoutConfig";
import SplitAction from "../../Logic/Osm/SplitAction";
2021-07-13 11:26:50 +02:00
export default class SplitRoadWizard extends Toggle {
private static splitLayout = new UIEventSource(SplitRoadWizard.GetSplitLayout())
2021-07-13 11:26:50 +02:00
/**
* A UI Element used for splitting roads
*
* @param id: The id of the road to remove
*/
constructor(id: string) {
2021-07-13 16:11:57 +02:00
const t = Translations.t.split;
2021-07-13 11:26:50 +02:00
// Contains the points on the road that are selected to split on - contains geojson points with extra properties such as 'location' with the distance along the linestring
const splitPoints = new UIEventSource<{feature: any, freshness: Date}[]>([]);
2021-07-13 16:11:57 +02:00
// Toggle variable between show split button and map
const splitClicked = new UIEventSource<boolean>(false);
2021-07-13 16:11:57 +02:00
// Minimap on which you can select the points to be splitted
const miniMap = new Minimap({background: State.state.backgroundLayer});
miniMap.SetStyle("width: 100%; height: 50rem;");
// Define how a cut is displayed on the map
// Load the road with given id on the minimap
const roadElement = State.state.allElements.ContainingFeatures.get(id)
const splitAction = new SplitAction(roadElement)
2021-07-13 16:11:57 +02:00
const roadEventSource = new UIEventSource([{feature: roadElement, freshness: new Date()}]);
// Datalayer displaying the road and the cut points (if any)
new ShowDataLayer(roadEventSource, miniMap.leafletMap, State.state.layoutToUse, false, true);
new ShowDataLayer(splitPoints, miniMap.leafletMap, SplitRoadWizard.splitLayout, false, false)
2021-07-13 16:11:57 +02:00
/**
* Handles a click on the overleaf map.
* Finds the closest intersection with the road and adds a point there, ready to confirm the cut.
* @param coordinates Clicked location, [lon, lat]
*/
function onMapClick(coordinates) {
// Get nearest point on the road
const pointOnRoad = GeoOperations.nearestPoint(roadElement, coordinates); // pointOnRoad is a geojson
// Update point properties to let it match the layer
pointOnRoad.properties._cutposition = "yes";
pointOnRoad["_matching_layer_id"] = "splitpositions";
2021-07-13 16:11:57 +02:00
// let the state remember the point, to be able to retrieve it later by id
State.state.allElements.addOrGetElement(pointOnRoad);
// Add it to the list of all points and notify observers
splitPoints.data.push({feature: pointOnRoad, freshness: new Date()}); // show the point on the data layer
splitPoints.ping(); // not updated using .setData, so manually ping observers
2021-07-13 16:11:57 +02:00
}
2021-07-13 11:26:50 +02:00
2021-07-13 16:11:57 +02:00
// When clicked, pass clicked location coordinates to onMapClick function
miniMap.leafletMap.addCallbackAndRunD(
(leafletMap) => leafletMap.on("click", (mouseEvent: LeafletMouseEvent) => {
onMapClick([mouseEvent.latlng.lng, mouseEvent.latlng.lat])
}))
// Toggle between splitmap
2021-07-13 11:26:50 +02:00
const splitButton = new SubtleButton(Svg.scissors_ui(), "Split road");
splitButton.onClick(
() => {
splitClicked.setData(true)
}
)
2021-07-13 16:11:57 +02:00
// Only show the splitButton if logged in, else show login prompt
const splitToggle = new Toggle(
splitButton,
t.loginToSplit.Clone().onClick(State.state.osmConnection.AttemptLogin),
State.state.osmConnection.isLoggedIn)
2021-07-13 11:26:50 +02:00
2021-07-13 16:11:57 +02:00
// Save button
const saveButton = new Button("Split here", () => splitAction.DoSplit(splitPoints.data));
2021-07-13 16:11:57 +02:00
saveButton.SetClass("block btn btn-primary");
const disabledSaveButton = new Button("Split here", undefined);
disabledSaveButton.SetClass("block btn btn-disabled");
// Only show the save button if there are split points defined
const saveToggle = new Toggle(disabledSaveButton, saveButton, splitPoints.map((data) => data.length === 0))
2021-07-13 11:26:50 +02:00
2021-07-13 16:11:57 +02:00
const cancelButton = new Button("Cancel", () => {
splitClicked.setData(false);
splitPoints.setData([]);
2021-07-13 16:11:57 +02:00
});
2021-07-13 11:26:50 +02:00
2021-07-13 16:11:57 +02:00
cancelButton.SetClass("block btn btn-secondary");
2021-07-13 11:26:50 +02:00
2021-07-13 16:11:57 +02:00
const splitTitle = t.splitTitle;
2021-07-13 11:26:50 +02:00
2021-07-13 16:11:57 +02:00
const mapView = new Combine([splitTitle, miniMap, cancelButton, saveToggle]);
super(mapView, splitToggle, splitClicked);
2021-07-13 11:26:50 +02:00
}
private static GetSplitLayout(): LayoutConfig {
return new LayoutConfig({
maintainer: "mapcomplete",
language: [],
startLon: 0,
startLat: 0,
description: undefined,
icon: "", startZoom: 0,
title: "Split locations",
version: "",
id: "splitpositions",
layers: [{id: "splitpositions", source: {osmTags: "_cutposition=yes"}, icon: "./assets/svg/plus.svg"}]
}, true, "split road wizard layout")
}
2021-07-13 11:26:50 +02:00
}