我正在研究JavaFX应用程序,在我的场景中是显示在JavaFX中创建的密码提示,它使用两个选项OK
和Cancel
获取密码。我已经返回了用户输入的密码。
我的显示密码对话框的类是 -
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(""));
答案 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对话框中输入了正确的密码。在输入正确的密码之前,不会显示访问授予阶段。
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();
}
}
上述程序仅为了演示目的而将密码打印到屏幕和控制台,显示或记录密码不是您在实际应用程序中要做的事情。