package javax.swing.text.html; import gnu.javax.swing.text.html.CombinedAttributes; import gnu.javax.swing.text.html.ImageViewIconFactory; import java.awt.Graphics; import java.awt.Image; import java.awt.MediaTracker; import java.awt.Rectangle; import java.awt.Shape; import java.net.MalformedURLException; import java.net.URL; import javax.swing.Icon; import javax.swing.ImageIcon; import javax.swing.text.AttributeSet; import javax.swing.text.BadLocationException; import javax.swing.text.Document; import javax.swing.text.Element; import javax.swing.text.View; import javax.swing.text.Position.Bias; import javax.swing.text.html.HTML.Attribute; /** * A view, representing a single image, represented by the HTML IMG tag. * * @author Audrius Meskauskas (AudriusA@Bioinformatics.org) */ public class ImageView extends View { /** * True if the image loads synchronuosly (on demand). By default, the image * loads asynchronuosly. */ boolean loadOnDemand; /** * The image icon, wrapping the image, */ ImageIcon imageIcon; /** * The image state. */ byte imageState = MediaTracker.LOADING; /** * Creates the image view that represents the given element. * * @param element the element, represented by this image view. */ public ImageView(Element element) { super(element); } /** * Load or reload the image. This method initiates the image reloading. After * the image is ready, the repaint event will be scheduled. The current image, * if it already exists, will be discarded. * * @param itsTime * also load if the "on demand" property is set */ void reloadImage(boolean itsTime) { URL url = getImageURL(); if (url == null) imageState = (byte) MediaTracker.ERRORED; else if (!(loadOnDemand && !itsTime)) imageIcon = new ImageIcon(url); else imageState = (byte) MediaTracker.LOADING; } /** * Get the image alignment. This method works handling standart alignment * attributes in the HTML IMG tag (align = top bottom middle left right). * Depending from the parameter, either horizontal or vertical alingment * information is returned. * * @param axis - * either X_AXIS or Y_AXIS */ public float getAlignment(int axis) { AttributeSet attrs = getAttributes(); Object al = attrs.getAttribute(Attribute.ALIGN); // Default is top left aligned. if (al == null) return 0.0f; String align = al.toString(); if (axis == View.X_AXIS) { if (align.equals("middle")) return 0.5f; else if (align.equals("left")) return 0.0f; else if (align.equals("right")) return 1.0f; else return 0.0f; } else if (axis == View.Y_AXIS) { if (align.equals("middle")) return 0.5f; else if (align.equals("top")) return 0.0f; else if (align.equals("bottom")) return 1.0f; else return 0.0f; } else throw new IllegalArgumentException("axis " + axis); } /** * Get the text that should be shown as the image replacement and also as the * image tool tip text. The method returns the value of the attribute, having * the name {@link Attribute#ALT}. If there is no such attribute, the image * name from the url is returned. If the URL is not available, the empty * string is returned. */ public String getAltText() { Object rt = getAttributes().getAttribute(Attribute.ALT); if (rt != null) return rt.toString(); else { URL u = getImageURL(); if (u == null) return ""; else return u.getFile(); } } /** * Returns the combination of the document and the style sheet attributes. */ public AttributeSet getAttributes() { StyleSheet styles = getStyleSheet(); if (styles == null) return super.getAttributes(); else return CombinedAttributes.combine(super.getAttributes(), styles.getViewAttributes(this)); } /** * Get the image to render. May return null if the image is not yet loaded. */ public Image getImage() { if (imageIcon == null) return null; else return imageIcon.getImage(); } /** * Get the URL location of the image to render. If this method returns null, * the "no image" icon is rendered instead. By defaul, url must be present as * the "src" property of the IMG tag. If it is missing, null is returned and * the "no image" icon is rendered. * * @return the URL location of the image to render. */ public URL getImageURL() { Element el = getElement(); String src = (String) el.getAttributes().getAttribute(Attribute.SRC); URL url = null; if (src != null) { URL base = ((HTMLDocument) getDocument()).getBase(); try { url = new URL(base, src); } catch (MalformedURLException ex) { // Return null. } } return url; } /** * Get the icon that should be displayed while the image is loading and hence * not yet available. * * @return an icon, showing a non broken sheet of paper with image. */ public Icon getLoadingImageIcon() { return ImageViewIconFactory.getLoadingImageIcon(); } /** * Get the image loading strategy. * * @return false (default) if the image is loaded when the view is * constructed, true if the image is only loaded on demand when * rendering. */ public boolean getLoadsSynchronously() { return loadOnDemand; } /** * Get the icon that should be displayed when the image is not available. * * @return an icon, showing a broken sheet of paper with image. */ public Icon getNoImageIcon() { return ImageViewIconFactory.getNoImageIcon(); } /** * Get the preferred span of the image along the axis. The image size is first * requested to the attributes {@link Attribute#WIDTH} and * {@link Attribute#HEIGHT}. If they are missing, and the image is already * loaded, the image size is returned. If there are no attributes, and the * image is not loaded, zero is returned. * * @param axis - * either X_AXIS or Y_AXIS * @return either width of height of the image, depending on the axis. */ public float getPreferredSpan(int axis) { AttributeSet attrs = getAttributes(); Image image = getImage(); if (axis == View.X_AXIS) { Object w = attrs.getAttribute(Attribute.WIDTH); if (w != null) return Integer.parseInt(w.toString()); else if (image != null) return image.getWidth(getContainer()); else return getNoImageIcon().getIconWidth(); } else if (axis == View.Y_AXIS) { Object w = attrs.getAttribute(Attribute.HEIGHT); if (w != null) return Integer.parseInt(w.toString()); else if (image != null) return image.getHeight(getContainer()); else return getNoImageIcon().getIconHeight(); } else throw new IllegalArgumentException("axis " + axis); } /** * Get the associated style sheet from the document. * * @return the associated style sheet. */ protected StyleSheet getStyleSheet() { Document d = getElement().getDocument(); if (d instanceof HTMLDocument) return ((HTMLDocument) d).getStyleSheet(); else return null; } /** * Get the tool tip text. This is overridden to return the value of the * {@link #getAltText()}. The parameters are ignored. * * @return that is returned by getAltText(). */ public String getToolTipText(float x, float y, Shape shape) { return getAltText(); } /** * Paints the image or one of the two image state icons. The image is resized * to the shape bounds. If there is no image available, the alternative text * is displayed besides the image state icon. * * @param g * the Graphics, used for painting. * @param bounds * the bounds of the region where the image or replacing icon must be * painted. */ public void paint(Graphics g, Shape bounds) { Rectangle r = bounds.getBounds(); if (imageIcon == null) { // Loading image on demand, rendering the loading icon so far. reloadImage(true); // The reloadImage sets the imageIcon, unless the URL is broken // or malformed. if (imageIcon != null) { if (imageIcon.getImageLoadStatus() != MediaTracker.COMPLETE) { // Render "not ready" icon, unless the image is ready // immediately. renderIcon(g, r, getLoadingImageIcon()); // Add the listener to repaint when the icon will be ready. imageIcon.setImageObserver(getContainer()); return; } } else { renderIcon(g, r, getNoImageIcon()); return; } } imageState = (byte) imageIcon.getImageLoadStatus(); switch (imageState) { case MediaTracker.ABORTED: case MediaTracker.ERRORED: renderIcon(g, r, getNoImageIcon()); break; case MediaTracker.LOADING: // If the image is not loaded completely, we still render it, as the // partial image may be available. case MediaTracker.COMPLETE: { // Paint the scaled image. Image scaled = imageIcon.getImage().getScaledInstance( r.width, r.height, Image.SCALE_DEFAULT); ImageIcon painter = new ImageIcon(scaled); painter.paintIcon(getContainer(), g, r.x, r.y); } break; } } /** * Render "no image" icon and the alternative "no image" text. The text is * rendered right from the icon and is aligned to the icon bottom. */ private void renderIcon(Graphics g, Rectangle bounds, Icon icon) { Shape current = g.getClip(); try { g.setClip(bounds); if (icon != null) { icon.paintIcon(getContainer(), g, bounds.x, bounds.y); g.drawString(getAltText(), bounds.x + icon.getIconWidth(), bounds.y + icon.getIconHeight()); } } finally { g.setClip(current); } } /** * Set if the image should be loaded only when needed (synchronuosly). By * default, the image loads asynchronuosly. If the image is not yet ready, the * icon, returned by the {@link #getLoadingImageIcon()}, is displayed. */ public void setLoadsSynchronously(boolean load_on_demand) { loadOnDemand = load_on_demand; } /** * Update all cached properties from the attribute set, returned by the * {@link #getAttributes}. */ protected void setPropertiesFromAttributes() { // In the current implementation, nothing is cached yet, unless the image // itself. imageIcon = null; } /** * Maps the picture co-ordinates into the image position in the model. As the * image is not divideable, this is currently implemented always to return the * start offset. */ public int viewToModel(float x, float y, Shape shape, Bias[] bias) { return getStartOffset(); } /** * This is currently implemented always to return the area of the image view, * as the image is not divideable by character positions. * * @param pos character position * @param area of the image view * @param bias bias * * @return the shape, where the given character position should be mapped. */ public Shape modelToView(int pos, Shape area, Bias bias) throws BadLocationException { return area; } /** * Starts loading the image asynchronuosly. If the image must be loaded * synchronuosly instead, the {@link #setLoadsSynchronously} must be * called before calling this method. The passed parameters are not used. */ public void setSize(float width, float height) { if (imageIcon == null) reloadImage(false); } }