簡體   English   中英

更新一個 JavaFX 狀態 label

[英]Updating a JavaFX status label

我有一個使用模型/視圖/控制器的 JavaFX 程序,我希望長時間運行 model 來更新視圖上的狀態 label。 我發現有人建議使用時間線class 來執行此操作。 我實現它時期望每一秒狀態 Label 都會更新。 但是,僅顯示最終狀態。 我究竟做錯了什么?

我的 controller 看起來像:

@FXML
private Button pullApplicantsButton;
@FXML
private Label statusLabel;
@FXML
private DatePicker orientationDate;
@FXML
private Spinner numberOfApplicants;

@FXML
private void pullApplicants() throws Exception {

    SelectApplicantsModel selectApplicantsModel = new SelectApplicantsModel(orientationDate.getValue() , ( int ) numberOfApplicants.getValue() , this.statusLabel);
    selectApplicantsModel.process();
}

我的 model 看起來像:

public SelectApplicantsModel(LocalDate nextOrientationDate, int numberOfApplicants , Label statusLabel ) throws FileNotFoundException {

    this.nextOrientationDate = nextOrientationDate;
    this.numberOfApplicants = numberOfApplicants;
    this.statusLabel = statusLabel;
}

public void process() throws Exception {

    Timeline timeline = new Timeline(
            new KeyFrame(Duration.seconds( 1 ) , event -> {
                statusLabel.setText( programStatus );
            })
    );
    timeline.setCycleCount( Animation.INDEFINITE );
    timeline.play();
    programStatus = "starting";
    changeSearchStringToIncludeOrientationDate(nextOrientationDate);
    MemberClicks memberClicks = new MemberClicks();
    programStatus = "retrieving profiles";
    JsonArray applicantProfilesJsonArray = memberClicks.getProfiles(searchJsonArray);
    programStatus = "converting profiles";

視圖看起來像:

  <Label text="Picks the next 30 applicants for the upcoming orientation.  Applicants whose Memberclick's OrientationDate matches the next orientation date get priority, followed by those with the oldest normalized application date." wrapText="true" GridPane.columnIndex="0" GridPane.columnSpan="2" GridPane.rowIndex="0" />
  <Label text="Date of next orientation:" GridPane.columnIndex="0" GridPane.rowIndex="1" />
  <DatePicker fx:id="orientationDate" editable="false" GridPane.columnIndex="1" GridPane.rowIndex="1" />
  <Label text="Number of applicants to pull:" GridPane.columnIndex="0" GridPane.rowIndex="2" />
  <Spinner fx:id="numberOfApplicants" editable="false" GridPane.columnIndex="1" GridPane.rowIndex="2" />
  <Button fx:id="pullApplicantsButton" mnemonicParsing="false" onAction="#pullApplicants" text="Pull Applicants" GridPane.columnIndex="0" GridPane.rowIndex="4" />
  <Button fx:id="closeWindowButton" mnemonicParsing="false" onAction="#closeWindow" text="Close Window" GridPane.columnIndex="1" GridPane.rowIndex="4" />
  <Label fx:id="statusLabel" text="" wrapText="true" GridPane.columnIndex="0" GridPane.columnSpan="2" GridPane.rowIndex="5" />

您可以根據自己的目的使用Task ,它在后台線程中完成所有工作,這樣 GUI 線程就不會被阻塞。 這是一個最小的例子:

Controller Class:

package sample;

import javafx.fxml.FXML;
import javafx.scene.control.Label;

public class Controller {

    @FXML
    private Label statusLabel;

    @FXML
    public void handleStartBtnClick() {
        MyTask myTask = new MyTask();
        statusLabel.textProperty().bind(myTask.messageProperty());
        new Thread(myTask).start();
    }
}

我的任務 Class:

package sample;

import javafx.concurrent.Task;

public class MyTask extends Task<Void> {

    @Override
    protected Void call() throws Exception {

        updateMessage("starting");

        // while (...) {
        // do something:
        // changeSearchStringToIncludeOrientationDate(nextOrientationDate);
        // MemberClicks memberClicks = new MemberClicks();

        Thread.sleep(1000); // just for demonstration purpose

        // Update the status:
        updateMessage("retrieving profiles");

        Thread.sleep(1000);

        // Do next step:
        // ...
        updateMessage("converting profiles");
        Thread.sleep(1000);

        // } End of while loop

        return null;
    }

    @Override
    protected void succeeded() {
        updateMessage("succeeded");
    }
}

FXML 文件:

<?xml version="1.0" encoding="UTF-8"?>

<?import javafx.scene.control.Button?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.layout.VBox?>


<VBox xmlns="http://javafx.com/javafx/11.0.1" xmlns:fx="http://javafx.com/fxml/1" fx:controller="sample.Controller">
    <children>
        <Button onAction="#handleStartBtnClick" text="Start background task"/>
        <Label fx:id="statusLabel" text="Status"/>
    </children>
</VBox>

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM