Bitmap.java revision f12f6f058f169ead83a546a1c3a51e49452a30b3
19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/*
29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2006 The Android Open Source Project
39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License.
69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at
79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and
149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License.
159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpackage android.graphics;
189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.os.Parcel;
208cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chenimport android.os.Parcelable;
212784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackbornimport android.util.DisplayMetrics;
228cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chenimport java.io.OutputStream;
239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.nio.Buffer;
249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.nio.ByteBuffer;
259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.nio.IntBuffer;
268cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chenimport java.nio.ShortBuffer;
279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpublic final class Bitmap implements Parcelable {
29e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy
309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Indicates that the bitmap was created for an unknown pixel density.
329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3311ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * @see Bitmap#getDensity()
3411ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * @see Bitmap#setDensity(int)
359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3611ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    public static final int DENSITY_NONE = 0;
3711ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn
38ce0537b80087a6225273040a987414b1dd081aa0Romain Guy    /**
39ce0537b80087a6225273040a987414b1dd081aa0Romain Guy     * Note:  mNativeBitmap is used by FaceDetector_jni.cpp
40ce0537b80087a6225273040a987414b1dd081aa0Romain Guy     * Don't change/rename without updating FaceDetector_jni.cpp
41ce0537b80087a6225273040a987414b1dd081aa0Romain Guy     *
42ce0537b80087a6225273040a987414b1dd081aa0Romain Guy     * @hide
43ce0537b80087a6225273040a987414b1dd081aa0Romain Guy     */
44ce0537b80087a6225273040a987414b1dd081aa0Romain Guy    public final int mNativeBitmap;
458cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
46e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy    /**
47e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy     * Backing buffer for the Bitmap.
48e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy     * Made public for quick access from drawing methods -- do NOT modify
49e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy     * from outside this class.
50e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy     *
51e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy     * @hide
52e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy     */
53e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy    public byte[] mBuffer;
54e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy
552361098da3b9d9c3eeed410dc72ba62c0e9177cfRomain Guy    @SuppressWarnings({"FieldCanBeLocal", "UnusedDeclaration"}) // Keep to finalize native resources
56e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy    private final BitmapFinalizer mFinalizer;
57e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy
589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private final boolean mIsMutable;
599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private byte[] mNinePatchChunk;   // may be null
609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mWidth = -1;
619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mHeight = -1;
629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private boolean mRecycled;
639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6411ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    // Package-scoped for fast access.
6596e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn    /*package*/ int mDensity = sDefaultDensity = getDefaultDensity();
669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6711ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    private static volatile Matrix sScaleMatrix;
689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6996e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn    private static volatile int sDefaultDensity = -1;
7002890fd0f98b3b8d98baf0bda1ea906afd723d8bRomain Guy
7196e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn    /**
7296e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * For backwards compatibility, allows the app layer to change the default
7396e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * density when running old apps.
7496e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * @hide
7596e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     */
7696e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn    public static void setDefaultDensity(int density) {
7796e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn        sDefaultDensity = density;
7896e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn    }
7996e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn
8096e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn    /*package*/ static int getDefaultDensity() {
8196e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn        if (sDefaultDensity >= 0) {
8296e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn            return sDefaultDensity;
8396e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn        }
8496e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn        sDefaultDensity = DisplayMetrics.DENSITY_DEVICE;
8596e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn        return sDefaultDensity;
8696e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn    }
8796e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn
889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @noinspection UnusedDeclaration
909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*  Private constructor that must received an already allocated native
929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        bitmap int (pointer).
939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        This can be called from JNI code.
959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    */
967f9f99ea11051614a7727dfb9f9578b518e76e3cXavier Ducrohet    /*package*/ Bitmap(int nativeBitmap, byte[] buffer, boolean isMutable, byte[] ninePatchChunk,
97e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy            int density) {
989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (nativeBitmap == 0) {
999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new RuntimeException("internal error: native bitmap is 0");
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1018cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
102e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy        mBuffer = buffer;
1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // we delete this in our finalizer
1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mNativeBitmap = nativeBitmap;
105e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy        mFinalizer = new BitmapFinalizer(nativeBitmap);
106e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mIsMutable = isMutable;
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mNinePatchChunk = ninePatchChunk;
109de0dfb7b65a02d4dd74c271b558adee0973fc267Dianne Hackborn        if (density >= 0) {
110de0dfb7b65a02d4dd74c271b558adee0973fc267Dianne Hackborn            mDensity = density;
111de0dfb7b65a02d4dd74c271b558adee0973fc267Dianne Hackborn        }
1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
11511ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * <p>Returns the density for this bitmap.</p>
1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
11796e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * <p>The default density is the same density as the current display,
11896e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * unless the current application does not support different screen
11996e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * densities in which case it is
12096e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * {@link android.util.DisplayMetrics#DENSITY_DEFAULT}.  Note that
12196e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * compatibility mode is determined by the application that was initially
12296e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * loaded into a process -- applications that share the same process should
12396e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * all have the same compatibility, or ensure they explicitly set the
12496e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * density of their bitmaps appropriately.</p>
1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
12696e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * @return A scaling factor of the default density or {@link #DENSITY_NONE}
1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *         if the scaling factor is unknown.
1289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
12911ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * @see #setDensity(int)
130a53b828635fce8b6b2d3e3377d74d72070056623Dianne Hackborn     * @see android.util.DisplayMetrics#DENSITY_DEFAULT
13111ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * @see android.util.DisplayMetrics#densityDpi
13211ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * @see #DENSITY_NONE
1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
13411ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    public int getDensity() {
13511ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn        return mDensity;
1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
13911ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * <p>Specifies the density for this bitmap.  When the bitmap is
14011ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * drawn to a Canvas that also has a density, it will be scaled
14111ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * appropriately.</p>
1429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
14311ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * @param density The density scaling factor to use with this bitmap or
14411ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     *        {@link #DENSITY_NONE} if the density is unknown.
1459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
14611ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * @see #getDensity()
147a53b828635fce8b6b2d3e3377d74d72070056623Dianne Hackborn     * @see android.util.DisplayMetrics#DENSITY_DEFAULT
14811ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * @see android.util.DisplayMetrics#densityDpi
14911ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * @see #DENSITY_NONE
1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
15111ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    public void setDensity(int density) {
15211ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn        mDensity = density;
1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
15411ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Sets the nine patch chunk.
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param chunk The definition of the nine patch
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @hide
1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setNinePatchChunk(byte[] chunk) {
1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mNinePatchChunk = chunk;
1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1658cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
167a383b3f646cbc1fd870f2694d4fac24340834916Patrick Dubroy     * Free the native object associated with this bitmap, and clear the
168a383b3f646cbc1fd870f2694d4fac24340834916Patrick Dubroy     * reference to the pixel data. This will not free the pixel data synchronously;
169a383b3f646cbc1fd870f2694d4fac24340834916Patrick Dubroy     * it simply allows it to be garbage collected if there are no other references.
170a383b3f646cbc1fd870f2694d4fac24340834916Patrick Dubroy     * The bitmap is marked as "dead", meaning it will throw an exception if
171a383b3f646cbc1fd870f2694d4fac24340834916Patrick Dubroy     * getPixels() or setPixels() is called, and will draw nothing. This operation
172a383b3f646cbc1fd870f2694d4fac24340834916Patrick Dubroy     * cannot be reversed, so it should only be called if you are sure there are no
1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * further uses for the bitmap. This is an advanced call, and normally need
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * not be called, since the normal GC process will free up this memory when
1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * there are no more references to this bitmap.
1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void recycle() {
1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (!mRecycled) {
179e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy            mBuffer = null;
1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            nativeRecycle(mNativeBitmap);
1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mNinePatchChunk = null;
1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mRecycled = true;
1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns true if this bitmap has been recycled. If so, then it is an error
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * to try to access its pixels, and the bitmap will not draw.
1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return true if the bitmap has been recycled
1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public final boolean isRecycled() {
1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mRecycled;
1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1958cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1970bbae0836426ba2704e38e7f90a9d0ca502ab71dRomain Guy     * Returns the generation ID of this bitmap. The generation ID changes
1980bbae0836426ba2704e38e7f90a9d0ca502ab71dRomain Guy     * whenever the bitmap is modified. This can be used as an efficient way to
1990bbae0836426ba2704e38e7f90a9d0ca502ab71dRomain Guy     * check if a bitmap has changed.
2000bbae0836426ba2704e38e7f90a9d0ca502ab71dRomain Guy     *
2010bbae0836426ba2704e38e7f90a9d0ca502ab71dRomain Guy     * @return The current generation ID for this bitmap.
2020bbae0836426ba2704e38e7f90a9d0ca502ab71dRomain Guy     */
2030bbae0836426ba2704e38e7f90a9d0ca502ab71dRomain Guy    public int getGenerationId() {
2040bbae0836426ba2704e38e7f90a9d0ca502ab71dRomain Guy        return nativeGenerationId(mNativeBitmap);
2050bbae0836426ba2704e38e7f90a9d0ca502ab71dRomain Guy    }
2060bbae0836426ba2704e38e7f90a9d0ca502ab71dRomain Guy
2070bbae0836426ba2704e38e7f90a9d0ca502ab71dRomain Guy    /**
2089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * This is called by methods that want to throw an exception if the bitmap
2099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * has already been recycled.
2109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void checkRecycled(String errorMessage) {
2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mRecycled) {
2139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalStateException(errorMessage);
2149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2168cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Common code for checking that x and y are >= 0
2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param x x coordinate to ensure is >= 0
2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param y y coordinate to ensure is >= 0
2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static void checkXYSign(int x, int y) {
2249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (x < 0) {
2259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException("x must be >= 0");
2269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (y < 0) {
2289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException("y must be >= 0");
2299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Common code for checking that width and height are > 0
2349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param width  width to ensure is > 0
2369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param height height to ensure is > 0
2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static void checkWidthHeight(int width, int height) {
2399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (width <= 0) {
2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException("width must be > 0");
2419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (height <= 0) {
2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException("height must be > 0");
2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
247676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy    /**
248676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy     * Possible bitmap configurations. A bitmap configuration describes
249676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy     * how pixels are stored. This affects the quality (color depth) as
250676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy     * well as the ability to display transparent/translucent colors.
251676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy     */
2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public enum Config {
2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // these native values must match up with the enum in SkBitmap.h
254676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy
255676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy        /**
256676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * Each pixel is stored as a single translucency (alpha) channel.
257676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * This is very useful to efficiently store masks for instance.
258676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * No color information is stored.
259676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * With this configuration, each pixel requires 1 byte of memory.
260676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         */
2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ALPHA_8     (2),
262676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy
263676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy        /**
264676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * Each pixel is stored on 2 bytes and only the RGB channels are
265676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * encoded: red is stored with 5 bits of precision (32 possible
266676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * values), green is stored with 6 bits of precision (64 possible
267676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * values) and blue is stored with 5 bits of precision.
268676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         *
269676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * This configuration can produce slight visual artifacts depending
270676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * on the configuration of the source. For instance, without
271676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * dithering, the result might show a greenish tint. To get better
272676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * results dithering should be applied.
273676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         *
274676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * This configuration may be useful when using opaque bitmaps
275676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * that do not require high color fidelity.
276676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         */
2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        RGB_565     (4),
278676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy
279676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy        /**
280676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * Each pixel is stored on 2 bytes. The three RGB color channels
281676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * and the alpha channel (translucency) are stored with a 4 bits
282676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * precision (16 possible values.)
283676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         *
284676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * This configuration is mostly useful if the application needs
285676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * to store translucency information but also needs to save
286676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * memory.
287676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         *
288676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * It is recommended to use {@link #ARGB_8888} instead of this
289676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * configuration.
290676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         *
291676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * @deprecated Because of the poor quality of this configuration,
292676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         *             it is advised to use {@link #ARGB_8888} instead.
293676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         */
294676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy        @Deprecated
2959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ARGB_4444   (5),
296676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy
297676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy        /**
298676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * Each pixel is stored on 4 bytes. Each channel (RGB and alpha
299676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * for translucency) is stored with 8 bits of precision (256
300676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * possible values.)
301676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         *
302676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * This configuration is very flexible and offers the best
303676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         * quality. It should be used whenever possible.
304676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy         */
3059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ARGB_8888   (6);
3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
307676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy        final int nativeInt;
308676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy
309676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy        @SuppressWarnings({"deprecation"})
310676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy        private static Config sConfigs[] = {
311676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy            null, null, ALPHA_8, null, RGB_565, ARGB_4444, ARGB_8888
312676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy        };
313676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy
3149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Config(int ni) {
3159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            this.nativeInt = ni;
3169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3178cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
318676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy        static Config nativeToConfig(int ni) {
3199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return sConfigs[ni];
3209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3228cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
3239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Copy the bitmap's pixels into the specified buffer (allocated by the
3259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * caller). An exception is thrown if the buffer is not large enough to
3269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * hold all of the pixels (taking into account the number of bytes per
3279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * pixel) or if the Buffer subclass is not one of the support types
3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * (ByteBuffer, ShortBuffer, IntBuffer).
3299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void copyPixelsToBuffer(Buffer dst) {
3319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int elements = dst.remaining();
3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int shift;
3339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (dst instanceof ByteBuffer) {
3349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            shift = 0;
3359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (dst instanceof ShortBuffer) {
3369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            shift = 1;
3379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (dst instanceof IntBuffer) {
3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            shift = 2;
3399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
3409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new RuntimeException("unsupported Buffer subclass");
3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3428cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        long bufferSize = (long)elements << shift;
344f7f9d9c39df22ad6929f001f07588469f77e8bf5Jesse Wilson        long pixelSize = getByteCount();
3458cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (bufferSize < pixelSize) {
3479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new RuntimeException("Buffer not large enough for pixels");
3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3498cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        nativeCopyPixelsToBuffer(mNativeBitmap, dst);
3518cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // now update the buffer's position
3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int position = dst.position();
3549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        position += pixelSize >> shift;
3559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dst.position(position);
3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Copy the pixels from the buffer, beginning at the current position,
3609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * overwriting the bitmap's pixels. The data in the buffer is not changed
3619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * in any way (unlike setPixels(), which converts from unpremultipled 32bit
3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * to whatever the bitmap's native format is.
3639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void copyPixelsFromBuffer(Buffer src) {
3659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkRecycled("copyPixelsFromBuffer called on recycled bitmap");
3668cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
3679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int elements = src.remaining();
3689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int shift;
3699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (src instanceof ByteBuffer) {
3709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            shift = 0;
3719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (src instanceof ShortBuffer) {
3729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            shift = 1;
3739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (src instanceof IntBuffer) {
3749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            shift = 2;
3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
3769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new RuntimeException("unsupported Buffer subclass");
3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3788cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
3799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        long bufferBytes = (long)elements << shift;
380f7f9d9c39df22ad6929f001f07588469f77e8bf5Jesse Wilson        long bitmapBytes = getByteCount();
3818cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
3829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (bufferBytes < bitmapBytes) {
3839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new RuntimeException("Buffer not large enough for pixels");
3849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3858cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        nativeCopyPixelsFromBuffer(mNativeBitmap, src);
3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3888cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
3899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Tries to make a new bitmap based on the dimensions of this bitmap,
3919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * setting the new bitmap's config to the one specified, and then copying
3929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * this bitmap's pixels into the new bitmap. If the conversion is not
39396e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * supported, or the allocator fails, then this returns NULL.  The returned
39496e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * bitmap initially has the same density as the original.
3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param config    The desired config for the resulting bitmap
3979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param isMutable True if the resulting bitmap should be mutable (i.e.
3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                  its pixels can be modified)
3999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return the new bitmap, or null if the copy could not be made.
4009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
4019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public Bitmap copy(Config config, boolean isMutable) {
4029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkRecycled("Can't copy a recycled bitmap");
40396e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn        Bitmap b = nativeCopy(mNativeBitmap, config.nativeInt, isMutable);
40496e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn        if (b != null) {
40596e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn            b.mDensity = mDensity;
40696e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn        }
40796e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn        return b;
4089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
41002d9102b3bdebac6989c2b7d12c080fb7c1f8e2bBrad Fitzpatrick    /**
411f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     * Creates a new bitmap, scaled from an existing bitmap, when possible. If the
412f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     * specified width and height are the same as the current width and height of
413f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     * the source btimap, the source bitmap is returned and now new bitmap is
414f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     * created.
41502d9102b3bdebac6989c2b7d12c080fb7c1f8e2bBrad Fitzpatrick     *
41602d9102b3bdebac6989c2b7d12c080fb7c1f8e2bBrad Fitzpatrick     * @param src       The source bitmap.
41702d9102b3bdebac6989c2b7d12c080fb7c1f8e2bBrad Fitzpatrick     * @param dstWidth  The new bitmap's desired width.
41802d9102b3bdebac6989c2b7d12c080fb7c1f8e2bBrad Fitzpatrick     * @param dstHeight The new bitmap's desired height.
41902d9102b3bdebac6989c2b7d12c080fb7c1f8e2bBrad Fitzpatrick     * @param filter    true if the source should be filtered.
420f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     * @return The new scaled bitmap or the source bitmap if no scaling is required.
42102d9102b3bdebac6989c2b7d12c080fb7c1f8e2bBrad Fitzpatrick     */
422f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy    public static Bitmap createScaledBitmap(Bitmap src, int dstWidth, int dstHeight,
423f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy            boolean filter) {
4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Matrix m;
4259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        synchronized (Bitmap.class) {
4269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // small pool of just 1 matrix
4279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            m = sScaleMatrix;
4289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            sScaleMatrix = null;
4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (m == null) {
4329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            m = new Matrix();
4339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4348cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
4359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        final int width = src.getWidth();
4369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        final int height = src.getHeight();
4379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        final float sx = dstWidth  / (float)width;
4389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        final float sy = dstHeight / (float)height;
4399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        m.setScale(sx, sy);
4409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Bitmap b = Bitmap.createBitmap(src, 0, 0, width, height, m, filter);
4419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        synchronized (Bitmap.class) {
4439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // do we need to check for null? why not just assign everytime?
4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (sScaleMatrix == null) {
4459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                sScaleMatrix = m;
4469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
4479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4498cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen        return b;
4509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4518cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
4529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
4539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns an immutable bitmap from the source bitmap. The new bitmap may
45496e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * be the same object as source, or a copy may have been made.  It is
45596e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * initialized with the same density as the original bitmap.
4569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
4579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static Bitmap createBitmap(Bitmap src) {
4589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return createBitmap(src, 0, 0, src.getWidth(), src.getHeight());
4599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
4629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns an immutable bitmap from the specified subset of the source
4639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * bitmap. The new bitmap may be the same object as source, or a copy may
464f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     * have been made. It is initialized with the same density as the original
465f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     * bitmap.
4669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
4679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param source   The bitmap we are subsetting
4689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param x        The x coordinate of the first pixel in source
4699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param y        The y coordinate of the first pixel in source
4709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param width    The number of pixels in each row
4719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param height   The number of rows
472f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     * @return A copy of a subset of the source bitmap or the source bitmap itself.
4739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
4749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static Bitmap createBitmap(Bitmap source, int x, int y, int width, int height) {
4759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return createBitmap(source, x, y, width, height, null, false);
4769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4778cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
4789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
4799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns an immutable bitmap from subset of the source bitmap,
480f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     * transformed by the optional matrix. The new bitmap may be the
481f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     * same object as source, or a copy may have been made. It is
48296e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * initialized with the same density as the original bitmap.
483f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     *
484f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     * If the source bitmap is immutable and the requested subset is the
485f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     * same as the source bitmap itself, then the source bitmap is
486f12f6f058f169ead83a546a1c3a51e49452a30b3Romain Guy     * returned and no new bitmap is created.
4879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
4889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param source   The bitmap we are subsetting
4899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param x        The x coordinate of the first pixel in source
4909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param y        The y coordinate of the first pixel in source
4919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param width    The number of pixels in each row
4929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param height   The number of rows
49360b88edea7132ddce90f2dced07c6706f1502270Ken Shirriff     * @param m        Optional matrix to be applied to the pixels
4949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param filter   true if the source should be filtered.
4959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                   Only applies if the matrix contains more than just
4969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                   translation.
4979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return A bitmap that represents the specified subset of source
4989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws IllegalArgumentException if the x, y, width, height values are
4999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *         outside of the dimensions of the source bitmap.
5009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static Bitmap createBitmap(Bitmap source, int x, int y, int width, int height,
5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            Matrix m, boolean filter) {
5039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkXYSign(x, y);
5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkWidthHeight(width, height);
5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (x + width > source.getWidth()) {
5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException("x + width must be <= bitmap.width()");
5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (y + height > source.getHeight()) {
5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException("y + height must be <= bitmap.height()");
5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // check if we can just return our argument unchanged
5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (!source.isMutable() && x == 0 && y == 0 && width == source.getWidth() &&
5159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                height == source.getHeight() && (m == null || m.isIdentity())) {
5169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return source;
5179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5188cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
5199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int neww = width;
5209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int newh = height;
5219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Canvas canvas = new Canvas();
5229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Bitmap bitmap;
5239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Paint paint;
5249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Rect srcR = new Rect(x, y, x + width, y + height);
5269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        RectF dstR = new RectF(0, 0, width, height);
5279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
528feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy        Config newConfig = Config.ARGB_8888;
529feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy        final Config config = source.getConfig();
530feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy        // GIF files generate null configs, assume ARGB_8888
531feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy        if (config != null) {
532feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy            switch (config) {
533feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy                case RGB_565:
534feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy                    newConfig = Config.RGB_565;
535feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy                    break;
536feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy                case ALPHA_8:
537feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy                    newConfig = Config.ALPHA_8;
538feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy                    break;
539676b17391fb6583e5df944c36b1bd4c1be835689Romain Guy                //noinspection deprecation
540feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy                case ARGB_4444:
541feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy                case ARGB_8888:
542feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy                default:
543feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy                    newConfig = Config.ARGB_8888;
544feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy                    break;
545feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy            }
546feeea8f07c854e6b0ae2fec3aa2d239c17f32b5bRomain Guy        }
547d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy
5489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (m == null || m.isIdentity()) {
549d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy            bitmap = createBitmap(neww, newh, newConfig, source.hasAlpha());
5509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            paint = null;   // not needed
5519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
552d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy            final boolean transformed = !m.rectStaysRect();
553d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy
5549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            RectF deviceR = new RectF();
5559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            m.mapRect(deviceR, dstR);
556d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy
5579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            neww = Math.round(deviceR.width());
5589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            newh = Math.round(deviceR.height());
559d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy
560d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy            bitmap = createBitmap(neww, newh, transformed ? Config.ARGB_8888 : newConfig,
561d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy                    transformed || source.hasAlpha());
562d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy
5639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            canvas.translate(-deviceR.left, -deviceR.top);
5649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            canvas.concat(m);
565d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy
5669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            paint = new Paint();
5679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            paint.setFilterBitmap(filter);
568d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy            if (transformed) {
5699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                paint.setAntiAlias(true);
5709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
57296e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn
5739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // The new bitmap was created from a known bitmap source so assume that
57411ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn        // they use the same density
57511ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn        bitmap.mDensity = source.mDensity;
57696e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn
57796e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn        canvas.setBitmap(bitmap);
57896e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn        canvas.drawBitmap(source, srcR, dstR, paint);
5796311d0a079702b29984c0d31937345be105e1a5eDianne Hackborn        canvas.setBitmap(null);
5809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return bitmap;
5829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5838cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
5849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
58596e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * Returns a mutable bitmap with the specified width and height.  Its
58696e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * initial density is as per {@link #getDensity}.
5879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
5889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param width    The width of the bitmap
5899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param height   The height of the bitmap
5909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param config   The bitmap config to create.
5919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws IllegalArgumentException if the width or height are <= 0
5929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static Bitmap createBitmap(int width, int height, Config config) {
594d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy        return createBitmap(width, height, config, true);
595d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy    }
596d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy
597d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy    /**
598d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy     * Returns a mutable bitmap with the specified width and height.  Its
599d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy     * initial density is as per {@link #getDensity}.
600d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy     *
601d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy     * @param width    The width of the bitmap
602d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy     * @param height   The height of the bitmap
603d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy     * @param config   The bitmap config to create.
604d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy     * @param hasAlpha If the bitmap is ARGB_8888 this flag can be used to mark the
605d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy     *                 bitmap as opaque. Doing so will clear the bitmap in black
606d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy     *                 instead of transparent.
607d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy     *
608d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy     * @throws IllegalArgumentException if the width or height are <= 0
609d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy     */
610d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy    private static Bitmap createBitmap(int width, int height, Config config, boolean hasAlpha) {
611e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy        if (width <= 0 || height <= 0) {
612e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy            throw new IllegalArgumentException("width and height must be > 0");
613e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy        }
6149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Bitmap bm = nativeCreate(null, 0, width, width, height, config.nativeInt, true);
615d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy        if (config == Config.ARGB_8888 && !hasAlpha) {
6161373a8eb581fe3c8e9a036e69042015f98a7e346Christopher Tate            nativeErase(bm.mNativeBitmap, 0xff000000);
617d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy            nativeSetHasAlpha(bm.mNativeBitmap, hasAlpha);
618d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy        } else {
6191373a8eb581fe3c8e9a036e69042015f98a7e346Christopher Tate            // No need to initialize it to zeroes; it is backed by a VM byte array
6201373a8eb581fe3c8e9a036e69042015f98a7e346Christopher Tate            // which is by definition preinitialized to all zeroes.
6211373a8eb581fe3c8e9a036e69042015f98a7e346Christopher Tate            //
6221373a8eb581fe3c8e9a036e69042015f98a7e346Christopher Tate            //nativeErase(bm.mNativeBitmap, 0);
623d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy        }
6249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return bm;
6259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6268cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
6279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
6289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns a immutable bitmap with the specified width and height, with each
62996e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * pixel value set to the corresponding value in the colors array.  Its
63096e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * initial density is as per {@link #getDensity}.
6319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
6329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param colors   Array of {@link Color} used to initialize the pixels.
6339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param offset   Number of values to skip before the first color in the
6349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 array of colors.
6359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param stride   Number of colors in the array between rows (must be >=
6369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 width or <= -width).
6379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param width    The width of the bitmap
6389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param height   The height of the bitmap
6399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param config   The bitmap config to create. If the config does not
6409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 support per-pixel alpha (e.g. RGB_565), then the alpha
6419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 bytes in the colors[] will be ignored (assumed to be FF)
6429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws IllegalArgumentException if the width or height are <= 0, or if
6439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *         the color array's length is less than the number of pixels.
6449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
6459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static Bitmap createBitmap(int colors[], int offset, int stride,
6469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int width, int height, Config config) {
6479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkWidthHeight(width, height);
6499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (Math.abs(stride) < width) {
6509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException("abs(stride) must be >= width");
6519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int lastScanline = offset + (height - 1) * stride;
6539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int length = colors.length;
6549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (offset < 0 || (offset + width > length) || lastScanline < 0 ||
6559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                (lastScanline + width > length)) {
6569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new ArrayIndexOutOfBoundsException();
6579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
658e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy        if (width <= 0 || height <= 0) {
659e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy            throw new IllegalArgumentException("width and height must be > 0");
660e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy        }
6619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return nativeCreate(colors, offset, stride, width, height,
6629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            config.nativeInt, false);
6639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
6669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns a immutable bitmap with the specified width and height, with each
66796e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * pixel value set to the corresponding value in the colors array.  Its
66896e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * initial density is as per {@link #getDensity}.
6699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
6709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param colors   Array of {@link Color} used to initialize the pixels.
6719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 This array must be at least as large as width * height.
6729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param width    The width of the bitmap
6739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param height   The height of the bitmap
6749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param config   The bitmap config to create. If the config does not
6759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 support per-pixel alpha (e.g. RGB_565), then the alpha
6769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 bytes in the colors[] will be ignored (assumed to be FF)
6779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws IllegalArgumentException if the width or height are <= 0, or if
6789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *         the color array's length is less than the number of pixels.
6799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
6809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static Bitmap createBitmap(int colors[], int width, int height, Config config) {
6819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return createBitmap(colors, 0, width, width, height, config);
6829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
6859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns an optional array of private data, used by the UI system for
6869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * some bitmaps. Not intended to be called by applications.
6879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
6889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public byte[] getNinePatchChunk() {
6899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mNinePatchChunk;
6909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
6939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Specifies the known formats a bitmap can be compressed into
6949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
6959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public enum CompressFormat {
6969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        JPEG    (0),
6972305ac9e4a262ed09fd034ae417e9b1dda4c0ccbVikas Arora        PNG     (1),
6982305ac9e4a262ed09fd034ae417e9b1dda4c0ccbVikas Arora        WEBP    (2);
6999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        CompressFormat(int nativeInt) {
7019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            this.nativeInt = nativeInt;
7029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        final int nativeInt;
7049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
7079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Number of bytes of temp storage we use for communicating between the
7089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * native compressor and the java OutputStream.
7099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
7109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private final static int WORKING_COMPRESS_STORAGE = 4096;
7119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
7139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Write a compressed version of the bitmap to the specified outputstream.
7149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * If this returns true, the bitmap can be reconstructed by passing a
7159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * corresponding inputstream to BitmapFactory.decodeStream(). Note: not
7169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * all Formats support all bitmap configs directly, so it is possible that
7179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * the returned bitmap from BitmapFactory could be in a different bitdepth,
7189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * and/or may have lost per-pixel alpha (e.g. JPEG only supports opaque
7199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * pixels).
7209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
7219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param format   The format of the compressed image
7229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param quality  Hint to the compressor, 0-100. 0 meaning compress for
7239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 small size, 100 meaning compress for max quality. Some
7249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 formats, like PNG which is lossless, will ignore the
7259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 quality setting
7269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param stream   The outputstream to write the compressed data.
7279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return true if successfully compressed to the specified stream.
7289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
7299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public boolean compress(CompressFormat format, int quality, OutputStream stream) {
7309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkRecycled("Can't compress a recycled bitmap");
7319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // do explicit check before calling the native method
7329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (stream == null) {
7339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new NullPointerException();
7349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (quality < 0 || quality > 100) {
7369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException("quality must be 0..100");
7379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return nativeCompress(mNativeBitmap, format.nativeInt, quality,
7399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                              stream, new byte[WORKING_COMPRESS_STORAGE]);
7409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7418cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
7429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
7439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns true if the bitmap is marked as mutable (i.e. can be drawn into)
7449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
7459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public final boolean isMutable() {
7469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mIsMutable;
7479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /** Returns the bitmap's width */
7509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public final int getWidth() {
7519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mWidth == -1 ? mWidth = nativeWidth(mNativeBitmap) : mWidth;
7529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /** Returns the bitmap's height */
7559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public final int getHeight() {
7569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mHeight == -1 ? mHeight = nativeHeight(mNativeBitmap) : mHeight;
7579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7588cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
7599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
76011ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * Convenience for calling {@link #getScaledWidth(int)} with the target
76111ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * density of the given {@link Canvas}.
7629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
7632784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn    public int getScaledWidth(Canvas canvas) {
76411ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn        return scaleFromDensity(getWidth(), mDensity, canvas.mDensity);
7652784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn    }
7662784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn
7672784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn    /**
76811ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * Convenience for calling {@link #getScaledHeight(int)} with the target
76911ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * density of the given {@link Canvas}.
7702784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn     */
7712784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn    public int getScaledHeight(Canvas canvas) {
77211ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn        return scaleFromDensity(getHeight(), mDensity, canvas.mDensity);
77311ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    }
77411ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn
77511ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    /**
77611ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * Convenience for calling {@link #getScaledWidth(int)} with the target
77711ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * density of the given {@link DisplayMetrics}.
77811ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     */
77911ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    public int getScaledWidth(DisplayMetrics metrics) {
78011ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn        return scaleFromDensity(getWidth(), mDensity, metrics.densityDpi);
78111ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    }
78211ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn
78311ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    /**
78411ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * Convenience for calling {@link #getScaledHeight(int)} with the target
78511ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * density of the given {@link DisplayMetrics}.
78611ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     */
78711ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    public int getScaledHeight(DisplayMetrics metrics) {
78811ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn        return scaleFromDensity(getHeight(), mDensity, metrics.densityDpi);
7892784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn    }
7902784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn
7912784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn    /**
7922784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn     * Convenience method that returns the width of this bitmap divided
7932784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn     * by the density scale factor.
7942784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn     *
79511ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * @param targetDensity The density of the target canvas of the bitmap.
7962784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn     * @return The scaled width of this bitmap, according to the density scale factor.
7972784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn     */
79811ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    public int getScaledWidth(int targetDensity) {
79911ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn        return scaleFromDensity(getWidth(), mDensity, targetDensity);
8009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
8039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Convenience method that returns the height of this bitmap divided
8049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * by the density scale factor.
8059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
80611ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * @param targetDensity The density of the target canvas of the bitmap.
8079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return The scaled height of this bitmap, according to the density scale factor.
8089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
80911ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    public int getScaledHeight(int targetDensity) {
81011ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn        return scaleFromDensity(getHeight(), mDensity, targetDensity);
81111ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    }
81211ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn
81311ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    /**
81411ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     * @hide
81511ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn     */
81611ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn    static public int scaleFromDensity(int size, int sdensity, int tdensity) {
81711ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn        if (sdensity == DENSITY_NONE || sdensity == tdensity) {
81811ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn            return size;
8192784ff0af88128f66ae690b73d48fb7e4a211e68Dianne Hackborn        }
82011ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn
82111ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn        // Scale by tdensity / sdensity, rounding up.
822366a84056c9a695ee6702d9d30bf9f3b521ba7ccRomain Guy        return ((size * tdensity) + (sdensity >> 1)) / sdensity;
8239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
82411ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn
8259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
8269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Return the number of bytes between rows in the bitmap's pixels. Note that
8279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * this refers to the pixels as stored natively by the bitmap. If you call
8289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * getPixels() or setPixels(), then the pixels are uniformly treated as
8299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * 32bit values, packed according to the Color class.
8309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
8319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return number of bytes between rows of the native bitmap pixels.
8329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
8339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public final int getRowBytes() {
8349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return nativeRowBytes(mNativeBitmap);
8359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8368cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
8379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
838f7f9d9c39df22ad6929f001f07588469f77e8bf5Jesse Wilson     * Returns the number of bytes used to store this bitmap's pixels.
839f7f9d9c39df22ad6929f001f07588469f77e8bf5Jesse Wilson     */
840f7f9d9c39df22ad6929f001f07588469f77e8bf5Jesse Wilson    public final int getByteCount() {
841f7f9d9c39df22ad6929f001f07588469f77e8bf5Jesse Wilson        // int result permits bitmaps up to 46,340 x 46,340
842f7f9d9c39df22ad6929f001f07588469f77e8bf5Jesse Wilson        return getRowBytes() * getHeight();
843f7f9d9c39df22ad6929f001f07588469f77e8bf5Jesse Wilson    }
844f7f9d9c39df22ad6929f001f07588469f77e8bf5Jesse Wilson
845f7f9d9c39df22ad6929f001f07588469f77e8bf5Jesse Wilson    /**
8469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * If the bitmap's internal config is in one of the public formats, return
8479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * that config, otherwise return null.
8489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
8499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public final Config getConfig() {
8509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return Config.nativeToConfig(nativeConfig(mNativeBitmap));
8519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
853a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed    /** Returns true if the bitmap's config supports per-pixel alpha, and
854a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     * if the pixels may contain non-opaque alpha values. For some configs,
855a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     * this is always false (e.g. RGB_565), since they do not support per-pixel
856a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     * alpha. However, for configs that do, the bitmap may be flagged to be
857a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     * known that all of its pixels are opaque. In this case hasAlpha() will
858a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     * also return false. If a config such as ARGB_8888 is not so flagged,
859a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     * it will return true by default.
860a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     */
8619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public final boolean hasAlpha() {
8629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return nativeHasAlpha(mNativeBitmap);
8639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
866a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     * Tell the bitmap if all of the pixels are known to be opaque (false)
867a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     * or if some of the pixels may contain non-opaque alpha values (true).
868366a84056c9a695ee6702d9d30bf9f3b521ba7ccRomain Guy     * Note, for some configs (e.g. RGB_565) this call is ignored, since it
869366a84056c9a695ee6702d9d30bf9f3b521ba7ccRomain Guy     * does not support per-pixel alpha values.
870a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     *
871a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     * This is meant as a drawing hint, as in some cases a bitmap that is known
872a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     * to be opaque can take a faster drawing case than one that may have
873a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     * non-opaque per-pixel alpha values.
874a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed     */
875a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed    public void setHasAlpha(boolean hasAlpha) {
876a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed        nativeSetHasAlpha(mNativeBitmap, hasAlpha);
877a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed    }
878a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed
879a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed    /**
8809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Fills the bitmap's pixels with the specified {@link Color}.
8819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
8829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws IllegalStateException if the bitmap is not mutable.
8839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
8849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void eraseColor(int c) {
8859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkRecycled("Can't erase a recycled bitmap");
8869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (!isMutable()) {
8879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalStateException("cannot erase immutable bitmaps");
8889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        nativeErase(mNativeBitmap, c);
8909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
8939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns the {@link Color} at the specified location. Throws an exception
8949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * if x or y are out of bounds (negative or >= to the width or height
8959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * respectively).
8969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
8979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param x    The x coordinate (0...width-1) of the pixel to return
8989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param y    The y coordinate (0...height-1) of the pixel to return
8999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return     The argb {@link Color} at the specified coordinate
9009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws IllegalArgumentException if x, y exceed the bitmap's bounds
9019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
9029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public int getPixel(int x, int y) {
9039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkRecycled("Can't call getPixel() on a recycled bitmap");
9049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkPixelAccess(x, y);
9059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return nativeGetPixel(mNativeBitmap, x, y);
9069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9078cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
9089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
9099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns in pixels[] a copy of the data in the bitmap. Each value is
9109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * a packed int representing a {@link Color}. The stride parameter allows
9119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * the caller to allow for gaps in the returned pixels array between
9129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * rows. For normal packed results, just pass width for the stride value.
9139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
9149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param pixels   The array to receive the bitmap's colors
9159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param offset   The first index to write into pixels[]
9169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param stride   The number of entries in pixels[] to skip between
9179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 rows (must be >= bitmap's width). Can be negative.
9189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param x        The x coordinate of the first pixel to read from
9199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 the bitmap
9209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param y        The y coordinate of the first pixel to read from
9219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 the bitmap
9229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param width    The number of pixels to read from each row
9239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param height   The number of rows to read
9249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws IllegalArgumentException if x, y, width, height exceed the
9259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *         bounds of the bitmap, or if abs(stride) < width.
9269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws ArrayIndexOutOfBoundsException if the pixels array is too small
9279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *         to receive the specified number of pixels.
9289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
9299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void getPixels(int[] pixels, int offset, int stride,
9309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                          int x, int y, int width, int height) {
9319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkRecycled("Can't call getPixels() on a recycled bitmap");
9329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (width == 0 || height == 0) {
9339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return; // nothing to do
9349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkPixelsAccess(x, y, width, height, offset, stride, pixels);
9369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        nativeGetPixels(mNativeBitmap, pixels, offset, stride,
9379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        x, y, width, height);
9389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9398cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
9409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
9419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Shared code to check for illegal arguments passed to getPixel()
9429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * or setPixel()
9439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param x x coordinate of the pixel
9449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param y y coordinate of the pixel
9459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
9469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void checkPixelAccess(int x, int y) {
9479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkXYSign(x, y);
9489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (x >= getWidth()) {
9499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException("x must be < bitmap.width()");
9509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (y >= getHeight()) {
9529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException("y must be < bitmap.height()");
9539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
9579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Shared code to check for illegal arguments passed to getPixels()
9589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * or setPixels()
9599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
9609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param x left edge of the area of pixels to access
9619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param y top edge of the area of pixels to access
9629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param width width of the area of pixels to access
9639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param height height of the area of pixels to access
9649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param offset offset into pixels[] array
9659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param stride number of elements in pixels[] between each logical row
9669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param pixels array to hold the area of pixels being accessed
9679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    */
9689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void checkPixelsAccess(int x, int y, int width, int height,
9699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                   int offset, int stride, int pixels[]) {
9709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkXYSign(x, y);
9719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (width < 0) {
9729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException("width must be >= 0");
9739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (height < 0) {
9759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException("height must be >= 0");
9769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (x + width > getWidth()) {
9789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException(
9799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    "x + width must be <= bitmap.width()");
9809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (y + height > getHeight()) {
9829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException(
9839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    "y + height must be <= bitmap.height()");
9849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (Math.abs(stride) < width) {
9869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalArgumentException("abs(stride) must be >= width");
9879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int lastScanline = offset + (height - 1) * stride;
9899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int length = pixels.length;
9909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (offset < 0 || (offset + width > length)
9919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                || lastScanline < 0
9929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                || (lastScanline + width > length)) {
9939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new ArrayIndexOutOfBoundsException();
9949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9968cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
9979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
9989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Write the specified {@link Color} into the bitmap (assuming it is
9999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * mutable) at the x,y coordinate.
10009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
10019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param x     The x coordinate of the pixel to replace (0...width-1)
10029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param y     The y coordinate of the pixel to replace (0...height-1)
10039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param color The {@link Color} to write into the bitmap
10049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws IllegalStateException if the bitmap is not mutable
10059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws IllegalArgumentException if x, y are outside of the bitmap's
10069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *         bounds.
10079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
10089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setPixel(int x, int y, int color) {
10099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkRecycled("Can't call setPixel() on a recycled bitmap");
10109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (!isMutable()) {
10119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalStateException();
10129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkPixelAccess(x, y);
10149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        nativeSetPixel(mNativeBitmap, x, y, color);
10159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10168cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
10179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
10189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Replace pixels in the bitmap with the colors in the array. Each element
10198cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen     * in the array is a packed int prepresenting a {@link Color}
10209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
10219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param pixels   The colors to write to the bitmap
10229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param offset   The index of the first color to read from pixels[]
10239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param stride   The number of colors in pixels[] to skip between rows.
10249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 Normally this value will be the same as the width of
10259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 the bitmap, but it can be larger (or negative).
10269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param x        The x coordinate of the first pixel to write to in
10279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 the bitmap.
10289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param y        The y coordinate of the first pixel to write to in
10299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 the bitmap.
10309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param width    The number of colors to copy from pixels[] per row
10319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param height   The number of rows to write to the bitmap
10329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws IllegalStateException if the bitmap is not mutable
10339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws IllegalArgumentException if x, y, width, height are outside of
10349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *         the bitmap's bounds.
10359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws ArrayIndexOutOfBoundsException if the pixels array is too small
10369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *         to receive the specified number of pixels.
10379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
10389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setPixels(int[] pixels, int offset, int stride,
10399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                          int x, int y, int width, int height) {
10409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkRecycled("Can't call setPixels() on a recycled bitmap");
10419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (!isMutable()) {
10429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new IllegalStateException();
10439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (width == 0 || height == 0) {
10459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return; // nothing to do
10469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkPixelsAccess(x, y, width, height, offset, stride, pixels);
10489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        nativeSetPixels(mNativeBitmap, pixels, offset, stride,
10499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        x, y, width, height);
10509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10518cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
10529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final Parcelable.Creator<Bitmap> CREATOR
10539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            = new Parcelable.Creator<Bitmap>() {
10549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
10559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Rebuilds a bitmap previously stored with writeToParcel().
10569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
10579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @param p    Parcel object to read the bitmap from
10589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @return a new bitmap created from the data in the parcel
10599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
10609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public Bitmap createFromParcel(Parcel p) {
10619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            Bitmap bm = nativeCreateFromParcel(p);
10629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (bm == null) {
10639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                throw new RuntimeException("Failed to unparcel Bitmap");
10649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
10659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return bm;
10669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public Bitmap[] newArray(int size) {
10689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return new Bitmap[size];
10699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    };
10719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
10739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * No special parcel contents.
10749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
10759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public int describeContents() {
1076b6377170960d40e66858d8b4d335a95eac773762Bart Sears        return 0;
10779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
10809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Write the bitmap and its pixels to the parcel. The bitmap can be
10819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * rebuilt from the parcel by calling CREATOR.createFromParcel().
10829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param p    Parcel object to write the bitmap data into
10839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
10849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void writeToParcel(Parcel p, int flags) {
10859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkRecycled("Can't parcel a recycled bitmap");
1086de0dfb7b65a02d4dd74c271b558adee0973fc267Dianne Hackborn        if (!nativeWriteToParcel(mNativeBitmap, mIsMutable, mDensity, p)) {
10879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new RuntimeException("native writeToParcel failed");
10889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
10929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns a new bitmap that captures the alpha values of the original.
10939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * This may be drawn with Canvas.drawBitmap(), where the color(s) will be
10949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * taken from the paint that is passed to the draw call.
10959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
10969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return new bitmap containing the alpha channel of the original bitmap.
10979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
10989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public Bitmap extractAlpha() {
10999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return extractAlpha(null, null);
11009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11018cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
11029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
11039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns a new bitmap that captures the alpha values of the original.
11049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * These values may be affected by the optional Paint parameter, which
11059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * can contain its own alpha, and may also contain a MaskFilter which
11069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * could change the actual dimensions of the resulting bitmap (e.g.
11079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * a blur maskfilter might enlarge the resulting bitmap). If offsetXY
11089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * is not null, it returns the amount to offset the returned bitmap so
11099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * that it will logically align with the original. For example, if the
11109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * paint contains a blur of radius 2, then offsetXY[] would contains
11119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * -2, -2, so that drawing the alpha bitmap offset by (-2, -2) and then
11129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * drawing the original would result in the blur visually aligning with
11139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * the original.
111496e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     *
111596e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     * <p>The initial density of the returned bitmap is the same as the original's.
111696e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn     *
11179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param paint Optional paint used to modify the alpha values in the
11189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *              resulting bitmap. Pass null for default behavior.
11199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param offsetXY Optional array that returns the X (index 0) and Y
11209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 (index 1) offset needed to position the returned bitmap
11219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *                 so that it visually lines up with the original.
11229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return new bitmap containing the (optionally modified by paint) alpha
11239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *         channel of the original bitmap. This may be drawn with
11249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *         Canvas.drawBitmap(), where the color(s) will be taken from the
11259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *         paint that is passed to the draw call.
11269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
11279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public Bitmap extractAlpha(Paint paint, int[] offsetXY) {
11289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        checkRecycled("Can't extractAlpha on a recycled bitmap");
11299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int nativePaint = paint != null ? paint.mNativePaint : 0;
11309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Bitmap bm = nativeExtractAlpha(mNativeBitmap, nativePaint, offsetXY);
11319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (bm == null) {
11329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new RuntimeException("Failed to extractAlpha on Bitmap");
11339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
113496e240f25a97c10bba863df328ed73a82c34ff61Dianne Hackborn        bm.mDensity = mDensity;
11359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return bm;
11369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11388cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen    /**
113976d1e01d5e65c4631c827831e98ad4e300d99eabMike Reed     *  Given another bitmap, return true if it has the same dimensions, config,
114076d1e01d5e65c4631c827831e98ad4e300d99eabMike Reed     *  and pixel data as this bitmap. If any of those differ, return false.
114176d1e01d5e65c4631c827831e98ad4e300d99eabMike Reed     *  If other is null, return false.
114276d1e01d5e65c4631c827831e98ad4e300d99eabMike Reed     */
114376d1e01d5e65c4631c827831e98ad4e300d99eabMike Reed    public boolean sameAs(Bitmap other) {
1144366a84056c9a695ee6702d9d30bf9f3b521ba7ccRomain Guy        return this == other || (other != null && nativeSameAs(mNativeBitmap, other.mNativeBitmap));
114576d1e01d5e65c4631c827831e98ad4e300d99eabMike Reed    }
114676d1e01d5e65c4631c827831e98ad4e300d99eabMike Reed
114776d1e01d5e65c4631c827831e98ad4e300d99eabMike Reed    /**
11488cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen     * Rebuilds any caches associated with the bitmap that are used for
11498cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen     * drawing it. In the case of purgeable bitmaps, this call will attempt to
11508cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen     * ensure that the pixels have been decoded.
11518cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen     * If this is called on more than one bitmap in sequence, the priority is
11528cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen     * given in LRU order (i.e. the last bitmap called will be given highest
11538cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen     * priority).
11548cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen     *
11558cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen     * For bitmaps with no associated caches, this call is effectively a no-op,
11568cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen     * and therefore is harmless.
11578cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen     */
11588cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen    public void prepareToDraw() {
11598cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen        nativePrepareToDraw(mNativeBitmap);
11608cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen    }
11618cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
1162e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy    private static class BitmapFinalizer {
1163e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy        private final int mNativeBitmap;
116402890fd0f98b3b8d98baf0bda1ea906afd723d8bRomain Guy
1165e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy        BitmapFinalizer(int nativeBitmap) {
1166e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy            mNativeBitmap = nativeBitmap;
116702890fd0f98b3b8d98baf0bda1ea906afd723d8bRomain Guy        }
116802890fd0f98b3b8d98baf0bda1ea906afd723d8bRomain Guy
116902890fd0f98b3b8d98baf0bda1ea906afd723d8bRomain Guy        @Override
1170e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy        public void finalize() {
1171366a84056c9a695ee6702d9d30bf9f3b521ba7ccRomain Guy            try {
1172366a84056c9a695ee6702d9d30bf9f3b521ba7ccRomain Guy                super.finalize();
1173366a84056c9a695ee6702d9d30bf9f3b521ba7ccRomain Guy            } catch (Throwable t) {
1174366a84056c9a695ee6702d9d30bf9f3b521ba7ccRomain Guy                // Ignore
1175366a84056c9a695ee6702d9d30bf9f3b521ba7ccRomain Guy            } finally {
1176366a84056c9a695ee6702d9d30bf9f3b521ba7ccRomain Guy                nativeDestructor(mNativeBitmap);
1177366a84056c9a695ee6702d9d30bf9f3b521ba7ccRomain Guy            }
11789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
11799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11808cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
11819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    //////////// native methods
11829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native Bitmap nativeCreate(int[] colors, int offset,
11849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                              int stride, int width, int height,
11859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                            int nativeConfig, boolean mutable);
11869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native Bitmap nativeCopy(int srcBitmap, int nativeConfig,
11879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                            boolean isMutable);
11889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native void nativeDestructor(int nativeBitmap);
11899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native void nativeRecycle(int nativeBitmap);
11909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native boolean nativeCompress(int nativeBitmap, int format,
11929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                            int quality, OutputStream stream,
11939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                            byte[] tempStorage);
11949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native void nativeErase(int nativeBitmap, int color);
11959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native int nativeWidth(int nativeBitmap);
11969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native int nativeHeight(int nativeBitmap);
11979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native int nativeRowBytes(int nativeBitmap);
11989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native int nativeConfig(int nativeBitmap);
11999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native boolean nativeHasAlpha(int nativeBitmap);
12008cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
12019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native int nativeGetPixel(int nativeBitmap, int x, int y);
12029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native void nativeGetPixels(int nativeBitmap, int[] pixels,
12039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                               int offset, int stride, int x,
12049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                               int y, int width, int height);
12058cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen
12069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native void nativeSetPixel(int nativeBitmap, int x, int y,
12079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                              int color);
12089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native void nativeSetPixels(int nativeBitmap, int[] colors,
12099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                               int offset, int stride, int x,
12109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                               int y, int width, int height);
12119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native void nativeCopyPixelsToBuffer(int nativeBitmap,
12129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                                        Buffer dst);
12139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native void nativeCopyPixelsFromBuffer(int nb, Buffer src);
12140bbae0836426ba2704e38e7f90a9d0ca502ab71dRomain Guy    private static native int nativeGenerationId(int nativeBitmap);
12159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native Bitmap nativeCreateFromParcel(Parcel p);
12179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // returns true on success
12189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native boolean nativeWriteToParcel(int nativeBitmap,
12199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                                      boolean isMutable,
1220de0dfb7b65a02d4dd74c271b558adee0973fc267Dianne Hackborn                                                      int density,
12219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                                      Parcel p);
12229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // returns a new bitmap built from the native bitmap's alpha, and the paint
12239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static native Bitmap nativeExtractAlpha(int nativeBitmap,
12249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                                    int nativePaint,
12259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                                    int[] offsetXY);
12269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12278cdcb12752b716d0407733fecefcf1d9e926310aWei-Ta Chen    private static native void nativePrepareToDraw(int nativeBitmap);
1228a78b0a2d9ebb38b86ed802b3d86de07d0b301262Mike Reed    private static native void nativeSetHasAlpha(int nBitmap, boolean hasAlpha);
122976d1e01d5e65c4631c827831e98ad4e300d99eabMike Reed    private static native boolean nativeSameAs(int nb0, int nb1);
1230d90f23e24a4d1768d5a7ed0e7072e67af6330a45Romain Guy
12319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /* package */ final int ni() {
12329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mNativeBitmap;
12339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1235