Java Tip 48: How to create a reusable MessageBox class
Display a message on your browser screen, with Java 1.1 or later
As a programmer and consultant, I frequently field questions (often on newsgroups) such as: “Why does my modal dialog box keep locking up?”, “How do I extract which button the user clicked in my OKCancel
dialog?” and “Why doesn’t Java have a standard MessageBox?” In this Java tip, I’ve provided one solution that takes into account all of these questions. So what’s this all-encompassing answer? A reusable MessageBox
class (for Java 1.1 or later versions) that allows you to display a message and clickable buttons in a browser window and gather a response from the user. It comes in the form of a JavaBean.
The bare minimum
Let’s consider what we want from a use case point of view. A use case is a series of steps an actor performs to accomplish a goal. Putting our analyst hats on, it would be nice if we had a MessageBox
class that allowed us to ask a user a question and get a response, say, as follows:
MessageBox box = new MessageBox(this);
box.setTitle("Delete Confirmation");
box.addChoice("Yes", "DeleteConfirmYes");
box.addChoice("No", "DeleteConfirmNo");
box.addChoice("Help", "DeleteConfirmHelp");
box.setCloseWindowCommand("DeleteConfirmNo");
box.ask("Do you really want to delete this customer?");
The above code is a use case at the lowest possible level. Note that we can configure the MessageBox
for a wide variety of uses — not just a limited one that provides the responses “Yes,” “No,” or “Cancel.” It’s a common beginner’s error to build a MessageBox
that handles only a few combinations of buttons. But once you begin to understand configurability, as is demonstrated in this simple MessageBox
, you are on the path to designing reusable classes.
To receive notification that a button was clicked, we must implement ActionListener
and test for the action command in public void actionPerformed(Action evt)
.
Adding features
For MessageBox
to be a top-notch reusable class, we need a few more features. For example, what if we have a Frame
and are opening a modal dialog box by using MessageBox
? Shouldn’t we provide MessageBox
with our Frame
so that when the MessageBox
is gone, the focus will return to the Frame
? What we need to add is the following optional use case feature:
box.setFrame(myFrame);
With GUIs getting more polished all the time on the Web, how can we snazz up our MessageBox
and provide the user with a more conceptual ease of use? One way to achieve this is by allowing an image to be displayed along with the message. For this, we need to add an additional optional use case feature:
box.useImageCanvas(lightBulbImage);
But this means the client must create the image, and often the client simply wants to use a standard image in the same directory as MessageBox
. In this case, we would like an easier method:
box.useImageCanvas("LightBulb.gif");
What if we find ourselves frequently using MessageBox
to as questions that demand yes and no answers, thus creating “Yes” and “No” answer boxes? What if, even more often, we ask questions that are best answered with “Okay”? In that case, the more useful features would be:
box.askYesNo("Is Java now the defacto 3GL for smart developers?");
and:
box.askOkay("James Gosling come here I need you.");
Additional requirements are:
-
The dialog should not deadlock the thread that called it (see below for a section on what deadlocking is)
-
The window should close itself when a button is clicked
-
The dialog should center itself on the screen for easy reading
- The dialog should be modal, whether or not a
Frame
is provided. By modal, we mean that users can only click in theMessageBox
window, nowhere else in the application
Finally, what you’ve been waiting for: MessageBox
code
Now that we have our requirements down, we can reveal the fabulous MessageBox
.
Examine the source code for MessageBox
in a separate window. As this code listing is too long to include in this tip, we will examine only the code highlights. MessageBox
uses another reusable class: ImageCanvas. Note the class declaration:
public class MessageBox implements Runnable,
ActionListener, WindowListener, KeyListener {
and the most important method:
public void ask(String message) {
if (frame == null) {
frame = new Frame();
frameNotProvided = true;
} else {
frameNotProvided = false;
}
dialog = new Dialog(frame, true); // Modal
dialog.addWindowListener(this);
dialog.addKeyListener(this);
dialog.setTitle(title);
dialog.setLayout(new BorderLayout(5, 5));
Panel messagePanel = createMultiLinePanel(message);
if (imageCanvas == null) {
dialog.add("Center", messagePanel);
} else {
Panel centerPanel = new Panel();
centerPanel.add(imageCanvas);
centerPanel.add(messagePanel);
dialog.add("Center", centerPanel);
}
dialog.add("South", buttonPanel);
dialog.pack();
enforceMinimumSize(dialog, 200, 100);
centerWindow(dialog);
Toolkit.getDefaultToolkit().beep();
// Start a new thread to show the dialog
Thread thread = new Thread(this);
thread.start();
}
We implement the listeners so as to receive these events, and implement Runnable
so we can create a fine and dandy Java thread. Let’s study the related methods:
public void run() {
dialog.setVisible(true);
}
It couldn’t get much simpler, could it? Notice in ask()
, we start a new thread that causes run()
to be called, and this shows the dialog. This is how we avoid deadlock, which we’ll now pause for a few Web seconds to discuss.
Deadlock: A definition
All Java code runs in a thread or in threads. When starting a Java program by calling a main()
, for example, the Java runtime creates a thread and calls main()
within that thread. Typically, the main()
method will instantiate an entry-point class, which will initialize the system and present a Frame
or Dialog
to the user. The initial thread dies when the main()
method has finished running. The reason the Java runtime itself doesn’t end is because the AWT has spawned one or more user threads to manage AWT behavior, including user input via buttons and such.
When the user clicks a button, the underlying “AWT thread” dispatches an ActionEvent
to the button’s ActionListener
s that have the method actionPerformed(ActionEvent evt)
. Now, suppose in actionPerformed()
, you decide to open a modal dialog box to ask the user something. When the modal dialog box is shown on the screen, the code blocks. (“Blocks” means a thread is waiting for notification to proceed, which, in the case of a modal dialog box, will not happen until the window is closed.) This means that the AWT thread that invoked actionPerformed()
is waiting for the method to return. That AWT thread now is unavailable to process user input, such as on the dialog box we just opened — so your application is deadlocked. Shucks.
To avoid this deadlock catastrophe, either switch to a “better” language or use the advanced features of Java (which makes Java the better language). Simply show the modal dialog box in a new thread, and all is peaches and roses in Javaland. This is what we have done in the code above. This type of deadlock is common until one understands its cause and has a simple solution for preventing it.
Conclusion
The rest of MessageBox
is self-explanatory. Study the MessageBox
code and the MessageBoxTest
application and have fun.
JavaWorld would like to pass on your Java Tip to the rest of the Java world. Write up your coolest tips and tricks now, and send them to [email protected]. You may find yourself an author in JavaWorld with your helpful hints chosen as the next Java Tip!