blob: 436aab1cdfe71e5d446100792e9b736360f870ad [file] [log] [blame]
/*
Licensed to the Apache Software Foundation (ASF) under one or more
contributor license agreements. See the NOTICE file distributed with
this work for additional information regarding copyright ownership.
The ASF licenses this file to You under the Apache License, Version 2.0
(the "License"); you may not use this file except in compliance with
the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package org.apache.batik.bridge;
import java.awt.Cursor;
import java.util.List;
import org.apache.batik.anim.dom.SVGOMAElement;
import org.apache.batik.anim.dom.SVGOMAnimationElement;
import org.apache.batik.anim.dom.SVGOMDocument;
import org.apache.batik.dom.events.AbstractEvent;
import org.apache.batik.dom.events.NodeEventTarget;
import org.apache.batik.gvt.GraphicsNode;
import org.apache.batik.util.ParsedURL;
import org.apache.batik.util.XMLConstants;
import org.w3c.dom.Element;
import org.w3c.dom.events.Event;
import org.w3c.dom.events.EventListener;
import org.w3c.dom.svg.SVGAElement;
/**
* Bridge class for the <a> element.
*
* @author <a href="mailto:tkormann@apache.org">Thierry Kormann</a>
* @version $Id$
*/
public class SVGAElementBridge extends SVGGElementBridge {
protected AnchorListener al;
protected CursorMouseOverListener bl;
protected CursorMouseOutListener cl;
/**
* Constructs a new bridge for the &lt;a> element.
*/
public SVGAElementBridge() {}
/**
* Returns 'a'.
*/
public String getLocalName() {
return SVG_A_TAG;
}
/**
* Returns a new instance of this bridge.
*/
public Bridge getInstance() {
return new SVGAElementBridge();
}
/**
* Builds using the specified BridgeContext and element, the
* specified graphics node.
*
* @param ctx the bridge context to use
* @param e the element that describes the graphics node to build
* @param node the graphics node to build
*/
public void buildGraphicsNode(BridgeContext ctx,
Element e,
GraphicsNode node) {
super.buildGraphicsNode(ctx, e, node);
if (ctx.isInteractive()) {
NodeEventTarget target = (NodeEventTarget)e;
CursorHolder ch = new CursorHolder(CursorManager.DEFAULT_CURSOR);
al = new AnchorListener(ctx.getUserAgent(), ch);
target.addEventListenerNS
(XMLConstants.XML_EVENTS_NAMESPACE_URI, SVG_EVENT_CLICK,
al, false, null);
ctx.storeEventListenerNS
(target,
XMLConstants.XML_EVENTS_NAMESPACE_URI, SVG_EVENT_CLICK,
al, false);
bl = new CursorMouseOverListener(ctx.getUserAgent(), ch);
target.addEventListenerNS
(XMLConstants.XML_EVENTS_NAMESPACE_URI, SVG_EVENT_MOUSEOVER,
bl, false, null);
ctx.storeEventListenerNS
(target,
XMLConstants.XML_EVENTS_NAMESPACE_URI, SVG_EVENT_MOUSEOVER,
bl, false);
cl = new CursorMouseOutListener(ctx.getUserAgent(), ch);
target.addEventListenerNS
(XMLConstants.XML_EVENTS_NAMESPACE_URI, SVG_EVENT_MOUSEOUT,
cl, false, null);
ctx.storeEventListenerNS
(target,
XMLConstants.XML_EVENTS_NAMESPACE_URI, SVG_EVENT_MOUSEOUT,
cl, false);
}
}
public void dispose() {
NodeEventTarget target = (NodeEventTarget)e;
if (al != null) {
target.removeEventListenerNS
(XMLConstants.XML_EVENTS_NAMESPACE_URI, SVG_EVENT_CLICK,
al, false);
al = null;
}
if (bl != null) {
target.removeEventListenerNS
(XMLConstants.XML_EVENTS_NAMESPACE_URI, SVG_EVENT_MOUSEOVER,
bl, false);
bl = null;
}
if (cl != null) {
target.removeEventListenerNS
(XMLConstants.XML_EVENTS_NAMESPACE_URI, SVG_EVENT_MOUSEOUT,
cl, false);
cl = null;
}
super.dispose();
}
/**
* Returns true as the &lt;a> element is a container.
*/
public boolean isComposite() {
return true;
}
public static class CursorHolder {
Cursor cursor = null;
public CursorHolder(Cursor c) {
cursor = c;
}
public void holdCursor(Cursor c) {
cursor = c;
}
public Cursor getCursor() {
return cursor;
}
}
/**
* To handle a click on an anchor.
*/
public static class AnchorListener implements EventListener {
protected UserAgent userAgent;
protected CursorHolder holder;
public AnchorListener(UserAgent ua, CursorHolder ch) {
userAgent = ua;
holder = ch;
}
public void handleEvent(Event evt) {
if (!(evt instanceof AbstractEvent)) return;
final AbstractEvent ae = (AbstractEvent)evt;
List l = ae.getDefaultActions();
if (l != null) {
for (Object o : l) {
if (o instanceof AnchorDefaultActionable)
return; // only one anchor in default list...
}
}
SVGAElement elt = (SVGAElement)evt.getCurrentTarget();
ae.addDefaultAction(new AnchorDefaultActionable
(elt, userAgent, holder));
}
}
public static class AnchorDefaultActionable implements Runnable {
protected SVGOMAElement elt;
protected UserAgent userAgent;
protected CursorHolder holder;
public AnchorDefaultActionable(SVGAElement e,
UserAgent ua,
CursorHolder ch) {
elt = (SVGOMAElement) e;
userAgent = ua;
holder = ch;
}
public void run() {
userAgent.setSVGCursor(holder.getCursor());
String href = elt.getHref().getAnimVal();
ParsedURL purl = new ParsedURL(elt.getBaseURI(), href);
SVGOMDocument doc = (SVGOMDocument) elt.getOwnerDocument();
ParsedURL durl = doc.getParsedURL();
if (purl.sameFile(durl)) {
String frag = purl.getRef();
if (frag != null && frag.length() != 0) {
Element refElt = doc.getElementById(frag);
if (refElt instanceof SVGOMAnimationElement) {
SVGOMAnimationElement aelt =
(SVGOMAnimationElement) refElt;
float t = aelt.getHyperlinkBeginTime();
if (Float.isNaN(t)) {
aelt.beginElement();
} else {
doc.getRootElement().setCurrentTime(t);
}
return;
}
}
}
userAgent.openLink(elt);
}
}
/**
* To handle a mouseover on an anchor and set the cursor.
*/
public static class CursorMouseOverListener implements EventListener {
protected UserAgent userAgent;
protected CursorHolder holder;
public CursorMouseOverListener(UserAgent ua, CursorHolder ch) {
userAgent = ua;
holder = ch;
}
public void handleEvent(Event evt) {
if (!(evt instanceof AbstractEvent)) return;
final AbstractEvent ae = (AbstractEvent)evt;
List l = ae.getDefaultActions();
if (l != null) {
for (Object o : l) {
if (o instanceof MouseOverDefaultActionable)
return; // only one anchor in default list...
}
}
Element target = (Element)ae.getTarget();
SVGAElement elt = (SVGAElement)ae.getCurrentTarget();
ae.addDefaultAction(new MouseOverDefaultActionable
(target, elt, userAgent, holder));
}
}
public static class MouseOverDefaultActionable implements Runnable {
protected Element target;
protected SVGAElement elt;
protected UserAgent userAgent;
protected CursorHolder holder;
public MouseOverDefaultActionable(Element t,
SVGAElement e,
UserAgent ua,
CursorHolder ch) {
target = t;
elt = e;
userAgent = ua;
holder = ch;
}
public void run() {
// Only modify the cursor if the target's cursor property is
// 'auto'. Note that we do not need to check the value of
// anchor element as the target's cursor value is resulting
// from the CSS cascade which has accounted for inheritance.
// This means that our behavior is to set the cursor to a
// hand cursor for any content on which the cursor property is
// 'auto' inside an anchor element. If, for example, the
// content was:
// <a cusor="wait">
// <g cursor="auto">
// <rect />
// </g>
// </a>
//
// The cursor on the inside rect will be set to the hand cursor and
// not the wait cursor
//
if (CSSUtilities.isAutoCursor(target)) {
holder.holdCursor(CursorManager.DEFAULT_CURSOR);
// The target's cursor value is 'auto': use the hand cursor
userAgent.setSVGCursor(CursorManager.ANCHOR_CURSOR);
}
//
// In all cases, display the href in the userAgent
//
if (elt != null) {
String href = elt.getHref().getAnimVal();
userAgent.displayMessage(href);
}
}
}
/**
* To handle a mouseout on an anchor and set the cursor.
*/
public static class CursorMouseOutListener implements EventListener {
protected UserAgent userAgent;
protected CursorHolder holder;
public CursorMouseOutListener(UserAgent ua, CursorHolder ch) {
userAgent = ua;
holder = ch;
}
public void handleEvent(Event evt) {
if (!(evt instanceof AbstractEvent)) return;
final AbstractEvent ae = (AbstractEvent)evt;
List l = ae.getDefaultActions();
if (l != null) {
for (Object o : l) {
if (o instanceof MouseOutDefaultActionable)
return; // only one anchor in default list...
}
}
SVGAElement elt = (SVGAElement)evt.getCurrentTarget();
ae.addDefaultAction(new MouseOutDefaultActionable
(elt, userAgent, holder));
}
}
public static class MouseOutDefaultActionable implements Runnable {
protected SVGAElement elt;
protected UserAgent userAgent;
protected CursorHolder holder;
public MouseOutDefaultActionable(SVGAElement e,
UserAgent ua,
CursorHolder ch) {
elt = e;
userAgent = ua;
holder = ch;
}
public void run() {
// No need to set the cursor on out events: this is taken care of
// by the BridgeContext(?)
// Hide the href in the userAgent
if (elt != null) {
userAgent.displayMessage("");
}
}
}
}