/* * Copyright (c) 2022, 2023, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ import java.awt.AWTException; import java.awt.BorderLayout; import java.awt.Dimension; import java.awt.GraphicsConfiguration; import java.awt.GraphicsDevice; import java.awt.GraphicsEnvironment; import java.awt.Image; import java.awt.Insets; import java.awt.Point; import java.awt.Rectangle; import java.awt.Robot; import java.awt.Toolkit; import java.awt.Window; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.awt.event.WindowAdapter; import java.awt.event.WindowEvent; import java.awt.event.WindowListener; import java.awt.image.RenderedImage; import java.io.File; import java.io.IOException; import java.lang.reflect.InvocationTargetException; import java.util.ArrayList; import java.util.Arrays; import java.util.Collection; import java.util.List; import java.util.Objects; import java.util.concurrent.CountDownLatch; import java.util.concurrent.TimeUnit; import java.util.concurrent.atomic.AtomicInteger; import javax.imageio.ImageIO; import javax.swing.JButton; import javax.swing.JComboBox; import javax.swing.JComponent; import javax.swing.JDialog; import javax.swing.JEditorPane; import javax.swing.JFrame; import javax.swing.JLabel; import javax.swing.JOptionPane; import javax.swing.JPanel; import javax.swing.JScrollPane; import javax.swing.JTextArea; import javax.swing.Timer; import javax.swing.text.JTextComponent; import javax.swing.text.html.HTMLEditorKit; import javax.swing.text.html.StyleSheet; import static java.util.Collections.unmodifiableList; import static javax.swing.SwingUtilities.invokeAndWait; import static javax.swing.SwingUtilities.isEventDispatchThread; /** * Provides a framework for manual tests to display test instructions and * Pass/Fail buttons. *

* Instructions for the user can be either plain text or HTML as supported * by Swing. If the instructions start with {@code }, the * instructions are displayed as HTML. *

* A simple test would look like this: *

{@code
 * public class SampleManualTest {
 *     private static final String INSTRUCTIONS =
 *             "Click Pass, or click Fail if the test failed.";
 *
 *     public static void main(String[] args) throws Exception {
 *         PassFailJFrame.builder()
 *                       .instructions(INSTRUCTIONS)
 *                       .testUI(() -> createTestUI())
 *                       .build()
 *                       .awaitAndCheck();
 *     }
 *
 *     private static Window createTestUI() {
 *         JFrame testUI = new JFrame("Test UI");
 *         testUI.setSize(250, 150);
 *         return testUI;
 *     }
 * }
 * }
*

* The above example uses the {@link Builder Builder} to set the parameters of * the instruction frame. It is the recommended way. *

* The framework will create instruction UI, it will call * the provided {@code createTestUI} on the Event Dispatch Thread (EDT), * and it will automatically position the test UI and make it visible. *

* The {@code Builder.testUI} methods accept interfaces which create one window * or a list of windows if the test needs multiple windows, * or directly a single window, an array of windows or a list of windows. *

* Alternatively, use one of the {@code PassFailJFrame} constructors to * create an object, then create secondary test UI, register it * with {@code PassFailJFrame}, position it and make it visible. * The following sample demonstrates it: *

{@code
 * public class SampleOldManualTest {
 *     private static final String INSTRUCTIONS =
 *             "Click Pass, or click Fail if the test failed.";
 *
 *     public static void main(String[] args) throws Exception {
 *         PassFailJFrame passFail = new PassFailJFrame(INSTRUCTIONS);
 *
 *         SwingUtilities.invokeAndWait(() -> createTestUI());
 *
 *         passFail.awaitAndCheck();
 *     }
 *
 *     private static void createTestUI() {
 *         JFrame testUI = new JFrame("Test UI");
 *         testUI.setSize(250, 150);
 *         PassFailJFrame.addTestWindow(testUI);
 *         PassFailJFrame.positionTestWindow(testUI, PassFailJFrame.Position.HORIZONTAL);
 *         testUI.setVisible(true);
 *     }
 * }
 * }
*

* Use methods of the {@code Builder} class or constructors of the * {@code PassFailJFrame} class to control other parameters: *

*/ public final class PassFailJFrame { private static final String TITLE = "Test Instruction Frame"; private static final long TEST_TIMEOUT = 5; private static final int ROWS = 10; private static final int COLUMNS = 40; /** * Prefix for the user-provided failure reason. */ private static final String FAILURE_REASON = "Failure Reason:\n"; /** * The failure reason message when the user didn't provide one. */ private static final String EMPTY_REASON = "(no reason provided)"; private static final List windowList = new ArrayList<>(); private static final CountDownLatch latch = new CountDownLatch(1); private static TimeoutHandler timeoutHandler; /** * The description of why the test fails. *

* Note: do not use this field directly, * use the {@link #setFailureReason(String) setFailureReason} and * {@link #getFailureReason() getFailureReason} methods to modify and * to read its value. */ private static String failureReason; private static final AtomicInteger imgCounter = new AtomicInteger(0); private static JFrame frame; private static Robot robot; public enum Position {HORIZONTAL, VERTICAL, TOP_LEFT_CORNER} public PassFailJFrame(String instructions) throws InterruptedException, InvocationTargetException { this(instructions, TEST_TIMEOUT); } public PassFailJFrame(String instructions, long testTimeOut) throws InterruptedException, InvocationTargetException { this(TITLE, instructions, testTimeOut); } public PassFailJFrame(String title, String instructions, long testTimeOut) throws InterruptedException, InvocationTargetException { this(title, instructions, testTimeOut, ROWS, COLUMNS); } /** * Constructs a JFrame with a given title & serves as test instructional * frame where the user follows the specified test instruction in order * to test the test case & mark the test pass or fail. If the expected * result is seen then the user click on the 'Pass' button else click * on the 'Fail' button and the reason for the failure should be * specified in the JDialog JTextArea. * * @param title title of the Frame. * @param instructions the instruction for the tester on how to test * and what is expected (pass) and what is not * expected (fail). * @param testTimeOut test timeout where time is specified in minutes. * @param rows number of visible rows of the JTextArea where the * instruction is show. * @param columns Number of columns of the instructional * JTextArea * @throws InterruptedException exception thrown when thread is * interrupted * @throws InvocationTargetException if an exception is thrown while * creating the test instruction frame on * EDT */ public PassFailJFrame(String title, String instructions, long testTimeOut, int rows, int columns) throws InterruptedException, InvocationTargetException { this(title, instructions, testTimeOut, rows, columns, false); } /** * Constructs a JFrame with a given title & serves as test instructional * frame where the user follows the specified test instruction in order * to test the test case & mark the test pass or fail. If the expected * result is seen then the user click on the 'Pass' button else click * on the 'Fail' button and the reason for the failure should be * specified in the JDialog JTextArea. *

* The test instruction frame also provides a way for the tester to take * a screenshot (full screen or individual frame) if this feature * is enabled by passing {@code true} as {@code enableScreenCapture} * parameter. * * @param title title of the Frame. * @param instructions the instruction for the tester on how to test * and what is expected (pass) and what is not * expected (fail). * @param testTimeOut test timeout where time is specified in minutes. * @param rows number of visible rows of the JTextArea where the * instruction is show. * @param columns Number of columns of the instructional * JTextArea * @param enableScreenCapture if set to true, 'Capture Screen' button & its * associated UIs are added to test instruction * frame * @throws InterruptedException exception thrown when thread is * interrupted * @throws InvocationTargetException if an exception is thrown while * creating the test instruction frame on * EDT */ public PassFailJFrame(String title, String instructions, long testTimeOut, int rows, int columns, boolean enableScreenCapture) throws InterruptedException, InvocationTargetException { invokeOnEDT(() -> createUI(title, instructions, testTimeOut, rows, columns, enableScreenCapture)); } private PassFailJFrame(Builder builder) throws InterruptedException, InvocationTargetException { this(builder.title, builder.instructions, builder.testTimeOut, builder.rows, builder.columns, builder.screenCapture); if (builder.windowListCreator != null) { invokeOnEDT(() -> builder.testWindows = builder.windowListCreator.createTestUI()); if (builder.testWindows == null) { throw new IllegalStateException("Window list creator returned null list"); } } if (builder.testWindows != null) { if (builder.testWindows.isEmpty()) { throw new IllegalStateException("Window list is empty"); } addTestWindow(builder.testWindows); builder.testWindows .forEach(w -> w.addWindowListener(windowClosingHandler)); if (builder.positionWindows != null) { positionInstructionFrame(builder.position); invokeOnEDT(() -> { builder.positionWindows .positionTestWindows(unmodifiableList(builder.testWindows), builder.instructionUIHandler); }); } else if (builder.testWindows.size() == 1) { Window window = builder.testWindows.get(0); positionTestWindow(window, builder.position); } else { positionTestWindow(null, builder.position); } } showAllWindows(); } /** * Performs an operation on EDT. If called on EDT, invokes {@code run} * directly, otherwise wraps into {@code invokeAndWait}. * * @param doRun an operation to run on EDT * @throws InterruptedException if we're interrupted while waiting for * the event dispatching thread to finish executing * {@code doRun.run()} * @throws InvocationTargetException if an exception is thrown while * running {@code doRun} * @see javax.swing.SwingUtilities#invokeAndWait(Runnable) */ private static void invokeOnEDT(Runnable doRun) throws InterruptedException, InvocationTargetException { if (isEventDispatchThread()) { doRun.run(); } else { invokeAndWait(doRun); } } private static void createUI(String title, String instructions, long testTimeOut, int rows, int columns, boolean enableScreenCapture) { frame = new JFrame(title); frame.setLayout(new BorderLayout()); JLabel testTimeoutLabel = new JLabel("", JLabel.CENTER); timeoutHandler = new TimeoutHandler(testTimeoutLabel, testTimeOut); frame.add(testTimeoutLabel, BorderLayout.NORTH); JTextComponent text = instructions.startsWith("") ? configureHTML(instructions, rows, columns) : configurePlainText(instructions, rows, columns); text.setEditable(false); frame.add(new JScrollPane(text), BorderLayout.CENTER); JButton btnPass = new JButton("Pass"); btnPass.addActionListener((e) -> { latch.countDown(); timeoutHandler.stop(); }); JButton btnFail = new JButton("Fail"); btnFail.addActionListener((e) -> { requestFailureReason(); timeoutHandler.stop(); }); JPanel buttonsPanel = new JPanel(); buttonsPanel.add(btnPass); buttonsPanel.add(btnFail); if (enableScreenCapture) { buttonsPanel.add(createCapturePanel()); } frame.addWindowListener(windowClosingHandler); frame.add(buttonsPanel, BorderLayout.SOUTH); frame.pack(); frame.setLocationRelativeTo(null); addTestWindow(frame); } private static JTextComponent configurePlainText(String instructions, int rows, int columns) { JTextArea text = new JTextArea(instructions, rows, columns); text.setLineWrap(true); text.setWrapStyleWord(true); return text; } private static JTextComponent configureHTML(String instructions, int rows, int columns) { JEditorPane text = new JEditorPane("text/html", instructions); text.putClientProperty(JEditorPane.HONOR_DISPLAY_PROPERTIES, Boolean.TRUE); // Set preferred size as if it were JTextArea text.setPreferredSize(new JTextArea(rows, columns).getPreferredSize()); HTMLEditorKit kit = (HTMLEditorKit) text.getEditorKit(); StyleSheet styles = kit.getStyleSheet(); // Reduce the default margins styles.addRule("ol, ul { margin-left-ltr: 20; margin-left-rtl: 20 }"); // Make the size of code blocks the same as other text styles.addRule("code { font-size: inherit }"); return text; } /** * Creates a test UI window. */ @FunctionalInterface public interface WindowCreator { /** * Creates a window for test UI. * This method is called by the framework on the EDT. * @return a test UI window */ Window createTestUI(); } /** * Creates a list of test UI windows. */ @FunctionalInterface public interface WindowListCreator { /** * Creates one or more windows for test UI. * This method is called by the framework on the EDT. * @return a list of test UI windows */ List createTestUI(); } /** * Positions test UI windows. */ @FunctionalInterface public interface PositionWindows { /** * Positions test UI windows. * This method is called by the framework on the EDT after * the instruction UI frame was positioned on the screen. *

* The list of the test windows contains the windows * that were passed to the framework via the * {@link Builder#testUI(Window...) testUI(Window...)} method or * that were created with {@code WindowCreator} * or {@code WindowListCreator} which were passed via * {@link Builder#testUI(WindowCreator) testUI(WindowCreator)} or * {@link Builder#testUI(WindowListCreator) testUI(WindowListCreator)} * correspondingly. * * @param testWindows the list of test windows * @param instructionUI information about the instruction frame */ void positionTestWindows(List testWindows, InstructionUI instructionUI); } /** * Provides information about the instruction frame. */ public interface InstructionUI { /** * {@return the location of the instruction frame} */ Point getLocation(); /** * {@return the size of the instruction frame} */ Dimension getSize(); /** * {@return the bounds of the instruction frame} */ Rectangle getBounds(); /** * Allows to change the location of the instruction frame. * * @param location the new location of the instruction frame */ void setLocation(Point location); /** * Allows to change the location of the instruction frame. * * @param x the x coordinate of the new location * @param y the y coordinate of the new location */ void setLocation(int x, int y); /** * Returns the specified position that was used to set * the initial location of the instruction frame. * * @return the specified position * * @see Position */ Position getPosition(); } private static final class TimeoutHandler implements ActionListener { private final long endTime; private final Timer timer; private final JLabel label; public TimeoutHandler(final JLabel label, final long testTimeOut) { endTime = System.currentTimeMillis() + TimeUnit.MINUTES.toMillis(testTimeOut); this.label = label; timer = new Timer(1000, this); timer.start(); updateTime(testTimeOut); } @Override public void actionPerformed(ActionEvent e) { long leftTime = endTime - System.currentTimeMillis(); if (leftTime < 0) { timer.stop(); setFailureReason(FAILURE_REASON + "Timeout - User did not perform testing."); latch.countDown(); } updateTime(leftTime); } private void updateTime(final long leftTime) { if (leftTime < 0) { label.setText("Test timeout: 00:00:00"); return; } long hours = leftTime / 3_600_000; long minutes = (leftTime - hours * 3_600_000) / 60_000; long seconds = (leftTime - hours * 3_600_000 - minutes * 60_000) / 1_000; label.setText(String.format("Test timeout: %02d:%02d:%02d", hours, minutes, seconds)); } public void stop() { timer.stop(); } } private static final class WindowClosingHandler extends WindowAdapter { @Override public void windowClosing(WindowEvent e) { setFailureReason(FAILURE_REASON + "User closed a window"); latch.countDown(); } } private static final WindowListener windowClosingHandler = new WindowClosingHandler(); private static JComponent createCapturePanel() { JComboBox screenShortType = new JComboBox<>(CaptureType.values()); JButton capture = new JButton("ScreenShot"); capture.addActionListener((e) -> captureScreen((CaptureType) screenShortType.getSelectedItem())); JPanel panel = new JPanel(); panel.add(screenShortType); panel.add(capture); return panel; } private enum CaptureType { FULL_SCREEN("Capture Full Screen"), WINDOWS("Capture Individual Frame"); private final String type; CaptureType(String type) { this.type = type; } @Override public String toString() { return type; } } private static Robot createRobot() { if (robot == null) { try { robot = new Robot(); } catch (AWTException e) { String errorMsg = "Failed to create an instance of Robot."; JOptionPane.showMessageDialog(frame, errorMsg, "Failed", JOptionPane.ERROR_MESSAGE); forceFail(errorMsg + e.getMessage()); } } return robot; } private static void captureScreen(Rectangle bounds) { Robot robot = createRobot(); List imageList = robot.createMultiResolutionScreenCapture(bounds) .getResolutionVariants(); Image image = imageList.get(imageList.size() - 1); File file = new File("CaptureScreen_" + imgCounter.incrementAndGet() + ".png"); try { ImageIO.write((RenderedImage) image, "png", file); } catch (IOException e) { throw new RuntimeException(e); } } private static void captureScreen(CaptureType type) { switch (type) { case FULL_SCREEN: Arrays.stream(GraphicsEnvironment.getLocalGraphicsEnvironment() .getScreenDevices()) .map(GraphicsDevice::getDefaultConfiguration) .map(GraphicsConfiguration::getBounds) .forEach(PassFailJFrame::captureScreen); break; case WINDOWS: windowList.stream() .filter(Window::isShowing) .map(Window::getBounds) .forEach(PassFailJFrame::captureScreen); break; default: throw new IllegalStateException("Unexpected value of capture type"); } JOptionPane.showMessageDialog(frame, "Screen Captured Successfully", "Screen Capture", JOptionPane.INFORMATION_MESSAGE); } /** * Sets the failure reason which describes why the test fails. * This method ensures the {@code failureReason} field does not change * after it's set to a non-{@code null} value. * @param reason the description of why the test fails * @throws IllegalArgumentException if the {@code reason} parameter * is {@code null} */ private static synchronized void setFailureReason(final String reason) { if (reason == null) { throw new IllegalArgumentException("The failure reason must not be null"); } if (failureReason == null) { failureReason = reason; } } /** * {@return the description of why the test fails} */ private static synchronized String getFailureReason() { return failureReason; } /** * Wait for the user decision i,e user selects pass or fail button. * If user does not select pass or fail button then the test waits for * the specified timeoutMinutes period and the test gets timeout. * Note: This method should be called from main() thread * * @throws InterruptedException exception thrown when thread is * interrupted * @throws InvocationTargetException if an exception is thrown while * disposing of frames on EDT */ public void awaitAndCheck() throws InterruptedException, InvocationTargetException { if (isEventDispatchThread()) { throw new IllegalStateException("awaitAndCheck() should not be called on EDT"); } latch.await(); invokeAndWait(PassFailJFrame::disposeWindows); String failure = getFailureReason(); if (failure != null) { throw new RuntimeException(failure); } System.out.println("Test passed!"); } /** * Requests the description of the test failure reason from the tester. */ private static void requestFailureReason() { final JDialog dialog = new JDialog(frame, "Test Failure ", true); dialog.setTitle("Failure reason"); JPanel jPanel = new JPanel(new BorderLayout()); JTextArea jTextArea = new JTextArea(5, 20); JButton okButton = new JButton("OK"); okButton.addActionListener((ae) -> { String text = jTextArea.getText(); setFailureReason(FAILURE_REASON + (!text.isEmpty() ? text : EMPTY_REASON)); dialog.setVisible(false); }); jPanel.add(new JScrollPane(jTextArea), BorderLayout.CENTER); JPanel okayBtnPanel = new JPanel(); okayBtnPanel.add(okButton); jPanel.add(okayBtnPanel, BorderLayout.SOUTH); dialog.add(jPanel); dialog.setLocationRelativeTo(frame); dialog.pack(); dialog.setVisible(true); // Ensure the test fails even if the dialog is closed // without clicking the OK button setFailureReason(FAILURE_REASON + EMPTY_REASON); dialog.dispose(); latch.countDown(); } /** * Disposes of all the windows. It disposes of the test instruction frame * and all other windows added via {@link #addTestWindow(Window)}. */ private static synchronized void disposeWindows() { windowList.forEach(Window::dispose); } private static void positionInstructionFrame(final Position position) { Dimension screenSize = Toolkit.getDefaultToolkit().getScreenSize(); // Get the screen insets to position the frame by taking into // account the location of taskbar or menu bar on screen. GraphicsConfiguration gc = GraphicsEnvironment.getLocalGraphicsEnvironment() .getDefaultScreenDevice() .getDefaultConfiguration(); Insets screenInsets = Toolkit.getDefaultToolkit().getScreenInsets(gc); switch (position) { case HORIZONTAL: int newX = ((screenSize.width / 2) - frame.getWidth()); frame.setLocation((newX + screenInsets.left), (frame.getY() + screenInsets.top)); break; case VERTICAL: int newY = ((screenSize.height / 2) - frame.getHeight()); frame.setLocation((frame.getX() + screenInsets.left), (newY + screenInsets.top)); break; case TOP_LEFT_CORNER: frame.setLocation(screenInsets.left, screenInsets.top); break; } syncLocationToWindowManager(); } /** * Approximately positions the instruction frame relative to the test * window as specified by the {@code position} parameter. If {@code testWindow} * is {@code null}, only the instruction frame is positioned according to * {@code position} parameter. *

This method should be called before making the test window visible * to avoid flickering.

* * @param testWindow test window that the test created. * May be {@code null}. * * @param position position must be one of: * */ public static void positionTestWindow(Window testWindow, Position position) { positionInstructionFrame(position); if (testWindow != null) { switch (position) { case HORIZONTAL: case TOP_LEFT_CORNER: testWindow.setLocation((frame.getX() + frame.getWidth() + 5), frame.getY()); break; case VERTICAL: testWindow.setLocation(frame.getX(), (frame.getY() + frame.getHeight() + 5)); break; } } // make instruction frame visible after updating // frame & window positions frame.setVisible(true); } /** * Ensures the frame location is updated by the window manager * if it adjusts the frame location after {@code setLocation}. * * @see #positionTestWindow */ private static void syncLocationToWindowManager() { Toolkit.getDefaultToolkit().sync(); try { Thread.sleep(500); } catch (InterruptedException e) { e.printStackTrace(); } } /** * Returns the current position and size of the test instruction frame. * This method can be used in scenarios when custom positioning of * multiple test windows w.r.t test instruction frame is necessary, * at test-case level and the desired configuration is not available * as a {@code Position} option. * * @return Rectangle bounds of test instruction frame * @see #positionTestWindow * * @throws InterruptedException exception thrown when thread is * interrupted * @throws InvocationTargetException if an exception is thrown while * obtaining frame bounds on EDT */ public static Rectangle getInstructionFrameBounds() throws InterruptedException, InvocationTargetException { final Rectangle[] bounds = {null}; invokeOnEDT(() -> bounds[0] = frame != null ? frame.getBounds() : null); return bounds[0]; } /** * Add the testWindow to the windowList so that test instruction frame * and testWindow and any other windows used in this test is disposed * via disposeWindows(). * * @param testWindow testWindow that needs to be disposed */ public static synchronized void addTestWindow(Window testWindow) { windowList.add(testWindow); } /** * Adds a collection of test windows to the windowList to be disposed of * when the test completes. * * @param testWindows the collection of test windows to be disposed of */ public static synchronized void addTestWindow(Collection testWindows) { windowList.addAll(testWindows); } /** * Displays all the windows in {@code windowList}. * * @throws InterruptedException if the thread is interrupted while * waiting for the event dispatch thread to finish running * the {@link #showUI() showUI} * @throws InvocationTargetException if an exception is thrown while * the event dispatch thread executes {@code showUI} */ private static void showAllWindows() throws InterruptedException, InvocationTargetException { invokeOnEDT(PassFailJFrame::showUI); } /** * Displays all the windows in {@code windowList}; it has to be called on * the EDT — use {@link #showAllWindows() showAllWindows} to ensure it. */ private static synchronized void showUI() { windowList.forEach(w -> w.setVisible(true)); } /** * Forcibly pass the test. *

The sample usage: *


     *      PrinterJob pj = PrinterJob.getPrinterJob();
     *      if (pj == null || pj.getPrintService() == null) {
     *          System.out.println(""Printer not configured or available.");
     *          PassFailJFrame.forcePass();
     *      }
     * 
*/ public static void forcePass() { latch.countDown(); } /** * Forcibly fail the test. */ public static void forceFail() { forceFail("forceFail called"); } /** * Forcibly fail the test and provide a reason. * * @param reason the reason why the test is failed */ public static void forceFail(String reason) { setFailureReason(FAILURE_REASON + reason); latch.countDown(); } public static final class Builder { private String title; private String instructions; private long testTimeOut; private int rows; private int columns; private boolean screenCapture; private List testWindows; private WindowListCreator windowListCreator; private PositionWindows positionWindows; private InstructionUI instructionUIHandler; private Position position; public Builder title(String title) { this.title = title; return this; } public Builder instructions(String instructions) { this.instructions = instructions; return this; } public Builder testTimeOut(long testTimeOut) { this.testTimeOut = testTimeOut; return this; } public Builder rows(int rows) { this.rows = rows; return this; } public Builder columns(int columns) { this.columns = columns; return this; } public Builder screenCapture() { this.screenCapture = true; return this; } /** * Adds a {@code WindowCreator} which the framework will use * to create the test UI window. * * @param windowCreator a {@code WindowCreator} * to create the test UI window * @return this builder * @throws IllegalArgumentException if {@code windowCreator} is {@code null} * @throws IllegalStateException if a window creator * or a list of test windows is already set */ public Builder testUI(WindowCreator windowCreator) { if (windowCreator == null) { throw new IllegalArgumentException("The window creator can't be null"); } checkWindowsLists(); this.windowListCreator = () -> List.of(windowCreator.createTestUI()); return this; } /** * Adds a {@code WindowListCreator} which the framework will use * to create a list of test UI windows. * * @param windowListCreator a {@code WindowListCreator} * to create test UI windows * @return this builder * @throws IllegalArgumentException if {@code windowListCreator} is {@code null} * @throws IllegalStateException if a window creator * or a list of test windows is already set */ public Builder testUI(WindowListCreator windowListCreator) { if (windowListCreator == null) { throw new IllegalArgumentException("The window list creator can't be null"); } checkWindowsLists(); this.windowListCreator = windowListCreator; return this; } /** * Adds an already created test UI window. * The window is positioned and shown automatically. * * @param window a test UI window * @return this builder */ public Builder testUI(Window window) { return testUI(List.of(window)); } /** * Adds an array of already created test UI windows. * * @param windows an array of test UI windows * @return this builder */ public Builder testUI(Window... windows) { return testUI(List.of(windows)); } /** * Adds a list of already created test UI windows. * * @param windows a list of test UI windows * @return this builder * @throws IllegalArgumentException if {@code windows} is {@code null} * or the list contains {@code null} * @throws IllegalStateException if a window creator * or a list of test windows is already set */ public Builder testUI(List windows) { if (windows == null) { throw new IllegalArgumentException("The list of windows can't be null"); } if (windows.stream() .anyMatch(Objects::isNull)) { throw new IllegalArgumentException("The list of windows can't contain null"); } checkWindowsLists(); this.testWindows = windows; return this; } /** * Verifies the state of window list and window creator. * * @throws IllegalStateException if a windows list creator * or a list of test windows is already set */ private void checkWindowsLists() { if (windowListCreator != null) { throw new IllegalStateException("Window list creator is already set"); } if (testWindows != null) { throw new IllegalStateException("The list of test windows is already set"); } } public Builder positionTestUI(PositionWindows positionWindows) { this.positionWindows = positionWindows; return this; } public Builder position(Position position) { this.position = position; return this; } public PassFailJFrame build() throws InterruptedException, InvocationTargetException { validate(); return new PassFailJFrame(this); } private void validate() { if (title == null) { title = TITLE; } if (instructions == null || instructions.isEmpty()) { throw new IllegalStateException("Please provide the test " + "instructions for this manual test"); } if (testTimeOut == 0L) { testTimeOut = TEST_TIMEOUT; } if (rows == 0) { rows = ROWS; } if (columns == 0) { columns = COLUMNS; } if (position == null && (testWindows != null || windowListCreator != null)) { position = Position.HORIZONTAL; } if (positionWindows != null) { if (testWindows == null && windowListCreator == null) { throw new IllegalStateException("To position windows, " + "provide an a list of windows to the builder"); } instructionUIHandler = new InstructionUIHandler(); } } private final class InstructionUIHandler implements InstructionUI { @Override public Point getLocation() { return frame.getLocation(); } @Override public Dimension getSize() { return frame.getSize(); } @Override public Rectangle getBounds() { return frame.getBounds(); } @Override public void setLocation(Point location) { setLocation(location.x, location.y); } @Override public void setLocation(int x, int y) { frame.setLocation(x, y); } @Override public Position getPosition() { return position; } } } public static Builder builder() { return new Builder(); } }