从javafx平台runlater返回结果

时间:2012-12-10 07:20:24

标签: javafx-2 javafx

我正在研究JavaFX应用程序,在我的场景中是显示在JavaFX中创建的密码提示,它使用两个选项OKCancel获取密码。我已经返回了用户输入的密码。

我的显示密码对话框的类是 -

public static String showPasswordDialog(String title, String message, Stage parentStage, double w, double h) {
    try {
        Stage stage = new Stage();
        PasswordDialogController controller = (PasswordDialogController) Utility.replaceScene("Password.fxml", stage);
        passwordDialogController.init(stage, message, "/images/password.png");
        if (parentStage != null) {
            stage.initOwner(parentStage);
        }
        stage.initModality(Modality.WINDOW_MODAL);
        stage.initStyle(StageStyle.UTILITY);
        stage.setResizable(false);
        stage.setWidth(w);
        stage.setHeight(h);                
        stage.showAndWait();
        return controller.getPassword(); 
    } catch (Exception ex) {
         return null;
    }

我的代码显示密码提示的位置如下,实际上此提示将显示在其他UI上,因此我需要将其包含在Platform.runlater()内,否则会抛出Not on FX application thread。我需要这个密码提示才能显示,直到我得到正确的密码提示。如果我在runlater中显示密码,我怎样才能获得密码值。

还有其他更好的方法吗?

final String sPassword = null;

          do {
            Platform.runLater(new Runnable() {

                @Override
                public void run() {
                     sPassword = JavaFXDialog.showPasswordDialog(sTaskName + "Password", "Enter the password:", parentStage, 400.0, 160.0);
                }
            });

            if (sPassword == null) {
                System.out.println("Entering password cancelled.");
                throw new Exception("Cancel");
            }
        } while (sPassword.equalsIgnoreCase(""));

2 个答案:

答案 0 :(得分:33)

我建议将代码包装在FutureTask对象中。 FutureTask是一个有用的构造(除其他外),用于在一个线程上执行一部分代码(通常是工作者,在您的情况下是事件队列)并在另一个线程上安全地检索它。 FutureTask#get将阻止FutureTask#run被调用,因此您的密码提示符如下所示:

final FutureTask query = new FutureTask(new Callable() {
    @Override
    public Object call() throws Exception {
        return queryPassword();
    }
});
Platform.runLater(query);
System.out.println(query.get());

FutureTask实现Runnable时,您可以将其直接传递给Platform#runLater(...)queryPassword()将在事件队列中被忽略,并且随后调用get block直到该方法完成。当然,您需要在循环中调用此代码,直到密码实际匹配为止。

答案 1 :(得分:10)

重要

此代码用于特定情况,当您拥有不在JavaFX应用程序线程上的代码并且您想要调用JavaFX应用程序线程上的代码以向用户显示GUI时,然后在继续处理JavaFX应用程序线程之前从该GUI获取结果。

当您在下面的代码段中调用CountdownLatch.await时,您不能在JavaFX应用程序线程上。如果在JavaFX Application线程上调用CountDownLatch.await,则会使应用程序死锁。除此之外,如果您已经在JavaFX应用程序线程上,则无需调用Platform.runLater来在JavaFX应用程序线程上执行某些操作。

大多数情况下,您知道自己是否使用JavaFX应用程序线程。如果您不确定,可以致电Platform.isFxApplicationThread()查看您的主题。


使用CountDownLatch的替代方法。我更喜欢Sarcan的方法; - )

final CountDownLatch latch = new CountDownLatch(1);
final StringProperty passwordProperty = new SimpleStringProperty();
Platform.runLater(new Runnable() {
    @Override public void run() {
        passwordProperty.set(queryPassword());
        latch.countDown();
    }
});
latch.await();      
System.out.println(passwordProperty.get());

这是一些可执行的示例代码,演示如何使用CountdownLatch暂停执行非JavaFX应用程序线程,直到JavaFX对话框检索到结果,然后非JavaFX应用程序线程可以访问该结果。

应用程序阻止应用程序的JavaFX启动程序线程继续,直到用户在JavaFX对话框中输入了正确的密码。在输入正确的密码之前,不会显示访问授予阶段。

enterpassword access granted

import javafx.application.*;
import javafx.beans.property.*;
import javafx.event.ActionEvent;
import javafx.event.EventHandler;
import javafx.geometry.Pos;
import javafx.scene.Scene;
import javafx.scene.control.*;
import javafx.scene.layout.*;
import javafx.scene.text.TextAlignment;
import javafx.stage.*;

import java.util.concurrent.CountDownLatch;

public class PasswordPrompter extends Application {
  final StringProperty passwordProperty = new SimpleStringProperty();
  @Override public void init() {
    final CountDownLatch latch = new CountDownLatch(1);

    Platform.runLater(new Runnable() {
      @Override public void run() {
        passwordProperty.set(new PasswordPrompt(null).getPassword());
        latch.countDown();
      }
    });

    try {
      latch.await();
    } catch (InterruptedException e) {
      Platform.exit();
    }

    System.out.println(passwordProperty.get());
  }

  @Override public void start(final Stage stage) {
    Label welcomeMessage = new Label("Access Granted\nwith password\n" + passwordProperty.get());
    welcomeMessage.setTextAlignment(TextAlignment.CENTER);

    StackPane layout = new StackPane();
    layout.setStyle("-fx-background-color: cornsilk; -fx-padding: 20px;");
    layout.getChildren().setAll(welcomeMessage);
    stage.setScene(new Scene(layout));

    stage.show();
  }

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

class PasswordPrompt {
  final Window owner;

  PasswordPrompt(Window owner) {
    this.owner = owner;
  }

  public String getPassword() {
    final Stage dialog = new Stage();
    dialog.setTitle("Pass is sesame");
    dialog.initOwner(owner);
    dialog.initStyle(StageStyle.UTILITY);
    dialog.initModality(Modality.WINDOW_MODAL);
    dialog.setOnCloseRequest(new EventHandler<WindowEvent>() {
      @Override public void handle(WindowEvent windowEvent) {
        Platform.exit();
      }
    });

    final TextField textField = new TextField();
    textField.setPromptText("Enter sesame");
    final Button submitButton = new Button("Submit");
    submitButton.setDefaultButton(true);
    submitButton.setOnAction(new EventHandler<ActionEvent>() {
      @Override public void handle(ActionEvent t) {
        if ("sesame".equals(textField.getText())) {
          dialog.close();
        }
      }
    });

    final VBox layout = new VBox(10);
    layout.setAlignment(Pos.CENTER_RIGHT);
    layout.setStyle("-fx-background-color: azure; -fx-padding: 10;");
    layout.getChildren().setAll(textField, submitButton);

    dialog.setScene(new Scene(layout));
    dialog.showAndWait();

    return textField.getText();
  }
}

上述程序仅为了演示目的而将密码打印到屏幕和控制台,显示或记录密码不是您在实际应用程序中要做的事情。