Java Popup
Volume Number: | | 12
|
Issue Number: | | 7
|
Column Tag: | | Getting Started
|
A Java Popup Menu URL Launcher
By Dave Mark
Note: Source code files accompanying article are located on MacTech CD-ROM or source code disks.
Since last months column, Ive been consumed with Java. Ive been reading every Java book I could get my hands on, including the galleys for Learn Java on the Macintosh (Addison-Wesley), Java in a Nutshell (OReilly & Associates), Active Java (Addison-Wesley), Java Essentials for C and C++ Programmers (Addison-Wesley), and Teach Yourself Java for Macintosh In 21 Days (Hayden). There are lots of Java books out there, some of them much better than others.
Beware of Java books written to pre-release versions of the Java SDK from Sun. Much has changed between the pre-release and release versions and, as a slew of messages on the net will attest, those changes will frustrate you as you attempt to get the book sample code to work.
The applets from Learn Java on the Macintosh and Java in a Nutshell have been translated into Metrowerks Java project format. You can download the applets from Metrowerks Web site (www.metrowerks.com).
This Months Applets
One important use of Java is to extend the functionality of a Web site, beyond the capabilities offered by HTML. If youve ever been to Netscapes Web site, youve probably encountered the long list of download sites that allow you to download the latest version of a Netscape product (Figure 1).
If youve got a long list of URLs to put on your Web site, you can, of course, represent them just like Netscape did, as a long series of links. The downside to this approach is that the links can take up a fair amount of screen space. An alternative is to use a form CGI and represent the list as a popup menu of URLs.
Figure 1. The long list of sites to download the latest version of Netscape Navigator
Solving the problem using a CGI is fine, but I wanted to implement this popup solution as a Java applet instead. To do this, we need to take advantage of the existing Java applet model, as embodied in the java.applet.Applet class. To make this a little easier to understand, I developed the applet in two stages. Heres the first version:
import java.applet.*;
import java.awt.*;
import java.net.*;
public class popup extends Applet
{
private Choice urlChoices;
int numURLs;
public voidinit()
{
this.setBackground( Color.yellow );//java.awt.Color
this.setForeground( Color.red );//java.awt.Color
urlChoices = new Choice();
urlChoices.addItem( "www.metrowerks.com" );
urlChoices.addItem( "www.netscape.com" );
urlChoices.addItem( "www.mactech.com" );
urlChoices.setForeground( Color.green );
urlChoices.setBackground( Color.blue );
this.add( new Label( "Select URL: " ) );
this.add( urlChoices );
}
public boolean action( Event e, Object obj )
{
URL url;
if ( e.target == urlChoices )
{
try
{
url = new URL( obj.toString() );
}
catch( MalformedURLException err )
{
// Could print error message here.
return true;
}
this.getAppletContext().showDocument( url );
return true;
}
else
return super.action( e, obj );
}
}
Since Im using CodeWarrior, I started by creating a new project file called popup.µ using the Java applet stationery. Next, I created a new source code file named popup.java and added it to the project. I typed in my source code, saved it, then selected the .java file in the project window and compiled it. The compiler used my Java source code to produce a file named popup.class which contains the Java byte code that will be loaded by my HTML file.
Speaking of my HTML file, I used CodeWarrior to create a new file called popup.html and saved it in the same folder as my other project files. I didnt add the HTML file to the project. Heres the HTML:
<HTML>
<HEAD>
<TITLE>URL Launcher</TITLE>
</HEAD>
<BODY>
This applet creates a popup menu of URLs.
<H>R
<APPLET CODE="popup.class" WIDTH=300 HEIGHT=35>
</APPLET>
</BODY>
</HTML>
If youve never seen HTML before, its pretty straightforward. Basically, your HTML file contains the text you want to appear on your Web page, with a bunch of tags sprinkled throughout that define how the text looks. A tag is a pair of angle brackets that enclose a tag name, along with optional tag attributes. Some tags work in pairs. An end tag always starts with a '/'.
For example, to define the title of a particular page, youll start with the <title> tag, follow it with the title, then follow the text with the </title> tag. The <hr> tag places a horizontal rule on your page. Note that the <hr> tag doesnt require a corresponding </hr> tag, since there is no text that needs to be surrounded. By the way, HTML tags are case-insensitive.
The <applet> tag is the tag were interested in in this column. The CODE attribute specifies the name of the class file we want launched. In this case, we want the Java byte codes in popup.class executed.
Drag the HTML file on top of your favorite Java applet runner. If you have a Java-capable browser (like Netscape Atlas), you can open the HTML file and see how your applet looks on an actual Web page. Alternatively, if you run the applet using an applet runner, youll see the applet portion of the Web page only. Most applet runners only understand the applet tag and will ignore the rest of the HTML.
Figure 2 shows the popup applet as viewed in an early beta of Netscape Atlas. As long as your Internet connection is live, selecting a URL from the popup menu transfers you to that URL.
Figure 2. Our popup applet in action
Lets take a look at the source code...
The Popup.java Source Code
As I already mentioned, this applet extends the java.applet.Applet class. Go into the API Documentation folder that came with your development environment and use your Web browser to open the file java.applet.Applet.html. As youll see in the figure at the top of the page, java.applet.Applet is derived from java.awt.Panel, which is derived from java.awt.Container, which is derived from java.awt.Component. java.awt.Component is where all the action is. Thats where youll find all the interesting methods used by most applets. Spend a little quality time with java.awt.Component.html. It will definitely pay off.
The first few lines make the classes in java.applet.*, java.awt.*, and java.net.* available to our program. This allows us to refer to Applet instead of java.applet.Applet and URL instead of java.net.URL. You get the idea.
import java.applet.*;
import java.awt.*;
import java.net.*;
Our popup class extends the Applet class. We declare the variable urlChoices inside our class. urlChoices is a reference to a Choice (Javas term for a popup menu).
public class popup extends Applet
{
private Choice urlChoices;
The first method in our applet is the init() method. init() will be called automatically when our applet is started up. Well use this method to set up our Choice popup menu. Well start off by setting the applet background color to yellow and the foreground color to red. Note that this refers to the applet. If you take a look at Figure 2, youll see that the background is yellow and that the popup menus label text is red. I know yellow and red might not be a particularly good looking combination, but it does make the point, right?
public voidinit()
{
this.setBackground( Color.yellow );//java.awt.Color
this.setForeground( Color.red );//java.awt.Color
The next line uses new to create a new Choice object. Note that urlChoices is set to null until the new Choice is created.
urlChoices = new Choice();
Next, the Choice classes addItem() method is called to add three URLs to the project. The popup menus foreground and background colors are set, too, but these calls dont seem to have any effect.
urlChoices.addItem( "www.metrowerks.com" );
urlChoices.addItem( "www.netscape.com" );
urlChoices.addItem( "www.mactech.com" );
urlChoices.setForeground( Color.green );
urlChoices.setBackground( Color.blue );
Finally, a new Label object is created and the Label and Choice objects are added to the applet.
this.add( new Label( "Select URL: " ) );
this.add( urlChoices );
}
The action() method is called whenever an event occurs that relates to the applet. Well declare a URL reference to hold the URL we want to transfer to.
public boolean action( Event e, Object obj )
{
URL url;
The Event object e contains the event description we are responding to. Check out java.awt.Event to get a sense of Java event handling. e.target specifies the target object of the event. If the target is our popup menu, urlChoices, well handle the event. Otherwise, well pass the event on to our superclass and return whatever value super.action() returns.
If urlChoices was the target, well try creating a new URL object. Some Java objects require that you catch exceptions thrown by the object. To get a sense of this, go to the java.net.URL.html page. Note that all of the constructors throw the MalformedURLException error. You must catch this exception or your code wont compile. We ignore the error, but we could print an error message or, better yet, put up a dialog letting users know that the URL they selected isnt quite right.
If you throw an exception, youll create a new java.lang.exception object (or some class that extends throwable). But when you catch an exception, the exception object is passed to your catch block, as err was in this case. Take a look at java.lang.Throwable.html, especially the getMessage() method.
if ( e.target == urlChoices )
{
try
{
Since the event target is our popup menu, the object will be the menu itself. By calling the method toString() (inherited from Object), well get a string with the popup menus current setting. Well then use that string to initialize the URL.
url = new URL( obj.toString() );
}
catch( MalformedURLException err )
{
// Could print error message here.
return true;
}
Assuming the URL object was created okay, well call getAppletContext() to get our applets context, then use that object to call showDocument(), which will tell our browser to jump to a different URL.
this.getAppletContext().showDocument( url );
return true;
}
else
return super.action( e, obj );
}
}
Same Applet, Using Parameters
Heres a second version of the applet that gets its URLs from the HTML file. To see how this works, take a look at the second version of our HTML file:
<HTML>
<HEAD>
<TITLE>URL Launcher</TITLE>
</HEAD>
<BODY>
This applet takes a series of parameters and turns them into a popup
menu of URLs. The PARAM with the name numURLs tells you how many URLs
are included. PARAM "1" contains the 1st URL, PARAM "2" contains the
next URL, etc.
<H>R
<APPLET code="popup.class" width=300 height=35>
<PARAM name="numURLs" value="3">
<PARAM name="1" value="http://www.metrowerks.com">
<PARAM name="2" value="http://www.mactech.com">
<PARAM name="3" value="http://www.netscape.com">
</APPLET>
</BODY>
</HTML>
Notice that weve added a series of <PARAM> tags between the <APPLET> and </APPLET> tags. Each PARAM passes a named parameter to our applet. The numURLs parameter tells our applet how many URLs we are passing in. The 1 parameter contains the first URL, 2 contains the second URL, etc.
Heres the second version of our applet:
import java.applet.*;
import java.awt.*;
import java.net.*;
public class popup extends Applet
{
private Choice urlChoices;
public voidinit()
{
int numURLs = 0;
this.setBackground( Color.yellow );//java.awt.Color
this.setForeground( Color.red );//java.awt.Color
urlChoices = new Choice();
urlChoices.setForeground( Color.green );
urlChoices.setBackground( Color.blue );
String numURLsString = this.getParameter( "numURLs" );
if ( numURLsString == null )
{
this.showStatus( "null or missing numURLs Parameter!" );
}
else
{
try
{
numURLs = Integer.parseInt( numURLsString );
}
catch( NumberFormatException err )
{
numURLs = 0;
this.showStatus( "Bad numURLs Parameter: " +
numURLsString );
}
String urlString;
for ( int i=1; i<=numURLs; i++ )
{
urlString = this.getParameter( Integer.toString( i ) );
if ( urlString == null )
this.showStatus( "Missing Parameter: " + i );
else
urlChoices.addItem( urlString );
}
}
this.add( new Label( "Select URL: " ) );
this.add( urlChoices );
}
public boolean action( Event e, Object obj )
{
URL url;
if ( e.target == urlChoices )
{
try
{
url = new URL( obj.toString() );
}
catch( MalformedURLException err )
{
// Could print error message here.
return true;
}
this.getAppletContext().showDocument( url );
return true;
}
else
return super.action( e, obj );
}
}
Basically, the difference here is in this chunk of code from init():
String numURLsString = this.getParameter( "numURLs" );
if ( numURLsString == null )
{
this.showStatus( "null or missing numURLs Parameter!" );
}
else
{
try
{
numURLs = Integer.parseInt( numURLsString );
}
catch( NumberFormatException err )
{
numURLs = 0;
this.showStatus( "Bad numURLs Parameter: " +
numURLsString );
}
String urlString;
for ( int i=1; i<=numURLs; i++ )
{
urlString = this.getParameter( Integer.toString( i ) );
if ( urlString == null )
this.showStatus( "Missing Parameter: " + i );
else
urlChoices.addItem( urlString );
}
}
this.add( new Label( "Select URL: " ) );
this.add( urlChoices );
We start by calling getParameter() to retrieve the numURLs parameter. If the parameter is missing, we call showStatus() to display an error message in the browsers status message area. If the parameter was there, well try parsing an integer value from the string, placing the value in numURLs. Assuming we get a proper number, well step from 1 to numURLs.
Inside the loop, well convert the current index into a string (using Integer.toString()) and pass that string into getParameter(). If that parameter exists, well add that parameter into the popup menu.
Till Next Month...
Want more Java? Want more PowerPlant? Want some TCL? What the heck do you want? Send me some email. Let me know. Otherwise, Ill just keep writing about Java until I get sick of it! <g> By the way, I hope that you will take a moment and visit www.zumacafe.com, the Web site for a brand new internet cafe in Santa Fe, New Mexico. Zumas is the brainchild of Robin Williams, a brilliant writer and someone who has given a great deal to the Macintosh community. Check out the site and, if you get to Santa Fe, drop by Zumas and say hello!