java - 为什么 XYChart 中轴的父节点不是图表本身? (JavaFX)

标签 java javafx charts

我一直在尝试使用 XYCharts 做一些工作,特别是在它们上面或周围放置东西。我一直需要使用与整个图表定义的坐标空间相关的轴边界框的坐标。但是,我意识到 .getBoundsInParent() 方法在应用于轴时没有返回我所期望的结果。这似乎是因为轴父级不是图表,而且我不知道父级是什么。例如,以下 boolean 值是 false,这对我来说似乎是糟糕的设计(假设图表是 LineChart 的实例):

chart.getXAxis().getParent() == 图表

这是怎么回事?谢谢。

最佳答案

没有直接的 API 来获取图表绘图区域的边界。一种方法是使用查找,但这不是很可靠。

如果您希望在图表绘图区域中添加其他内容,建议的方法是重写图表的 layoutPlotChildren 方法,如 this question .

在图表区域之外添加内容的最佳方法可能是子类化 Region 并重写 layoutChildren 方法来布局图表和所需的额外内容。您可以通过使用 ValueAxis.getDisplayPosition(value) 获取相对于轴的坐标来确定特定“图表值”的坐标,然后使用一些转换来获取相对于区域的这些坐标.

这是一个简单的示例:

import java.util.List;
import java.util.Random;
import java.util.stream.Collectors;
import java.util.stream.Stream;

import javafx.application.Application;
import javafx.geometry.Point2D;
import javafx.scene.Scene;
import javafx.scene.chart.LineChart;
import javafx.scene.chart.NumberAxis;
import javafx.scene.chart.XYChart.Data;
import javafx.scene.chart.XYChart.Series;
import javafx.scene.control.Tooltip;
import javafx.scene.layout.BorderPane;
import javafx.scene.layout.Region;
import javafx.scene.paint.Color;
import javafx.scene.shape.Rectangle;
import javafx.stage.Stage;

public class LineChartWithRightPanel extends Application {

    private final Color[] RECT_COLORS = 
            new Color[] {Color.RED, Color.ORANGE, Color.YELLOW, Color.GREEN, Color.BLUE};
    private final double RECT_WIDTH = 20 ;

    @Override
    public void start(Stage primaryStage) {

        NumberAxis xAxis = new NumberAxis();
        NumberAxis yAxis = new NumberAxis(0, 1, 0.2);
        LineChart<Number, Number> chart = new LineChart<>(xAxis, yAxis);

        Region chartWithPanel = new Region() {

            private List<Rectangle> rectangles = 
                    Stream.of(RECT_COLORS)
                    .map(c -> new Rectangle(RECT_WIDTH, 0, c))
                    .collect(Collectors.toList());

            {
                getChildren().add(chart);
                getChildren().addAll(rectangles);
            }

            @Override
            protected void layoutChildren() {
                double w = getWidth();
                double h = getHeight();
                chart.resizeRelocate(0, 0, w-RECT_WIDTH, h);
                chart.layout();

                double chartMinY = yAxis.getLowerBound();
                double chartMaxY = yAxis.getUpperBound();
                double chartYRange = chartMaxY - chartMinY ;

                for (int i = 0 ; i < rectangles.size(); i++) {

                    // easier ways to do this in this example, 
                    // but this technique shows how to map "chart coords"
                    // to coords in this pane:

                    double rectMinYInChart = chartMinY + i * chartYRange / rectangles.size();
                    double rectMaxYInChart = chartMinY + (i+1) * chartYRange / rectangles.size();
                    double rectMinYInAxis = yAxis.getDisplayPosition(rectMinYInChart);
                    double rectMaxYInAxis = yAxis.getDisplayPosition(rectMaxYInChart);
                    double rectMinY = sceneToLocal(yAxis.localToScene(new Point2D(0, rectMinYInAxis))).getY();
                    double rectMaxY = sceneToLocal(yAxis.localToScene(new Point2D(0, rectMaxYInAxis))).getY();

                    rectangles.get(i).setHeight(Math.abs(rectMaxY - rectMinY));
                    rectangles.get(i).setX(w - RECT_WIDTH);
                    rectangles.get(i).setY(Math.min(rectMinY, rectMaxY));

                    Tooltip.install(rectangles.get(i), 
                            new Tooltip(String.format("%.2f - %.2f", rectMinYInChart, rectMaxYInChart)));
                }
            }
        };

        Series<Number, Number> series = new Series<>();
        series.setName("Data");
        Random rng = new Random();
        for (int i = 0; i < 20; i++) {
            series.getData().add(new Data<>(i, rng.nextDouble()));
        }
        chart.getData().add(series);

        BorderPane root = new BorderPane(chartWithPanel);
        Scene scene = new Scene(root);
        primaryStage.setScene(scene);
        primaryStage.show();
    }


    public static void main(String[] args) {
        launch(args);
    }
}

Screenshot of chart with colored bands on right

请注意,您可以使用相同的想法获取整个绘图区域的边界:

double minXInAxis = xAxis.getDisplayPosition(xAxis.getLowerBound());
double maxXInAxis = xAxis.getDisplayPosition(xAxis.getUpperBound());
double minYInAxis = yAxis.getDisplayPosition(yAxis.getLowerBound());
double maxYInAxis = yAxis.getDisplayPosition(yAxis.getUpperBound());

double minX = sceneToLocal(xAxis.localToScene(new Point2D(minXInAxis,0))).getX();
double maxX = sceneToLocal(xAxis.localToScene(new Point2D(maxXInAxis,0))).getX();
double minY = sceneToLocal(yAxis.localToScene(new Point2D(0,minYInAxis))).getY();
double maxY = sceneToLocal(yAxis.localToScene(new Point2D(0,maxYInAxis))).getY();

然后minXmaxXminYmaxY定义绘图区域的边界。

关于java - 为什么 XYChart 中轴的父节点不是图表本身? (JavaFX),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40590431/

相关文章:

java - JavaFX WebView 中的 Youtube

javascript - 如何从 ReCharts 折线图中删除点?

c# - 无法将数据表绑定(bind)到图表控件

java - 在重新添加之前立即从 HashSet 中删除元素的可能原因是什么?

java - 为什么我们可以使用 'this' 作为实例方法参数?

JAVA FX - 从输入中获取大写字母的数量

javascript - AmCharts - 缩放到选择捕捉到最近的点而不是选定的日期范围

java - 将 TextView 设置为导入的字体时强制关闭应用程序

java - Android:定期(在后台)打开 tcp 连接约 2 分钟的最有效方法

java - 是否可以在同一框架中打开新的 fxml 文件