// MainFrame - run an Applet as an application
//
// Copyright (C) 1996 by Jef Poskanzer <jef@acme.com>.  All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions
// are met:
// 1. Redistributions of source code must retain the above copyright
//    notice, this list of conditions and the following disclaimer.
// 2. Redistributions in binary form must reproduce the above copyright
//    notice, this list of conditions and the following disclaimer in the
//    documentation and/or other materials provided with the distribution.
//
// THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
// OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
// HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
// LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
// OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
// SUCH DAMAGE.
//
// Visit the ACME Labs Java page for up-to-date versions of this and other
// fine Java utilities: http://www.acme.com/java/
//
// PS DON'T COPY THIS VERSION - TAKE JEF'S INSTEAD.
// I HACKED IT TO NOT REQUIRE sun.* CLASSES THAT DON'T EXIST ON MY MACHINE
// BUT LOST AUDIO FUNCTIONALITY IN THE PROCESS.
// D. Eppstein, UC Irvine, http://www.ics.uci.edu/~eppstein/, 25 Jun 1997

package Acme;

import java.applet.*;
import java.awt.*;
import java.awt.image.*;
import java.net.*;
import java.io.*;
import java.util.*;

/// Run an Applet as an application.
// <P>
// Using this class you can add a trivial main program to any Applet
// and run it directly, as well as from a browser or the appletviewer.
// And unlike some versions of this concept, MainFrame implements both
// images and sound.
// <P>
// Sample main program:
// <BLOCKQUOTE><PRE>
// public static void main( String[] args )
//     {
//     new Acme.MainFrame( new ThisApplet(), args, 400, 400 );
//     }
// </PRE></BLOCKQUOTE>
// The only methods you need to know about are the constructors.
// <P>
// You can specify Applet parameters on the command line, as name=value.
// For instance, the equivalent of:
// <BLOCKQUOTE><PRE>
// &lt;PARAM NAME="pause" VALUE="200"&gt;
// </PRE></BLOCKQUOTE>
// would just be:
// <BLOCKQUOTE><PRE>
// pause=200
// </PRE></BLOCKQUOTE>
// You can also specify three special parameters:
// <BLOCKQUOTE><PRE>
// width=N          Width of the Applet.
// height=N         Height of the Applet.
// barebones=true   Leave off the menu bar and status area.
// </PRE></BLOCKQUOTE>
// <P>
// <A HREF="/resources/classes/Acme/MainFrame.java">Fetch the software.</A><BR>
// <A HREF="/resources/classes/Acme.tar.Z">Fetch the entire Acme package.</A>

