我的可视化中有一个 JavaFX BubbleChart,我需要能够在图表的每个气泡中创建/显示文本。在我的可视化中,我有许多 XYChart.Series,每个系列只有 1 个气泡。对于每个系列,我都会执行“series.setName("xxxx");” (其中 xxxx = 唯一的系列名称)并且我需要能够在气泡内显示该系列名称。
我已经为显示系列名称的气泡图实现了工具提示(鼠标悬停事件),但我还需要在气泡内显示文本,而不需要将鼠标悬停在上面。
为了有代码可以使用,这里有一个 5 系列的基本示例。我该如何在每个气泡中添加文本?
谢谢。
public class bubbleChartTest extends Application {
@Override
public void start(Stage stage) {
final NumberAxis xAxis = new NumberAxis(0, 10, 1);
final NumberAxis yAxis = new NumberAxis(0, 10, 1);
final BubbleChart<Number, Number> bc = new BubbleChart<Number, Number>(xAxis, yAxis);
xAxis.setLabel("X Axis");
xAxis.setMinorTickCount(2);
yAxis.setLabel("Y Axis");
yAxis.setTickLabelGap(2);
bc.setTitle("Bubble Chart StackOverflow Example");
XYChart.Series<Number, Number> series1 = new XYChart.Series<Number, Number>();
series1.setName("Series 1");
series1.getData().add(new XYChart.Data<Number, Number>(3, 7, 1.5));
XYChart.Series<Number, Number> series2 = new XYChart.Series<Number, Number>();
series2.setName("Series 2");
series2.getData().add(new XYChart.Data<Number, Number>(8, 3, 1));
XYChart.Series<Number, Number> series3 = new XYChart.Series<Number, Number>();
series3.setName("Series 3");
series3.getData().add(new XYChart.Data<Number, Number>(1, 9, 2));
XYChart.Series<Number, Number> series4 = new XYChart.Series<Number, Number>();
series4.setName("Series 4");
series4.getData().add(new XYChart.Data<Number, Number>(4, 1, 0.5));
XYChart.Series<Number, Number> series5 = new XYChart.Series<Number, Number>();
series5.setName("Series 5");
series5.getData().add(new XYChart.Data<Number, Number>(9, 9, 3));
Scene scene = new Scene(bc);
bc.getData().addAll(series1, series2, series3, series4, series5);
stage.setScene(scene);
stage.show();
for(XYChart.Series<Number, Number> series : bc.getData()) {
for(XYChart.Data<Number, Number> data : series.getData()) {
Tooltip.install(data.getNode(), new Tooltip(series.getName()));
}
}
}
public static void main(String[] args) {
launch(args);
}
}
最佳答案
从数据中得到的节点是一个Stackpane,Stackpane的形状是一个椭圆形。您可能需要该椭圆在 x 方向上的半径并向堆栈 Pane 添加标签。但需要设置Label的minWidth属性,否则只会显示三个点。并且您需要一个属性来保存动态字体大小,因为如果您想调整图表大小,它应该看起来很漂亮。
您不需要太多代码即可完成这项工作:
for (XYChart.Series<Number, Number> series : bc.getData()) {
for (XYChart.Data<Number, Number> data : series.getData()) {
Node bubble = data.getNode();
if (bubble != null && bubble instanceof StackPane) {
StackPane region = (StackPane) bubble;
if (region.getShape() != null && region.getShape() instanceof Ellipse) {
Ellipse ellipse = (Ellipse) region.getShape();
DoubleProperty fontSize = new SimpleDoubleProperty(10);
Label label = new Label(series.getName());
label.setAlignment(Pos.CENTER);
label.minWidthProperty().bind(ellipse.radiusXProperty());
//fontSize.bind(Bindings.when(ellipse.radiusXProperty().lessThan(40)).then(6).otherwise(10));
fontSize.bind(Bindings.divide(ellipse.radiusXProperty(), 5));
label.styleProperty().bind(Bindings.concat("-fx-font-size:", fontSize.asString(), ";"));
region.getChildren().add(label);
}
}
}
}
更新
James_D 提到,在更改形状(例如形状)的情况下,循环不太稳健。所以我对其进行了一些更改以请求椭圆实例。这有点像 BubbleChart 中的原始布局PlotChildren 方法。
关于java - 在 JavaFX BubbleChart 的气泡内创建文本,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32742368/