com.alee.extended.image.WebImage Maven / Gradle / Ivy
/*
* This file is part of WebLookAndFeel library.
*
* WebLookAndFeel library is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* WebLookAndFeel 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with WebLookAndFeel library. If not, see .
*/
package com.alee.extended.image;
import com.alee.laf.WebLookAndFeel;
import com.alee.utils.GraphicsUtils;
import com.alee.utils.ImageUtils;
import com.alee.utils.SwingUtils;
import javax.swing.*;
import java.awt.*;
import java.awt.geom.Rectangle2D;
import java.awt.image.BufferedImage;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.net.URL;
/**
* This component allows you to display images in many different ways.
* This component uses less resources than a label and has a few optimization.
*
* @author Mikle Garin
*/
public class WebImage extends JComponent implements SwingConstants
{
/**
* Image source.
*/
private BufferedImage image;
/**
* Cached disabled image version.
*/
private BufferedImage disabledImage;
/**
* How image should be displayed.
*/
private DisplayType displayType;
/**
* Image horizontal alignment.
* Doesn't affect anything in case fitComponent display type is used.
*/
private int horizontalAlignment;
/**
* Image vertical alignment.
* Doesn't affect anything in case fitComponent display type is used.
*/
private int verticalAlignment;
/**
* Image transparency.
*/
private float transparency;
/**
* Image margins.
*/
private Insets margin;
/**
* Last cached image size.
* This is used to determine when image component was resized since last paint call.
*/
private Dimension lastDimention = null;
/**
* Last cached image preview.
* This variable is used when actual painted image is smaller than source image.
* In that case source image is getting scaled and saved into this variable.
*/
private BufferedImage lastPreviewImage = null;
/**
* Constructs an empty image component.
*/
public WebImage ()
{
this ( ( Image ) null );
}
/**
* Constructs component with an image loaded from the specified path.
*
* @param src path to image
*/
public WebImage ( final String src )
{
this ( ImageUtils.loadImage ( src ) );
}
/**
* Constructs component with an image loaded from package near specified class.
*
* @param nearClass class near which image is located
* @param src image file location
*/
public WebImage ( final Class nearClass, final String src )
{
this ( ImageUtils.loadImage ( nearClass, src ) );
}
/**
* Constructs component with an image loaded from the specified url.
*
* @param url image url
*/
public WebImage ( final URL url )
{
this ( ImageUtils.loadImage ( url ) );
}
/**
* Constructs component with an image retrieved from the specified icon.
*
* @param icon icon to process
*/
public WebImage ( final Icon icon )
{
this ( ImageUtils.getBufferedImage ( icon ) );
}
/**
* Constructs component with an image retrieved from the specified image icon.
*
* @param icon image icon to process
*/
public WebImage ( final ImageIcon icon )
{
this ( icon.getImage () );
}
/**
* Constructs component with a specified image.
*
* @param image image
*/
public WebImage ( final Image image )
{
this ( ImageUtils.getBufferedImage ( image ) );
}
/**
* Constructs component with a specified image.
*
* @param image image
*/
public WebImage ( final BufferedImage image )
{
super ();
this.image = image;
this.disabledImage = null;
this.displayType = DisplayType.preferred;
this.horizontalAlignment = CENTER;
this.verticalAlignment = CENTER;
this.transparency = 1f;
SwingUtils.setOrientation ( this );
setOpaque ( false );
addPropertyChangeListener ( WebLookAndFeel.ENABLED_PROPERTY, new PropertyChangeListener ()
{
@Override
public void propertyChange ( final PropertyChangeEvent evt )
{
if ( !isEnabled () )
{
calculateDisabledImage ();
repaint ();
}
else
{
clearDisabledImage ();
repaint ();
}
}
} );
}
/**
* Updates cached disabled image.
*/
protected void calculateDisabledImage ()
{
disabledImage = image != null ? ImageUtils.createDisabledCopy ( image ) : null;
lastPreviewImage = null;
}
/**
* Clears cached disabled image
*/
private void clearDisabledImage ()
{
if ( disabledImage != null )
{
disabledImage.flush ();
disabledImage = null;
}
lastPreviewImage = null;
}
/**
* Returns image width or -1 if image was not set.
*
* @return image width or -1 if image was not set
*/
public int getImageWidth ()
{
return image != null ? image.getWidth () : -1;
}
/**
* Returns image height or -1 if image was not set.
*
* @return image height or -1 if image was not set
*/
public int getImageHeight ()
{
return image != null ? image.getHeight () : -1;
}
/**
* Returns current image.
*
* @return image
*/
public BufferedImage getImage ()
{
return image;
}
/**
* Changes image to new one taken from specified icon.
*
* @param icon icon to process
* @return this image component
*/
public WebImage setIcon ( final Icon icon )
{
setImage ( ImageUtils.getBufferedImage ( icon ) );
return this;
}
/**
* Changes image to new one taken from specified image icon.
*
* @param icon image icon to process
* @return this image component
*/
public WebImage setIcon ( final ImageIcon icon )
{
setImage ( icon.getImage () );
return this;
}
/**
* Changes image to the specified one.
*
* @param image new image
* @return this image component
*/
public WebImage setImage ( final Image image )
{
setImage ( ImageUtils.getBufferedImage ( image ) );
return this;
}
/**
* Changes image to the specified one.
*
* @param image new image
* @return this image component
*/
public WebImage setImage ( final BufferedImage image )
{
this.image = image;
if ( !isEnabled () )
{
calculateDisabledImage ();
}
revalidate ();
repaint ();
return this;
}
/**
* Returns image display type.
*
* @return image display type
*/
public DisplayType getDisplayType ()
{
return displayType;
}
/**
* Changes image display type.
*
* @param displayType new image display type
* @return this image component
*/
public WebImage setDisplayType ( final DisplayType displayType )
{
this.displayType = displayType;
updateView ();
return this;
}
/**
* Returns image horizontal alignment.
*
* @return image horizontal alignment
*/
public int getHorizontalAlignment ()
{
return horizontalAlignment;
}
/**
* Changes image horizontal alignment to the specified one.
*
* @param horizontalAlignment new image horizontal alignment
* @return this image component
*/
public WebImage setHorizontalAlignment ( final int horizontalAlignment )
{
this.horizontalAlignment = horizontalAlignment;
updateView ();
return this;
}
/**
* Returns image vertical alignment.
*
* @return image vertical alignment
*/
public int getVerticalAlignment ()
{
return verticalAlignment;
}
/**
* Changes image vertical alignment to the specified one.
*
* @param verticalAlignment new image vertical alignment
* @return this image component
*/
public WebImage setVerticalAlignment ( final int verticalAlignment )
{
this.verticalAlignment = verticalAlignment;
updateView ();
return this;
}
/**
* Returns image transparency.
*
* @return image transparency
*/
public float getTransparency ()
{
return transparency;
}
/**
* Changes image transparency.
*
* @param transparency new image transparency
* @return this image component
*/
public WebImage setTransparency ( final float transparency )
{
this.transparency = transparency;
updateView ();
return this;
}
/**
* Updates image component view.
*/
protected void updateView ()
{
if ( isShowing () )
{
repaint ();
}
}
/**
* Returns image margin.
*
* @return image margin
*/
public Insets getMargin ()
{
return margin;
}
/**
* Changes image margin.
*
* @param margin new image margin
* @return this image component
*/
public WebImage setMargin ( final Insets margin )
{
this.margin = margin;
updateBorder ();
return this;
}
/**
* Changes image margin.
*
* @param top top margin
* @param left left margin
* @param bottom bottom margin
* @param right right margin
* @return this image component
*/
public WebImage setMargin ( final int top, final int left, final int bottom, final int right )
{
return setMargin ( new Insets ( top, left, bottom, right ) );
}
/**
* Changes image margin.
*
* @param spacing side spacing
* @return this image component
*/
public WebImage setMargin ( final int spacing )
{
return setMargin ( spacing, spacing, spacing, spacing );
}
/**
* Updates image component border.
*/
protected void updateBorder ()
{
if ( margin != null )
{
setBorder ( BorderFactory.createEmptyBorder ( margin.top, margin.left, margin.bottom, margin.right ) );
}
else
{
setBorder ( null );
}
}
/**
* Paints image component.
*
* @param g graphics
*/
@Override
protected void paintComponent ( final Graphics g )
{
super.paintComponent ( g );
if ( transparency <= 0f )
{
return;
}
final Graphics2D g2d = ( Graphics2D ) g;
final Composite oc = GraphicsUtils.setupAlphaComposite ( g2d, transparency, transparency < 1f );
// todo Optimize for repaint (check if image is out of repainted/clipped bounds)
final BufferedImage currentImage = getCurrentImage ();
if ( currentImage != null )
{
final Insets insets = getInsets ();
if ( getSize ().equals ( getRequiredSize () ) )
{
// Drawing image when it is currently at preferred size
g2d.drawImage ( currentImage, insets.left, insets.top, null );
}
else
{
switch ( displayType )
{
case preferred:
{
// Drawing preferred sized image at specified side
final int x = horizontalAlignment == LEFT ? insets.left :
( horizontalAlignment == RIGHT ? getWidth () - currentImage.getWidth () - insets.right :
getCenterX ( insets ) - currentImage.getWidth () / 2 );
final int y = verticalAlignment == TOP ? insets.top :
( verticalAlignment == BOTTOM ? getHeight () - currentImage.getHeight () - insets.bottom :
getCenterY ( insets ) - currentImage.getHeight () / 2 );
g2d.drawImage ( currentImage, x, y, null );
break;
}
case fitComponent:
{
// Drawing sized to fit object image
final BufferedImage preview = getPreviewImage ( insets );
g2d.drawImage ( preview, getCenterX ( insets ) - preview.getWidth () / 2,
getCenterY ( insets ) - preview.getHeight () / 2, null );
break;
}
case repeat:
{
// Drawing repeated in background image
final int x = horizontalAlignment == LEFT ? insets.left :
( horizontalAlignment == RIGHT ? getWidth () - currentImage.getWidth () - insets.right :
getCenterX ( insets ) - currentImage.getWidth () / 2 );
final int y = verticalAlignment == TOP ? insets.top :
( verticalAlignment == BOTTOM ? getHeight () - currentImage.getHeight () - insets.bottom :
getCenterY ( insets ) - currentImage.getHeight () / 2 );
g2d.setPaint ( new TexturePaint ( currentImage,
new Rectangle2D.Double ( x, y, currentImage.getWidth (), currentImage.getHeight () ) ) );
g2d.fillRect ( insets.left, insets.top, getWidth () - insets.left - insets.right,
getHeight () - insets.top - insets.bottom );
break;
}
}
}
}
GraphicsUtils.restoreComposite ( g2d, oc, transparency < 1f );
}
/**
* Returns image component center X coordinate.
*
* @param insets image component insets
* @return image component center X coordinate
*/
protected int getCenterX ( final Insets insets )
{
return insets.left + ( getWidth () - insets.left - insets.right ) / 2;
}
/**
* Returns image component center Y coordinate.
*
* @param insets image component insets
* @return image component center Y coordinate
*/
protected int getCenterY ( final Insets insets )
{
return insets.top + ( getHeight () - insets.top - insets.bottom ) / 2;
}
/**
* Returns preview image for specified insets.
*
* @param insets image component insets
* @return preview image
*/
protected BufferedImage getPreviewImage ( final Insets insets )
{
if ( image.getWidth () > getWidth () || image.getHeight () > getHeight () )
{
final Dimension size = getSize ();
size.setSize ( size.width - insets.left - insets.right, size.height - insets.top - insets.bottom );
if ( lastPreviewImage == null || lastDimention != null && !lastDimention.equals ( size ) )
{
if ( lastPreviewImage != null )
{
lastPreviewImage.flush ();
lastPreviewImage = null;
}
lastPreviewImage = ImageUtils.createPreviewImage ( getCurrentImage (), size );
lastDimention = getSize ();
}
return lastPreviewImage;
}
else
{
return image;
}
}
/**
* Returns currently displayed image.
*
* @return currently displayed image
*/
protected BufferedImage getCurrentImage ()
{
return !isEnabled () && disabledImage != null ? disabledImage : image;
}
/**
* Returns preferred size of image component.
*
* @return preferred size of image component
*/
@Override
public Dimension getPreferredSize ()
{
if ( isPreferredSizeSet () )
{
return super.getPreferredSize ();
}
else
{
return getRequiredSize ();
}
}
/**
* Returns component size required to fully show the image.
*
* @return component size required to fully show the image
*/
protected Dimension getRequiredSize ()
{
final Insets insets = getInsets ();
return new Dimension ( insets.left + ( image != null ? image.getWidth () : 0 ) + insets.right,
insets.top + ( image != null ? image.getHeight () : 0 ) + insets.bottom );
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy