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

Fresh From the Land Down Under – The Tou...
After a two week hiatus, we are back with another episode of The TouchArcade Show. Eli is fresh off his trip to Australia, which according to him is very similar to America but more upside down. Also kangaroos all over. Other topics this week... | Read more »
TouchArcade Game of the Week: ‘Dungeon T...
I’m a little conflicted on this week’s pick. Pretty much everyone knows the legend of Dungeon Raid, the match-3 RPG hybrid that took the world by storm way back in 2011. Everyone at the time was obsessed with it, but for whatever reason the... | Read more »
SwitchArcade Round-Up: Reviews Featuring...
Hello gentle readers, and welcome to the SwitchArcade Round-Up for July 19th, 2024. In today’s article, we finish up the week with the unusual appearance of a review. I’ve spent my time with Hot Lap Racing, and I’m ready to give my verdict. After... | Read more »
Draknek Interview: Alan Hazelden on Thin...
Ever since I played my first release from Draknek & Friends years ago, I knew I wanted to sit down with Alan Hazelden and chat about the team, puzzle games, and much more. | Read more »
The Latest ‘Marvel Snap’ OTA Update Buff...
I don’t know about all of you, my fellow Marvel Snap (Free) players, but these days when I see a balance update I find myself clenching my… teeth and bracing for the impact to my decks. They’ve been pretty spicy of late, after all. How will the... | Read more »
‘Honkai Star Rail’ Version 2.4 “Finest D...
HoYoverse just announced the Honkai Star Rail (Free) version 2.4 “Finest Duel Under the Pristine Blue" update alongside a surprising collaboration. Honkai Star Rail 2.4 follows the 2.3 “Farewell, Penacony" update. Read about that here. | Read more »
‘Vampire Survivors+’ on Apple Arcade Wil...
Earlier this month, Apple revealed that poncle’s excellent Vampire Survivors+ () would be heading to Apple Arcade as a new App Store Great. I reached out to poncle to check in on the DLC for Vampire Survivors+ because only the first two DLCs were... | Read more »
Homerun Clash 2: Legends Derby opens for...
Since launching in 2018, Homerun Clash has performed admirably for HAEGIN, racking up 12 million players all eager to prove they could be the next baseball champions. Well, the title will soon be up for grabs again, as Homerun Clash 2: Legends... | Read more »
‘Neverness to Everness’ Is a Free To Pla...
Perfect World Games and Hotta Studio (Tower of Fantasy) announced a new free to play open world RPG in the form of Neverness to Everness a few days ago (via Gematsu). Neverness to Everness has an urban setting, and the two reveal trailers for it... | Read more »
Meditative Puzzler ‘Ouros’ Coming to iOS...
Ouros is a mediative puzzle game from developer Michael Kamm that launched on PC just a couple of months back, and today it has been revealed that the title is now heading to iOS and Android devices next month. Which is good news I say because this... | Read more »

Price Scanner via MacPrices.net

Amazon is still selling 16-inch MacBook Pros...
Prime Day in July is over, but Amazon is still selling 16-inch Apple MacBook Pros for $500-$600 off MSRP. Shipping is free. These are the lowest prices available this weekend for new 16″ Apple... Read more
Walmart continues to sell clearance 13-inch M...
Walmart continues to offer clearance, but new, Apple 13″ M1 MacBook Airs (8GB RAM, 256GB SSD) online for $699, $300 off original MSRP, in Space Gray, Silver, and Gold colors. These are new MacBooks... Read more
Apple is offering steep discounts, up to $600...
Apple has standard-configuration 16″ M3 Max MacBook Pros available, Certified Refurbished, starting at $2969 and ranging up to $600 off MSRP. Each model features a new outer case, shipping is free,... Read more
Save up to $480 with these 14-inch M3 Pro/M3...
Apple has 14″ M3 Pro and M3 Max MacBook Pros in stock today and available, Certified Refurbished, starting at $1699 and ranging up to $480 off MSRP. Each model features a new outer case, shipping is... Read more
Amazon has clearance 9th-generation WiFi iPad...
Amazon has Apple’s 9th generation 10.2″ WiFi iPads on sale for $80-$100 off MSRP, starting only $249. Their prices are the lowest available for new iPads anywhere: – 10″ 64GB WiFi iPad (Space Gray or... Read more
Apple is offering a $50 discount on 2nd-gener...
Apple has Certified Refurbished White and Midnight HomePods available for $249, Certified Refurbished. That’s $50 off MSRP and the lowest price currently available for a full-size Apple HomePod today... Read more
The latest MacBook Pro sale at Amazon: 16-inc...
Amazon is offering instant discounts on 16″ M3 Pro and 16″ M3 Max MacBook Pros ranging up to $400 off MSRP as part of their early July 4th sale. Shipping is free. These are the lowest prices... Read more
14-inch M3 Pro MacBook Pros with 36GB of RAM...
B&H Photo has 14″ M3 Pro MacBook Pros with 36GB of RAM and 512GB or 1TB SSDs in stock today and on sale for $200 off Apple’s MSRP, each including free 1-2 day shipping: – 14″ M3 Pro MacBook Pro (... Read more
14-inch M3 MacBook Pros with 16GB of RAM on s...
B&H Photo has 14″ M3 MacBook Pros with 16GB of RAM and 512GB or 1TB SSDs in stock today and on sale for $150-$200 off Apple’s MSRP, each including free 1-2 day shipping: – 14″ M3 MacBook Pro (... Read more
Amazon is offering $170-$200 discounts on new...
Amazon is offering a $170-$200 discount on every configuration and color of Apple’s M3-powered 15″ MacBook Airs. Prices start at $1129 for models with 8GB of RAM and 256GB of storage: – 15″ M3... Read more

Jobs Board

*Apple* Systems Engineer - Chenega Corporati...
…LLC,** a **Chenega Professional Services** ' company, is looking for a ** Apple Systems Engineer** to support the Information Technology Operations and Maintenance Read more
Solutions Engineer - *Apple* - SHI (United...
**Job Summary** An Apple Solution Engineer's primary role is tosupport SHI customers in their efforts to select, deploy, and manage Apple operating systems and Read more
*Apple* / Mac Administrator - JAMF Pro - Ame...
Amentum is seeking an ** Apple / Mac Administrator - JAMF Pro** to provide support with the Apple Ecosystem to include hardware and software to join our team and Read more
Operations Associate - *Apple* Blossom Mall...
Operations Associate - Apple Blossom Mall Location:Winchester, VA, United States (https://jobs.jcp.com/jobs/location/191170/winchester-va-united-states) - Apple Read more
Cashier - *Apple* Blossom Mall - JCPenney (...
Cashier - Apple Blossom Mall Location:Winchester, VA, United States (https://jobs.jcp.com/jobs/location/191170/winchester-va-united-states) - Apple Blossom Mall Read more
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.