【问题标题】:JavaFX Charting - CategoryAixis Resize IssueJavaFX 图表 - CategoryAxis 调整大小问题
【发布时间】:2015-10-06 20:21:30
【问题描述】:

我有一个用户可调整大小的图表。当图表变小时,CategoryAxis 会旋转,您将无法再看到大多数类别标签。这是一个显示问题的 gif:

有什么方法可以阻止标签旋转?

我知道我可以向旋转属性添加一个侦听器,如果旋转发生变化,我可以将其旋转回 0。但是,当我这样做时,它不会阻止间距调整,因此标签会被切断(您只会看到标签的最后几个字符)。

这里是包含的 gif 的代码,你会在调整窗口大小时看到问题:

import javafx.application.Application;
import javafx.scene.Scene;
import javafx.scene.chart.BarChart;
import javafx.scene.chart.CategoryAxis;
import javafx.scene.chart.NumberAxis;
import javafx.scene.chart.XYChart;
import javafx.stage.Stage;

public class HorizontalBarExample extends Application {
    @Override
    public void start(Stage stage)  {
        NumberAxis xAxis = new NumberAxis();
        CategoryAxis yAxis = new CategoryAxis();
        BarChart<Number, String> bc = new BarChart<Number, String>(xAxis, yAxis);
        bc.setBarGap(0d);
        bc.setCategoryGap(0);

        xAxis.setTickLabelRotation(90);
        yAxis.tickLabelRotationProperty().set(0d);

        XYChart.Series<Number, String> series1 = new XYChart.Series<>();
        series1.setName("example");

        for (int i = 0; i < 10; i++)
            series1.getData().add(new XYChart.Data<Number, String>(Math.random() * 5000, "long data label number" + i));

        bc.getData().add(series1);

        Scene scene = new Scene(bc, 800, 600);
        stage.setScene(scene);
        stage.show();
    }

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

【问题讨论】:

  • 我想说,这是一个错误:如果空间变小以水平显示所有标签,无论类别轴应该做什么,旋转都不是可行的选项之一(每 n 次可能- 或不 - 是或不显示任何内容,直到鼠标悬停在类别上,然后仅显示该类别)

标签: charts javafx-8


【解决方案1】:

这个问题不容易解决 (IMO)。

发生了什么

Java 8u60 的 CategoryAxis 代码在内部维护一个私有 effectiveTickLabelRotation 成员。由于内部实现,此成员有时可以覆盖您为公开可用的tickLabelRotationProperty 设置的任何值。所以你对功能并没有太多的控制权。

各种(失败)尝试使用公共 API 修复它

它覆盖的一种方式是将类别轴设置为 autoRanging,因此您可以将 autoRanging 设置为 false,也可以使用 CategoryAxis::setCategory 方法手动设置类别。这种解决了这个问题,因为当图形变小时不使用有效旋转,它尊重您想要的旋转并且文本保持垂直。

但是,即使关闭自动测距并手动设置类别,实现中仍有其他怪癖妨碍了合理的结果。 CategoryAxis 的内部布局算法仍然认为类别标签已经旋转,因此当图表变小时它没有为标签分配足够的空间。 CategoryAxis 类是最终类,因此布局逻辑不能在子类中被覆盖。快速破解将设置轴的最小宽度yAxis.setMinWidth(200);。不幸的是,CategoryAxis 布局算法也不遵守最小宽度设置。

获得此功能的选项

简而言之,它有点坏了……你要么:

  1. 按原样接受默认行为,或者
  2. log a bug 请求修复,或者
  3. 您将 CategoryAxis 代码复制到一个新类,例如Horizo​​ntallyLabelledCategoryAxis,进行一些修改以允许布局算法按您的意愿工作,并使用该新类代替原始 CategoryAxis。

选项 3 有点棘手,但可行。

建议的解决方法

综上所述,如果您的 UI 可以接受,我建议的方法(例如解决方法)就是不要让图表变得足够小以至于其布局搞砸。

VBox chartHolder = new VBox(bc);
VBox.setVgrow(bc, Priority.ALWAYS);
bc.setMinHeight(300);
Scene scene = new Scene(chartHolder, 800, 600);

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-02-22
    • 2012-05-08
    • 2014-11-05
    • 2017-05-14
    • 1970-01-01
    相关资源
    最近更新 更多