/*
* Copyright (C) 2010 The Android Open Source Project
*
* Licensed 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 com.android.layoutlib.bridge.impl;
import com.android.ide.common.rendering.api.LayoutLog;
import com.android.layoutlib.bridge.Bridge;
import android.graphics.Bitmap_Delegate;
import android.graphics.Canvas;
import android.graphics.ColorFilter_Delegate;
import android.graphics.Paint;
import android.graphics.Paint_Delegate;
import android.graphics.Rect;
import android.graphics.RectF;
import android.graphics.Region;
import android.graphics.Region_Delegate;
import android.graphics.Shader_Delegate;
import android.graphics.Xfermode_Delegate;
import java.awt.AlphaComposite;
import java.awt.Color;
import java.awt.Composite;
import java.awt.Graphics2D;
import java.awt.RenderingHints;
import java.awt.Shape;
import java.awt.geom.AffineTransform;
import java.awt.geom.Area;
import java.awt.geom.Rectangle2D;
import java.awt.image.BufferedImage;
import java.util.ArrayList;
/**
* Class representing a graphics context snapshot, as well as a context stack as a linked list.
*
* This is based on top of {@link Graphics2D} but can operate independently if none are available
* yet when setting transforms and clip information.
*
* This allows for drawing through {@link #draw(Drawable, Paint_Delegate, boolean, boolean)} and
* {@link #draw(Drawable)}
*
* Handling of layers (created with {@link Canvas#saveLayer(RectF, Paint, int)}) is handled through
* a list of Graphics2D for each layers. The class actually maintains a list of {@link Layer}
* for each layer. Doing a save() will duplicate this list so that each graphics2D object
* ({@link Layer#getGraphics()}) is configured only for the new snapshot.
*/
public class GcSnapshot {
private final GcSnapshot mPrevious;
private final int mFlags;
/** list of layers. The first item in the list is always the */
private final ArrayList mLayers = new ArrayList();
/** temp transform in case transformation are set before a Graphics2D exists */
private AffineTransform mTransform = null;
/** temp clip in case clipping is set before a Graphics2D exists */
private Area mClip = null;
// local layer data
/** a local layer created with {@link Canvas#saveLayer(RectF, Paint, int)}.
* If this is null, this does not mean there's no layer, just that the snapshot is not the
* one that created the layer.
* @see #getLayerSnapshot()
*/
private final Layer mLocalLayer;
private final Paint_Delegate mLocalLayerPaint;
private final Rect mLayerBounds;
public interface Drawable {
void draw(Graphics2D graphics, Paint_Delegate paint);
}
/**
* Class containing information about a layer.
*
* This contains graphics, bitmap and layer information.
*/
private static class Layer {
private final Graphics2D mGraphics;
private final Bitmap_Delegate mBitmap;
private final BufferedImage mImage;
/** the flags that were used to configure the layer. This is never changed, and passed
* as is when {@link #makeCopy()} is called */
private final int mFlags;
/** the original content of the layer when the next object was created. This is not
* passed in {@link #makeCopy()} and instead is recreated when a new layer is added
* (depending on its flags) */
private BufferedImage mOriginalCopy;
/**
* Creates a layer with a graphics and a bitmap. This is only used to create
* the base layer.
*
* @param graphics the graphics
* @param bitmap the bitmap
*/
Layer(Graphics2D graphics, Bitmap_Delegate bitmap) {
mGraphics = graphics;
mBitmap = bitmap;
mImage = mBitmap.getImage();
mFlags = 0;
}
/**
* Creates a layer with a graphics and an image. If the image belongs to a
* {@link Bitmap_Delegate} (case of the base layer), then
* {@link Layer#Layer(Graphics2D, Bitmap_Delegate)} should be used.
*
* @param graphics the graphics the new graphics for this layer
* @param image the image the image from which the graphics came
* @param flags the flags that were used to save this layer
*/
Layer(Graphics2D graphics, BufferedImage image, int flags) {
mGraphics = graphics;
mBitmap = null;
mImage = image;
mFlags = flags;
}
/** The Graphics2D, guaranteed to be non null */
Graphics2D getGraphics() {
return mGraphics;
}
/** The BufferedImage, guaranteed to be non null */
BufferedImage getImage() {
return mImage;
}
/** Returns the layer save flags. This is only valid for additional layers.
* For the base layer this will always return 0;
* For a given layer, all further copies of this {@link Layer} object in new snapshots
* will always return the same value.
*/
int getFlags() {
return mFlags;
}
Layer makeCopy() {
if (mBitmap != null) {
return new Layer((Graphics2D) mGraphics.create(), mBitmap);
}
return new Layer((Graphics2D) mGraphics.create(), mImage, mFlags);
}
/** sets an optional copy of the original content to be used during restore */
void setOriginalCopy(BufferedImage image) {
mOriginalCopy = image;
}
BufferedImage getOriginalCopy() {
return mOriginalCopy;
}
void change() {
if (mBitmap != null) {
mBitmap.change();
}
}
/**
* Sets the clip for the graphics2D object associated with the layer.
* This should be used over the normal Graphics2D setClip method.
*
* @param clipShape the shape to use a the clip shape.
*/
void setClip(Shape clipShape) {
// because setClip is only guaranteed to work with rectangle shape,
// first reset the clip to max and then intersect the current (empty)
// clip with the shap.
mGraphics.setClip(null);
mGraphics.clip(clipShape);
}
/**
* Clips the layer with the given shape. This performs an intersect between the current
* clip shape and the given shape.
* @param shape the new clip shape.
*/
public void clip(Shape shape) {
mGraphics.clip(shape);
}
}
/**
* Creates the root snapshot associating it with a given bitmap.
*
* If bitmap is null, then {@link GcSnapshot#setBitmap(Bitmap_Delegate)} must be
* called before the snapshot can be used to draw. Transform and clip operations are permitted
* before.
*
* @param bitmap the image to associate to the snapshot or null.
* @return the root snapshot
*/
public static GcSnapshot createDefaultSnapshot(Bitmap_Delegate bitmap) {
GcSnapshot snapshot = new GcSnapshot();
if (bitmap != null) {
snapshot.setBitmap(bitmap);
}
return snapshot;
}
/**
* Saves the current state according to the given flags and returns the new current snapshot.
*
* This is the equivalent of {@link Canvas#save(int)}
*
* @param flags the save flags.
* @return the new snapshot
*
* @see Canvas#save(int)
*/
public GcSnapshot save(int flags) {
return new GcSnapshot(this, null /*layerbounds*/, null /*paint*/, flags);
}
/**
* Saves the current state and creates a new layer, and returns the new current snapshot.
*
* This is the equivalent of {@link Canvas#saveLayer(RectF, Paint, int)}
*
* @param layerBounds the layer bounds
* @param paint the Paint information used to blit the layer back into the layers underneath
* upon restore
* @param flags the save flags.
* @return the new snapshot
*
* @see Canvas#saveLayer(RectF, Paint, int)
*/
public GcSnapshot saveLayer(RectF layerBounds, Paint_Delegate paint, int flags) {
return new GcSnapshot(this, layerBounds, paint, flags);
}
/**
* Creates the root snapshot.
* {@link #setGraphics2D(Graphics2D)} will have to be called on it when possible.
*/
private GcSnapshot() {
mPrevious = null;
mFlags = 0;
mLocalLayer = null;
mLocalLayerPaint = null;
mLayerBounds = null;
}
/**
* Creates a new {@link GcSnapshot} on top of another one, with a layer data to be restored
* into the main graphics when {@link #restore()} is called.
*
* @param previous the previous snapshot head.
* @param layerBounds the region of the layer. Optional, if null, this is a normal save()
* @param paint the Paint information used to blit the layer back into the layers underneath
* upon restore
* @param flags the flags regarding what should be saved.
*/
private GcSnapshot(GcSnapshot previous, RectF layerBounds, Paint_Delegate paint, int flags) {
assert previous != null;
mPrevious = previous;
mFlags = flags;
// make a copy of the current layers before adding the new one.
// This keeps the same BufferedImage reference but creates new Graphics2D for this
// snapshot.
// It does not copy whatever original copy the layers have, as they will be done
// only if the new layer doesn't clip drawing to itself.
for (Layer layer : mPrevious.mLayers) {
mLayers.add(layer.makeCopy());
}
if (layerBounds != null) {
// get the current transform
AffineTransform matrix = mLayers.get(0).getGraphics().getTransform();
// transform the layerBounds with the current transform and stores it into a int rect
RectF rect2 = new RectF();
mapRect(matrix, rect2, layerBounds);
mLayerBounds = new Rect();
rect2.round(mLayerBounds);
// get the base layer (always at index 0)
Layer baseLayer = mLayers.get(0);
// create the image for the layer
BufferedImage layerImage = new BufferedImage(
baseLayer.getImage().getWidth(),
baseLayer.getImage().getHeight(),
(mFlags & Canvas.HAS_ALPHA_LAYER_SAVE_FLAG) != 0 ?
BufferedImage.TYPE_INT_ARGB :
BufferedImage.TYPE_INT_RGB);
// create a graphics for it so that drawing can be done.
Graphics2D layerGraphics = layerImage.createGraphics();
// because this layer inherits the current context for transform and clip,
// set them to one from the base layer.
AffineTransform currentMtx = baseLayer.getGraphics().getTransform();
layerGraphics.setTransform(currentMtx);
// create a new layer for this new layer and add it to the list at the end.
mLayers.add(mLocalLayer = new Layer(layerGraphics, layerImage, flags));
// set the clip on it.
Shape currentClip = baseLayer.getGraphics().getClip();
mLocalLayer.setClip(currentClip);
// if the drawing is not clipped to the local layer only, we save the current content
// of all other layers. We are only interested in the part that will actually
// be drawn, so we create as small bitmaps as we can.
// This is so that we can erase the drawing that goes in the layers below that will
// be coming from the layer itself.
if ((mFlags & Canvas.CLIP_TO_LAYER_SAVE_FLAG) == 0) {
int w = mLayerBounds.width();
int h = mLayerBounds.height();
for (int i = 0 ; i < mLayers.size() - 1 ; i++) {
Layer layer = mLayers.get(i);
BufferedImage image = new BufferedImage(w, h, BufferedImage.TYPE_INT_ARGB);
Graphics2D graphics = image.createGraphics();
graphics.drawImage(layer.getImage(),
0, 0, w, h,
mLayerBounds.left, mLayerBounds.top,
mLayerBounds.right, mLayerBounds.bottom,
null);
graphics.dispose();
layer.setOriginalCopy(image);
}
}
} else {
mLocalLayer = null;
mLayerBounds = null;
}
mLocalLayerPaint = paint;
}
public void dispose() {
for (Layer layer : mLayers) {
layer.getGraphics().dispose();
}
if (mPrevious != null) {
mPrevious.dispose();
}
}
/**
* Restores the top {@link GcSnapshot}, and returns the next one.
*/
public GcSnapshot restore() {
return doRestore();
}
/**
* Restores the {@link GcSnapshot} to saveCount.
* @param saveCount the saveCount or -1 to only restore 1.
*
* @return the new head of the Gc snapshot stack.
*/
public GcSnapshot restoreTo(int saveCount) {
return doRestoreTo(size(), saveCount);
}
public int size() {
if (mPrevious != null) {
return mPrevious.size() + 1;
}
return 1;
}
/**
* Link the snapshot to a Bitmap_Delegate.
*
* This is only for the case where the snapshot was created with a null image when calling
* {@link #createDefaultSnapshot(Bitmap_Delegate)}, and is therefore not yet linked to
* a previous snapshot.
*
* If any transform or clip information was set before, they are put into the Graphics object.
* @param bitmap the bitmap to link to.
*/
public void setBitmap(Bitmap_Delegate bitmap) {
// create a new Layer for the bitmap. This will be the base layer.
Graphics2D graphics2D = bitmap.getImage().createGraphics();
Layer baseLayer = new Layer(graphics2D, bitmap);
// Set the current transform and clip which can either come from mTransform/mClip if they
// were set when there was no bitmap/layers or from the current base layers if there is
// one already.
graphics2D.setTransform(getTransform());
// reset mTransform in case there was one.
mTransform = null;
baseLayer.setClip(getClip());
// reset mClip in case there was one.
mClip = null;
// replace whatever current layers we have with this.
mLayers.clear();
mLayers.add(baseLayer);
}
public void translate(float dx, float dy) {
if (mLayers.size() > 0) {
for (Layer layer : mLayers) {
layer.getGraphics().translate(dx, dy);
}
} else {
if (mTransform == null) {
mTransform = new AffineTransform();
}
mTransform.translate(dx, dy);
}
}
public void rotate(double radians) {
if (mLayers.size() > 0) {
for (Layer layer : mLayers) {
layer.getGraphics().rotate(radians);
}
} else {
if (mTransform == null) {
mTransform = new AffineTransform();
}
mTransform.rotate(radians);
}
}
public void scale(float sx, float sy) {
if (mLayers.size() > 0) {
for (Layer layer : mLayers) {
layer.getGraphics().scale(sx, sy);
}
} else {
if (mTransform == null) {
mTransform = new AffineTransform();
}
mTransform.scale(sx, sy);
}
}
public AffineTransform getTransform() {
if (mLayers.size() > 0) {
// all graphics2D in the list have the same transform
return mLayers.get(0).getGraphics().getTransform();
} else {
if (mTransform == null) {
mTransform = new AffineTransform();
}
return mTransform;
}
}
public void setTransform(AffineTransform transform) {
if (mLayers.size() > 0) {
for (Layer layer : mLayers) {
layer.getGraphics().setTransform(transform);
}
} else {
if (mTransform == null) {
mTransform = new AffineTransform();
}
mTransform.setTransform(transform);
}
}
public boolean clip(Shape shape, int regionOp) {
// Simple case of intersect with existing layers.
// Because Graphics2D#setClip works a bit peculiarly, we optimize
// the case of clipping by intersection, as it's supported natively.
if (regionOp == Region.Op.INTERSECT.nativeInt && mLayers.size() > 0) {
for (Layer layer : mLayers) {
layer.clip(shape);
}
Shape currentClip = getClip();
return currentClip != null && currentClip.getBounds().isEmpty() == false;
}
Area area = null;
if (regionOp == Region.Op.REPLACE.nativeInt) {
area = new Area(shape);
} else {
area = Region_Delegate.combineShapes(getClip(), shape, regionOp);
}
assert area != null;
if (mLayers.size() > 0) {
if (area != null) {
for (Layer layer : mLayers) {
layer.setClip(area);
}
}
Shape currentClip = getClip();
return currentClip != null && currentClip.getBounds().isEmpty() == false;
} else {
if (area != null) {
mClip = area;
} else {
mClip = new Area();
}
return mClip.getBounds().isEmpty() == false;
}
}
public boolean clipRect(float left, float top, float right, float bottom, int regionOp) {
return clip(new Rectangle2D.Float(left, top, right - left, bottom - top), regionOp);
}
/**
* Returns the current clip, or null if none have been setup.
*/
public Shape getClip() {
if (mLayers.size() > 0) {
// they all have the same clip
return mLayers.get(0).getGraphics().getClip();
} else {
return mClip;
}
}
private GcSnapshot doRestoreTo(int size, int saveCount) {
if (size <= saveCount) {
return this;
}
// restore the current one first.
GcSnapshot previous = doRestore();
if (size == saveCount + 1) { // this was the only one that needed restore.
return previous;
} else {
return previous.doRestoreTo(size - 1, saveCount);
}
}
/**
* Executes the Drawable's draw method, with a null paint delegate.
*
* Note that the method can be called several times if there are more than one active layer.
*/
public void draw(Drawable drawable) {
draw(drawable, null, false /*compositeOnly*/, false /*forceSrcMode*/);
}
/**
* Executes the Drawable's draw method.
*
* Note that the method can be called several times if there are more than one active layer.
* @param compositeOnly whether the paint is used for composite only. This is typically
* the case for bitmaps.
* @param forceSrcMode if true, this overrides the composite to be SRC
*/
public void draw(Drawable drawable, Paint_Delegate paint, boolean compositeOnly,
boolean forceSrcMode) {
int forceMode = forceSrcMode ? AlphaComposite.SRC : 0;
// the current snapshot may not have a mLocalLayer (ie it was created on save() instead
// of saveLayer(), but that doesn't mean there's no layer.
// mLayers however saves all the information we need (flags).
if (mLayers.size() == 1) {
// no layer, only base layer. easy case.
drawInLayer(mLayers.get(0), drawable, paint, compositeOnly, forceMode);
} else {
// draw in all the layers until the layer save flags tells us to stop (ie drawing
// in that layer is limited to the layer itself.
int flags;
int i = mLayers.size() - 1;
do {
Layer layer = mLayers.get(i);
drawInLayer(layer, drawable, paint, compositeOnly, forceMode);
// then go to previous layer, only if there are any left, and its flags
// doesn't restrict drawing to the layer itself.
i--;
flags = layer.getFlags();
} while (i >= 0 && (flags & Canvas.CLIP_TO_LAYER_SAVE_FLAG) == 0);
}
}
private void drawInLayer(Layer layer, Drawable drawable, Paint_Delegate paint,
boolean compositeOnly, int forceMode) {
Graphics2D originalGraphics = layer.getGraphics();
if (paint == null) {
drawOnGraphics((Graphics2D) originalGraphics.create(), drawable,
null /*paint*/, layer);
} else {
ColorFilter_Delegate filter = paint.getColorFilter();
if (filter == null || !filter.isSupported()) {
// get a Graphics2D object configured with the drawing parameters.
Graphics2D configuredGraphics = createCustomGraphics(originalGraphics, paint,
compositeOnly, forceMode);
drawOnGraphics(configuredGraphics, drawable, paint, layer);
return;
}
int width = layer.getImage().getWidth();
int height = layer.getImage().getHeight();
// Create a temporary image to which the color filter will be applied.
BufferedImage image = new BufferedImage(width, height,
BufferedImage.TYPE_INT_ARGB);
Graphics2D imageBaseGraphics = (Graphics2D) image.getGraphics();
// Configure the Graphics2D object with drawing parameters and shader.
Graphics2D imageGraphics = createCustomGraphics(
imageBaseGraphics, paint, compositeOnly,
AlphaComposite.SRC_OVER);
// get a Graphics2D object configured with the drawing parameters, but no shader.
Graphics2D configuredGraphics = createCustomGraphics(originalGraphics, paint,
true /*compositeOnly*/, forceMode);
try {
// The main draw operation.
drawable.draw(imageGraphics, paint);
// Apply the color filter.
filter.applyFilter(imageGraphics, width, height);
// Draw the tinted image on the main layer.
configuredGraphics.drawImage(image, 0, 0, null);
layer.change();
} finally {
// dispose Graphics2D objects
imageGraphics.dispose();
imageBaseGraphics.dispose();
configuredGraphics.dispose();
}
}
}
private void drawOnGraphics(Graphics2D g, Drawable drawable, Paint_Delegate paint,
Layer layer) {
try {
drawable.draw(g, paint);
layer.change();
} finally {
g.dispose();
}
}
private GcSnapshot doRestore() {
if (mPrevious != null) {
if (mLocalLayer != null) {
// prepare to blit the layers in which we have draw, in the layer beneath
// them, starting with the top one (which is the current local layer).
int i = mLayers.size() - 1;
int flags;
do {
Layer dstLayer = mLayers.get(i - 1);
restoreLayer(dstLayer);
flags = dstLayer.getFlags();
i--;
} while (i > 0 && (flags & Canvas.CLIP_TO_LAYER_SAVE_FLAG) == 0);
}
// if this snapshot does not save everything, then set the previous snapshot
// to this snapshot content
// didn't save the matrix? set the current matrix on the previous snapshot
if ((mFlags & Canvas.MATRIX_SAVE_FLAG) == 0) {
AffineTransform mtx = getTransform();
for (Layer layer : mPrevious.mLayers) {
layer.getGraphics().setTransform(mtx);
}
}
// didn't save the clip? set the current clip on the previous snapshot
if ((mFlags & Canvas.CLIP_SAVE_FLAG) == 0) {
Shape clip = getClip();
for (Layer layer : mPrevious.mLayers) {
layer.setClip(clip);
}
}
}
for (Layer layer : mLayers) {
layer.getGraphics().dispose();
}
return mPrevious;
}
private void restoreLayer(Layer dstLayer) {
Graphics2D baseGfx = dstLayer.getImage().createGraphics();
// if the layer contains an original copy this means the flags
// didn't restrict drawing to the local layer and we need to make sure the
// layer bounds in the layer beneath didn't receive any drawing.
// so we use the originalCopy to erase the new drawings in there.
BufferedImage originalCopy = dstLayer.getOriginalCopy();
if (originalCopy != null) {
Graphics2D g = (Graphics2D) baseGfx.create();
g.setComposite(AlphaComposite.Src);
g.drawImage(originalCopy,
mLayerBounds.left, mLayerBounds.top, mLayerBounds.right, mLayerBounds.bottom,
0, 0, mLayerBounds.width(), mLayerBounds.height(),
null);
g.dispose();
}
// now draw put the content of the local layer onto the layer,
// using the paint information
Graphics2D g = createCustomGraphics(baseGfx, mLocalLayerPaint,
true /*alphaOnly*/, 0 /*forceMode*/);
g.drawImage(mLocalLayer.getImage(),
mLayerBounds.left, mLayerBounds.top, mLayerBounds.right, mLayerBounds.bottom,
mLayerBounds.left, mLayerBounds.top, mLayerBounds.right, mLayerBounds.bottom,
null);
g.dispose();
baseGfx.dispose();
}
/**
* Creates a new {@link Graphics2D} based on the {@link Paint} parameters.
* The object must be disposed ({@link Graphics2D#dispose()}) after being used.
*/
private Graphics2D createCustomGraphics(Graphics2D original, Paint_Delegate paint,
boolean compositeOnly, int forceMode) {
// make new one graphics
Graphics2D g = (Graphics2D) original.create();
// configure it
if (paint.isAntiAliased()) {
g.setRenderingHint(
RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
g.setRenderingHint(
RenderingHints.KEY_TEXT_ANTIALIASING, RenderingHints.VALUE_TEXT_ANTIALIAS_ON);
}
// set the shader first, as it'll replace the color if it can be used it.
boolean customShader = false;
if (!compositeOnly) {
customShader = setShader(g, paint);
// set the stroke
g.setStroke(paint.getJavaStroke());
}
// set the composite.
setComposite(g, paint, compositeOnly || customShader, forceMode);
return g;
}
private boolean setShader(Graphics2D g, Paint_Delegate paint) {
Shader_Delegate shaderDelegate = paint.getShader();
if (shaderDelegate != null) {
if (shaderDelegate.isSupported()) {
java.awt.Paint shaderPaint = shaderDelegate.getJavaPaint();
assert shaderPaint != null;
if (shaderPaint != null) {
g.setPaint(shaderPaint);
return true;
}
} else {
Bridge.getLog().fidelityWarning(LayoutLog.TAG_SHADER,
shaderDelegate.getSupportMessage(),
null /*throwable*/, null /*data*/);
}
}
// if no shader, use the paint color
g.setColor(new Color(paint.getColor(), true /*hasAlpha*/));
return false;
}
private void setComposite(Graphics2D g, Paint_Delegate paint, boolean usePaintAlpha,
int forceMode) {
// the alpha for the composite. Always opaque if the normal paint color is used since
// it contains the alpha
int alpha = usePaintAlpha ? paint.getAlpha() : 0xFF;
if (forceMode != 0) {
g.setComposite(AlphaComposite.getInstance(forceMode, (float) alpha / 255.f));
return;
}
Xfermode_Delegate xfermodeDelegate = paint.getXfermode();
if (xfermodeDelegate != null) {
if (xfermodeDelegate.isSupported()) {
Composite composite = xfermodeDelegate.getComposite(alpha);
assert composite != null;
if (composite != null) {
g.setComposite(composite);
return;
}
} else {
Bridge.getLog().fidelityWarning(LayoutLog.TAG_XFERMODE,
xfermodeDelegate.getSupportMessage(),
null /*throwable*/, null /*data*/);
}
}
// if there was no custom xfermode, but we have alpha (due to a shader and a non
// opaque alpha channel in the paint color), then we create an AlphaComposite anyway
// that will handle the alpha.
if (alpha != 0xFF) {
g.setComposite(AlphaComposite.getInstance(
AlphaComposite.SRC_OVER, (float) alpha / 255.f));
}
}
private void mapRect(AffineTransform matrix, RectF dst, RectF src) {
// array with 4 corners
float[] corners = new float[] {
src.left, src.top,
src.right, src.top,
src.right, src.bottom,
src.left, src.bottom,
};
// apply the transform to them.
matrix.transform(corners, 0, corners, 0, 4);
// now put the result in the rect. We take the min/max of Xs and min/max of Ys
dst.left = Math.min(Math.min(corners[0], corners[2]), Math.min(corners[4], corners[6]));
dst.right = Math.max(Math.max(corners[0], corners[2]), Math.max(corners[4], corners[6]));
dst.top = Math.min(Math.min(corners[1], corners[3]), Math.min(corners[5], corners[7]));
dst.bottom = Math.max(Math.max(corners[1], corners[3]), Math.max(corners[5], corners[7]));
}
}