我是一名新的JavaFX用户,我在实习项目中遇到了一些问题。我必须制作一个GUI,绘制一个光谱并在每个峰的顶部添加标签(x值)。用户可以放大图表,因此必须更新标签。
我使用第一种方法根据这篇文章添加标签:write text on a chart但是我有一些麻烦,只有一半的标签是可见的(见图片)并且缩放后标签位置的更新不好。我检查了每个标签的价值和位置,它们很好,如果我在我的图表上应用了一个动作,标签的后半部分出现了,所以我不知道这个误解的原因。我的更新(标签位置)功能是用于缩放的上行yaxis的监听器,但是为了获得良好的位置我需要手动使用我的更新按钮,如果有人有想法,我找不到解决方案?
我目前使用的是anchorpane,使用stackpane会更好吗?
我放了一部分代码:
public class PaneController implements Initializable{
@FXML
private LineChart<Number, Number> lineChart;
private ObservableList<Annotation> annotationNodes = FXCollections.observableArrayList();
@Override
public void initialize(URL location, ResourceBundle resources) {
// TODO Auto-generated method stub
lineChart.setLegendVisible(false);
lineChart.setCreateSymbols(false);
lineChart.setData(createData());
//create labels
createLabels(lineChart.getData());
//frame listener
((AnchorPane)lineChart.getParent()).heightProperty().addListener((num)-> update());
//axis listener
((NumberAxis)lineChart.getYAxis()).upperBoundProperty().addListener((num)-> update());
//update listener on middle button
lineChart.getParent().addEventHandler(MouseEvent.MOUSE_CLICKED, new EventHandler<MouseEvent>() {
@Override
public void handle(MouseEvent event) {
if( event.getButton() == MouseButton.MIDDLE){
update();
}
}
});
MyJFXChartUtil.setupZooming( lineChart );
}
/** create data for the line chart */
public ObservableList<Series<Number,Number>> createData(){
ObservableList<Series<Number,Number>> data = FXCollections.observableArrayList();
for(int i=1;i<=10;i++){
XYChart.Series<Number, Number> peakSelect = new XYChart.Series<>();
peakSelect.getData().add(new XYChart.Data<Number, Number>(i*2,0));
peakSelect.getData().add(new XYChart.Data<Number, Number>(i*2,i));
data.add(peakSelect);
}
return data;
}
/** place a x-value label on each peak */
private void createLabels(ObservableList<Series<Number,Number>> data){
Pane pane = (Pane) lineChart.getParent();
//clear old annotations
pane.getChildren().clear();
pane.getChildren().add(lineChart);
for(int i=0;i<lineChart.getData().size();i++){
for(XYChart.Data<Number, Number> value:lineChart.getData().get(i).getData()){
if(value.getYValue().intValue()>0){
//massAnnot
Annotation massAnnot = new Annotation(new Label(""+value.getXValue()), value.getXValue().doubleValue(), value.getYValue().doubleValue());
annotationNodes.add(massAnnot);
//labelAnnot
Annotation labelAnnot = new Annotation(new Label(""+(i+1)), value.getXValue().doubleValue(), 11);
annotationNodes.add(labelAnnot);
}
}
}
//add node to parent
for (Annotation annot : annotationNodes) {
pane.getChildren().add(annot.getLabel());
}
}
/** update position of labels */
private void update(){
//clear nodes and add the linechart as children
NumberAxis xAxis = (NumberAxis) lineChart.getXAxis();
NumberAxis yAxis = (NumberAxis) lineChart.getYAxis();
System.out.println(xAxis.getUpperBound()+" update "+yAxis.getUpperBound());
Pane pane = (Pane) lineChart.getParent();
pane.getChildren().clear();
pane.getChildren().add(lineChart);
int i = 0;
for (Annotation annot : annotationNodes) {
Node node = annot.getLabel();
if(i%2==0){//massAnnot
double x = lineChart.getXAxis().localToParent(xAxis.getDisplayPosition(annot.getX()), 0).getX() + lineChart.getPadding().getLeft();
double y = yAxis.localToParent(0,yAxis.getDisplayPosition(annot.getY())).getY() + lineChart.getPadding().getTop();
node.setLayoutX(x);
node.setLayoutY(y - node.prefHeight(Integer.MAX_VALUE));
}
else{//labelAnnot
double x = lineChart.getXAxis().localToParent(xAxis.getDisplayPosition(annot.getX()), 0).getX() + lineChart.getPadding().getLeft();
double y = yAxis.localToParent(0,yAxis.getDisplayPosition(annot.getY())).getY() + lineChart.getPadding().getTop();
node.setLayoutX(x);
node.setLayoutY(y - node.prefHeight(Integer.MAX_VALUE));
}
node.autosize();
pane.getChildren().add(node);
}
}
class Annotation{
private Label label;
private double x;
private double y;
public Annotation(Label label, double x, double y) {
this.label = label;
this.x = x;
this.y = y;
}
public Label getLabel(){
return this.label;
}
public double getX(){
return this.x;
}
public double getY(){
return this.y;
}
}
}
&#13;
public class Main extends Application {
@Override
public void start(Stage primaryStage) {
try {
FXMLLoader loader = new FXMLLoader();
loader.setLocation(Main.class.getResource("Pane.fxml"));
AnchorPane rootPane = (AnchorPane) loader.load();
Scene scene = new Scene(rootPane);
primaryStage.setScene(scene);
primaryStage.setTitle("Test");
PaneController controller = loader.getController();
primaryStage.show();
} catch(Exception e) {
e.printStackTrace();
}
}
public static void main(String[] args) {
launch(args);
}
}
&#13;
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.chart.*?>
<?import javafx.scene.control.*?>
<?import java.lang.*?>
<?import javafx.scene.layout.*?>
<AnchorPane xmlns="http://javafx.com/javafx/8.0.40" xmlns:fx="http://javafx.com/fxml/1" fx:controller="application.PaneController">
<children>
<LineChart fx:id="lineChart" prefHeight="400.0" prefWidth="500.0" AnchorPane.bottomAnchor="0.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" AnchorPane.topAnchor="0.0">
<xAxis>
<NumberAxis side="BOTTOM" />
</xAxis>
<yAxis>
<NumberAxis side="LEFT" />
</yAxis>
</LineChart>
</children>
</AnchorPane>
&#13;
答案 0 :(得分:2)
似乎更新被调用为快速且轴未更新,因此当您更新标签位置时,没有更新任何布局值。您可以将代码更改为:
private void update() {
runIn(50, () -> {
/*your current code*/
}
}
private void runIn(int ms, Runnable callback) {
new Thread(() -> {
try {
Thread.sleep(ms);
} catch (InterruptedException e) { e.printStackTrace(); }
Platform.runLater(callback);
}).start();
}
这是有效的,因为我想最长的Pulse
(一个屏幕布局和渲染)需要多长时间。您可以通过将-Djavafx.pulseLogger=true
标志添加到JVM args来查看脉冲的持续时间。每个脉冲都会显示在控制台上。您正在寻找:
...
PULSE: 49 [15ms:29ms]
...
29ms
是布局和渲染场景所需的时间,因为50ms
比较长,当update()
运行时它获得了新值。但是,假设我们有一台慢速机器,脉冲可能需要60ms
,update()
可以获得正确的值。然后,您需要将50ms
更改为70ms
。最好在pulse
上找一个钩子,看看它何时结束,然后运行update()
。不幸的是,我不知道是否存在这样的钩子。