org.netbeans.modeler.util.SVGImageUtilities Maven / Gradle / Ivy
/**
* Copyright 2013-2022 Gaurav Gupta
*
* Licensed 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.netbeans.modeler.util;
import java.awt.Component;
import java.awt.Graphics;
import java.awt.HeadlessException;
import java.awt.Image;
import java.awt.MediaTracker;
import java.awt.Toolkit;
import java.awt.Transparency;
import java.awt.image.BufferedImage;
import java.awt.image.ColorModel;
import java.awt.image.FilteredImageSource;
import java.awt.image.ImageObserver;
import java.awt.image.ImageProducer;
import java.awt.image.RGBImageFilter;
import java.awt.image.WritableRaster;
import java.io.IOException;
import java.lang.ref.SoftReference;
import java.net.URL;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Hashtable;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.imageio.ImageIO;
import javax.imageio.ImageReadParam;
import javax.imageio.ImageReader;
import javax.imageio.stream.ImageInputStream;
import javax.swing.Icon;
import javax.swing.ImageIcon;
import javax.swing.JLabel;
import javax.swing.UIManager;
import org.openide.util.ImageUtilities;
import org.openide.util.Lookup;
import org.openide.util.LookupEvent;
import org.openide.util.LookupListener;
import org.openide.util.NbBundle;
import org.openide.util.Parameters;
import org.openide.util.Utilities;
public class SVGImageUtilities {
/**
* separator for individual parts of tool tip text
*/
static final String TOOLTIP_SEPAR = "
"; // NOI18N
/**
* a value that indicates that the icon does not exists
*/
private static final ActiveRef NO_ICON = new ActiveRef(null, null, null);
private static final Map> cache = new HashMap>(128);
private static final Map> localizedCache = new HashMap>(128);
private static final Map> compositeCache = new HashMap>(128);
private static final Map> imageToolTipCache = new HashMap>(128);
private static RGBImageFilter imageIconFilter = null;
/**
* Resource paths for which we have had to strip initial slash.
*
* @see "#20072"
*/
private static final Set extraInitialSlashes = new HashSet();
private static volatile Object currentLoader;
private static Lookup.Result loaderQuery = null;
private static boolean noLoaderWarned = false;
private static final Component component = new Component() {
};
private static final MediaTracker tracker = new MediaTracker(component);
private static int mediaTrackerID;
private static ImageReader PNG_READER;
// private static ImageReader GIF_READER;
private static final Logger ERR = Logger.getLogger(ImageUtilities.class.getName());
private SVGImageUtilities() {
}
static {
ImageIO.setUseCache(false);
PNG_READER = ImageIO.getImageReadersByMIMEType("image/png").next();
// GIF_READER = ImageIO.getImageReadersByMIMEType("image/gif").next();
}
/**
* Loads an image from the specified resource ID. The image is loaded using
* the "system" classloader registered in Lookup.
*
* @param resourceID resource path of the icon (no initial slash)
* @return icon's Image, or null, if the icon cannot be loaded.
*/
public static final Image loadImage(String resourceID) {
return getIcon(resourceID, false);
}
/**
* Loads an image based on resource path. Exactly like
* {@link #loadImage(String)} but may do a localized search. For example,
* requesting org/netbeans/modules/foo/resources/foo.gif
* might actually find
* org/netbeans/modules/foo/resources/foo_ja.gif
* or org/netbeans/modules/foo/resources/foo_mybranding.gif.
*
*
* Caching of loaded images can be used internally to improve performance.
*
* Since version 8.12 the returned image object responds to call
* image.getProperty("url", null)
by returning the internal
* {@link URL} of the found and loaded resource
.
*
* @param resource resource path of the image (no initial slash)
* @param localized true for localized search
* @return icon's Image or null if the icon cannot be loaded
*/
public static final Image loadImage(String resource, boolean localized) {
return getIcon(resource, localized);
}
/**
* Loads an icon based on resource path. Similar to
* {@link #loadImage(String, boolean)}, returns ImageIcon instead of Image.
*
* @param resource resource path of the icon (no initial slash)
* @param localized localized resource should be used
* @return ImageIcon or null, if the icon cannot be loaded.
* @since 7.22
*/
public static final ImageIcon loadImageIcon(String resource, boolean localized) {
Image image = getIcon(resource, localized);
if (image == null) {
return null;
}
RGBImageFilter imageFilter = getImageIconFilter();
if (null != imageFilter) {
image = Toolkit.getDefaultToolkit().createImage(new FilteredImageSource(image.getSource(), imageFilter));
}
return (ImageIcon) image2Icon(image);
}
private static RGBImageFilter getImageIconFilter() {
if (null == imageIconFilter) {
Object obj = UIManager.get("nb.imageicon.filter"); //NOI18N
if (obj instanceof RGBImageFilter) {
imageIconFilter = (RGBImageFilter) obj;
}
}
return imageIconFilter;
}
/**
* This method merges two images into the new one. The second image is drawn
* over the first one with its top-left corner at x, y. Images need not be
* of the same size. New image will have a size of max(second image size +
* top-left corner, first image size). Method is used mostly when second
* image contains transparent pixels (e.g. for badging). Method that
* attempts to find the merged image in the cache first, then creates the
* image if it was not found.
*
* @param image1 underlying image
* @param image2 second image
* @param x x position of top-left corner
* @param y y position of top-left corner
* @return new merged image
*/
public static final Image mergeImages(Image image1, Image image2, int x, int y) {
if (image1 == null || image2 == null) {
throw new NullPointerException();
}
CompositeImageKey k = new CompositeImageKey(image1, image2, x, y);
Image cached;
synchronized (compositeCache) {
ActiveRef r = compositeCache.get(k);
if (r != null) {
cached = r.get();
if (cached != null) {
return cached;
}
}
cached = doMergeImages(image1, image2, x, y);
compositeCache.put(k, new ActiveRef(cached, compositeCache, k));
return cached;
}
}
/**
* Converts given image to an icon.
*
* @param image to be converted
* @return icon corresponding icon
*/
public static final Icon image2Icon(Image image) {
if (image instanceof ToolTipImage) {
return ((ToolTipImage) image).getIcon();
} else {
return new ImageIcon(image);
}
}
/**
* Converts given icon to a {@link java.awt.Image}.
*
* @param icon {@link javax.swing.Icon} to be converted.
*/
public static final Image icon2Image(Icon icon) {
if (icon instanceof ImageIcon) {
return ((ImageIcon) icon).getImage();
} else {
ToolTipImage image = new ToolTipImage("", icon.getIconWidth(), icon.getIconHeight(), BufferedImage.TYPE_INT_ARGB);
Graphics g = image.getGraphics();
icon.paintIcon(new JLabel(), g, 0, 0);
g.dispose();
return image;
}
}
/**
* Assign tool tip text to given image (creates new or returns cached,
* original remains unmodified) Text can contain HTML tags e.g.
* "<b>my</b> text"
*
* @param image image to which tool tip should be set
* @param text tool tip text
* @return Image with attached tool tip
*/
public static final Image assignToolTipToImage(Image image, String text) {
Parameters.notNull("image", image);
Parameters.notNull("text", text);
ToolTipImageKey key = new ToolTipImageKey(image, text);
Image cached;
synchronized (imageToolTipCache) {
ActiveRef r = imageToolTipCache.get(key);
if (r != null) {
cached = r.get();
if (cached != null) {
return cached;
}
}
cached = ToolTipImage.createNew(text, image, null);
imageToolTipCache.put(key, new ActiveRef(cached, imageToolTipCache, key));
return cached;
}
}
/**
* Get tool tip text for given image
*
* @param image image which is asked for tool tip text
* @return String containing attached tool tip text
*/
public static final String getImageToolTip(Image image) {
if (image instanceof ToolTipImage) {
return ((ToolTipImage) image).toolTipText;
} else {
return "";
}
}
/**
* Add text to tool tip for given image (creates new or returns cached,
* original remains unmodified) Text can contain HTML tags e.g.
* "<b>my</b> text"
*
* @param text text to add to tool tip
* @return Image with attached tool tip
*/
public static final Image addToolTipToImage(Image image, String text) {
if (image instanceof ToolTipImage) {
ToolTipImage tti = (ToolTipImage) image;
StringBuilder str = new StringBuilder(tti.toolTipText);
if (str.length() > 0 && text.length() > 0) {
str.append(TOOLTIP_SEPAR);
}
str.append(text);
return assignToolTipToImage(image, str.toString());
} else {
return assignToolTipToImage(image, text);
}
}
/**
* Creates disabled (color saturation lowered) icon. Icon image conversion
* is performed lazily.
*
* @param icon original icon used for conversion
* @return less saturated Icon
* @since 7.28
*/
public static Icon createDisabledIcon(Icon icon) {
Parameters.notNull("icon", icon);
return new LazyDisabledIcon(icon2Image(icon));
}
/**
* Creates disabled (color saturation lowered) image.
*
* @param image original image used for conversion
* @return less saturated Image
* @since 7.28
*/
public static Image createDisabledImage(Image image) {
Parameters.notNull("image", image);
return LazyDisabledIcon.createDisabledImage(image);
}
/**
* Get the class loader from lookup. Since this is done very frequently, it
* is wasteful to query lookup each time. Instead, remember the last result
* and just listen for changes.
*/
static ClassLoader getLoader() {
Object is = currentLoader;
if (is instanceof ClassLoader) {
return (ClassLoader) is;
}
currentLoader = Thread.currentThread();
if (loaderQuery == null) {
loaderQuery = Lookup.getDefault().lookup(new Lookup.Template(ClassLoader.class));
loaderQuery.addLookupListener(
new LookupListener() {
@Override
public void resultChanged(LookupEvent ev) {
ERR.fine("Loader cleared"); // NOI18N
currentLoader = null;
}
});
}
Iterator it = loaderQuery.allInstances().iterator();
if (it.hasNext()) {
ClassLoader toReturn = (ClassLoader) it.next();
if (currentLoader == Thread.currentThread()) {
currentLoader = toReturn;
}
ERR.fine("Loader computed: " + currentLoader); // NOI18N
return toReturn;
} else {
if (!noLoaderWarned) {
noLoaderWarned = true;
ERR.warning(
"No ClassLoader instance found in " + Lookup.getDefault() // NOI18N
);
}
return null;
}
}
static Image getIcon(String resource, boolean localized) {
if (localized) {
synchronized (localizedCache) {
ActiveRef ref = localizedCache.get(resource);
Image img = null;
// no icon for this name (already tested)
if (ref == NO_ICON) {
return null;
}
if (ref != null) {
// then it is SoftRefrence
img = ref.get();
}
// icon found
if (img != null) {
return img;
}
// find localized or base image
ClassLoader loader = getLoader();
// we'll keep the String probably for long time, optimize it
resource = new String(resource).intern(); // NOPMD
String base;
String ext;
int idx = resource.lastIndexOf('.');
if ((idx != -1) && (idx > resource.lastIndexOf('/'))) {
base = resource.substring(0, idx);
ext = resource.substring(idx);
} else {
base = resource;
ext = ""; // NOI18N
}
// #31008. [PENDING] remove in case package cache is precomputed
java.net.URL baseurl = (loader != null) ? loader.getResource(resource) // NOPMD
: ImageUtilities.class.getClassLoader().getResource(resource);
Iterator it = NbBundle.getLocalizingSuffixes();
while (it.hasNext()) {
String suffix = it.next();
Image i;
if (suffix.length() == 0) {
i = getIcon(resource, loader, false);
} else {
i = getIcon(base + suffix + ext, loader, true);
}
if (i != null) {
localizedCache.put(resource, new ActiveRef(i, localizedCache, resource));
return i;
}
}
localizedCache.put(resource, NO_ICON);
return null;
}
} else {
return getIcon(resource, getLoader(), false);
}
}
/**
* Finds image for given resource.
*
* @param name name of the resource
* @param loader classloader to use for locating it, or null to use
* classpath
* @param localizedQuery whether the name contains some localization suffix
* and is not optimized/interned
*/
private static Image getIcon(String name, ClassLoader loader, boolean localizedQuery) {
ActiveRef ref = cache.get(name);
Image img = null;
// no icon for this name (already tested)
if (ref == NO_ICON) {
return null;
}
if (ref != null) {
img = ref.get();
}
// icon found
if (img != null) {
return img;
}
synchronized (cache) {
// again under the lock
ref = cache.get(name);
// no icon for this name (already tested)
if (ref == NO_ICON) {
return null;
}
if (ref != null) {
// then it is SoftRefrence
img = ref.get();
}
if (img != null) {
// cannot be NO_ICON, since it never disappears from the map.
return img;
}
// path for bug in classloader
String n;
boolean warn;
if (name.startsWith("/")) { // NOI18N
warn = true;
n = name.substring(1);
} else {
warn = false;
n = name;
}
// we have to load it
java.net.URL url = (loader != null) ? loader.getResource(n)
: ImageUtilities.class.getClassLoader().getResource(n);
// img = (url == null) ? null : Toolkit.getDefaultToolkit().createImage(url);
Image result = null;
try {
if (url != null) {
if (name.endsWith(".png")) {
ImageInputStream stream = ImageIO.createImageInputStream(url.openStream());
ImageReadParam param = PNG_READER.getDefaultReadParam();
try {
PNG_READER.setInput(stream, true, true);
result = PNG_READER.read(0, param);
} catch (IOException ioe1) {
ERR.log(Level.INFO, "Image " + name + " is not PNG", ioe1);
}
stream.close();
}
/*
else if (name.endsWith(".gif")) {
ImageInputStream stream = ImageIO.createImageInputStream(url.openStream());
ImageReadParam param = GIF_READER.getDefaultReadParam();
try {
GIF_READER.setInput(stream, true, true);
result = GIF_READER.read(0, param);
}
catch (IOException ioe1) {
ERR.log(Level.INFO, "Image "+name+" is not GIF", ioe1);
}
stream.close();
}
*/
if (result == null) {
result = ImageIO.read(url);
}
}
} catch (IOException ioe) {
ERR.log(Level.WARNING, "Cannot load " + name + " image", ioe);
}
if (result != null) {
if (warn && extraInitialSlashes.add(name)) {
ERR.warning(
"Initial slashes in Utilities.loadImage deprecated (cf. #20072): "
+ name); // NOI18N
}
// Image img2 = toBufferedImage(result);
ERR.log(Level.FINE, "loading icon {0} = {1}", new Object[]{n, result});
name = new String(name).intern(); // NOPMD
result = ToolTipImage.createNew("", result, url);
cache.put(name, new ActiveRef(result, cache, name));
return result;
} else { // no icon found
if (!localizedQuery) {
cache.put(name, NO_ICON);
}
return null;
}
}
}
/**
* The method creates a BufferedImage which represents the same Image as the
* parameter but consumes less memory.
*/
static final Image toBufferedImage(Image img) {
// load the image
new javax.swing.ImageIcon(img, "");
if (img.getHeight(null) * img.getWidth(null) > 24 * 24) {
return img;
}
java.awt.image.BufferedImage rep = createBufferedImage(img.getWidth(null), img.getHeight(null));
java.awt.Graphics g = rep.createGraphics();
g.drawImage(img, 0, 0, null);
g.dispose();
img.flush();
return rep;
}
private static void ensureLoaded(Image image) {
if ((Toolkit.getDefaultToolkit().checkImage(image, -1, -1, null)
& (ImageObserver.ALLBITS | ImageObserver.FRAMEBITS)) != 0) {
return;
}
synchronized (tracker) {
int id = ++mediaTrackerID;
tracker.addImage(image, id);
try {
tracker.waitForID(id, 0);
} catch (InterruptedException e) {
System.out.println("INTERRUPTED while loading Image");
}
assert (tracker.statusID(id, false) == MediaTracker.COMPLETE) : "Image loaded";
tracker.removeImage(image, id);
}
}
private static final Image doMergeImages(Image image1, Image image2, int x, int y) {
ensureLoaded(image1);
ensureLoaded(image2);
int w = Math.max(image1.getWidth(null), x + image2.getWidth(null));
int h = Math.max(image1.getHeight(null), y + image2.getHeight(null));
boolean bitmask = (image1 instanceof Transparency) && ((Transparency) image1).getTransparency() != Transparency.TRANSLUCENT
&& (image2 instanceof Transparency) && ((Transparency) image2).getTransparency() != Transparency.TRANSLUCENT;
StringBuilder str = new StringBuilder(image1 instanceof ToolTipImage ? ((ToolTipImage) image1).toolTipText : "");
if (image2 instanceof ToolTipImage) {
String toolTip = ((ToolTipImage) image2).toolTipText;
if (str.length() > 0 && toolTip.length() > 0) {
str.append(TOOLTIP_SEPAR);
}
str.append(toolTip);
}
ColorModel model = colorModel(bitmask ? Transparency.BITMASK : Transparency.TRANSLUCENT);
ToolTipImage buffImage = new ToolTipImage(str.toString(),
model, model.createCompatibleWritableRaster(w, h), model.isAlphaPremultiplied(), null, null);
java.awt.Graphics g = buffImage.createGraphics();
g.drawImage(image1, 0, 0, null);
g.drawImage(image2, x, y, null);
g.dispose();
return buffImage;
}
/**
* Creates BufferedImage with Transparency.TRANSLUCENT
*/
static final java.awt.image.BufferedImage createBufferedImage(int width, int height) {
if (Utilities.isMac()) {
return new BufferedImage(width, height, BufferedImage.TYPE_INT_ARGB_PRE);
}
ColorModel model = colorModel(java.awt.Transparency.TRANSLUCENT);
java.awt.image.BufferedImage buffImage = new java.awt.image.BufferedImage(
model, model.createCompatibleWritableRaster(width, height), model.isAlphaPremultiplied(), null);
return buffImage;
}
static private ColorModel colorModel(int transparency) {
ColorModel model;
try {
model = java.awt.GraphicsEnvironment.getLocalGraphicsEnvironment()
.getDefaultScreenDevice().getDefaultConfiguration()
.getColorModel(transparency);
} catch (ArrayIndexOutOfBoundsException aioobE) {
//#226279
model = ColorModel.getRGBdefault();
} catch (HeadlessException he) {
model = ColorModel.getRGBdefault();
}
return model;
}
/**
* Key used for composite images -- it holds image identities
*/
private static class CompositeImageKey {
Image baseImage;
Image overlayImage;
int x;
int y;
CompositeImageKey(Image base, Image overlay, int x, int y) {
this.x = x;
this.y = y;
this.baseImage = base;
this.overlayImage = overlay;
}
@Override
public boolean equals(Object other) {
if (!(other instanceof CompositeImageKey)) {
return false;
}
CompositeImageKey k = (CompositeImageKey) other;
return (x == k.x) && (y == k.y) && (baseImage == k.baseImage) && (overlayImage == k.overlayImage);
}
@Override
public int hashCode() {
int hash = ((x << 3) ^ y) << 4;
hash = hash ^ baseImage.hashCode() ^ overlayImage.hashCode();
return hash;
}
@Override
public String toString() {
return "Composite key for " + baseImage + " + " + overlayImage + " at [" + x + ", " + y + "]"; // NOI18N
}
}
/**
* Key used for ToolTippedImage
*/
private static class ToolTipImageKey {
Image image;
String str;
ToolTipImageKey(Image image, String str) {
this.image = image;
this.str = str;
}
@Override
public boolean equals(Object other) {
if (!(other instanceof ToolTipImageKey)) {
return false;
}
ToolTipImageKey k = (ToolTipImageKey) other;
return (str.equals(k.str)) && (image == k.image);
}
@Override
public int hashCode() {
int hash = image.hashCode() ^ str.hashCode();
return hash;
}
@Override
public String toString() {
return "ImageStringKey for " + image + " + " + str; // NOI18N
}
}
/**
* Cleaning reference.
*/
private static final class ActiveRef extends SoftReference implements Runnable {
private final Map> holder;
private final T key;
public ActiveRef(Image o, Map> holder, T key) {
super(o, Utilities.activeReferenceQueue());
this.holder = holder;
this.key = key;
}
@Override
public void run() {
synchronized (holder) {
holder.remove(key);
}
}
}
// end of ActiveRef
/**
* Image with tool tip text (for icons with badges)
*/
private static class ToolTipImage extends BufferedImage implements Icon {
final String toolTipText;
ImageIcon imageIcon;
final URL url;
public static ToolTipImage createNew(String toolTipText, Image image, URL url) {
SVGImageUtilities.ensureLoaded(image);
boolean bitmask = (image instanceof Transparency) && ((Transparency) image).getTransparency() != Transparency.TRANSLUCENT;
ColorModel model = colorModel(bitmask ? Transparency.BITMASK : Transparency.TRANSLUCENT);
int w = image.getWidth(null);
int h = image.getHeight(null);
ToolTipImage newImage = new ToolTipImage(
toolTipText,
model,
model.createCompatibleWritableRaster(w, h),
model.isAlphaPremultiplied(), null, url);
java.awt.Graphics g = newImage.createGraphics();
g.drawImage(image, 0, 0, null);
g.dispose();
return newImage;
}
public ToolTipImage(
String toolTipText, ColorModel cm, WritableRaster raster,
boolean isRasterPremultiplied, Hashtable, ?> properties, URL url) {
super(cm, raster, isRasterPremultiplied, properties);
this.toolTipText = toolTipText;
this.url = url;
}
public ToolTipImage(String toolTipText, int width, int height, int imageType) {
super(width, height, imageType);
this.toolTipText = toolTipText;
this.url = null;
}
synchronized ImageIcon getIcon() {
if (imageIcon == null) {
imageIcon = new ImageIcon(this);
}
return imageIcon;
}
@Override
public int getIconHeight() {
return super.getHeight();
}
@Override
public int getIconWidth() {
return super.getWidth();
}
@Override
public void paintIcon(Component c, Graphics g, int x, int y) {
g.drawImage(this, x, y, null);
}
@Override
public Object getProperty(String name, ImageObserver observer) {
if ("url".equals(name)) { // NOI18N
if (url != null) {
return url;
} else {
return imageIcon.getImage().getProperty("url", observer);
}
}
return super.getProperty(name, observer);
}
}
private static class LazyDisabledIcon implements Icon {
/**
* Shared instance of filter for disabled icons
*/
private static final RGBImageFilter DISABLED_BUTTON_FILTER = new DisabledButtonFilter();
private Image img;
private Icon disabledIcon;
public LazyDisabledIcon(Image img) {
assert null != img;
this.img = img;
}
@Override
public void paintIcon(Component c, Graphics g, int x, int y) {
getDisabledIcon().paintIcon(c, g, x, y);
}
@Override
public int getIconWidth() {
return getDisabledIcon().getIconWidth();
}
@Override
public int getIconHeight() {
return getDisabledIcon().getIconHeight();
}
private synchronized Icon getDisabledIcon() {
if (null == disabledIcon) {
disabledIcon = new ImageIcon(createDisabledImage(img));
}
return disabledIcon;
}
static Image createDisabledImage(Image img) {
ImageProducer prod = new FilteredImageSource(img.getSource(), DISABLED_BUTTON_FILTER);
return Toolkit.getDefaultToolkit().createImage(prod);
}
}
private static class DisabledButtonFilter extends RGBImageFilter {
DisabledButtonFilter() {
canFilterIndexColorModel = true;
}
@Override
public int filterRGB(int x, int y, int rgb) {
// Reduce the color bandwidth in quarter (>> 2) and Shift 0x88.
return (rgb & 0xff000000) + 0x888888 + ((((rgb >> 16) & 0xff) >> 2) << 16) + ((((rgb >> 8) & 0xff) >> 2) << 8) + (((rgb) & 0xff) >> 2);
}
// override the superclass behaviour to not pollute
// the heap with useless properties strings. Saves tens of KBs
@Override
public void setProperties(Hashtable props) {
props = (Hashtable) props.clone();
consumer.setProperties(props);
}
}
}