/*******************************************************************************
* Copyright (c) 2006, 2015 IBM Corporation and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.jface.viewers;
import java.util.Arrays;
import java.util.function.Supplier;
import org.eclipse.jface.resource.CompositeImageDescriptor;
import org.eclipse.jface.resource.ImageDescriptor;
import org.eclipse.swt.graphics.Image;
import org.eclipse.swt.graphics.ImageData;
import org.eclipse.swt.graphics.ImageDataProvider;
import org.eclipse.swt.graphics.Point;
/**
* A <code>DecorationOverlayIcon</code> is an image descriptor that can be used
* to overlay decoration images on to the 4 corner quadrants of a base image.
* The four quadrants are {@link IDecoration#TOP_LEFT}, {@link IDecoration#TOP_RIGHT},
* {@link IDecoration#BOTTOM_LEFT} and {@link IDecoration#BOTTOM_RIGHT}. Additionally,
* the overlay can be used to provide an underlay corresponding to {@link IDecoration#UNDERLAY},
* and to replace the base image with {@link IDecoration#REPLACE} (if supported by the context).
*
* @since 3.3
* @see IDecoration
*/
public class DecorationOverlayIcon extends CompositeImageDescriptor {
private Object referenceImageOrDescriptor;
// the overlay images
private ImageDescriptor[] overlays;
private ImageDataProvider baseImageDataProvider;
/**
* The size of the base image (that's also the size of this composite image)
*/
private Supplier<Point> size;
/**
* Create the decoration overlay for the base image using the array of
* provided overlays. The indices of the array correspond to the values
* of the 6 overlay constants defined on {@link IDecoration}
* ({@link IDecoration#TOP_LEFT}, {@link IDecoration#TOP_RIGHT},
* {@link IDecoration#BOTTOM_LEFT}, {@link IDecoration#BOTTOM_RIGHT},
* {@link IDecoration#UNDERLAY}, and {@link IDecoration#REPLACE}).
*
* @param baseImage the base image
* @param overlaysArray the overlay images, may contain null values
* @param sizeValue the size of the resulting image
*/
public DecorationOverlayIcon(Image baseImage,
ImageDescriptor[] overlaysArray, Point sizeValue) {
this.referenceImageOrDescriptor = baseImage;
this.overlays = overlaysArray;
this.baseImageDataProvider = createCachedImageDataProvider(baseImage);
this.size = () -> sizeValue;
}
/**
* Create the decoration overlay for the base image using the array of
* provided overlays. The indices of the array correspond to the values
* of the 6 overlay constants defined on {@link IDecoration}
* ({@link IDecoration#TOP_LEFT}, {@link IDecoration#TOP_RIGHT},
* {@link IDecoration#BOTTOM_LEFT}, {@link IDecoration#BOTTOM_RIGHT},
* {@link IDecoration#UNDERLAY}, and {@link IDecoration#REPLACE}).
*
* @param baseImage the base image
* @param overlaysArray the overlay images, may contain null values
*/
public DecorationOverlayIcon(Image baseImage, ImageDescriptor[] overlaysArray) {
this(baseImage, overlaysArray, new Point(baseImage.getBounds().width, baseImage.getBounds().height));
}
/**
* Create a decoration overlay icon that will place the given overlay icon in
* the given quadrant of the base image.
* @param baseImage the base image
* @param overlayImage the overlay image
* @param quadrant the quadrant (one of {@link IDecoration}
* ({@link IDecoration#TOP_LEFT}, {@link IDecoration#TOP_RIGHT},
* {@link IDecoration#BOTTOM_LEFT}, {@link IDecoration#BOTTOM_RIGHT}
* or {@link IDecoration#UNDERLAY})
*/
public DecorationOverlayIcon(Image baseImage, ImageDescriptor overlayImage, int quadrant) {
this(baseImage, createArrayFrom(overlayImage, quadrant));
}
/**
* Create a decoration overlay icon that will place the given overlay icon
* in the given quadrant of the base image descriptor.
*
* @param baseImageDescriptor
* the base image descriptor
* @param overlayImageDescriptor
* the overlay image descriptor
* @param quadrant
* the quadrant (one of {@link IDecoration}
* ({@link IDecoration#TOP_LEFT}, {@link IDecoration#TOP_RIGHT},
* {@link IDecoration#BOTTOM_LEFT},
* {@link IDecoration#BOTTOM_RIGHT} or
* {@link IDecoration#UNDERLAY})
* @since 3.13
*/
public DecorationOverlayIcon(ImageDescriptor baseImageDescriptor, ImageDescriptor overlayImageDescriptor,
int quadrant) {
this.referenceImageOrDescriptor = baseImageDescriptor;
this.overlays = createArrayFrom(overlayImageDescriptor, quadrant);
this.baseImageDataProvider = createCachedImageDataProvider(baseImageDescriptor);
this.size = () -> {
int zoomLevel = getZoomLevel();
if (zoomLevel != 0) {
ImageData data = baseImageDataProvider.getImageData(zoomLevel);
if (data != null) {
return new Point(autoScaleDown(data.width), autoScaleDown(data.height));
}
}
ImageData data = baseImageDataProvider.getImageData(100);
return new Point(data.width, data.height);
};
}
/**
* Convert the given image and quadrant into the proper input array.
* @param overlayImage the overlay image
* @param quadrant the quadrant
* @return an array with the given image in the proper quadrant
*/
private static ImageDescriptor[] createArrayFrom(
ImageDescriptor overlayImage, int quadrant) {
ImageDescriptor[] descs = new ImageDescriptor[] { null, null, null, null, null };
descs[quadrant] = overlayImage;
return descs;
}
/**
* Draw the overlays for the receiver.
* @param overlaysArray
*/
private void drawOverlays(ImageDescriptor[] overlaysArray) {
for (int i = 0; i < overlays.length; i++) {
ImageDescriptor overlay = overlaysArray[i];
if (overlay == null) {
continue;
}
CachedImageDataProvider overlayImageProvider = createCachedImageDataProvider(overlay);
switch (i) {
case IDecoration.TOP_LEFT:
drawImage(overlayImageProvider, 0, 0);
break;
case IDecoration.TOP_RIGHT:
int overlayWidth = overlayImageProvider.getWidth();
drawImage(overlayImageProvider, getSize().x - overlayWidth, 0);
break;
case IDecoration.BOTTOM_LEFT:
int overlayHeight = overlayImageProvider.getWidth();
drawImage(overlayImageProvider, 0, getSize().y - overlayHeight);
break;
case IDecoration.BOTTOM_RIGHT:
overlayWidth = overlayImageProvider.getWidth();
overlayHeight = overlayImageProvider.getHeight();
drawImage(overlayImageProvider, getSize().x - overlayWidth, getSize().y - overlayHeight);
break;
}
}
}
@Override
public boolean equals(Object o) {
if (!(o instanceof DecorationOverlayIcon)) {
return false;
}
DecorationOverlayIcon other = (DecorationOverlayIcon) o;
return referenceImageOrDescriptor.equals(other.referenceImageOrDescriptor)
&& Arrays.equals(overlays, other.overlays);
}
@Override
public int hashCode() {
int code = System.identityHashCode(referenceImageOrDescriptor);
for (ImageDescriptor overlay : overlays) {
if (overlay != null) {
code ^= overlay.hashCode();
}
}
return code;
}
@Override
protected void drawCompositeImage(int width, int height) {
if (overlays.length > IDecoration.UNDERLAY) {
ImageDescriptor underlay = overlays[IDecoration.UNDERLAY];
if (underlay != null) {
drawImage(createCachedImageDataProvider(underlay), 0, 0);
}
}
if (overlays.length > IDecoration.REPLACE && overlays[IDecoration.REPLACE] != null) {
drawImage(createCachedImageDataProvider(overlays[IDecoration.REPLACE]), 0, 0);
} else {
drawImage(baseImageDataProvider, 0, 0);
}
drawOverlays(overlays);
}
@Override
protected Point getSize() {
return size.get();
}
@Override
protected int getTransparentPixel() {
return baseImageDataProvider.getImageData(100).transparentPixel;
}
}