public class MainFrame extends Frame

    implements Runnable, AppletStub, AppletContext
    {

    private String[] args = null;
    private static int instances = 0;
    private String name;
    private boolean barebones = false;
    private Applet applet;
    private Label label = null;
    private Dimension appletSize;

    private static final String PARAM_PROP_PREFIX = "parameter.";

    /// Constructor with everything specified.
    public MainFrame(
	Applet applet, String[] args, int width, int height )
	{
	build( applet, args, width, height );
	}

    /// Constructor with no default width/height.
    public MainFrame( Applet applet, String[] args )
	{
	build( applet, args, -1, -1 );
	}

    /// Constructor with no arg parsing.
    public MainFrame( Applet applet, int width, int height )
	{
	build( applet, null, width, height );
	}

// Use menu item not name, allow caller to change strings -- DE
public static MenuItem quitItem = null;
public static MenuItem closeItem = null;
public static MenuItem newItem = null;
public static MenuItem undoItem = null;
public static MenuItem cloneItem = null;

    // Internal constructor routine.
    private void build(
	Applet applet, String[] args, int width, int height )
	{
	++instances;
	this.applet = applet;
	this.args = args;
	applet.setStub( this );
	name = applet.getClass().getName();
	setTitle( name );

	// Set up properties.
	Properties props = System.getProperties();
	props.put( "browser", "Acme.MainFrame" );
	props.put( "browser.version", "11jul96" );
	props.put( "browser.vendor", "Acme Laboratories" );
	props.put( "browser.vendor.url", "http://www.acme.com/" );

	// Turn args into parameters by way of the properties list.
	if ( args != null )
	    parseArgs( args, props );

	// If width and height are specified in the parameters, override
	// the compiled-in values.
	String widthStr = getParameter( "width" );
	if ( widthStr != null )
	    width = Integer.parseInt( widthStr );
	String heightStr = getParameter( "height" );
	if ( heightStr != null )
	    height = Integer.parseInt( heightStr );

	// Were width and height specified somewhere?
	if ( width == -1 || height == -1 )
	    {
	    System.err.println( "Width and height must be specified." );
	    return;
	    }

	// Do we want to run bare-bones?
	String bonesStr = getParameter( "barebones" );
	if ( bonesStr != null && bonesStr.equals( "true" ) )
	    barebones = true;

	if ( ! barebones )
	    {
	    // Make menu bar.
	    MenuBar mb = new MenuBar();
// Make more maclike -- DE
//	    Menu m = new Menu( "Applet" );
//	    m.add( new MenuItem( "Restart" ) );
//	    m.add( new MenuItem( "Clone" ) );
//	    m.add( new MenuItem( "Close" ) );
//	    m.add( new MenuItem( "Quit" ) );
Menu m = new Menu("File");
if (quitItem == null) quitItem = new MenuItem("Quit");
//if (closeItem == null) closeItem = new MenuItem("Close");
//if (cloneItem == null) cloneItem = new MenuItem("Clone");
if (newItem == null) newItem = new MenuItem("Restart");
m.add(quitItem);
//m.add(closeItem);
//m.add(cloneItem);
m.add(newItem);
mb.add(m);
m = new Menu("Edit");
if (undoItem == null) undoItem = new MenuItem("Undo");
m.add(undoItem);
	    mb.add( m );
	    setMenuBar( mb );
	    }

	// Lay out components.
	setLayout( new BorderLayout() );
	add( "Center", applet );
	if ( ! barebones )
	    {
	    Panel borderPanel =
		new Acme.Widgets.BorderPanel( Acme.Widgets.BorderPanel.RAISED );
	    borderPanel.setLayout( new BorderLayout() );
	    label = new Label( "" );
	    borderPanel.add( "Center", label );
	    add( "South", borderPanel );
	    }

	// Set up size.
	pack();
	validate();
	appletSize = applet.size();
	applet.resize( width, height );
	show();

	// Start a separate thread to call the applet's init() and start()
	// methods, in case they take a long time.
	(new Thread( this )).start();
	}
    
    // Turn command-line arguments into Applet parameters, by way of the
    // properties list.
    private static void parseArgs( String[] args, Properties props )
	{
	for ( int i = 0; i < args.length; ++i )
	    {
	    String arg = args[i];
	    int ind = arg.indexOf( '=' );
	    if ( ind == -1 )
		props.put( PARAM_PROP_PREFIX + arg.toLowerCase(), "" );
	    else
		props.put(
		    PARAM_PROP_PREFIX + arg.substring( 0, ind ).toLowerCase(),
		    arg.substring( ind + 1 ) );
	    }
	}

    /// Event handler for the menu bar.
    public boolean handleEvent( Event evt )
	{
	switch ( evt.id )
	    {
	    case Event.ACTION_EVENT:
// Cleaner menu handling -- DE
MenuItem m = (MenuItem) evt.target;
if (m == newItem)
//	    if ( evt.arg.equals( "Restart" ) )
		{
		applet.stop();
//		applet.destroy();
//		Thread thread = new Thread( this );
//		thread.start();
applet.start();
		}
else if (m == cloneItem)
//	    else if ( evt.arg.equals( "Clone" ) )
		{
		try
		    {
		    new MainFrame(
			(Applet) applet.getClass().newInstance(), args,
			appletSize.width, appletSize.height );
		    }
		catch ( IllegalAccessException e )
		    {
		    showStatus( e.getMessage() );
		    }
		catch ( InstantiationException e )
		    {
		    showStatus( e.getMessage() );
		    }
		}
//	    else if ( evt.arg.equals( "Close" ) )
else if (m == closeItem)
		{
		hide();
		remove( applet );
		applet.stop();
		applet.destroy();
		if ( label != null )
		    remove( label );
		dispose();
		--instances;
		if ( instances == 0 )
		    System.exit( 0 );
		}
else if (m == quitItem)
//	    else if ( evt.arg.equals( "Quit" ) )
		System.exit( 0 );
else if (m == undoItem) ((Undo)applet).undo();
	    break;

	    case Event.WINDOW_DESTROY:
	    System.exit( 0 );
	    break;
	    }
	return super.handleEvent( evt );
	}
    

    // Methods from Runnable.

    /// Separate thread to call the applet's init() and start() methods.
    public void run()
	{
// I don't want this crap on the status line [DE]
//	showStatus( name + " initializing..." );
	applet.init();
	validate();
//	showStatus( name + " starting..." );
	applet.start();
	validate();
//	showStatus( name + " running..." );
	}


    // Methods from AppletStub.

    public boolean isActive()
	{
	return true;
	}
    
    public URL getDocumentBase()
	{
	// Returns the current directory.
	String dir = System.getProperty( "user.dir" );
	String urlDir = dir.replace( File.separatorChar, '/' );
	try
	    {
// DE 25 Sep 1997:
// If urlDir already ends in a slash, don't add another one.
//
//		    return new URL( "file:" + urlDir + "/" );
if (!urlDir.endsWith("/")) urlDir = urlDir + "/";
return new URL( "file:" + urlDir);
	    }
	catch ( MalformedURLException e )
	    {
	    return null;
	    }
	}
    
    public URL getCodeBase()
	{
	// Hack: loop through each item in CLASSPATH, checking if
	// the appropriately named .class file exists there.  But
	// this doesn't account for .zip files.
	String path = System.getProperty( "java.class.path" );
	Enumeration st = new StringTokenizer( path, ":" );
	while ( st.hasMoreElements() )
	    {
	    String dir = (String) st.nextElement();
	    String filename = dir + File.separatorChar + name + ".class";
	    File file = new File( filename );
	    if ( file.exists() )
		{
		String urlDir = dir.replace( File.separatorChar, '/' );
		try
		    {
// DE 25 Sep 1997:
// If urlDir already ends in a slash, don't add another one.
//
//		    return new URL( "file:" + urlDir + "/" );
if (!urlDir.endsWith("/")) urlDir = urlDir + "/";
return new URL( "file:" + urlDir);
		    }
		catch ( MalformedURLException e )
		    {
		    return null;
		    }
		}
	    }
	return null;
	}
    
    public String getParameter( String name )
	{
	// Return a parameter via the munged names in the properties list.
	return System.getProperty( PARAM_PROP_PREFIX + name.toLowerCase() );
	}
    
    public void appletResize( int width, int height )
	{
	// Change the frame's size by the same amount that the applet's
	// size is changing.
	Dimension frameSize = size();
	frameSize.width += width - appletSize.width;
	frameSize.height += height - appletSize.height;
	resize( frameSize );
	appletSize = applet.size();
	}

    public AppletContext getAppletContext()
	{
	return this;
	}
    

    // Methods from AppletContext.

    public AudioClip getAudioClip( URL url )
	{
	// This is an internal undocumented routine.  However, it
	// also provides needed functionality not otherwise available.
	// I suspect that in a future release, JavaSoft will add an
	// audio content handler which encapsulates this, and then
	// we can just do a getContent just like for images.
//	return new sun.applet.AppletAudioClip( url );
// not available on the mac - fortunately not needed
// D. Eppstein, 25 Jun 1997
return null;
	}

   public Image getImage( URL url )
	{
		Toolkit tk = Toolkit.getDefaultToolkit();

// Attempt to avoid fileAccess message to Java Console in MRJ 2.0 - DE
		String s = url.toString();
		if (s.startsWith("file:"))
			return tk.getImage(s.substring(5));
		else return tk.getImage(url);

//	try
//	    {
//	    ImageProducer prod = (ImageProducer) url.getContent();
//	    return tk.createImage( prod );
//	    }
//	catch ( IOException e )
//	    {
//	    return null;
//	    }
	}
    
    public Applet getApplet( String name )
	{
	// Returns this Applet or nothing.
	if ( name.equals( this.name ) )
	    return applet;
	return null;
	}
    
    public Enumeration getApplets()
	{
	// Just yields this applet.
	Vector v = new Vector();
	v.addElement( applet );
	return v.elements();
	}
    
    public void showDocument( URL url )
	{
	// Ignore.
	}
    
    public void showDocument( URL url, String target )
	{
	// Ignore.
	}
    
    public void showStatus( String status )
	{
	if ( label != null )
	    label.setText( status );
	}

    }
