/* * BioJava development code * * This code may be freely distributed and modified under the * terms of the GNU Lesser General Public Licence. This should * be distributed with the code. If you do not have a copy, * see: * * http://www.gnu.org/copyleft/lesser.html * * Copyright for this code is held jointly by the individual * authors. These should be listed in @author doc comments. * * For more information on the BioJava project and its aims, * or to join the biojava-l mailing list, visit the home page * at: * * http://www.biojava.org/ * */ package org.biojava.bio.gui.sequence; import java.awt.Graphics2D; import java.awt.Rectangle; import java.awt.event.MouseEvent; import java.awt.geom.AffineTransform; import java.io.Serializable; import java.net.URL; import java.util.Iterator; import org.biojava.bio.seq.Feature; import org.biojava.bio.seq.FeatureHolder; import org.biojava.bio.symbol.Location; import org.biojava.utils.net.URLFactory; /** *

ZiggyImapRenderer is a decorator for * ZiggyFeatureRenderer which adds the ability to create * HTML image map coordinates which correspond to the feature * rendering produced by the ZiggyFeatureRenderer. These * coordinates correspond to the block regions of the image rather * than the "elbow" sections which join the locations.

* * @author Keith James * @since 1.3 */ public class ZiggyImapRenderer implements ImageMapRenderer, Serializable { private ZiggyFeatureRenderer renderer; private ImageMap imageMap; private URLFactory urlFactory; /** * Creates a new ZiggyImapRenderer. * * @param renderer a ZiggyFeatureRenderer. * @param imageMap an ImageMap. * @param urlFactory an URLFactory which should be * capable of creating a suitable URL from each * Feature on the Sequence to be * rendered. */ public ZiggyImapRenderer(ZiggyFeatureRenderer renderer, ImageMap imageMap, URLFactory urlFactory) { this.renderer = renderer; this.imageMap = imageMap; this.urlFactory = urlFactory; } /** * getImageMap returns the current image map. * * @return an ImageMap. */ public ImageMap getImageMap() { return imageMap; } /** * setImageMap sets the current image map. * * @param imageMap an ImageMap. */ public void setImageMap(ImageMap imageMap) { this.imageMap = imageMap; } /** *

renderImageMap writes a set of image map * coordinates corresponding to the rectangle sections drawn by * the renderer. All the block regions of the image receive the * same URL. The hotspots created by this method have the rendered * Feature set as their user object.

* *

This method is called by renderFeature when * a raster image is rendered.

* * @param g2 a Graphics2D. * @param f a Feature. * @param context a SequenceRenderContext. */ public void renderImageMap(Graphics2D g2, Feature f, SequenceRenderContext context) { Rectangle bounds = g2.getDeviceConfiguration().getBounds(); // Safe to cast as bounds come from raster int mapWidth = (int) bounds.getWidth(); int mapHeight = (int) bounds.getHeight(); URL url = urlFactory.createURL(f); double depth = renderer.getBlockDepth(); AffineTransform t = g2.getTransform(); double transX = t.getTranslateX(); double transY = t.getTranslateY(); int min, max, dif, x1, y1, x2, y2; double posXW, posYN, width, height; Location location = f.getLocation(); for (Iterator li = location.blockIterator(); li.hasNext();) { Location loc = (Location) li.next(); min = loc.getMin(); max = loc.getMax(); dif = max - min; if (context.getDirection() == SequenceRenderContext.HORIZONTAL) { posXW = context.sequenceToGraphics(min); posYN = 0.0; width = Math.max(((double) (dif + 1)) * context.getScale(), 1.0); height = depth; } else { posXW = 0.0; posYN = context.sequenceToGraphics(min); width = depth; height = Math.max(((double) (dif + 1)) * context.getScale(), 1.0); } // Apply translation and round x1 = (int) Math.floor(posXW + transX); y1 = (int) Math.floor(posYN + transY); x2 = (int) Math.floor(posXW + width + transX); y2 = (int) Math.floor(posYN + height + transY); // If the whole rectangle is outside the image then ignore // it if (! (x1 > mapWidth || y1 > mapHeight || x2 < 0 || y2 < 0)) { // Constrain to image size x1 = Math.max(x1, 0); y1 = Math.max(y1, 0); x2 = Math.min(x2, mapWidth); y2 = Math.min(y2, mapHeight); Integer [] coordinates = new Integer[4]; coordinates[0] = new Integer(x1); coordinates[1] = new Integer(y1); coordinates[2] = new Integer(x2); coordinates[3] = new Integer(y2); imageMap.addHotSpot(new ImageMap.HotSpot(ImageMap.RECT, url, coordinates, f)); } } } public void renderFeature(Graphics2D g2, Feature f, SequenceRenderContext context) { renderImageMap(g2, f, context); renderer.renderFeature(g2, f, context); } public double getDepth(SequenceRenderContext context) { return renderer.getDepth(context); } public FeatureHolder processMouseEvent(FeatureHolder holder, SequenceRenderContext context, MouseEvent mEvent) { return renderer.processMouseEvent(holder, context, mEvent); } }