org.jdesktop.swingx.hyperlink.HyperlinkAction Maven / Gradle / Ivy
/*
* Copyright 2009 Sun Microsystems, Inc., 4150 Network Circle,
* Santa Clara, California 95054, U.S.A. All rights reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*
*/
package org.jdesktop.swingx.hyperlink;
import java.awt.Desktop;
import java.awt.GraphicsEnvironment;
import java.awt.HeadlessException;
import java.awt.Desktop.Action;
import java.awt.event.ActionEvent;
import java.io.IOException;
import java.net.URI;
import java.util.logging.Logger;
/**
* A implementation wrapping Desktop
actions BROWSE and MAIL, that is URI-related.
*
* @author Jeanette Winzenburg
*/
public class HyperlinkAction extends AbstractHyperlinkAction {
private static final Logger LOG = Logger.getLogger(HyperlinkAction.class.getName());
private Action desktopAction;
private URIVisitor visitor;
/**
* Factory method to create and return a HyperlinkAction for the given uri. Tries
* to guess the appropriate type from the uri. If uri is not null and has a
* scheme of mailto, create one of type Mail. In all other cases, creates one
* for BROWSE.
*
* @param uri to uri to create a HyperlinkAction for, maybe null.
* @return a HyperlinkAction for the given URI.
* @throws HeadlessException if {@link
* GraphicsEnvironment#isHeadless()} returns {@code true}
* @throws UnsupportedOperationException if the current platform doesn't support
* Desktop
*/
public static HyperlinkAction createHyperlinkAction(URI uri) {
Action type = isMailURI(uri) ? Action.MAIL : Action.BROWSE;
return createHyperlinkAction(uri, type);
}
/**
* Creates and returns a HyperlinkAction with the given target and action type.
*
* @param uri the target uri, maybe null.
* @param desktopAction the type of desktop action this class should perform, must be BROWSE or MAIL
* @return a HyperlinkAction
* @throws HeadlessException if {@link GraphicsEnvironment#isHeadless()} returns {@code true}
* @throws UnsupportedOperationException if the current platform doesn't support Desktop
* @throws IllegalArgumentException if unsupported action type
*/
public static HyperlinkAction createHyperlinkAction(URI uri, Action desktopAction) {
return new HyperlinkAction(uri, desktopAction);
}
/**
* @param uri
* @return
*/
private static boolean isMailURI(URI uri) {
return uri != null && "mailto".equalsIgnoreCase(uri.getScheme());
}
/**
* Instantiates a HyperlinkAction with action type BROWSE.
*
* @throws HeadlessException if {@link
* GraphicsEnvironment#isHeadless()} returns {@code true}
* @throws UnsupportedOperationException if the current platform doesn't support
* Desktop
* @throws IllegalArgumentException if unsupported action type
*/
public HyperlinkAction() {
this(Action.BROWSE);
}
/**
* Instantiates a HyperlinkAction with the given action type.
*
* @param desktopAction the type of desktop action this class should perform, must be
* BROWSE or MAIL
* @throws HeadlessException if {@link
* GraphicsEnvironment#isHeadless()} returns {@code true}
* @throws UnsupportedOperationException if the current platform doesn't support
* Desktop
* @throws IllegalArgumentException if unsupported action type
*/
public HyperlinkAction(Action desktopAction) {
this(null, desktopAction);
}
/**
*
* @param uri the target uri, maybe null.
* @param desktopAction the type of desktop action this class should perform, must be
* BROWSE or MAIL
* @throws HeadlessException if {@link
* GraphicsEnvironment#isHeadless()} returns {@code true}
* @throws UnsupportedOperationException if the current platform doesn't support
* Desktop
* @throws IllegalArgumentException if unsupported action type
*/
public HyperlinkAction(URI uri, Action desktopAction) {
super();
if (!Desktop.isDesktopSupported()) {
throw new UnsupportedOperationException("Desktop API is not " +
"supported on the current platform");
}
if (desktopAction != Desktop.Action.BROWSE && desktopAction != Desktop.Action.MAIL) {
throw new IllegalArgumentException("Illegal action type: " + desktopAction +
". Must be BROWSE or MAIL");
}
this.desktopAction = desktopAction;
getURIVisitor();
setTarget(uri);
}
/**
* {@inheritDoc}
*
* Implemented to perform the appropriate Desktop action if supported on the current
* target. Sets the visited property to true if the desktop action doesn't throw
* an exception or to false if it did.
*
* Does nothing if the action isn't supported.
*/
@Override
public void actionPerformed(ActionEvent e) {
if (!getURIVisitor().isEnabled(getTarget())) return;
try {
getURIVisitor().visit(getTarget());
setVisited(true);
} catch (IOException e1) {
setVisited(false);
LOG.fine("can't visit Desktop " + e);
}
}
/**
* @return Action
*/
public Action getDesktopAction() {
return desktopAction;
}
@Override
protected void installTarget() {
// doohh ... this is called from super's constructor before we are
// fully initialized
if (visitor == null) return;
super.installTarget();
updateEnabled();
}
/**
*
*/
private void updateEnabled() {
setEnabled(getURIVisitor().isEnabled(getTarget()));
}
/**
* @return
*/
private URIVisitor getURIVisitor() {
if (visitor == null) {
visitor = createURIVisitor();
}
return visitor;
}
/**
* for testing
*/
void setURIVisitor(URIVisitor visitor) {
this.visitor = visitor;
}
/**
* @return
*/
private URIVisitor createURIVisitor() {
return getDesktopAction() == Action.BROWSE ?
new BrowseVisitor() : new MailVisitor();
}
/**
* Thin wrapper around Desktop functionality to allow uniform handling of
* different actions in HyperlinkAction.
*
*/
abstract class URIVisitor {
protected boolean desktopSupported = Desktop.isDesktopSupported();
/**
* Returns a boolean indicating whether the action is supported on the
* given URI. This implementation returns true if both the Desktop is
* generally supported and isActionSupported()
.
*
* PENDING JW: hmm ... which class exactly has to check for valid combination
* of Action and URI?
*
* @param uri
* @return
*
* @see #isActionSupported()
*/
public boolean isEnabled(URI uri) {
return desktopSupported && isActionSupported();
}
/**
* Visits the given URI via Desktop functionality. Must not be called if not
* enabled.
*
* @param uri the URI to visit
* @throws IOException if the Desktop method throws IOException.
*
*/
public abstract void visit(URI uri) throws IOException;
/**
* Returns a boolean indicating if the action is supported by the current
* Desktop.
*
* @return true if the Action is supported by the current desktop, false
* otherwise.
*/
protected abstract boolean isActionSupported();
}
private class BrowseVisitor extends URIVisitor {
/**
* {@inheritDoc}
*
* Implemented to message the browse method of Desktop.
*/
@Override
public void visit(URI uri) throws IOException {
Desktop.getDesktop().browse(uri);
}
/**
* {@inheritDoc}
*
* Implemented to query the Desktop for support of BROWSE action.
*/
@Override
protected boolean isActionSupported() {
return Desktop.getDesktop().isSupported(Desktop.Action.BROWSE);
}
/**
* {@inheritDoc}
*
* Implemented to guard against null URI in addition to super.
*/
@Override
public boolean isEnabled(URI uri) {
return uri != null && super.isEnabled(uri);
}
}
private class MailVisitor extends URIVisitor {
/**
* {@inheritDoc}
*
* Implemented to message the mail function of Desktop.
*/
@Override
public void visit(URI uri) throws IOException {
if (uri == null) {
Desktop.getDesktop().mail();
} else {
Desktop.getDesktop().mail(uri);
}
}
/**
* {@inheritDoc}
*
* Implemented to query the Desktop for support of MAIL action.
*/
@Override
protected boolean isActionSupported() {
return Desktop.getDesktop().isSupported(Desktop.Action.MAIL);
}
}
}