com.google.code.appengine.awt.image.BufferedImage Maven / Gradle / Ivy
The newest version!
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.
*/
/**
* @author Igor V. Stolyarov
*/
package com.google.code.appengine.awt.image;
import java.util.Enumeration;
import java.util.Hashtable;
import java.util.Vector;
import org.apache.harmony.awt.gl.ImageSurface;
import org.apache.harmony.awt.gl.Surface;
import org.apache.harmony.awt.gl.image.BufferedImageSource;
import org.apache.harmony.awt.internal.nls.Messages;
import com.google.code.appengine.awt.Graphics;
import com.google.code.appengine.awt.Graphics2D;
import com.google.code.appengine.awt.GraphicsEnvironment;
import com.google.code.appengine.awt.Image;
import com.google.code.appengine.awt.Point;
import com.google.code.appengine.awt.Rectangle;
import com.google.code.appengine.awt.Transparency;
import com.google.code.appengine.awt.color.ColorSpace;
import com.google.code.appengine.awt.image.BufferedImage;
import com.google.code.appengine.awt.image.ColorModel;
import com.google.code.appengine.awt.image.ComponentColorModel;
import com.google.code.appengine.awt.image.DataBuffer;
import com.google.code.appengine.awt.image.DirectColorModel;
import com.google.code.appengine.awt.image.ImageObserver;
import com.google.code.appengine.awt.image.ImageProducer;
import com.google.code.appengine.awt.image.IndexColorModel;
import com.google.code.appengine.awt.image.Raster;
import com.google.code.appengine.awt.image.RenderedImage;
import com.google.code.appengine.awt.image.SampleModel;
import com.google.code.appengine.awt.image.TileObserver;
import com.google.code.appengine.awt.image.WritableRaster;
import com.google.code.appengine.awt.image.WritableRenderedImage;
public class BufferedImage extends
Image implements WritableRenderedImage, Transparency{
public static final int TYPE_CUSTOM = 0;
public static final int TYPE_INT_RGB = 1;
public static final int TYPE_INT_ARGB = 2;
public static final int TYPE_INT_ARGB_PRE = 3;
public static final int TYPE_INT_BGR = 4;
public static final int TYPE_3BYTE_BGR = 5;
public static final int TYPE_4BYTE_ABGR = 6;
public static final int TYPE_4BYTE_ABGR_PRE = 7;
public static final int TYPE_USHORT_565_RGB = 8;
public static final int TYPE_USHORT_555_RGB = 9;
public static final int TYPE_BYTE_GRAY = 10;
public static final int TYPE_USHORT_GRAY = 11;
public static final int TYPE_BYTE_BINARY = 12;
public static final int TYPE_BYTE_INDEXED = 13;
private static final int ALPHA_MASK = 0xff000000;
private static final int RED_MASK = 0x00ff0000;
private static final int GREEN_MASK = 0x0000ff00;
private static final int BLUE_MASK = 0x000000ff;
private static final int RED_BGR_MASK = 0x000000ff;
private static final int GREEN_BGR_MASK = 0x0000ff00;
private static final int BLUE_BGR_MASK = 0x00ff0000;
private static final int RED_565_MASK = 0xf800;
private static final int GREEN_565_MASK = 0x07e0;
private static final int BLUE_565_MASK = 0x001f;
private static final int RED_555_MASK = 0x7c00;
private static final int GREEN_555_MASK = 0x03e0;
private static final int BLUE_555_MASK = 0x001f;
private ColorModel cm;
private final WritableRaster raster;
private final int imageType;
private Hashtable, ?> properties;
// Surface of the Buffered Image - used for blitting one Buffered Image
// on the other one or on the Component
private final ImageSurface imageSurf;
public BufferedImage(ColorModel cm, WritableRaster raster,
boolean isRasterPremultiplied, Hashtable, ?> properties) {
if (!cm.isCompatibleRaster(raster)) {
// awt.4D=The raster is incompatible with this ColorModel
throw new IllegalArgumentException(Messages.getString("awt.4D")); //$NON-NLS-1$
}
if (raster.getMinX() != 0 || raster.getMinY() != 0) {
// awt.228=minX or minY of this raster not equal to zero
throw new IllegalArgumentException(Messages.getString("awt.228")); //$NON-NLS-1$
}
this.cm = cm;
this.raster = raster;
this.properties = properties;
coerceData(isRasterPremultiplied);
imageType = Surface.getType(cm, raster);
imageSurf = createImageSurface(imageType);
}
public BufferedImage(int width, int height, int imageType,
IndexColorModel cm) {
switch (imageType) {
case TYPE_BYTE_BINARY:
if (cm.hasAlpha()) {
// awt.227=This image type can't have alpha
throw new IllegalArgumentException(Messages.getString("awt.227")); //$NON-NLS-1$
}
int pixel_bits = 0;
int mapSize = cm.getMapSize();
if (mapSize <= 2) {
pixel_bits = 1;
} else if (mapSize <= 4) {
pixel_bits = 2;
} else if (mapSize <= 16) {
pixel_bits = 4;
} else {
// awt.221=The imageType is TYPE_BYTE_BINARY and the color map has more than 16 entries
throw new IllegalArgumentException(Messages.getString("awt.221")); //$NON-NLS-1$
}
raster = Raster.createPackedRaster(DataBuffer.TYPE_BYTE, width,
height, 1, pixel_bits, null);
break;
case TYPE_BYTE_INDEXED:
raster = Raster.createInterleavedRaster(DataBuffer.TYPE_BYTE,
width, height, 1, null);
break;
default:
// awt.222=The imageType is not TYPE_BYTE_BINARY or TYPE_BYTE_INDEXED
throw new IllegalArgumentException(Messages.getString("awt.222")); //$NON-NLS-1$
}
if (!cm.isCompatibleRaster(raster)) {
// awt.223=The imageType is not compatible with ColorModel
throw new IllegalArgumentException(Messages.getString("awt.223")); //$NON-NLS-1$
}
this.cm = cm;
this.imageType = imageType;
imageSurf = createImageSurface(imageType);
}
public BufferedImage(int width, int height, int imageType) {
switch (imageType) {
case TYPE_INT_RGB:
cm = new DirectColorModel(24, RED_MASK, GREEN_MASK, BLUE_MASK);
raster = cm.createCompatibleWritableRaster(width, height);
break;
case TYPE_INT_ARGB:
cm = ColorModel.getRGBdefault();
raster = cm.createCompatibleWritableRaster(width, height);
break;
case TYPE_INT_ARGB_PRE:
cm = new DirectColorModel(
ColorSpace.getInstance(ColorSpace.CS_sRGB),
32,
RED_MASK,
GREEN_MASK,
BLUE_MASK,
ALPHA_MASK,
true,
DataBuffer.TYPE_INT);
raster = cm.createCompatibleWritableRaster(width, height);
break;
case TYPE_INT_BGR:
cm = new DirectColorModel(24,
RED_BGR_MASK,
GREEN_BGR_MASK,
BLUE_BGR_MASK);
raster = cm.createCompatibleWritableRaster(width, height);
break;
case TYPE_3BYTE_BGR: {
int bits[] = { 8, 8, 8 };
int bandOffsets[] = { 2, 1, 0 };
cm = new ComponentColorModel(
ColorSpace.getInstance(ColorSpace.CS_sRGB),
bits,
false,
false,
Transparency.OPAQUE,
DataBuffer.TYPE_BYTE);
raster = Raster.createInterleavedRaster(DataBuffer.TYPE_BYTE,
width, height, width * 3, 3, bandOffsets, null);
}
break;
case TYPE_4BYTE_ABGR: {
int bits[] = { 8, 8, 8, 8 };
int bandOffsets[] = { 3, 2, 1, 0 };
cm = new ComponentColorModel(
ColorSpace.getInstance(ColorSpace.CS_sRGB),
bits,
true,
false,
Transparency.TRANSLUCENT,
DataBuffer.TYPE_BYTE);
raster = Raster.createInterleavedRaster(DataBuffer.TYPE_BYTE,
width, height, width * 4, 4, bandOffsets, null);
}
break;
case TYPE_4BYTE_ABGR_PRE: {
int bits[] = { 8, 8, 8, 8 };
int bandOffsets[] = { 3, 2, 1, 0 };
cm = new ComponentColorModel(
ColorSpace.getInstance(ColorSpace.CS_sRGB),
bits,
true,
true,
Transparency.TRANSLUCENT, DataBuffer.TYPE_BYTE);
raster = Raster.createInterleavedRaster(DataBuffer.TYPE_BYTE,
width, height, width * 4, 4, bandOffsets, null);
}
break;
case TYPE_USHORT_565_RGB:
cm = new DirectColorModel(
ColorSpace.getInstance(ColorSpace.CS_sRGB),
16,
RED_565_MASK,
GREEN_565_MASK,
BLUE_565_MASK,
0,
false,
DataBuffer.TYPE_USHORT);
raster = cm.createCompatibleWritableRaster(width, height);
break;
case TYPE_USHORT_555_RGB:
cm = new DirectColorModel(
ColorSpace.getInstance(ColorSpace.CS_sRGB),
15,
RED_555_MASK,
GREEN_555_MASK,
BLUE_555_MASK,
0,
false,
DataBuffer.TYPE_USHORT);
raster = cm.createCompatibleWritableRaster(width, height);
break;
case TYPE_BYTE_GRAY: {
int bits[] = { 8 };
cm = new ComponentColorModel(
ColorSpace.getInstance(ColorSpace.CS_GRAY),
bits,
false,
false,
Transparency.OPAQUE,
DataBuffer.TYPE_BYTE);
raster = cm.createCompatibleWritableRaster(width, height);
}
break;
case TYPE_USHORT_GRAY: {
int bits[] = { 16 };
cm = new ComponentColorModel(
ColorSpace.getInstance(ColorSpace.CS_GRAY),
bits,
false,
false,
Transparency.OPAQUE,
DataBuffer.TYPE_USHORT);
raster = cm.createCompatibleWritableRaster(width, height);
}
break;
case TYPE_BYTE_BINARY: {
int colorMap[] = { 0, 0xffffff };
cm = new IndexColorModel(1, 2, colorMap, 0, false, -1,
DataBuffer.TYPE_BYTE);
raster = Raster.createPackedRaster(DataBuffer.TYPE_BYTE, width,
height, 1, 1, null);
}
break;
case TYPE_BYTE_INDEXED: {
int colorMap[] = new int[256];
int i = 0;
for (int r = 0; r < 256; r += 51) {
for (int g = 0; g < 256; g += 51) {
for (int b = 0; b < 256; b += 51) {
colorMap[i] = (r << 16) | (g << 8) | b;
i++;
}
}
}
int gray = 0x12;
for (; i < 256; i++, gray += 6) {
colorMap[i] = (gray << 16) | (gray << 8) | gray;
}
cm = new IndexColorModel(8, 256, colorMap, 0, false, -1,
DataBuffer.TYPE_BYTE);
raster = Raster.createInterleavedRaster(DataBuffer.TYPE_BYTE,
width, height, 1, null);
}
break;
default:
// awt.224=Unknown image type
throw new IllegalArgumentException(Messages.getString("awt.224")); //$NON-NLS-1$
}
this.imageType = imageType;
imageSurf = createImageSurface(imageType);
}
@Override
public Object getProperty(String name, ImageObserver observer) {
return getProperty(name);
}
public Object getProperty(String name) {
if(name == null) {
// awt.225=Property name is null
throw new NullPointerException(Messages.getString("awt.225")); //$NON-NLS-1$
}
if (properties == null) {
return Image.UndefinedProperty;
}
Object property = properties.get(name);
if (property == null) {
property = Image.UndefinedProperty;
}
return property;
}
public WritableRaster copyData(WritableRaster outRaster) {
if (outRaster == null) {
outRaster = Raster.createWritableRaster(raster.getSampleModel(),
new Point(raster.getSampleModelTranslateX(),
raster.getSampleModelTranslateY()));
}
int w = outRaster.getWidth();
int h = outRaster.getHeight();
int minX = outRaster.getMinX();
int minY = outRaster.getMinY();
Object data = null;
data = raster.getDataElements(minX, minY, w, h, data);
outRaster.setDataElements(minX, minY, w, h, data);
return outRaster;
}
public Raster getData(Rectangle rect) {
int minX = rect.x;
int minY = rect.y;
int w = rect.width;
int h = rect.height;
SampleModel sm = raster.getSampleModel();
SampleModel nsm = sm.createCompatibleSampleModel(w, h);
WritableRaster outr = Raster.createWritableRaster(nsm,
rect.getLocation());
Object data = null;
data = raster.getDataElements(minX, minY, w, h, data);
outr.setDataElements(minX, minY, w, h, data);
return outr;
}
public Vector getSources() {
return null;
}
public String[] getPropertyNames() {
if (properties == null) {
return null;
}
Vector v = new Vector();
for (Enumeration> e = properties.keys(); e.hasMoreElements();) {
try {
v.add((String) e.nextElement());
} catch (ClassCastException ex) {
}
}
int size = v.size();
if (size > 0) {
String names[] = new String[size];
for (int i = 0; i < size; i++) {
names[i] = v.elementAt(i);
}
return names;
}
return null;
}
@Override
public String toString() {
return "BufferedImage@" + Integer.toHexString(hashCode()) + //$NON-NLS-1$
": type = " + imageType + " " + cm + " " + raster; //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
}
public WritableRaster getWritableTile(int tileX, int tileY) {
return raster;
}
public WritableRaster getRaster() {
return raster;
}
public WritableRaster getAlphaRaster() {
return cm.getAlphaRaster(raster);
}
public void removeTileObserver(TileObserver to) {
}
public void addTileObserver(TileObserver to) {
}
public SampleModel getSampleModel() {
return raster.getSampleModel();
}
public void setData(Raster r) {
Rectangle from = r.getBounds();
Rectangle to = raster.getBounds();
Rectangle intersection = to.intersection(from);
int minX = intersection.x;
int minY = intersection.y;
int w = intersection.width;
int h = intersection.height;
Object data = null;
data = r.getDataElements(minX, minY, w, h, data);
raster.setDataElements(minX, minY, w, h, data);
}
public Raster getTile(int tileX, int tileY) {
if (tileX == 0 && tileY == 0) {
return raster;
}
// awt.226=Both tileX and tileY are not equal to 0
throw new ArrayIndexOutOfBoundsException(Messages.getString("awt.226")); //$NON-NLS-1$
}
public Raster getData() {
int w = raster.getWidth();
int h = raster.getHeight();
int minX = raster.getMinX();
int minY = raster.getMinY();
WritableRaster outr = Raster.createWritableRaster(
raster.getSampleModel(),
new Point(raster.getSampleModelTranslateX(),
raster.getSampleModelTranslateY()));
Object data = null;
data = raster.getDataElements(minX, minY, w, h, data);
outr.setDataElements(minX, minY, w, h, data);
return outr;
}
@Override
public ImageProducer getSource() {
return new BufferedImageSource(this, properties);
}
@Override
public int getWidth(ImageObserver observer) {
return raster.getWidth();
}
@Override
public int getHeight(ImageObserver observer) {
return raster.getHeight();
}
public ColorModel getColorModel() {
return cm;
}
public BufferedImage getSubimage(int x, int y, int w, int h) {
WritableRaster wr = raster.createWritableChild(x, y, w, h, 0, 0, null);
return new BufferedImage(cm, wr, cm.isAlphaPremultiplied(), properties);
}
public Point[] getWritableTileIndices() {
Point points[] = new Point[1];
points[0] = new Point(0, 0);
return points;
}
public Graphics2D createGraphics() {
GraphicsEnvironment ge =
GraphicsEnvironment.getLocalGraphicsEnvironment();
return ge.createGraphics(this);
}
@Override
public Graphics getGraphics() {
return createGraphics();
}
public void coerceData(boolean isAlphaPremultiplied) {
if (cm.hasAlpha() &&
cm.isAlphaPremultiplied() != isAlphaPremultiplied) {
cm = cm.coerceData(raster, isAlphaPremultiplied);
}
}
public int[] getRGB(int startX, int startY, int w, int h, int[] rgbArray,
int offset, int scansize) {
if (rgbArray == null) {
rgbArray = new int[offset + h * scansize];
}
int off = offset;
for (int y = startY; y < startY + h; y++, off += scansize) {
int i = off;
for (int x = startX; x < startX + w; x++, i++) {
rgbArray[i] = cm.getRGB(raster.getDataElements(x, y, null));
}
}
return rgbArray;
}
public void setRGB(int startX, int startY, int w, int h, int[] rgbArray,
int offset, int scansize) {
int off = offset;
for (int y = startY; y < startY + h; y++, off += scansize) {
int i = off;
for (int x = startX; x < startX + w; x++, i++) {
raster.setDataElements(x, y,
cm.getDataElements(rgbArray[i], null));
}
}
}
public synchronized void setRGB(int x, int y, int rgb) {
raster.setDataElements(x, y, cm.getDataElements(rgb, null));
}
public boolean isTileWritable(int tileX, int tileY) {
if (tileX == 0 && tileY == 0) {
return true;
}
// awt.226=Both tileX and tileY are not equal to 0
throw new IllegalArgumentException(Messages.getString("awt.226")); //$NON-NLS-1$
}
public void releaseWritableTile(int tileX, int tileY) {
}
public int getRGB(int x, int y) {
return cm.getRGB(raster.getDataElements(x, y, null));
}
public boolean isAlphaPremultiplied() {
return cm.isAlphaPremultiplied();
}
public boolean hasTileWriters() {
return true;
}
@Override
public void flush() {
imageSurf.dispose();
}
public int getWidth() {
return raster.getWidth();
}
public int getType() {
return imageType;
}
public int getTileWidth() {
return raster.getWidth();
}
public int getTileHeight() {
return raster.getHeight();
}
public int getTileGridYOffset() {
return raster.getSampleModelTranslateY();
}
public int getTileGridXOffset() {
return raster.getSampleModelTranslateX();
}
public int getNumYTiles() {
return 1;
}
public int getNumXTiles() {
return 1;
}
public int getMinY() {
return raster.getMinY();
}
public int getMinX() {
return raster.getMinX();
}
public int getMinTileY() {
return 0;
}
public int getMinTileX() {
return 0;
}
public int getHeight() {
return raster.getHeight();
}
private ImageSurface createImageSurface(int type) {
return new ImageSurface(getColorModel(), getRaster(), type);
}
ImageSurface getImageSurface() {
return imageSurf;
}
public int getTransparency() {
return cm.getTransparency();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy