/* |
|
* Copyright (c) 2005, 2018, Oracle and/or its affiliates. All rights reserved. |
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
|
* |
|
* This code is free software; you can redistribute it and/or modify it |
|
* under the terms of the GNU General Public License version 2 only, as |
|
* published by the Free Software Foundation. Oracle designates this |
|
* particular file as subject to the "Classpath" exception as provided |
|
* by Oracle in the LICENSE file that accompanied this code. |
|
* |
|
* This code is distributed in the hope that it will be useful, but WITHOUT |
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
|
* version 2 for more details (a copy is included in the LICENSE file that |
|
* accompanied this code). |
|
* |
|
* You should have received a copy of the GNU General Public License version |
|
* 2 along with this work; if not, write to the Free Software Foundation, |
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. |
|
* |
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
|
* or visit www.oracle.com if you need additional information or have any |
|
* questions. |
|
*/ |
|
package java.awt; |
|
import java.io.File; |
|
import java.io.IOException; |
|
import java.net.URISyntaxException; |
|
import java.net.URI; |
|
import java.net.URL; |
|
import java.net.MalformedURLException; |
|
import java.awt.AWTPermission; |
|
import java.awt.GraphicsEnvironment; |
|
import java.awt.HeadlessException; |
|
import java.awt.peer.DesktopPeer; |
|
import sun.awt.SunToolkit; |
|
import sun.awt.HeadlessToolkit; |
|
import java.io.FilePermission; |
|
import sun.security.util.SecurityConstants; |
|
/** |
|
* The {@code Desktop} class allows a Java application to launch |
|
* associated applications registered on the native desktop to handle |
|
* a {@link java.net.URI} or a file. |
|
* |
|
* <p> Supported operations include: |
|
* <ul> |
|
* <li>launching the user-default browser to show a specified |
|
* URI;</li> |
|
* <li>launching the user-default mail client with an optional |
|
* {@code mailto} URI;</li> |
|
* <li>launching a registered application to open, edit or print a |
|
* specified file.</li> |
|
* </ul> |
|
* |
|
* <p> This class provides methods corresponding to these |
|
* operations. The methods look for the associated application |
|
* registered on the current platform, and launch it to handle a URI |
|
* or file. If there is no associated application or the associated |
|
* application fails to be launched, an exception is thrown. |
|
* |
|
* <p> An application is registered to a URI or file type; for |
|
* example, the {@code "sxi"} file extension is typically registered |
|
* to StarOffice. The mechanism of registering, accessing, and |
|
* launching the associated application is platform-dependent. |
|
* |
|
* <p> Each operation is an action type represented by the {@link |
|
* Desktop.Action} class. |
|
* |
|
* <p> Note: when some action is invoked and the associated |
|
* application is executed, it will be executed on the same system as |
|
* the one on which the Java application was launched. |
|
* |
|
* @since 1.6 |
|
* @author Armin Chen |
|
* @author George Zhang |
|
*/ |
|
public class Desktop { |
|
/** |
|
* Represents an action type. Each platform supports a different |
|
* set of actions. You may use the {@link Desktop#isSupported} |
|
* method to determine if the given action is supported by the |
|
* current platform. |
|
* @see java.awt.Desktop#isSupported(java.awt.Desktop.Action) |
|
* @since 1.6 |
|
*/ |
|
public static enum Action { |
|
/** |
|
* Represents an "open" action. |
|
* @see Desktop#open(java.io.File) |
|
*/ |
|
OPEN, |
|
/** |
|
* Represents an "edit" action. |
|
* @see Desktop#edit(java.io.File) |
|
*/ |
|
EDIT, |
|
/** |
|
* Represents a "print" action. |
|
* @see Desktop#print(java.io.File) |
|
*/ |
|
PRINT, |
|
/** |
|
* Represents a "mail" action. |
|
* @see Desktop#mail() |
|
* @see Desktop#mail(java.net.URI) |
|
*/ |
|
MAIL, |
|
/** |
|
* Represents a "browse" action. |
|
* @see Desktop#browse(java.net.URI) |
|
*/ |
|
BROWSE |
|
}; |
|
private DesktopPeer peer; |
|
/** |
|
* Suppresses default constructor for noninstantiability. |
|
*/ |
|
private Desktop() { |
|
peer = Toolkit.getDefaultToolkit().createDesktopPeer(this); |
|
} |
|
/** |
|
* Returns the <code>Desktop</code> instance of the current |
|
* browser context. On some platforms the Desktop API may not be |
|
* supported; use the {@link #isDesktopSupported} method to |
|
* determine if the current desktop is supported. |
|
* @return the Desktop instance of the current browser context |
|
* @throws HeadlessException if {@link |
|
* GraphicsEnvironment#isHeadless()} returns {@code true} |
|
* @throws UnsupportedOperationException if this class is not |
|
* supported on the current platform |
|
* @see #isDesktopSupported() |
|
* @see java.awt.GraphicsEnvironment#isHeadless |
|
*/ |
|
public static synchronized Desktop getDesktop(){ |
|
if (GraphicsEnvironment.isHeadless()) throw new HeadlessException(); |
|
if (!Desktop.isDesktopSupported()) { |
|
throw new UnsupportedOperationException("Desktop API is not " + |
|
"supported on the current platform"); |
|
} |
|
sun.awt.AppContext context = sun.awt.AppContext.getAppContext(); |
|
Desktop desktop = (Desktop)context.get(Desktop.class); |
|
if (desktop == null) { |
|
desktop = new Desktop(); |
|
context.put(Desktop.class, desktop); |
|
} |
|
return desktop; |
|
} |
|
/** |
|
* Tests whether this class is supported on the current platform. |
|
* If it's supported, use {@link #getDesktop()} to retrieve an |
|
* instance. |
|
* |
|
* @return <code>true</code> if this class is supported on the |
|
* current platform; <code>false</code> otherwise |
|
* @see #getDesktop() |
|
*/ |
|
public static boolean isDesktopSupported(){ |
|
Toolkit defaultToolkit = Toolkit.getDefaultToolkit(); |
|
if (defaultToolkit instanceof SunToolkit) { |
|
return ((SunToolkit)defaultToolkit).isDesktopSupported(); |
|
} |
|
return false; |
|
} |
|
/** |
|
* Tests whether an action is supported on the current platform. |
|
* |
|
* <p>Even when the platform supports an action, a file or URI may |
|
* not have a registered application for the action. For example, |
|
* most of the platforms support the {@link Desktop.Action#OPEN} |
|
* action. But for a specific file, there may not be an |
|
* application registered to open it. In this case, {@link |
|
* #isSupported} may return {@code true}, but the corresponding |
|
* action method will throw an {@link IOException}. |
|
* |
|
* @param action the specified {@link Action} |
|
* @return <code>true</code> if the specified action is supported on |
|
* the current platform; <code>false</code> otherwise |
|
* @see Desktop.Action |
|
*/ |
|
public boolean isSupported(Action action) { |
|
return peer.isSupported(action); |
|
} |
|
/** |
|
* Checks if the file is a valid file and readable. |
|
* |
|
* @throws SecurityException If a security manager exists and its |
|
* {@link SecurityManager#checkRead(java.lang.String)} method |
|
* denies read access to the file |
|
* @throws NullPointerException if file is null |
|
* @throws IllegalArgumentException if file doesn't exist |
|
*/ |
|
private static void checkFileValidation(File file) { |
|
if (!file.exists()) { |
|
throw new IllegalArgumentException("The file: " |
|
+ file.getPath() + " doesn't exist."); |
|
} |
|
} |
|
/** |
|
* Checks if the action type is supported. |
|
* |
|
* @param actionType the action type in question |
|
* @throws UnsupportedOperationException if the specified action type is not |
|
* supported on the current platform |
|
*/ |
|
private void checkActionSupport(Action actionType){ |
|
if (!isSupported(actionType)) { |
|
throw new UnsupportedOperationException("The " + actionType.name() |
|
+ " action is not supported on the current platform!"); |
|
} |
|
} |
|
/** |
|
* Calls to the security manager's <code>checkPermission</code> method with |
|
* an <code>AWTPermission("showWindowWithoutWarningBanner")</code> |
|
* permission. |
|
*/ |
|
private void checkAWTPermission(){ |
|
SecurityManager sm = System.getSecurityManager(); |
|
if (sm != null) { |
|
sm.checkPermission(new AWTPermission( |
|
"showWindowWithoutWarningBanner")); |
|
} |
|
} |
|
/** |
|
* Launches the associated application to open the file. |
|
* |
|
* <p> If the specified file is a directory, the file manager of |
|
* the current platform is launched to open it. |
|
* |
|
* @param file the file to be opened with the associated application |
|
* @throws NullPointerException if {@code file} is {@code null} |
|
* @throws IllegalArgumentException if the specified file doesn't |
|
* exist |
|
* @throws UnsupportedOperationException if the current platform |
|
* does not support the {@link Desktop.Action#OPEN} action |
|
* @throws IOException if the specified file has no associated |
|
* application or the associated application fails to be launched |
|
* @throws SecurityException if a security manager exists and its |
|
* {@link java.lang.SecurityManager#checkRead(java.lang.String)} |
|
* method denies read access to the file, or it denies the |
|
* <code>AWTPermission("showWindowWithoutWarningBanner")</code> |
|
* permission, or the calling thread is not allowed to create a |
|
* subprocess |
|
* @see java.awt.AWTPermission |
|
*/ |
|
public void open(File file) throws IOException { |
|
file = new File(file.getPath()); |
|
checkAWTPermission(); |
|
checkExec(); |
|
checkActionSupport(Action.OPEN); |
|
checkFileValidation(file); |
|
peer.open(file); |
|
} |
|
/** |
|
* Launches the associated editor application and opens a file for |
|
* editing. |
|
* |
|
* @param file the file to be opened for editing |
|
* @throws NullPointerException if the specified file is {@code null} |
|
* @throws IllegalArgumentException if the specified file doesn't |
|
* exist |
|
* @throws UnsupportedOperationException if the current platform |
|
* does not support the {@link Desktop.Action#EDIT} action |
|
* @throws IOException if the specified file has no associated |
|
* editor, or the associated application fails to be launched |
|
* @throws SecurityException if a security manager exists and its |
|
* {@link java.lang.SecurityManager#checkRead(java.lang.String)} |
|
* method denies read access to the file, or {@link |
|
* java.lang.SecurityManager#checkWrite(java.lang.String)} method |
|
* denies write access to the file, or it denies the |
|
* <code>AWTPermission("showWindowWithoutWarningBanner")</code> |
|
* permission, or the calling thread is not allowed to create a |
|
* subprocess |
|
* @see java.awt.AWTPermission |
|
*/ |
|
public void edit(File file) throws IOException { |
|
file = new File(file.getPath()); |
|
checkAWTPermission(); |
|
checkExec(); |
|
checkActionSupport(Action.EDIT); |
|
file.canWrite(); |
|
checkFileValidation(file); |
|
peer.edit(file); |
|
} |
|
/** |
|
* Prints a file with the native desktop printing facility, using |
|
* the associated application's print command. |
|
* |
|
* @param file the file to be printed |
|
* @throws NullPointerException if the specified file is {@code |
|
* null} |
|
* @throws IllegalArgumentException if the specified file doesn't |
|
* exist |
|
* @throws UnsupportedOperationException if the current platform |
|
* does not support the {@link Desktop.Action#PRINT} action |
|
* @throws IOException if the specified file has no associated |
|
* application that can be used to print it |
|
* @throws SecurityException if a security manager exists and its |
|
* {@link java.lang.SecurityManager#checkRead(java.lang.String)} |
|
* method denies read access to the file, or its {@link |
|
* java.lang.SecurityManager#checkPrintJobAccess()} method denies |
|
* the permission to print the file, or the calling thread is not |
|
* allowed to create a subprocess |
|
*/ |
|
public void print(File file) throws IOException { |
|
file = new File(file.getPath()); |
|
checkExec(); |
|
SecurityManager sm = System.getSecurityManager(); |
|
if (sm != null) { |
|
sm.checkPrintJobAccess(); |
|
} |
|
checkActionSupport(Action.PRINT); |
|
checkFileValidation(file); |
|
peer.print(file); |
|
} |
|
/** |
|
* Launches the default browser to display a {@code URI}. |
|
* If the default browser is not able to handle the specified |
|
* {@code URI}, the application registered for handling |
|
* {@code URIs} of the specified type is invoked. The application |
|
* is determined from the protocol and path of the {@code URI}, as |
|
* defined by the {@code URI} class. |
|
* <p> |
|
* If the calling thread does not have the necessary permissions, |
|
* and this is invoked from within an applet, |
|
* {@code AppletContext.showDocument()} is used. Similarly, if the calling |
|
* does not have the necessary permissions, and this is invoked from within |
|
* a Java Web Started application, {@code BasicService.showDocument()} |
|
* is used. |
|
* |
|
* @param uri the URI to be displayed in the user default browser |
|
* @throws NullPointerException if {@code uri} is {@code null} |
|
* @throws UnsupportedOperationException if the current platform |
|
* does not support the {@link Desktop.Action#BROWSE} action |
|
* @throws IOException if the user default browser is not found, |
|
* or it fails to be launched, or the default handler application |
|
* failed to be launched |
|
* @throws SecurityException if a security manager exists and it |
|
* denies the |
|
* <code>AWTPermission("showWindowWithoutWarningBanner")</code> |
|
* permission, or the calling thread is not allowed to create a |
|
* subprocess; and not invoked from within an applet or Java Web Started |
|
* application |
|
* @throws IllegalArgumentException if the necessary permissions |
|
* are not available and the URI can not be converted to a {@code URL} |
|
* @see java.net.URI |
|
* @see java.awt.AWTPermission |
|
* @see java.applet.AppletContext |
|
*/ |
|
public void browse(URI uri) throws IOException { |
|
SecurityException securityException = null; |
|
try { |
|
checkAWTPermission(); |
|
checkExec(); |
|
} catch (SecurityException e) { |
|
securityException = e; |
|
} |
|
checkActionSupport(Action.BROWSE); |
|
if (uri == null) { |
|
throw new NullPointerException(); |
|
} |
|
if (securityException == null) { |
|
peer.browse(uri); |
|
return; |
|
} |
|
// Calling thread doesn't have necessary priviledges. |
|
// Delegate to DesktopBrowse so that it can work in |
|
// applet/webstart. |
|
URL url = null; |
|
try { |
|
url = uri.toURL(); |
|
} catch (MalformedURLException e) { |
|
throw new IllegalArgumentException("Unable to convert URI to URL", e); |
|
} |
|
sun.awt.DesktopBrowse db = sun.awt.DesktopBrowse.getInstance(); |
|
if (db == null) { |
|
// Not in webstart/applet, throw the exception. |
|
throw securityException; |
|
} |
|
db.browse(url); |
|
} |
|
/** |
|
* Launches the mail composing window of the user default mail |
|
* client. |
|
* |
|
* @throws UnsupportedOperationException if the current platform |
|
* does not support the {@link Desktop.Action#MAIL} action |
|
* @throws IOException if the user default mail client is not |
|
* found, or it fails to be launched |
|
* @throws SecurityException if a security manager exists and it |
|
* denies the |
|
* <code>AWTPermission("showWindowWithoutWarningBanner")</code> |
|
* permission, or the calling thread is not allowed to create a |
|
* subprocess |
|
* @see java.awt.AWTPermission |
|
*/ |
|
public void mail() throws IOException { |
|
checkAWTPermission(); |
|
checkExec(); |
|
checkActionSupport(Action.MAIL); |
|
URI mailtoURI = null; |
|
try{ |
|
mailtoURI = new URI("mailto:?"); |
|
peer.mail(mailtoURI); |
|
} catch (URISyntaxException e){ |
|
// won't reach here. |
|
} |
|
} |
|
/** |
|
* Launches the mail composing window of the user default mail |
|
* client, filling the message fields specified by a {@code |
|
* mailto:} URI. |
|
* |
|
* <p> A <code>mailto:</code> URI can specify message fields |
|
* including <i>"to"</i>, <i>"cc"</i>, <i>"subject"</i>, |
|
* <i>"body"</i>, etc. See <a |
|
* href="http://www.ietf.org/rfc/rfc2368.txt">The mailto URL |
|
* scheme (RFC 2368)</a> for the {@code mailto:} URI specification |
|
* details. |
|
* |
|
* @param mailtoURI the specified {@code mailto:} URI |
|
* @throws NullPointerException if the specified URI is {@code |
|
* null} |
|
* @throws IllegalArgumentException if the URI scheme is not |
|
* <code>"mailto"</code> |
|
* @throws UnsupportedOperationException if the current platform |
|
* does not support the {@link Desktop.Action#MAIL} action |
|
* @throws IOException if the user default mail client is not |
|
* found or fails to be launched |
|
* @throws SecurityException if a security manager exists and it |
|
* denies the |
|
* <code>AWTPermission("showWindowWithoutWarningBanner")</code> |
|
* permission, or the calling thread is not allowed to create a |
|
* subprocess |
|
* @see java.net.URI |
|
* @see java.awt.AWTPermission |
|
*/ |
|
public void mail(URI mailtoURI) throws IOException { |
|
checkAWTPermission(); |
|
checkExec(); |
|
checkActionSupport(Action.MAIL); |
|
if (mailtoURI == null) throw new NullPointerException(); |
|
if (!"mailto".equalsIgnoreCase(mailtoURI.getScheme())) { |
|
throw new IllegalArgumentException("URI scheme is not \"mailto\""); |
|
} |
|
peer.mail(mailtoURI); |
|
} |
|
private void checkExec() throws SecurityException { |
|
SecurityManager sm = System.getSecurityManager(); |
|
if (sm != null) { |
|
sm.checkPermission(new FilePermission("<<ALL FILES>>", |
|
SecurityConstants.FILE_EXECUTE_ACTION)); |
|
} |
|
} |
|
} |