BrowserLauncher.java |
package ij.plugin; import ij.IJ; import java.io.File; import java.io.FileNotFoundException; import java.io.IOException; import java.lang.reflect.Constructor; import java.lang.reflect.Field; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.net.URL; import java.net.URLClassLoader; /** * This plugin implements the File/Import/URL command and the commands in the Help menu that * open web pages. It is based on Eric Albert's cross-platform (Windows, Mac OS X and Unix) * BrowserLauncher class. * <p> * BrowserLauncher is a class that provides one static method, openURL, which opens the default * web browser for the current user of the system to the given URL. It may support other * protocols depending on the system -- mailto, ftp, etc. -- but that has not been rigorously * tested and is not guaranteed to work. * <p> * Yes, this is platform-specific code, and yes, it may rely on classes on certain platforms * that are not part of the standard JDK. What we're trying to do, though, is to take something * that's frequently desirable but inherently platform-specific -- opening a default browser -- * and allow programmers (you, for example) to do so without worrying about dropping into native * code or doing anything else similarly evil. * <p> * Anyway, this code is completely in Java and will run on all JDK 1.1-compliant systems without * modification or a need for additional libraries. All classes that are required on certain * platforms to allow this to run are dynamically loaded at runtime via reflection and, if not * found, will not cause this to do anything other than returning an error when opening the * browser. * <p> * This code is Copyright 1999-2001 by Eric Albert (ejalbert@cs.stanford.edu) and may be * redistributed or modified in any form without restrictions as long as the portion of this * comment from this paragraph through the end of the comment is not removed. The author * requests that he be notified of any application, applet, or other binary that makes use of * this code, but that's more out of curiosity than anything and is not required. This software * includes no warranty. The author is not repsonsible for any loss of data or functionality * or any adverse or unexpected effects of using this software. * <p> * Credits: * <br>Steven Spencer, JavaWorld magazine (<a href="http://www.javaworld.com/javaworld/javatips/jw-javatip66.html">Java Tip 66</a>) * <br>Thanks also to Ron B. Yeh, Eric Shapiro, Ben Engber, Paul Teitlebaum, Andrea Cantatore, * Larry Barowski, Trevor Bedzek, Frank Miedrich, and Ron Rabakukk * * @author Eric Albert (<a href="mailto:ejalbert@cs.stanford.edu">ejalbert@cs.stanford.edu</a>) * @version 1.4b1 (Released June 20, 2001) */ public class BrowserLauncher implements PlugIn { /** The com.apple.mrj.MRJFileUtils class */ private static Class mrjFileUtilsClass; /** The openURL method of com.apple.mrj.MRJFileUtils */ private static Method openURL; private static boolean error; static {loadClasses();} /** Opens the specified URL (default is the ImageJ home page). */ public void run(String theURL) { if (error) return; if (theURL==null || theURL.equals("")) theURL = IJ.URL; try {openURL(theURL);} catch (IOException e) {} } /** * Attempts to open the default web browser to the given URL. * @param url The URL to open * @throws IOException If the web browser could not be located or does not run */ public static void openURL(String url) throws IOException { String errorMessage = ""; if (IJ.isMacOSX()) { if (IJ.isJava16()) IJ.runMacro("exec('open', getArgument())",url); else { try { Method aMethod = mrjFileUtilsClass.getDeclaredMethod("sharedWorkspace", new Class[] {}); Object aTarget = aMethod.invoke( mrjFileUtilsClass, new Object[] {}); openURL.invoke(aTarget, new Object[] { new java.net.URL( url )}); } catch (Exception e) { errorMessage = ""+e; } } } else if (IJ.isWindows()) { String cmd = "rundll32 url.dll,FileProtocolHandler " + url; if (System.getProperty("os.name").startsWith("Windows 2000")) cmd = "rundll32 shell32.dll,ShellExec_RunDLL " + url; Process process = Runtime.getRuntime().exec(cmd); // This avoids a memory leak on some versions of Java on Windows. // That's hinted at in <http://developer.java.sun.com/developer/qow/archive/68/>. try { process.waitFor(); process.exitValue(); } catch (InterruptedException ie) { throw new IOException("InterruptedException while launching browser: " + ie.getMessage()); } } else { // Assume Linux or Unix // Based on BareBonesBrowserLaunch (http://www.centerkey.com/java/browser/) // The utility 'xdg-open' launches the URL in the user's preferred browser, // therefore we try to use it first, before trying to discover other browsers. String[] browsers = {"xdg-open", "netscape", "firefox", "konqueror", "mozilla", "opera", "epiphany", "lynx" }; String browserName = null; try { for (int count=0; count<browsers.length && browserName==null; count++) { String[] c = new String[] {"which", browsers[count]}; if (Runtime.getRuntime().exec(c).waitFor()==0) browserName = browsers[count]; } if (browserName==null) ij.IJ.error("BrowserLauncher", "Could not find a browser"); else Runtime.getRuntime().exec(new String[] {browserName, url}); } catch (Exception e) { throw new IOException("Exception while launching browser: " + e.getMessage()); } } } /** * Called by a static initializer to load any classes, fields, and methods * required at runtime to locate the user's web browser. */ private static void loadClasses() { if (IJ.isMacOSX() && !IJ.isJava16()) { try { if (new File("/System/Library/Java/com/apple/cocoa/application/NSWorkspace.class").exists()) { ClassLoader classLoader = new URLClassLoader(new URL[]{new File("/System/Library/Java").toURL()}); mrjFileUtilsClass = Class.forName("com.apple.cocoa.application.NSWorkspace", true, classLoader); } else mrjFileUtilsClass = Class.forName("com.apple.cocoa.application.NSWorkspace"); openURL = mrjFileUtilsClass.getDeclaredMethod("openURL", new Class[] { java.net.URL.class }); } catch (Exception e) { IJ.log("BrowserLauncher"+e); error = true; } } } }