18

我正在开发 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(""));
4

2 回答 2

43

我建议将代码包装在一个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 阻塞直到该方法完成。当然,您将希望循环调用此代码,直到密码实际匹配。

于 2012-12-10T16:03:04.800 回答
12

重要的

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

在下面的代码片段中调用 CountdownLatch.await 时,您不能在 JavaFX 应用程序线程上。如果您在 JavaFX 应用程序线程上调用 CountDownLatch.await,您将死锁您的应用程序。除此之外,如果您已经在 J​​avaFX 应用程序线程上,则不需要调用 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 对话框中输入正确的密码。在输入正确密码之前,不会显示访问权限阶段。

输入密码  授予访问权限

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();
  }
}

上述程序将密码打印到屏幕和控制台纯粹是出于演示目的,显示或记录密码不是您在真实应用程序中会做的事情。

于 2012-12-11T00:07:51.613 回答