/*

   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.

 */
package org.apache.batik.ext.awt.image.rendered;

import java.awt.Point;
import java.awt.Rectangle;
import java.awt.RenderingHints;
import java.awt.Transparency;
import java.awt.color.ColorSpace;
import java.awt.geom.AffineTransform;
import java.awt.geom.NoninvertibleTransformException;
import java.awt.geom.Point2D;
import java.awt.image.AffineTransformOp;
import java.awt.image.BufferedImage;
import java.awt.image.ColorModel;
import java.awt.image.ComponentColorModel;
import java.awt.image.DataBuffer;
import java.awt.image.DirectColorModel;
import java.awt.image.Raster;
import java.awt.image.SampleModel;
import java.awt.image.WritableRaster;

import org.apache.batik.ext.awt.image.GraphicsUtil;

This is an implementation of an affine operation as a RenderedImage. Right now the implementation makes use of the AffineBufferedImageOp to do the work. Eventually this may move to be more tiled in nature.
Author:Thomas DeWeese
Version:$Id: AffineRed.java 1808843 2017-09-19 08:37:38Z ssteiner $
/** * This is an implementation of an affine operation as a RenderedImage. * Right now the implementation makes use of the AffineBufferedImageOp * to do the work. Eventually this may move to be more tiled in nature. * * @author <a href="mailto:Thomas.DeWeeese@Kodak.com">Thomas DeWeese</a> * @version $Id: AffineRed.java 1808843 2017-09-19 08:37:38Z ssteiner $ */
public class AffineRed extends AbstractRed { RenderingHints hints; AffineTransform src2me; AffineTransform me2src; public AffineTransform getTransform() { return (AffineTransform)src2me.clone(); } public CachableRed getSource() { return (CachableRed)getSources().get(0); } public AffineRed(CachableRed src, AffineTransform src2me, RenderingHints hints) { super(); // We _must_ call init... this.src2me = src2me; this.hints = hints; try { me2src = src2me.createInverse(); } catch (NoninvertibleTransformException nite) { me2src = null; } // Calculate my bounds by applying the affine transform to // my input data..codec/ Rectangle srcBounds = src.getBounds(); // srcBounds.grow(-1,-1); Rectangle myBounds; myBounds = src2me.createTransformedShape(srcBounds).getBounds(); // If the output buffer is not premultiplied in certain cases it // fails to properly divide out the Alpha (it always does // the affine on premultiplied data), hence you get ugly // back aliasing effects... ColorModel cm = fixColorModel(src); // fix my sample model so it makes sense given my size. SampleModel sm = fixSampleModel(src, cm, myBounds); Point2D pt = new Point2D.Float(src.getTileGridXOffset(), src.getTileGridYOffset()); pt = src2me.transform(pt, null); // Finish initializing our base class... init(src, myBounds, cm, sm, (int)pt.getX(), (int)pt.getY(), null); } public WritableRaster copyData(WritableRaster wr) { // System.out.println("Affine CopyData:" + wr); // copyToRaster(wr); PadRed.ZeroRecter zr = PadRed.ZeroRecter.getZeroRecter(wr); zr.zeroRect(new Rectangle(wr.getMinX(), wr.getMinY(), wr.getWidth(), wr.getHeight())); genRect(wr); return wr; } public Raster getTile(int x, int y) { if (me2src == null) return null; int tx = tileGridXOff+x*tileWidth; int ty = tileGridYOff+y*tileHeight; Point pt = new Point(tx, ty); WritableRaster wr = Raster.createWritableRaster(sm, pt); genRect(wr); return wr; } public void genRect(WritableRaster wr) { if (me2src == null) return; Rectangle srcR = me2src.createTransformedShape(wr.getBounds()).getBounds(); // System.out.println("Affine wrR: " + wr.getBounds()); // System.out.println("Affine srcR: " + srcR); // Outset by two pixels so we get context for interpolation... srcR.setBounds(srcR.x-1, srcR.y-1, srcR.width+2, srcR.height+2); // Don't try and get data from src that it doesn't have... CachableRed src = (CachableRed)getSources().get(0); // Raster srcRas = src.getData(srcR); if ( ! srcR.intersects(src.getBounds()) ) return; Raster srcRas = src.getData(srcR.intersection(src.getBounds())); if (srcRas == null) return; // This works around the problem that the buffered ops // completely ignore the coords of the Rasters passed in. AffineTransform aff = (AffineTransform)src2me.clone(); // Translate what is at 0,0 (which will be what our current // minX/Y is) to our current minX,minY. aff.concatenate(AffineTransform.getTranslateInstance (srcRas.getMinX(), srcRas.getMinY())); Point2D srcPt = new Point2D.Float(wr.getMinX(), wr.getMinY()); srcPt = me2src.transform(srcPt, null); Point2D destPt = new Point2D.Double(srcPt.getX()-srcRas.getMinX(), srcPt.getY()-srcRas.getMinY()); destPt = aff.transform(destPt, null); // Translate what will be at minX,minY to zero, zero // which where java2d will think the real minX,minY is. aff.preConcatenate(AffineTransform.getTranslateInstance (-destPt.getX(), -destPt.getY())); AffineTransformOp op = new AffineTransformOp(aff, hints); BufferedImage srcBI, myBI; ColorModel srcCM = src.getColorModel(); ColorModel myCM = getColorModel(); WritableRaster srcWR = (WritableRaster)srcRas; // If the output buffer is not premultiplied in certain cases // it fails to properly divide out the Alpha (it always does // the affine on premultiplied data). We help it out by // premultiplying for it. srcCM = GraphicsUtil.coerceData(srcWR, srcCM, true); srcBI = new BufferedImage(srcCM, srcWR.createWritableTranslatedChild(0,0), srcCM.isAlphaPremultiplied(), null); myBI = new BufferedImage(myCM,wr.createWritableTranslatedChild(0,0), myCM.isAlphaPremultiplied(), null); op.filter(srcBI.getRaster(), myBI.getRaster()); // if ((count % 40) == 0) { // org.apache.batik.ImageDisplay.showImage("Src: " , srcBI); // org.apache.batik.ImageDisplay.showImage("Dst: " , myBI); // } // count++; } // int count=0; protected static ColorModel fixColorModel(CachableRed src) { ColorModel cm = src.getColorModel(); if (cm.hasAlpha()) { if (!cm.isAlphaPremultiplied()) cm = GraphicsUtil.coerceColorModel(cm, true); return cm; } ColorSpace cs = cm.getColorSpace(); int b = src.getSampleModel().getNumBands()+1; if (b == 4) { int [] masks = new int[4]; for (int i=0; i < b-1; i++) masks[i] = 0xFF0000 >> (8*i); masks[3] = 0xFF << (8*(b-1)); return new DirectColorModel(cs, 8*b, masks[0], masks[1], masks[2], masks[3], true, DataBuffer.TYPE_INT); } int [] bits = new int[b]; for (int i=0; i<b; i++) bits[i] = 8; return new ComponentColorModel(cs, bits, true, true, Transparency.TRANSLUCENT, DataBuffer.TYPE_INT); }
This function 'fixes' the source's sample model. right now it just ensures that the sample model isn't much larger than my width.
/** * This function 'fixes' the source's sample model. * right now it just ensures that the sample model isn't * much larger than my width. */
protected SampleModel fixSampleModel(CachableRed src, ColorModel cm, Rectangle bounds) { SampleModel sm = src.getSampleModel(); int defSz = AbstractTiledRed.getDefaultTileSize(); int w = sm.getWidth(); if (w < defSz) w = defSz; if (w > bounds.width) w = bounds.width; int h = sm.getHeight(); if (h < defSz) h = defSz; if (h > bounds.height) h = bounds.height; if ((w <= 0) || (h <= 0)) { w = 1; h = 1; } return cm.createCompatibleSampleModel(w, h); } }