JavaFX8:在折线图上添加标签并根据缩放

时间:2015-04-22 14:15:32

标签: javafx javafx-8 linechart

我是一名新的JavaFX用户,我在实习项目中遇到了一些问题。我必须制作一个GUI,绘制一个光谱并在每个峰的顶部添加标签(x值)。用户可以放大图表,因此必须更新标签。

我使用第一种方法根据这篇文章添加标签:write text on a chart但是我有一些麻烦,只有一半的标签是可见的(见图片)并且缩放后标签位置的更新不好。我检查了每个标签的价值和位置,它们很好,如果我在我的图表上应用了一个动作,标签的后半部分出现了,所以我不知道这个误解的原因。我的更新(标签位置)功能是用于缩放的上行yaxis的监听器,但是为了获得良好的位置我需要手动使用我的更新按钮,如果有人有想法,我找不到解决方案? half labels

我目前使用的是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;
&#13;
&#13;

1 个答案:

答案 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()运行时它获得了新值。但是,假设我们有一台慢速机器,脉冲可能需要60msupdate()可以获得正确的值。然后,您需要将50ms更改为70ms。最好在pulse上找一个钩子,看看它何时结束,然后运行update()。不幸的是,我不知道是否存在这样的钩子。