VAqua.src.org.violetlib.aqua.AquaIcon Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of vaqua Show documentation
Show all versions of vaqua Show documentation
An improved native Swing look and feel for macOS
The newest version!
/*
* Changes copyright (c) 2015-2021 Alan Snyder.
* All rights reserved.
*
* You may not use, copy or modify this file, except in compliance with the license agreement. For details see
* accompanying license terms.
*/
/*
* Copyright (c) 2011, 2012, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code 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
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package org.violetlib.aqua;
import java.awt.*;
import java.awt.image.BufferedImage;
import javax.swing.*;
import javax.swing.plaf.UIResource;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.violetlib.aqua.AquaUtils.RecyclableSingleton;
import org.violetlib.jnr.Painter;
public class AquaIcon {
public static final IconImageCreator alertCautionImageCreator = new IconImageCreator("caut", 64);
public static final IconImageCreator alertStopImageCreator = new IconImageCreator("stop", 64);
public static final IconImageCreator openFolderImageCreator = new IconImageCreator("ofld", 20);
private static class RecyclableIconImageSingleton extends RecyclableSingleton {
final IconImageCreator imageCreator;
public RecyclableIconImageSingleton(IconImageCreator imageCreator) {
this.imageCreator = imageCreator;
}
protected Image getInstance() {
return imageCreator.getImage();
}
}
private static final RecyclableIconImageSingleton cautionIcon = new RecyclableIconImageSingleton(alertCautionImageCreator);
private static final RecyclableIconImageSingleton stopIcon = new RecyclableIconImageSingleton(alertStopImageCreator);
private static final RecyclableIconImageSingleton openFolderIcon = new RecyclableIconImageSingleton(openFolderImageCreator);
private static final IconResourceLoader browserExpandArrowLoader = new IconResourceLoader("fc/ExpandArrowTemplate.png");
/**
* Create an image for an icon, if possible.
* @param i The icon (may be null).
* @return the corresponding image, or null if not possible.
*/
public static @Nullable Image getImageForIcon(@Nullable Icon i) {
if (i == null) {
return null;
}
if (i instanceof ImageIcon) {
return ((ImageIcon)i).getImage();
}
int w = i.getIconWidth();
int h = i.getIconHeight();
if (w <= 0 || h <= 0) {
return null;
}
// This could be any kind of icon, so we need to make a buffer for it, draw it and then pass the new image off to appkit.
BufferedImage image = new BufferedImage(w, h, BufferedImage.TYPE_INT_ARGB_PRE);
Graphics g = image.getGraphics();
i.paintIcon(null, g, 0, 0);
g.dispose();
return image;
}
public static @NotNull Icon createPressedDarkIcon(@NotNull Icon ic) {
Image im = getImageForIcon(ic);
if (im != null) {
Image pressedImage = AquaImageFactory.getProcessedImage(im, AquaImageFactory.DARKEN_FOR_PRESSED);
return new ImageIcon(pressedImage);
} else {
return ic;
}
}
public static @NotNull Icon createDisabledLightIcon(@NotNull Icon ic) {
Image im = getImageForIcon(ic);
if (im != null) {
Image disabledImage = AquaImageFactory.getProcessedImage(im, AquaImageFactory.LIGHTEN_FOR_DISABLED);
return new ImageIcon(disabledImage);
} else {
return ic;
}
}
public static abstract class CachingScalingIcon implements Icon, UIResource {
int width;
int height;
Image image;
public CachingScalingIcon(int width, int height) {
this.width = width;
this.height = height;
}
void setSize(int width, int height) {
this.width = width;
this.height = height;
this.image = null;
}
@Nullable Image getImage() {
if (image != null) {
return image;
}
if (!GraphicsEnvironment.isHeadless()) {
image = createImage();
}
return image;
}
abstract Image createImage();
public boolean hasIconRef() {
return getImage() != null;
}
public void paintIcon(Component c, Graphics g, int x, int y) {
g = g.create();
if (g instanceof Graphics2D) {
// improves icon rendering quality in Quartz
((Graphics2D)g).setRenderingHint(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY);
}
Image myImage = getImage();
if (myImage != null) {
g.drawImage(myImage, x, y, getIconWidth(), getIconHeight(), null);
}
g.dispose();
}
public int getIconWidth() {
return width;
}
public int getIconHeight() {
return height;
}
}
public static abstract class ScalingNativeRenderedIcon implements Icon, UIResource {
protected final int width;
protected final int height;
public ScalingNativeRenderedIcon(int width, int height) {
this.width = width;
this.height = height;
}
@Override
public void paintIcon(Component c, Graphics g, int x, int y) {
if (GraphicsEnvironment.isHeadless()) {
return;
}
g = g.create();
if (g instanceof Graphics2D) {
// improves icon rendering quality in Quartz
((Graphics2D) g).setRenderingHint(RenderingHints.KEY_RENDERING,
RenderingHints.VALUE_RENDER_QUALITY);
}
org.violetlib.jnr.Painter painter = getPainter(width, height);
g.clipRect(x, y, width, height);
painter.paint(g, x, y);
g.dispose();
}
public abstract Painter getPainter(int width, int height);
@Override
public int getIconWidth() {
return width;
}
@Override
public int getIconHeight() {
return height;
}
}
public static Image getStopIconImage() {
return stopIcon.getInstance();
}
public static Image getCautionIconImage() {
return cautionIcon.getInstance();
}
public static ImageIconUIResource getOpenFolderIcon() {
return new ImageIconUIResource(openFolderIcon.getInstance());
}
public static @NotNull ImageIcon getBrowserExpandArrowTemplate() {
return browserExpandArrowLoader.getIcon();
}
private static class IconResourceLoader {
private final @NotNull String resourceName;
private @Nullable ImageIcon result;
public IconResourceLoader(@NotNull String resourceName) {
this.resourceName = resourceName;
}
public @NotNull ImageIcon getIcon() {
if (result == null) {
result = AquaImageFactory.loadResource(resourceName);
if (result == null) {
throw new UnsupportedOperationException("Missing icon resource: " + resourceName);
}
}
return result;
}
}
private static class IconImageCreator {
private final String osTypeName;
private final int osType;
private final int size;
private Image result;
public IconImageCreator(String osTypeName, int size) {
this.osTypeName = osTypeName;
this.osType = OSTypeToInt(osTypeName);
this.size = size;
}
public Image getImage() {
if (result == null) {
AquaNativeSupport.load();
int[][] buffers = new int[2][];
if (!nativeRenderIcon(osType, buffers, size)) {
if (AquaImageFactory.debugNativeRendering) {
Utils.logDebug("Failed to render image for icon " + osTypeName);
}
throw new UnsupportedOperationException();
}
if (AquaImageFactory.debugNativeRendering) {
Utils.logDebug("Rendered image for icon " + osTypeName);
}
result = AquaMultiResolutionImage.createImage(size, size, buffers[0], buffers[1]);
}
return result;
}
}
@SuppressWarnings("deprecation")
public static int OSTypeToInt(String type) {
int result = 0;
byte b[] = { (byte) 0, (byte) 0, (byte) 0, (byte) 0 };
int len = type.length();
if (len > 0) {
if (len > 4) len = 4;
type.getBytes(0, len, b, 4 - len);
}
for (int i = 0; i < len; i++) {
if (i > 0) result <<= 8;
result |= (b[i] & 0xff);
}
return result;
}
/**
* Render a system icon.
*
* @param osType The OSType code that identifies the icon.
* @param buffers 1x and 2x rasters are stored here (2x is optional)
* @param size The width and height of the icon.
* @return true if successful, false otherwise.
*/
private static native boolean nativeRenderIcon(int osType, int[][] buffers, int size);
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy