JavaFX 2.x: Translate mouse coordinate into XYChart axis value

In JavaFX 2.x, I use XYChart, and I want to display the coordinate values โ€‹โ€‹of the axis (X, Y) of the chart when the mouse moves around the chart. I am setting up an event handler in a diagram to handle setOnMouseMoved events. However, I'm not sure how to convert the MouseEvent getX () value to the chart coordinate value?

+6
source share
2 answers

Use axis.getValueForDisplay (displayPosition) to determine the location of the mouse in the coordinates of the axis values:

xAxis.getValueForDisplay(mouseEvent.getX()), yAxis.getValueForDisplay(mouseEvent.getY()) 

Here is an example that reports the coordinates at which the mouse hangs in a line chart. Screen capture does not capture the mouse cursor - you just need to present it there; -)

hoverreporter

 import javafx.application.Application; import javafx.collections.FXCollections; import javafx.event.EventHandler; import javafx.geometry.Insets; import javafx.geometry.Pos; import javafx.scene.*; import javafx.scene.chart.*; import javafx.scene.control.Label; import javafx.scene.input.MouseEvent; import javafx.scene.layout.VBox; import javafx.stage.Stage; public class LineChartWithHoverCoords extends Application { @Override public void start(Stage stage) { stage.setTitle("Line Chart Sample"); final LineChart<Number, Number> lineChart = createChart(); Label cursorCoords = createCursorGraphCoordsMonitorLabel(lineChart); stage.setScene( new Scene( layoutScene( lineChart, cursorCoords ) ) ); stage.show(); } private VBox layoutScene(LineChart<Number, Number> lineChart, Label cursorCoords) { VBox layout = new VBox(10); layout.setPadding(new Insets(10)); layout.setAlignment(Pos.CENTER); layout.getChildren().setAll( cursorCoords, lineChart ); return layout; } private LineChart<Number, Number> createChart() { final NumberAxis xAxis = new NumberAxis(); final NumberAxis yAxis = new NumberAxis(); xAxis.setLabel("Number of Month"); final LineChart<Number,Number> lineChart = new LineChart<>(xAxis,yAxis); lineChart.setTitle("Stock Monitoring, 2010"); XYChart.Series<Number, Number> series = new XYChart.Series<>( "My portfolio", FXCollections.<XYChart.Data<Number, Number>>observableArrayList( new XYChart.Data<Number, Number>(1, 23), new XYChart.Data<Number, Number>(2, 14), new XYChart.Data<Number, Number>(3, 15), new XYChart.Data<Number, Number>(4, 24), new XYChart.Data<Number, Number>(5, 34), new XYChart.Data<Number, Number>(6, 36), new XYChart.Data<Number, Number>(7, 22), new XYChart.Data<Number, Number>(8, 45), new XYChart.Data<Number, Number>(9, 43), new XYChart.Data<Number, Number>(10, 17), new XYChart.Data<Number, Number>(11, 29), new XYChart.Data<Number, Number>(12, 25) ) ); lineChart.getData().add(series); return lineChart; } private Label createCursorGraphCoordsMonitorLabel(LineChart<Number, Number> lineChart) { final Axis<Number> xAxis = lineChart.getXAxis(); final Axis<Number> yAxis = lineChart.getYAxis(); final Label cursorCoords = new Label(); final Node chartBackground = lineChart.lookup(".chart-plot-background"); for (Node n: chartBackground.getParent().getChildrenUnmodifiable()) { if (n != chartBackground && n != xAxis && n != yAxis) { n.setMouseTransparent(true); } } chartBackground.setOnMouseEntered(new EventHandler<MouseEvent>() { @Override public void handle(MouseEvent mouseEvent) { cursorCoords.setVisible(true); } }); chartBackground.setOnMouseMoved(new EventHandler<MouseEvent>() { @Override public void handle(MouseEvent mouseEvent) { cursorCoords.setText( String.format( "(%.2f, %.2f)", xAxis.getValueForDisplay(mouseEvent.getX()), yAxis.getValueForDisplay(mouseEvent.getY()) ) ); } }); chartBackground.setOnMouseExited(new EventHandler<MouseEvent>() { @Override public void handle(MouseEvent mouseEvent) { cursorCoords.setVisible(false); } }); xAxis.setOnMouseEntered(new EventHandler<MouseEvent>() { @Override public void handle(MouseEvent mouseEvent) { cursorCoords.setVisible(true); } }); xAxis.setOnMouseMoved(new EventHandler<MouseEvent>() { @Override public void handle(MouseEvent mouseEvent) { cursorCoords.setText( String.format( "x = %.2f", xAxis.getValueForDisplay(mouseEvent.getX()) ) ); } }); xAxis.setOnMouseExited(new EventHandler<MouseEvent>() { @Override public void handle(MouseEvent mouseEvent) { cursorCoords.setVisible(false); } }); yAxis.setOnMouseEntered(new EventHandler<MouseEvent>() { @Override public void handle(MouseEvent mouseEvent) { cursorCoords.setVisible(true); } }); yAxis.setOnMouseMoved(new EventHandler<MouseEvent>() { @Override public void handle(MouseEvent mouseEvent) { cursorCoords.setText( String.format( "y = %.2f", yAxis.getValueForDisplay(mouseEvent.getY()) ) ); } }); yAxis.setOnMouseExited(new EventHandler<MouseEvent>() { @Override public void handle(MouseEvent mouseEvent) { cursorCoords.setVisible(false); } }); return cursorCoords; } public static void main(String[] args) { launch(args); } } 
+15
source

I wanted to write a comment, but I do not know how to do it. This is an additional comment, so the next guy who runs into the problem doesn't spend hours figuring out the reasons. The answer is correct, but be careful when using ImageCursor, it gives the wrong coordinates. After deactivating the scaling, scrolling, and dates and all my settings, I eventually found that using a custom cursor would ruin the coordinates of "getValueForDisplay".

+2
source

Source: https://habr.com/ru/post/944669/


All Articles