org.xhtmlrenderer.swing.DeferredImageReplacedElement Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of flying-saucer-core Show documentation
Show all versions of flying-saucer-core Show documentation
Flying Saucer is a CSS 2.1 renderer written in Java. This artifact contains the core rendering and layout code as well as Java2D output.
/*
* {{{ header & license
* Copyright (c) 2009 Patrick Wright
*
* This program 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 program 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 program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
* }}}
*/
package org.xhtmlrenderer.swing;
import org.xhtmlrenderer.resource.ImageResource;
import org.xhtmlrenderer.util.Configuration;
import org.xhtmlrenderer.util.ImageUtil;
import org.xhtmlrenderer.util.XRLog;
import javax.swing.*;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.util.logging.Level;
/**
* An DeferredImageReplacedElement is a {@link org.xhtmlrenderer.extend.ReplacedElement} that contains a {@link java.awt.Image}
* which by default is simply a transparent image scaled to the size provided to the constructor. The DeferredImageReplacedElement
* also has a reference to an ImageResource which points to the image which will be returned for this replaced element.
* That Image may be loaded some time after this DeferredImageReplacedElement is created. Calling getImage() on
* instances of DeferredImageReplacedElement will return either the original dummy image, or the actual image
* loaded into the ImageResource.
*/
public class DeferredImageReplacedElement extends ImageReplacedElement {
private Point _location = new Point(0, 0);
private final RepaintListener repaintListener;
private final int _targetHeight;
private final int _targetWidth;
private final boolean _doScaleImage;
private boolean _loaded;
private final ImageResource _imageResource;
/**
* Creates a new ImageReplacedElement and scales it to the size specified if either width or height has a valid
* value (values are greater than -1), otherwise original size is preserved. The idea is that the image was loaded at
* a certain size (that's the Image instance here) and that at the time we create the ImageReplacedElement
* we have a target W/H we want to use.
*/
public DeferredImageReplacedElement(ImageResource imageResource, RepaintListener repaintListener, int w, int h) {
this._imageResource = imageResource;
_loaded = false;
this.repaintListener = repaintListener;
if (w == -1 && h == -1) {
_doScaleImage = false;
_targetHeight = 1;
_targetWidth = 1;
} else {
_doScaleImage = true;
_targetHeight = Math.max(1, h);
_targetWidth = Math.max(1, w);
}
_image = ImageUtil.createCompatibleBufferedImage(_targetWidth, _targetHeight);
}
public int getIntrinsicHeight() {
return _loaded ? _image.getHeight(null) : _targetHeight;
}
public int getIntrinsicWidth() {
return _loaded ? _image.getWidth(null) : _targetWidth;
}
public Point getLocation() {
return _location;
}
public void setLocation(int x, int y) {
_location = new Point(x, y);
}
/**
* The image we're replacing.
*/
public Image getImage() {
if (!_loaded && _imageResource.isLoaded()) {
Image image = ((AWTFSImage) _imageResource.getImage()).getImage();
if (_doScaleImage && (_targetWidth > 0 || _targetHeight > 0)) {
int w = image.getWidth(null);
int h = image.getHeight(null);
int newW = _targetWidth;
int newH = _targetHeight;
if (newW == -1) {
newW = (int) (w * ((double) newH / h));
}
if (newH == -1) {
newH = (int) (h * ((double) newW / w));
}
if (w != newW || h != newH) {
if (image instanceof BufferedImage) {
image = ImageUtil.getScaledInstance((BufferedImage) image, newW, newH);
} else {
if (true) {
throw new RuntimeException("image is not a buffered image! " + _imageResource.getImageUri());
}
String scalingType = Configuration.valueFor("xr.image.scale", "HIGH").trim();
if (scalingType.equalsIgnoreCase("HIGH") || scalingType.equalsIgnoreCase("MID")) {
image = image.getScaledInstance(newW, newH, Image.SCALE_SMOOTH);
} else {
image = image.getScaledInstance(newW, newH, Image.SCALE_FAST);
}
}
}
_image = image;
} else {
_image = image;
}
_loaded = true;
XRLog.load(Level.FINE, "Icon: replaced image " + _imageResource.getImageUri() + ", repaint requested");
SwingUtilities.invokeLater(() -> repaintListener.repaintRequested(_doScaleImage));
}
return _image;
}
}