TweetFollow Us on Twitter

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 month’s column, I’ve been consumed with Java. I’ve 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 (O’Reilly & 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 Month’s Applets

One important use of Java is to extend the functionality of a Web site, beyond the capabilities offered by HTML. If you’ve ever been to Netscape’s Web site, you’ve probably encountered the long list of download sites that allow you to download the latest version of a Netscape product (Figure 1).

If you’ve 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. Here’s 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 I’m 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 didn’t add the HTML file to the project. Here’s 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 you’ve never seen HTML before, it’s 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, you’ll 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 doesn’t 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 we’re 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, you’ll 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

Let’s 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 you’ll 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. That’s where you’ll 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 (Java’s 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. We’ll use this method to set up our Choice popup menu. We’ll 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, you’ll see that the background is yellow and that the popup menu’s 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 menu’s foreground and background colors are set, too, but these calls don’t 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. We’ll 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, we’ll handle the event. Otherwise, we’ll pass the event on to our superclass and return whatever value super.action() returns.

If urlChoices was the target, we’ll 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 won’t 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 isn’t quite right.

If you throw an exception, you’ll 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), we’ll get a string with the popup menu’s current setting. We’ll 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, we’ll call getAppletContext() to get our applet’s 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

Here’s 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 we’ve 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.

Here’s 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 browser’s status message area. If the parameter was there, we’ll try parsing an integer value from the string, placing the value in numURLs. Assuming we get a proper number, we’ll step from 1 to numURLs.

Inside the loop, we’ll convert the current index into a string (using Integer.toString()) and pass that string into getParameter(). If that parameter exists, we’ll 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, I’ll 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. Zuma’s 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 Zuma’s and say hello!

 

Community Search:
MacTech Search:

Software Updates via MacUpdate

Latest Forum Discussions

See All

Top Mobile Game Discounts
Every day, we pick out a curated list of the best mobile discounts on the App Store and post them here. This list won't be comprehensive, but it every game on it is recommended. Feel free to check out the coverage we did on them in the links... | Read more »
Price of Glory unleashes its 1.4 Alpha u...
As much as we all probably dislike Maths as a subject, we do have to hand it to geometry for giving us the good old Hexgrid, home of some of the best strategy games. One such example, Price of Glory, has dropped its 1.4 Alpha update, stocked full... | Read more »
The SLC 2025 kicks off this month to cro...
Ever since the Solo Leveling: Arise Championship 2025 was announced, I have been looking forward to it. The promotional clip they released a month or two back showed crowds going absolutely nuts for the previous competitions, so imagine the... | Read more »
Dive into some early Magicpunk fun as Cr...
Excellent news for fans of steampunk and magic; the Precursor Test for Magicpunk MMORPG Crystal of Atlan opens today. This rather fancy way of saying beta test will remain open until March 5th and is available for PC - boo - and Android devices -... | Read more »
Prepare to get your mind melted as Evang...
If you are a fan of sci-fi shooters and incredibly weird, mind-bending anime series, then you are in for a treat, as Goddess of Victory: Nikke is gearing up for its second collaboration with Evangelion. We were also treated to an upcoming... | Read more »
Square Enix gives with one hand and slap...
We have something of a mixed bag coming over from Square Enix HQ today. Two of their mobile games are revelling in life with new events keeping them alive, whilst another has been thrown onto the ever-growing discard pile Square is building. I... | Read more »
Let the world burn as you have some fest...
It is time to leave the world burning once again as you take a much-needed break from that whole “hero” lark and enjoy some celebrations in Genshin Impact. Version 5.4, Moonlight Amidst Dreams, will see you in Inazuma to attend the Mikawa Flower... | Read more »
Full Moon Over the Abyssal Sea lands on...
Aether Gazer has announced its latest major update, and it is one of the loveliest event names I have ever heard. Full Moon Over the Abyssal Sea is an amazing name, and it comes loaded with two side stories, a new S-grade Modifier, and some fancy... | Read more »
Open your own eatery for all the forest...
Very important question; when you read the title Zoo Restaurant, do you also immediately think of running a restaurant in which you cook Zoo animals as the course? I will just assume yes. Anyway, come June 23rd we will all be able to start up our... | Read more »
Crystal of Atlan opens registration for...
Nuverse was prominently featured in the last month for all the wrong reasons with the USA TikTok debacle, but now it is putting all that behind it and preparing for the Crystal of Atlan beta test. Taking place between February 18th and March 5th,... | Read more »

Price Scanner via MacPrices.net

AT&T is offering a 65% discount on the ne...
AT&T is offering the new iPhone 16e for up to 65% off their monthly finance fee with 36-months of service. No trade-in is required. Discount is applied via monthly bill credits over the 36 month... Read more
Use this code to get a free iPhone 13 at Visi...
For a limited time, use code SWEETDEAL to get a free 128GB iPhone 13 Visible, Verizon’s low-cost wireless cell service, Visible. Deal is valid when you purchase the Visible+ annual plan. Free... Read more
M4 Mac minis on sale for $50-$80 off MSRP at...
B&H Photo has M4 Mac minis in stock and on sale right now for $50 to $80 off Apple’s MSRP, each including free 1-2 day shipping to most US addresses: – M4 Mac mini (16GB/256GB): $549, $50 off... Read more
Buy an iPhone 16 at Boost Mobile and get one...
Boost Mobile, an MVNO using AT&T and T-Mobile’s networks, is offering one year of free Unlimited service with the purchase of any iPhone 16. Purchase the iPhone at standard MSRP, and then choose... Read more
Get an iPhone 15 for only $299 at Boost Mobil...
Boost Mobile, an MVNO using AT&T and T-Mobile’s networks, is offering the 128GB iPhone 15 for $299.99 including service with their Unlimited Premium plan (50GB of premium data, $60/month), or $20... Read more
Unreal Mobile is offering $100 off any new iP...
Unreal Mobile, an MVNO using AT&T and T-Mobile’s networks, is offering a $100 discount on any new iPhone with service. This includes new iPhone 16 models as well as iPhone 15, 14, 13, and SE... Read more
Apple drops prices on clearance iPhone 14 mod...
With today’s introduction of the new iPhone 16e, Apple has discontinued the iPhone 14, 14 Pro, and SE. In response, Apple has dropped prices on unlocked, Certified Refurbished, iPhone 14 models to a... Read more
B&H has 16-inch M4 Max MacBook Pros on sa...
B&H Photo is offering a $360-$410 discount on new 16-inch MacBook Pros with M4 Max CPUs right now. B&H offers free 1-2 day shipping to most US addresses: – 16″ M4 Max MacBook Pro (36GB/1TB/... Read more
Amazon is offering a $100 discount on the M4...
Amazon has the M4 Pro Mac mini discounted $100 off MSRP right now. Shipping is free. Their price is the lowest currently available for this popular mini: – Mac mini M4 Pro (24GB/512GB): $1299, $100... Read more
B&H continues to offer $150-$220 discount...
B&H Photo has 14-inch M4 MacBook Pros on sale for $150-$220 off MSRP. B&H offers free 1-2 day shipping to most US addresses: – 14″ M4 MacBook Pro (16GB/512GB): $1449, $150 off MSRP – 14″ M4... Read more

Jobs Board

All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.