/* * Copyright 1997-2007 Sun Microsystems, Inc. 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. Sun designates this * particular file as subject to the "Classpath" exception as provided * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara, * CA 95054 USA or visit www.sun.com if you need additional information or * have any questions. */ /* * @author Charlton Innovations, Inc. * @author Jim Graham */ package sun.java2d.loops; import java.awt.Composite; import java.awt.Rectangle; import java.awt.image.ColorModel; import java.awt.image.DataBuffer; import java.awt.image.Raster; import java.awt.image.WritableRaster; import sun.awt.image.IntegerComponentRaster; import sun.java2d.SurfaceData; import sun.java2d.pipe.Region; import sun.java2d.pipe.SpanIterator; /** * CustomComponent, collection of GraphicsPrimitive * Basically, this collection of components performs conversion from * ANY to ANY via opaque copy */ public final class CustomComponent { public static void register() { // REMIND: This does not work for all destinations yet since // the screen SurfaceData objects do not implement getRaster Class owner = CustomComponent.class; GraphicsPrimitive[] primitives = { new GraphicsPrimitiveProxy(owner, "OpaqueCopyAnyToArgb", Blit.methodSignature, Blit.primTypeID, SurfaceType.Any, CompositeType.SrcNoEa, SurfaceType.IntArgb), new GraphicsPrimitiveProxy(owner, "OpaqueCopyArgbToAny", Blit.methodSignature, Blit.primTypeID, SurfaceType.IntArgb, CompositeType.SrcNoEa, SurfaceType.Any), new GraphicsPrimitiveProxy(owner, "XorCopyArgbToAny", Blit.methodSignature, Blit.primTypeID, SurfaceType.IntArgb, CompositeType.Xor, SurfaceType.Any), }; GraphicsPrimitiveMgr.register(primitives); } public static Region getRegionOfInterest(SurfaceData src, SurfaceData dst, Region clip, int srcx, int srcy, int dstx, int dsty, int w, int h) { /* * Intersect all of: * - operation area (dstx, dsty, w, h) * - destination bounds * - (translated) src bounds * - supplied clip (may be non-rectangular) * Intersect the rectangular regions first since those are * simpler operations. */ Region ret = Region.getInstanceXYWH(dstx, dsty, w, h); ret = ret.getIntersection(dst.getBounds()); Rectangle r = src.getBounds(); // srcxy in src space maps to dstxy in dst space r.translate(dstx - srcx, dsty - srcy); ret = ret.getIntersection(r); if (clip != null) { // Intersect with clip last since it may be non-rectangular ret = ret.getIntersection(clip); } return ret; } } /** * ANY format to ARGB format Blit */ class OpaqueCopyAnyToArgb extends Blit { OpaqueCopyAnyToArgb() { super(SurfaceType.Any, CompositeType.SrcNoEa, SurfaceType.IntArgb); } public void Blit(SurfaceData src, SurfaceData dst, Composite comp, Region clip, int srcx, int srcy, int dstx, int dsty, int w, int h) { Raster srcRast = src.getRaster(srcx, srcy, w, h); ColorModel srcCM = src.getColorModel(); Raster dstRast = dst.getRaster(dstx, dsty, w, h); IntegerComponentRaster icr = (IntegerComponentRaster) dstRast; int[] dstPix = icr.getDataStorage(); Region roi = CustomComponent.getRegionOfInterest(src, dst, clip, srcx, srcy, dstx, dsty, w, h); SpanIterator si = roi.getSpanIterator(); Object srcPix = null; int dstScan = icr.getScanlineStride(); // assert(icr.getPixelStride() == 1); srcx -= dstx; srcy -= dsty; int span[] = new int[4]; while (si.nextSpan(span)) { int rowoff = icr.getDataOffset(0) + span[1] * dstScan + span[0]; for (int y = span[1]; y < span[3]; y++) { int off = rowoff; for (int x = span[0]; x < span[2]; x++) { srcPix = srcRast.getDataElements(x+srcx, y+srcy, srcPix); dstPix[off++] = srcCM.getRGB(srcPix); } rowoff += dstScan; } } // Pixels in the dest were modified directly, we must // manually notify the raster that it was modified icr.markDirty(); // REMIND: We need to do something to make sure that dstRast // is put back to the destination (as in the native Release // function) // src.releaseRaster(srcRast); // NOP? // dst.releaseRaster(dstRast); } } /** * ARGB format to ANY format Blit */ class OpaqueCopyArgbToAny extends Blit { OpaqueCopyArgbToAny() { super(SurfaceType.IntArgb, CompositeType.SrcNoEa, SurfaceType.Any); } public void Blit(SurfaceData src, SurfaceData dst, Composite comp, Region clip, int srcx, int srcy, int dstx, int dsty, int w, int h) { Raster srcRast = src.getRaster(srcx, srcy, w, h); IntegerComponentRaster icr = (IntegerComponentRaster) srcRast; int[] srcPix = icr.getDataStorage(); WritableRaster dstRast = (WritableRaster) dst.getRaster(dstx, dsty, w, h); ColorModel dstCM = dst.getColorModel(); Region roi = CustomComponent.getRegionOfInterest(src, dst, clip, srcx, srcy, dstx, dsty, w, h); SpanIterator si = roi.getSpanIterator(); Object dstPix = null; int srcScan = icr.getScanlineStride(); // assert(icr.getPixelStride() == 1); srcx -= dstx; srcy -= dsty; int span[] = new int[4]; while (si.nextSpan(span)) { int rowoff = (icr.getDataOffset(0) + (srcy + span[1]) * srcScan + (srcx + span[0])); for (int y = span[1]; y < span[3]; y++) { int off = rowoff; for (int x = span[0]; x < span[2]; x++) { dstPix = dstCM.getDataElements(srcPix[off++], dstPix); dstRast.setDataElements(x, y, dstPix); } rowoff += srcScan; } } // REMIND: We need to do something to make sure that dstRast // is put back to the destination (as in the native Release // function) // src.releaseRaster(srcRast); // NOP? // dst.releaseRaster(dstRast); } } /** * ARGB format to ANY format Blit (pixels are XORed together with XOR pixel) */ class XorCopyArgbToAny extends Blit { XorCopyArgbToAny() { super(SurfaceType.IntArgb, CompositeType.Xor, SurfaceType.Any); } public void Blit(SurfaceData src, SurfaceData dst, Composite comp, Region clip, int srcx, int srcy, int dstx, int dsty, int w, int h) { Raster srcRast = src.getRaster(srcx, srcy, w, h); IntegerComponentRaster icr = (IntegerComponentRaster) srcRast; int[] srcPix = icr.getDataStorage(); WritableRaster dstRast = (WritableRaster) dst.getRaster(dstx, dsty, w, h); ColorModel dstCM = dst.getColorModel(); Region roi = CustomComponent.getRegionOfInterest(src, dst, clip, srcx, srcy, dstx, dsty, w, h); SpanIterator si = roi.getSpanIterator(); int xorrgb = ((XORComposite)comp).getXorColor().getRGB(); Object xorPixel = dstCM.getDataElements(xorrgb, null); Object srcPixel = null; Object dstPixel = null; int srcScan = icr.getScanlineStride(); // assert(icr.getPixelStride() == 1); srcx -= dstx; srcy -= dsty; int span[] = new int[4]; while (si.nextSpan(span)) { int rowoff = (icr.getDataOffset(0) + (srcy + span[1]) * srcScan + (srcx + span[0])); for (int y = span[1]; y < span[3]; y++) { int off = rowoff; for (int x = span[0]; x < span[2]; x++) { // REMIND: alpha bits of the destination pixel are // currently altered by the XOR operation, but // should be left untouched srcPixel = dstCM.getDataElements(srcPix[off++], srcPixel); dstPixel = dstRast.getDataElements(x, y, dstPixel); switch (dstCM.getTransferType()) { case DataBuffer.TYPE_BYTE: byte[] bytesrcarr = (byte[]) srcPixel; byte[] bytedstarr = (byte[]) dstPixel; byte[] bytexorarr = (byte[]) xorPixel; for (int i = 0; i < bytedstarr.length; i++) { bytedstarr[i] ^= bytesrcarr[i] ^ bytexorarr[i]; } break; case DataBuffer.TYPE_SHORT: case DataBuffer.TYPE_USHORT: short[] shortsrcarr = (short[]) srcPixel; short[] shortdstarr = (short[]) dstPixel; short[] shortxorarr = (short[]) xorPixel; for (int i = 0; i < shortdstarr.length; i++) { shortdstarr[i] ^= shortsrcarr[i] ^ shortxorarr[i]; } break; case DataBuffer.TYPE_INT: int[] intsrcarr = (int[]) srcPixel; int[] intdstarr = (int[]) dstPixel; int[] intxorarr = (int[]) xorPixel; for (int i = 0; i < intdstarr.length; i++) { intdstarr[i] ^= intsrcarr[i] ^ intxorarr[i]; } break; case DataBuffer.TYPE_FLOAT: float[] floatsrcarr = (float[]) srcPixel; float[] floatdstarr = (float[]) dstPixel; float[] floatxorarr = (float[]) xorPixel; for (int i = 0; i < floatdstarr.length; i++) { int v = (Float.floatToIntBits(floatdstarr[i]) ^ Float.floatToIntBits(floatsrcarr[i]) ^ Float.floatToIntBits(floatxorarr[i])); floatdstarr[i] = Float.intBitsToFloat(v); } break; case DataBuffer.TYPE_DOUBLE: double[] doublesrcarr = (double[]) srcPixel; double[] doubledstarr = (double[]) dstPixel; double[] doublexorarr = (double[]) xorPixel; for (int i = 0; i < doubledstarr.length; i++) { long v = (Double.doubleToLongBits(doubledstarr[i]) ^ Double.doubleToLongBits(doublesrcarr[i]) ^ Double.doubleToLongBits(doublexorarr[i])); doubledstarr[i] = Double.longBitsToDouble(v); } break; default: throw new InternalError("Unsupported XOR pixel type"); } dstRast.setDataElements(x, y, dstPixel); } rowoff += srcScan; } } // REMIND: We need to do something to make sure that dstRast // is put back to the destination (as in the native Release // function) // src.releaseRaster(srcRast); // NOP? // dst.releaseRaster(dstRast); } }