package com.twasyl.slideshowfx.utils; import javafx.scene.Node; import javafx.scene.control.Alert; import javafx.scene.control.ButtonType; import javafx.scene.control.Dialog; import java.util.Optional; import java.util.concurrent.ExecutionException; import java.util.concurrent.FutureTask; import java.util.logging.Level; import java.util.logging.Logger; /** * This class provides helper methods to build {@link javafx.scene.control.Dialog} for SlideshowFX. * * @author Thierry Wasylczenko * @version 1.0 * @since SlideshowFX 1.0 */ public class DialogHelper { private static final Logger LOGGER = Logger.getLogger(DialogHelper.class.getName()); /** * Show a confirmation alert with the given {@code title} and {@code text}. The method returns the answer of the * user. The answer can either be {@link javafx.scene.control.ButtonType#NO} or {@link javafx.scene.control.ButtonType#YES}. * * @param title The title of the alert. * @param text The text of the alert. * @return The answer of the user or {@code null} if no answer has been made. */ public static ButtonType showConfirmationAlert(final String title, final String text) { return displayDialog(buildAlert(Alert.AlertType.CONFIRMATION, title, text, ButtonType.NO, ButtonType.YES)); } /** * Show an alert with the given {@code title} and {@code text}. The method returns the answer of the user which can * only be {@link ButtonType#OK}. * * @param title The title of the alert. * @param text The text of the alert. * @return The answer of the user which can only be {@link ButtonType#OK}. */ public static ButtonType showAlert(final String title, final String text) { return displayDialog(buildAlert(Alert.AlertType.INFORMATION, title, text, ButtonType.OK)); } /** * Show an error alert with the given {@code title} and {@code text}. The method returns the answer of the user * which can only be {@link javafx.scene.control.ButtonType#OK}. * * @param title The title of the error. * @param text The text of the error. * @return The answer of the user or {@code null} if no answer has been made. */ public static ButtonType showError(final String title, final String text) { return displayDialog(buildAlert(Alert.AlertType.ERROR, title, text, ButtonType.OK)); } /** * Show an error alert with the given {@code title} and {@code content}. The method returns the answer of the user * which can only be {@link javafx.scene.control.ButtonType#OK}. * * @param title The title of the error. * @param content The content of the error. * @return The answer of the user or {@code null} if no answer has been made. */ public static ButtonType showError(final String title, final Node content) { return displayDialog(buildAlert(Alert.AlertType.ERROR, title, content, ButtonType.OK)); } /** * Show a dialog that can be cancelled with the given {@code title} and {@code content}. The method returns the * answer of the user which can be {@link javafx.scene.control.ButtonType#CANCEL} or * {@link javafx.scene.control.ButtonType#OK}. * * @param title The title of the dialog. * @param content The content of the dialog. * @return The answer of the user or {@code null} if no answer has been made. */ public static ButtonType showCancellableDialog(final String title, final Node content) { return displayDialog(buildDialog(title, content, ButtonType.CANCEL, ButtonType.OK)); } /** * Show a dialog that contains the given {@code title}, {@code content} and with the given {@code buttons}. * @param title The title of the dialog. * @param content The content of the dialog. * @param buttons The buttons of the dialog. * @return The answer of the user or {@code null} if no answer has been made. */ public static ButtonType showDialog(final String title, final Node content, ButtonType ... buttons) { return displayDialog(buildDialog(title, content, buttons)); } /** * Build an {@link javafx.scene.control.Alert Alert} object. This method ensures the alert is created in a JavaFX * application thread. If the alert can not be created then {@code null} is returned. * @param type The type of alert to create. * @param title The title of the alert. * @param text The text of this alert. * @param buttons The buttons the alert will contain. * @return A well created Alert or {@code null} if an error occurred during the creation of the alert. */ private static Alert buildAlert(final Alert.AlertType type, final String title, final String text, final ButtonType ... buttons) { final FutureTask<Alert> future = new FutureTask<>(() -> { final Alert alert = new Alert(type, text, buttons); alert.setGraphic(null); alert.setHeaderText(null); alert.setTitle(title); alert.getDialogPane().getStylesheets().add("/com/twasyl/slideshowfx/css/Default.css"); return alert; }); PlatformHelper.run(future); Alert alert = null; try { alert = future.get(); } catch (InterruptedException | ExecutionException e) { LOGGER.log(Level.SEVERE, "Can not build an alert", e); } return alert; } /** * Build an {@link javafx.scene.control.Alert Alert} object. This method ensures the alert is created in a JavaFX * application thread. If the alert can not be created then {@code null} is returned. * @param type The type of alert to create. * @param title The title of the alert. * @param content The content of this alert. * @param buttons The buttons the alert will contain. * @return A well created Alert or {@code null} if an error occurred during the creation of the alert. */ private static Alert buildAlert(final Alert.AlertType type, final String title, final Node content, final ButtonType ... buttons) { final FutureTask<Alert> future = new FutureTask<>(() -> { final Alert alert = new Alert(type, null, buttons); alert.setGraphic(null); alert.setHeaderText(null); alert.setTitle(title); alert.getDialogPane().setContent(content); alert.getDialogPane().getStylesheets().add("/com/twasyl/slideshowfx/css/Default.css"); return alert; }); PlatformHelper.run(future); Alert alert = null; try { alert = future.get(); } catch (InterruptedException | ExecutionException e) { LOGGER.log(Level.SEVERE, "Can not build an alert", e); } return alert; } /** * Build a {@link javafx.scene.control.Dialog Dialog} object. This method ensures the dialog is created in a JavaFX * application thread. If the dialog can not be created then {@code null} is returned. * @param title The title of the Dialog * @param content The content of the Dialog * @param buttons The type of buttons the Dialog will contain. * @return A well created Dialog or {@code null} if an error occurred during the creation of the Dialog. */ private static Dialog buildDialog(final String title, final Node content, final ButtonType ... buttons) { final FutureTask<Dialog> future = new FutureTask<>(() -> { final Dialog dialog = new Dialog(); dialog.setGraphic(null); dialog.setHeaderText(null); dialog.setTitle(title); dialog.getDialogPane().getButtonTypes().addAll(buttons); dialog.getDialogPane().setContent(content); dialog.getDialogPane().getStylesheets().add("/com/twasyl/slideshowfx/css/Default.css"); return dialog; }); PlatformHelper.run(future); Dialog dialog = null; try { dialog = future.get(); } catch (InterruptedException | ExecutionException e) { LOGGER.log(Level.SEVERE, "Can not build a Dialog", e); } return dialog; } /** * Show and wait for the response of the given {@code dialog}. This method ensures the dialog is displayed in the * JavaFX application thread. * @param dialog The dialog to show. * @return The answer of the user or {@code null} if no answer has been made. */ private static ButtonType displayDialog(Dialog<ButtonType> dialog) { Optional<ButtonType> response = null; if(dialog != null) { final FutureTask<Optional<ButtonType>> future = new FutureTask<>(() -> dialog.showAndWait()); PlatformHelper.run(future); try { response = future.get(); } catch (InterruptedException | ExecutionException e) { LOGGER.log(Level.SEVERE, "Can not show dialog", e); } } return response != null && response.isPresent() ? response.get() : null; } }