/*
 * Copyright (c) 2000, 2002, 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 sun.awt.image;
import java.awt.image.ColorModel;
import java.awt.image.DataBuffer;

This class provides utilities for converting between the standard rgb colorspace specification and the equivalent value for a pixel of a given surface type. The class was designed for use by the SurfaceType objects, since the conversion between pixel values and rgb values is inherently tied to the type of surface we are dealing with. Some conversions cannot be done automatically, however (for example, the AnyInt or AnyDCM surface types), so we require the caller to pass in a ColorModel object so that we can calculate the pixel values in these generic cases as well.
/** * This class provides utilities for converting between the standard * rgb colorspace specification and the equivalent value for a pixel * of a given surface type. The class was designed for use by the * SurfaceType objects, since the conversion between pixel values * and rgb values is inherently tied to the type of surface we are * dealing with. Some conversions cannot be done automatically, * however (for example, the AnyInt or AnyDCM surface types), so * we require the caller to pass in a ColorModel object so that * we can calculate the pixel values in these generic cases as well. */
public class PixelConverter {
Default object, used as a fallback for any surface types where we do not know enough about the surface to calculate the conversions directly. We use the ColorModel object to assist us in these cases.
/** * Default object, used as a fallback for any surface types where * we do not know enough about the surface to calculate the * conversions directly. We use the ColorModel object to assist * us in these cases. */
public static final PixelConverter instance = new PixelConverter(); protected int alphaMask = 0; protected PixelConverter() {} public int rgbToPixel(int rgb, ColorModel cm) { Object obj = cm.getDataElements(rgb, null); switch (cm.getTransferType()) { case DataBuffer.TYPE_BYTE: byte[] bytearr = (byte[]) obj; int pix = 0; switch(bytearr.length) { default: // bytearr.length >= 4 pix = bytearr[3] << 24; // FALLSTHROUGH case 3: pix |= (bytearr[2] & 0xff) << 16; // FALLSTHROUGH case 2: pix |= (bytearr[1] & 0xff) << 8; // FALLSTHROUGH case 1: pix |= (bytearr[0] & 0xff); } return pix; case DataBuffer.TYPE_SHORT: case DataBuffer.TYPE_USHORT: short[] shortarr = (short[]) obj; return (((shortarr.length > 1) ? shortarr[1] << 16 : 0) | shortarr[0] & 0xffff); case DataBuffer.TYPE_INT: return ((int[]) obj)[0]; default: return rgb; } } public int pixelToRgb(int pixel, ColorModel cm) { // REMIND: Not yet implemented return pixel; } public final int getAlphaMask() { return alphaMask; }
Subclasses of PixelConverter. These subclasses are specific to surface types where we can definitively calculate the conversions. Note that some conversions are lossy; that is, we cannot necessarily convert a value and then convert it back and wind up with the original value. For example, an rgb value that has an alpha != 1 cannot be converted to an Xrgb pixel without losing the information in the alpha component. The conversion strategies associated with the ThreeByte* and FourByte* surface types swap the components around due to the ordering used when the bytes are stored. The low order byte of a packed-byte pixel will be the first byte stored and the high order byte will be the last byte stored. For example, the ThreeByteBgr surface type is associated with an Xrgb conversion object because the three bytes are stored as follows: pixels[0] = b; // low order byte of an Xrgb pixel pixels[1] = g; pixels[2] = r; // high order byte of an Xrgb pixel
/** * Subclasses of PixelConverter. These subclasses are * specific to surface types where we can definitively * calculate the conversions. Note that some conversions * are lossy; that is, we cannot necessarily convert a * value and then convert it back and wind up with the * original value. For example, an rgb value that has * an alpha != 1 cannot be converted to an Xrgb pixel * without losing the information in the alpha component. * * The conversion strategies associated with the ThreeByte* * and FourByte* surface types swap the components around * due to the ordering used when the bytes are stored. The * low order byte of a packed-byte pixel will be the first * byte stored and the high order byte will be the last byte * stored. For example, the ThreeByteBgr surface type is * associated with an Xrgb conversion object because the * three bytes are stored as follows: * pixels[0] = b; // low order byte of an Xrgb pixel * pixels[1] = g; * pixels[2] = r; // high order byte of an Xrgb pixel */
public static class Rgbx extends PixelConverter { public static final PixelConverter instance = new Rgbx(); private Rgbx() {} public int rgbToPixel(int rgb, ColorModel cm) { return (rgb << 8); } public int pixelToRgb(int pixel, ColorModel cm) { return (0xff000000 | (pixel >> 8)); } } public static class Xrgb extends PixelConverter { public static final PixelConverter instance = new Xrgb(); private Xrgb() {} public int rgbToPixel(int rgb, ColorModel cm) { return rgb; } public int pixelToRgb(int pixel, ColorModel cm) { return (0xff000000 | pixel); } } public static class Argb extends PixelConverter { public static final PixelConverter instance = new Argb(); private Argb() { alphaMask = 0xff000000; } public int rgbToPixel(int rgb, ColorModel cm) { return rgb; } public int pixelToRgb(int pixel, ColorModel cm) { return pixel; } } public static class Ushort565Rgb extends PixelConverter { public static final PixelConverter instance = new Ushort565Rgb(); private Ushort565Rgb() {} public int rgbToPixel(int rgb, ColorModel cm) { return (((rgb >> (16 + 3 - 11)) & 0xf800) | ((rgb >> ( 8 + 2 - 5)) & 0x07e0) | ((rgb >> ( 0 + 3 - 0)) & 0x001f)); } public int pixelToRgb(int pixel, ColorModel cm) { int r, g, b; r = (pixel >> 11) & 0x1f; r = (r << 3) | (r >> 2); g = (pixel >> 5) & 0x3f; g = (g << 2) | (g >> 4); b = (pixel ) & 0x1f; b = (b << 3) | (b >> 2); return (0xff000000 | (r << 16) | (g << 8) | (b)); } } public static class Ushort555Rgbx extends PixelConverter { public static final PixelConverter instance = new Ushort555Rgbx(); private Ushort555Rgbx() {} public int rgbToPixel(int rgb, ColorModel cm) { return (((rgb >> (16 + 3 - 11)) & 0xf800) | ((rgb >> ( 8 + 3 - 6)) & 0x07c0) | ((rgb >> ( 0 + 3 - 1)) & 0x003e)); } public int pixelToRgb(int pixel, ColorModel cm) { int r, g, b; r = (pixel >> 11) & 0x1f; r = (r << 3) | (r >> 2); g = (pixel >> 6) & 0x1f; g = (g << 3) | (g >> 2); b = (pixel >> 1) & 0x1f; b = (b << 3) | (b >> 2); return (0xff000000 | (r << 16) | (g << 8) | (b)); } } public static class Ushort555Rgb extends PixelConverter { public static final PixelConverter instance = new Ushort555Rgb(); private Ushort555Rgb() {} public int rgbToPixel(int rgb, ColorModel cm) { return (((rgb >> (16 + 3 - 10)) & 0x7c00) | ((rgb >> ( 8 + 3 - 5)) & 0x03e0) | ((rgb >> ( 0 + 3 - 0)) & 0x001f)); } public int pixelToRgb(int pixel, ColorModel cm) { int r, g, b; r = (pixel >> 10) & 0x1f; r = (r << 3) | (r >> 2); g = (pixel >> 5) & 0x1f; g = (g << 3) | (g >> 2); b = (pixel ) & 0x1f; b = (b << 3) | (b >> 2); return (0xff000000 | (r << 16) | (g << 8) | (b)); } } public static class Ushort4444Argb extends PixelConverter { public static final PixelConverter instance = new Ushort4444Argb(); private Ushort4444Argb() { alphaMask = 0xf000; } public int rgbToPixel(int rgb, ColorModel cm) { // use upper 4 bits for each color // 0xAaRrGgBb -> 0x0000ARGB int a = (rgb >> 16) & 0xf000; int r = (rgb >> 12) & 0x0f00; int g = (rgb >> 8) & 0x00f0; int b = (rgb >> 4) & 0x000f; return (a | r | g | b); } public int pixelToRgb(int pixel, ColorModel cm) { int a, r, g, b; // replicate 4 bits for each color // 0xARGB -> 0xAARRGGBB a = pixel & 0xf000; a = ((pixel << 16) | (pixel << 12)) & 0xff000000; r = pixel & 0x0f00; r = ((pixel << 12) | (pixel << 8)) & 0x00ff0000; g = pixel & 0x00f0; g = ((pixel << 8) | (pixel << 4)) & 0x0000ff00; b = pixel & 0x000f; b = ((pixel << 4) | (pixel << 0)) & 0x000000ff; return (a | r | g | b); } } public static class Xbgr extends PixelConverter { public static final PixelConverter instance = new Xbgr(); private Xbgr() {} public int rgbToPixel(int rgb, ColorModel cm) { return (((rgb & 0xff) << 16) | (rgb & 0xff00) | ((rgb >> 16) & 0xff)); } public int pixelToRgb(int pixel, ColorModel cm) { return (0xff000000 | ((pixel & 0xff) << 16) | (pixel & 0xff00) | ((pixel >> 16) & 0xff)); } } public static class Bgrx extends PixelConverter { public static final PixelConverter instance = new Bgrx(); private Bgrx() {} public int rgbToPixel(int rgb, ColorModel cm) { return ((rgb << 24) | ((rgb & 0xff00) << 8) | ((rgb >> 8) & 0xff00)); } public int pixelToRgb(int pixel, ColorModel cm) { return (0xff000000 | ((pixel & 0xff00) << 8) | ((pixel >> 8) & 0xff00) | (pixel >>> 24)); } } public static class Rgba extends PixelConverter { public static final PixelConverter instance = new Rgba(); private Rgba() { alphaMask = 0x000000ff; } public int rgbToPixel(int rgb, ColorModel cm) { return ((rgb << 8) | (rgb >>> 24)); } public int pixelToRgb(int pixel, ColorModel cm) { return ((pixel << 24) | (pixel >>> 8)); } } public static class RgbaPre extends PixelConverter { public static final PixelConverter instance = new RgbaPre(); private RgbaPre() { alphaMask = 0x000000ff; } public int rgbToPixel(int rgb, ColorModel cm) { if ((rgb >> 24) == -1) { return ((rgb << 8) | (rgb >>> 24)); } int a = rgb >>> 24; int r = (rgb >> 16) & 0xff; int g = (rgb >> 8) & 0xff; int b = (rgb ) & 0xff; int a2 = a + (a >> 7); r = (r * a2) >> 8; g = (g * a2) >> 8; b = (b * a2) >> 8; return ((r << 24) | (g << 16) | (b << 8) | (a)); } public int pixelToRgb(int pixel, ColorModel cm) { int a = pixel & 0xff; if ((a == 0xff) || (a == 0)) { return ((pixel >>> 8) | (pixel << 24)); } int r = pixel >>> 24; int g = (pixel >> 16) & 0xff; int b = (pixel >> 8) & 0xff; r = ((r << 8) - r) / a; g = ((g << 8) - g) / a; b = ((b << 8) - b) / a; return ((r << 24) | (g << 16) | (b << 8) | (a)); } } public static class ArgbPre extends PixelConverter { public static final PixelConverter instance = new ArgbPre(); private ArgbPre() { alphaMask = 0xff000000; } public int rgbToPixel(int rgb, ColorModel cm) { if ((rgb >> 24) == -1) { return rgb; } int a = rgb >>> 24; int r = (rgb >> 16) & 0xff; int g = (rgb >> 8) & 0xff; int b = (rgb ) & 0xff; int a2 = a + (a >> 7); r = (r * a2) >> 8; g = (g * a2) >> 8; b = (b * a2) >> 8; return ((a << 24) | (r << 16) | (g << 8) | (b)); } public int pixelToRgb(int pixel, ColorModel cm) { int a = pixel >>> 24; if ((a == 0xff) || (a == 0)) { return pixel; } int r = (pixel >> 16) & 0xff; int g = (pixel >> 8) & 0xff; int b = (pixel ) & 0xff; r = ((r << 8) - r) / a; g = ((g << 8) - g) / a; b = ((b << 8) - b) / a; return ((a << 24) | (r << 16) | (g << 8) | (b)); } } public static class ArgbBm extends PixelConverter { public static final PixelConverter instance = new ArgbBm(); private ArgbBm() {} public int rgbToPixel(int rgb, ColorModel cm) { return (rgb | ((rgb >> 31) << 24)); } public int pixelToRgb(int pixel, ColorModel cm) { return ((pixel << 7) >> 7); } } public static class ByteGray extends PixelConverter { static final double RED_MULT = 0.299; static final double GRN_MULT = 0.587; static final double BLU_MULT = 0.114; public static final PixelConverter instance = new ByteGray(); private ByteGray() {} public int rgbToPixel(int rgb, ColorModel cm) { int red = (rgb >> 16) & 0xff; int grn = (rgb >> 8) & 0xff; int blu = (rgb ) & 0xff; return (int) (red * RED_MULT + grn * GRN_MULT + blu * BLU_MULT + 0.5); } public int pixelToRgb(int pixel, ColorModel cm) { return ((((((0xff << 8) | pixel) << 8) | pixel) << 8) | pixel); } } public static class UshortGray extends ByteGray { static final double SHORT_MULT = 257.0; // (65535.0 / 255.0); static final double USHORT_RED_MULT = RED_MULT * SHORT_MULT; static final double USHORT_GRN_MULT = GRN_MULT * SHORT_MULT; static final double USHORT_BLU_MULT = BLU_MULT * SHORT_MULT; public static final PixelConverter instance = new UshortGray(); private UshortGray() {} public int rgbToPixel(int rgb, ColorModel cm) { int red = (rgb >> 16) & 0xff; int grn = (rgb >> 8) & 0xff; int blu = (rgb ) & 0xff; return (int) (red * USHORT_RED_MULT + grn * USHORT_GRN_MULT + blu * USHORT_BLU_MULT + 0.5); } public int pixelToRgb(int pixel, ColorModel cm) { pixel = pixel >> 8; return ((((((0xff << 8) | pixel) << 8) | pixel) << 8) | pixel); } } }