ImageView.java revision fd52b18d9bf3cd62c7a07058536e9f97db65beea
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.widget;
189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
190f8555b72861b7b9a233ab65b14055c98b145c91Bjorn Bringertimport android.content.ContentResolver;
209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.Context;
219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.res.Resources;
229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.res.TypedArray;
239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.Bitmap;
249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.Canvas;
259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.ColorFilter;
269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.Matrix;
279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.PorterDuff;
289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.PorterDuffColorFilter;
299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.RectF;
309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.drawable.BitmapDrawable;
319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.drawable.Drawable;
329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.net.Uri;
339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.util.AttributeSet;
349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.util.Log;
352b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkeyimport android.view.RemotableViewMethod;
369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.View;
37fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onoratoimport android.view.ViewDebug;
389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.widget.RemoteViews.RemoteView;
399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/**
429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Displays an arbitrary image, such as an icon.  The ImageView class
439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * can load images from various sources (such as resources or content
449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * providers), takes care of computing its measurement from the image so that
459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * it can be used in any layout manager, and provides various display options
469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * such as scaling and tinting.
479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#ImageView_adjustViewBounds
499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#ImageView_src
509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#ImageView_maxWidth
519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#ImageView_maxHeight
529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#ImageView_tint
539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#ImageView_scaleType
549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#ImageView_cropToPadding
559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project@RemoteView
579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpublic class ImageView extends View {
589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // settable by the client
599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private Uri mUri;
609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mResource = 0;
619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private Matrix mMatrix;
629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private ScaleType mScaleType;
639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private boolean mHaveFrame = false;
649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private boolean mAdjustViewBounds = false;
659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mMaxWidth = Integer.MAX_VALUE;
669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mMaxHeight = Integer.MAX_VALUE;
679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // these are applied to the drawable
699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private ColorFilter mColorFilter;
709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mAlpha = 255;
719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mViewAlphaScale = 256;
722b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey    private boolean mColorMod = false;
739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private Drawable mDrawable = null;
759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int[] mState = null;
769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private boolean mMergeState = false;
779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mLevel = 0;
789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mDrawableWidth;
799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mDrawableHeight;
809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private Matrix mDrawMatrix = null;
819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // Avoid allocations...
839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private RectF mTempSrc = new RectF();
849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private RectF mTempDst = new RectF();
859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private boolean mCropToPadding;
879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
88fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato    private int mBaseline = -1;
89fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato    private boolean mBaselineAlignBottom = false;
909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static final ScaleType[] sScaleTypeArray = {
929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ScaleType.MATRIX,
939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ScaleType.FIT_XY,
949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ScaleType.FIT_START,
959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ScaleType.FIT_CENTER,
969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ScaleType.FIT_END,
979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ScaleType.CENTER,
989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ScaleType.CENTER_CROP,
999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ScaleType.CENTER_INSIDE
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    };
1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public ImageView(Context context) {
1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super(context);
1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        initImageView();
1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public ImageView(Context context, AttributeSet attrs) {
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        this(context, attrs, 0);
1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public ImageView(Context context, AttributeSet attrs, int defStyle) {
1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super(context, attrs, defStyle);
1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        initImageView();
1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        TypedArray a = context.obtainStyledAttributes(attrs,
1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                com.android.internal.R.styleable.ImageView, defStyle, 0);
1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Drawable d = a.getDrawable(com.android.internal.R.styleable.ImageView_src);
1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (d != null) {
1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            setImageDrawable(d);
1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
123fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato        mBaselineAlignBottom = a.getBoolean(
1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                com.android.internal.R.styleable.ImageView_baselineAlignBottom, false);
125fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato
126fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato        mBaseline = a.getDimensionPixelSize(
127fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato                com.android.internal.R.styleable.ImageView_baseline, -1);
128fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato
1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setAdjustViewBounds(
1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            a.getBoolean(com.android.internal.R.styleable.ImageView_adjustViewBounds,
1319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            false));
1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setMaxWidth(a.getDimensionPixelSize(
1349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                com.android.internal.R.styleable.ImageView_maxWidth, Integer.MAX_VALUE));
1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setMaxHeight(a.getDimensionPixelSize(
1379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                com.android.internal.R.styleable.ImageView_maxHeight, Integer.MAX_VALUE));
1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int index = a.getInt(com.android.internal.R.styleable.ImageView_scaleType, -1);
1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (index >= 0) {
1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            setScaleType(sScaleTypeArray[index]);
1429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int tint = a.getInt(com.android.internal.R.styleable.ImageView_tint, 0);
1459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (tint != 0) {
1462b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey            setColorFilter(tint);
1479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mCropToPadding = a.getBoolean(
1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                com.android.internal.R.styleable.ImageView_cropToPadding, false);
1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        a.recycle();
1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        //need inflate syntax/reader for matrix
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void initImageView() {
1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mMatrix     = new Matrix();
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mScaleType  = ScaleType.FIT_CENTER;
1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected boolean verifyDrawable(Drawable dr) {
1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mDrawable == dr || super.verifyDrawable(dr);
1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
168e213677037f836529efcc0ac201fc61dd95481c5Dianne Hackborn    public void jumpDrawablesToCurrentState() {
169e213677037f836529efcc0ac201fc61dd95481c5Dianne Hackborn        super.jumpDrawablesToCurrentState();
170e213677037f836529efcc0ac201fc61dd95481c5Dianne Hackborn        if (mDrawable != null) mDrawable.jumpToCurrentState();
171e213677037f836529efcc0ac201fc61dd95481c5Dianne Hackborn    }
172e213677037f836529efcc0ac201fc61dd95481c5Dianne Hackborn
173e213677037f836529efcc0ac201fc61dd95481c5Dianne Hackborn    @Override
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void invalidateDrawable(Drawable dr) {
1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (dr == mDrawable) {
1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            /* we invalidate the whole view in this case because it's very
1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project             * hard to know where the drawable actually is. This is made
1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project             * complicated because of the offsets and transformations that
1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project             * can be applied. In theory we could get the drawable's bounds
1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project             * and run them through the transformation and offsets, but this
1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project             * is probably not worth the effort.
1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project             */
1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            invalidate();
1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            super.invalidateDrawable(dr);
1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
188fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato
1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected boolean onSetAlpha(int alpha) {
1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (getBackground() == null) {
1929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int scale = alpha + (alpha >> 7);
1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mViewAlphaScale != scale) {
1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mViewAlphaScale = scale;
19542c79880b0c19dfbcd8589d89d35fcedb1a7c9daRomain Guy                mColorMod = true;
1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                applyColorMod();
1979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
1989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return true;
1999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return false;
2019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Set this to true if you want the ImageView to adjust its bounds
2059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * to preserve the aspect ratio of its drawable.
2069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param adjustViewBounds Whether to adjust the bounds of this view
2079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * to presrve the original aspect ratio of the drawable
2089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#ImageView_adjustViewBounds
2109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @android.view.RemotableViewMethod
2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setAdjustViewBounds(boolean adjustViewBounds) {
2139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mAdjustViewBounds = adjustViewBounds;
2149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (adjustViewBounds) {
2159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            setScaleType(ScaleType.FIT_CENTER);
2169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * An optional argument to supply a maximum width for this view. Only valid if
2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * {@link #setAdjustViewBounds} has been set to true. To set an image to be a maximum of 100 x
2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * 100 while preserving the original aspect ratio, do the following: 1) set adjustViewBounds to
2239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * true 2) set maxWidth and maxHeight to 100 3) set the height and width layout params to
2249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * WRAP_CONTENT.
2259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>
2279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Note that this view could be still smaller than 100 x 100 using this approach if the original
2289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * image is small. To set an image to a fixed size, specify that size in the layout params and
2299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * then use {@link #setScaleType} to determine how to fit the image within the bounds.
2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * </p>
2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param maxWidth maximum width for this view
2339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#ImageView_maxWidth
2359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @android.view.RemotableViewMethod
2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setMaxWidth(int maxWidth) {
2389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mMaxWidth = maxWidth;
2399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * An optional argument to supply a maximum height for this view. Only valid if
2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * {@link #setAdjustViewBounds} has been set to true. To set an image to be a maximum of 100 x
2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * 100 while preserving the original aspect ratio, do the following: 1) set adjustViewBounds to
2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * true 2) set maxWidth and maxHeight to 100 3) set the height and width layout params to
2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * WRAP_CONTENT.
2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * <p>
2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Note that this view could be still smaller than 100 x 100 using this approach if the original
2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * image is small. To set an image to a fixed size, specify that size in the layout params and
2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * then use {@link #setScaleType} to determine how to fit the image within the bounds.
2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * </p>
2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param maxHeight maximum height for this view
2559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#ImageView_maxHeight
2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @android.view.RemotableViewMethod
2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setMaxHeight(int maxHeight) {
2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mMaxHeight = maxHeight;
2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /** Return the view's drawable, or null if no drawable has been
2649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        assigned.
2659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    */
2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public Drawable getDrawable() {
2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mDrawable;
2689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Sets a drawable as the content of this ImageView.
272c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     *
273c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     * <p class="note">This does Bitmap reading and decoding on the UI
274c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     * thread, which can cause a latency hiccup.  If that's a concern,
275c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     * consider using {@link #setImageDrawable} or
276c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     * {@link #setImageBitmap} and
277c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     * {@link android.graphics.BitmapFactory} instead.</p>
278c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     *
2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param resId the resource identifier of the the drawable
280c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     *
2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#ImageView_src
2829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @android.view.RemotableViewMethod
2849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setImageResource(int resId) {
2859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mUri != null || mResource != resId) {
2869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            updateDrawable(null);
2879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mResource = resId;
2889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mUri = null;
2899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            resolveUri();
2909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            requestLayout();
2919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            invalidate();
2929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Sets the content of this ImageView to the specified Uri.
297c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     *
298c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     * <p class="note">This does Bitmap reading and decoding on the UI
299c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     * thread, which can cause a latency hiccup.  If that's a concern,
300c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     * consider using {@link #setImageDrawable} or
301c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     * {@link #setImageBitmap} and
302c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     * {@link android.graphics.BitmapFactory} instead.</p>
303c6b0b7755c7932136c3bcdadfb56657f1f611465Brad Fitzpatrick     *
3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param uri The Uri of an image
3059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @android.view.RemotableViewMethod
3079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setImageURI(Uri uri) {
3089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mResource != 0 ||
3099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                (mUri != uri &&
3109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                 (uri == null || mUri == null || !uri.equals(mUri)))) {
3119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            updateDrawable(null);
3129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mResource = 0;
3139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mUri = uri;
3149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            resolveUri();
3159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            requestLayout();
3169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            invalidate();
3179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Sets a drawable as the content of this ImageView.
3239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param drawable The drawable to set
3259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setImageDrawable(Drawable drawable) {
3279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDrawable != drawable) {
3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mResource = 0;
3299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mUri = null;
3309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            updateDrawable(drawable);
3319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            requestLayout();
3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            invalidate();
3339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Sets a Bitmap as the content of this ImageView.
3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param bm The bitmap to set
3409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @android.view.RemotableViewMethod
3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setImageBitmap(Bitmap bm) {
3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // if this is used frequently, may handle bitmaps explicitly
3449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // to reduce the intermediate drawable object
34511ea33471e1a14a8594f0b2cd012d86340dd3bd8Dianne Hackborn        setImageDrawable(new BitmapDrawable(mContext.getResources(), bm));
3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setImageState(int[] state, boolean merge) {
3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mState = state;
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mMergeState = merge;
3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDrawable != null) {
3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            refreshDrawableState();
3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            resizeFromDrawable();
3549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
3589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setSelected(boolean selected) {
3599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super.setSelected(selected);
3609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        resizeFromDrawable();
3619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3634df2423a947bcd3f024cc3d3a1a315a8dc428598The Android Open Source Project    /**
3644df2423a947bcd3f024cc3d3a1a315a8dc428598The Android Open Source Project     * Sets the image level, when it is constructed from a
3654df2423a947bcd3f024cc3d3a1a315a8dc428598The Android Open Source Project     * {@link android.graphics.drawable.LevelListDrawable}.
3664df2423a947bcd3f024cc3d3a1a315a8dc428598The Android Open Source Project     *
3674df2423a947bcd3f024cc3d3a1a315a8dc428598The Android Open Source Project     * @param level The new level for the image.
3684df2423a947bcd3f024cc3d3a1a315a8dc428598The Android Open Source Project     */
3699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @android.view.RemotableViewMethod
3709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setImageLevel(int level) {
3719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mLevel = level;
3729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDrawable != null) {
3739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDrawable.setLevel(level);
3749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            resizeFromDrawable();
3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Options for scaling the bounds of an image to the bounds of this view.
3809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public enum ScaleType {
3829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
3839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Scale using the image matrix when drawing. The image matrix can be set using
3849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * {@link ImageView#setImageMatrix(Matrix)}. From XML, use this syntax:
3859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * <code>android:scaleType="matrix"</code>.
3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        MATRIX      (0),
3889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
3899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Scale the image using {@link Matrix.ScaleToFit#FILL}.
3909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * From XML, use this syntax: <code>android:scaleType="fitXY"</code>.
3919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
3929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        FIT_XY      (1),
3939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
3949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Scale the image using {@link Matrix.ScaleToFit#START}.
3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * From XML, use this syntax: <code>android:scaleType="fitStart"</code>.
3969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
3979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        FIT_START   (2),
3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
3999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Scale the image using {@link Matrix.ScaleToFit#CENTER}.
4009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * From XML, use this syntax:
4019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * <code>android:scaleType="fitCenter"</code>.
4029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
4039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        FIT_CENTER  (3),
4049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
4059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Scale the image using {@link Matrix.ScaleToFit#END}.
4069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * From XML, use this syntax: <code>android:scaleType="fitEnd"</code>.
4079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
4089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        FIT_END     (4),
4099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
4109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Center the image in the view, but perform no scaling.
4119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * From XML, use this syntax: <code>android:scaleType="center"</code>.
4129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
4139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        CENTER      (5),
4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
4159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Scale the image uniformly (maintain the image's aspect ratio) so
4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * that both dimensions (width and height) of the image will be equal
4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * to or larger than the corresponding dimension of the view
4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * (minus padding). The image is then centered in the view.
4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * From XML, use this syntax: <code>android:scaleType="centerCrop"</code>.
4209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
4219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        CENTER_CROP (6),
4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Scale the image uniformly (maintain the image's aspect ratio) so
4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * that both dimensions (width and height) of the image will be equal
4259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * to or less than the corresponding dimension of the view
4269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * (minus padding). The image is then centered in the view.
4279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * From XML, use this syntax: <code>android:scaleType="centerInside"</code>.
4289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        CENTER_INSIDE (7);
4309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ScaleType(int ni) {
4329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            nativeInt = ni;
4339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        final int nativeInt;
4359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
4389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Controls how the image should be resized or moved to match the size
4399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * of this ImageView.
4409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
4419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param scaleType The desired scaling mode.
4429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
4439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#ImageView_scaleType
4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
4459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setScaleType(ScaleType scaleType) {
4469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (scaleType == null) {
4479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            throw new NullPointerException();
4489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mScaleType != scaleType) {
4519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mScaleType = scaleType;
4529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            setWillNotCacheDrawing(mScaleType == ScaleType.CENTER);
4549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            requestLayout();
4569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            invalidate();
4579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
4619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Return the current scale type in use by this ImageView.
4629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
4639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see ImageView.ScaleType
4649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
4659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#ImageView_scaleType
4669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
4679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public ScaleType getScaleType() {
4689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mScaleType;
4699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /** Return the view's optional matrix. This is applied to the
4729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        view's drawable when it is drawn. If there is not matrix,
4739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        this method will return null.
4749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Do not change this matrix in place. If you want a different matrix
4759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        applied to the drawable, be sure to call setImageMatrix().
4769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    */
4779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public Matrix getImageMatrix() {
4789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mMatrix;
4799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setImageMatrix(Matrix matrix) {
4829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // collaps null and identity to just null
4839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (matrix != null && matrix.isIdentity()) {
4849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            matrix = null;
4859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // don't invalidate unless we're actually changing our matrix
4889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (matrix == null && !mMatrix.isIdentity() ||
4899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                matrix != null && !mMatrix.equals(matrix)) {
4909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mMatrix.set(matrix);
491f88872655e85570f3b9bfebe78647ed36916b3f4Chih-Chung Chang            configureBounds();
4929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            invalidate();
4939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void resolveUri() {
4979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDrawable != null) {
4989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return;
4999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Resources rsrc = getResources();
5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (rsrc == null) {
5039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return;
5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Drawable d = null;
5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mResource != 0) {
5099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            try {
5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                d = rsrc.getDrawable(mResource);
5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } catch (Exception e) {
5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                Log.w("ImageView", "Unable to find resource: " + mResource, e);
5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // Don't try again.
5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mUri = null;
5159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (mUri != null) {
5170f8555b72861b7b9a233ab65b14055c98b145c91Bjorn Bringert            String scheme = mUri.getScheme();
5189d8d9c22bb98e656cb2e42673c4ba801821c4e5dBjorn Bringert            if (ContentResolver.SCHEME_ANDROID_RESOURCE.equals(scheme)) {
5199d8d9c22bb98e656cb2e42673c4ba801821c4e5dBjorn Bringert                try {
5209d8d9c22bb98e656cb2e42673c4ba801821c4e5dBjorn Bringert                    // Load drawable through Resources, to get the source density information
5219d8d9c22bb98e656cb2e42673c4ba801821c4e5dBjorn Bringert                    ContentResolver.OpenResourceIdResult r =
5229d8d9c22bb98e656cb2e42673c4ba801821c4e5dBjorn Bringert                            mContext.getContentResolver().getResourceId(mUri);
5239d8d9c22bb98e656cb2e42673c4ba801821c4e5dBjorn Bringert                    d = r.r.getDrawable(r.id);
5249d8d9c22bb98e656cb2e42673c4ba801821c4e5dBjorn Bringert                } catch (Exception e) {
5259d8d9c22bb98e656cb2e42673c4ba801821c4e5dBjorn Bringert                    Log.w("ImageView", "Unable to open content: " + mUri, e);
5269d8d9c22bb98e656cb2e42673c4ba801821c4e5dBjorn Bringert                }
5279d8d9c22bb98e656cb2e42673c4ba801821c4e5dBjorn Bringert            } else if (ContentResolver.SCHEME_CONTENT.equals(scheme)
5280f8555b72861b7b9a233ab65b14055c98b145c91Bjorn Bringert                    || ContentResolver.SCHEME_FILE.equals(scheme)) {
5299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                try {
5309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    d = Drawable.createFromStream(
5319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        mContext.getContentResolver().openInputStream(mUri),
5329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        null);
5339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                } catch (Exception e) {
5349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    Log.w("ImageView", "Unable to open content: " + mUri, e);
5359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
5369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else {
5379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                d = Drawable.createFromPath(mUri.toString());
5389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (d == null) {
5419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                System.out.println("resolveUri failed on bad bitmap uri: "
5429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                   + mUri);
5439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // Don't try again.
5449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mUri = null;
5459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
5479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return;
5489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        updateDrawable(d);
5519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
5549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public int[] onCreateDrawableState(int extraSpace) {
5559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mState == null) {
5569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return super.onCreateDrawableState(extraSpace);
5579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (!mMergeState) {
5589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return mState;
5599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
5609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return mergeDrawableStates(
5619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    super.onCreateDrawableState(extraSpace + mState.length), mState);
5629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void updateDrawable(Drawable d) {
5669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDrawable != null) {
5679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDrawable.setCallback(null);
5689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            unscheduleDrawable(mDrawable);
5699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDrawable = d;
5719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (d != null) {
5729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            d.setCallback(this);
5739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (d.isStateful()) {
5749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                d.setState(getDrawableState());
5759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            d.setLevel(mLevel);
5779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDrawableWidth = d.getIntrinsicWidth();
5789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDrawableHeight = d.getIntrinsicHeight();
5799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            applyColorMod();
5809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            configureBounds();
5819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void resizeFromDrawable() {
5859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Drawable d = mDrawable;
5869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (d != null) {
5879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int w = d.getIntrinsicWidth();
5889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (w < 0) w = mDrawableWidth;
5899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int h = d.getIntrinsicHeight();
5909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (h < 0) h = mDrawableHeight;
5919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (w != mDrawableWidth || h != mDrawableHeight) {
5929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDrawableWidth = w;
5939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDrawableHeight = h;
5949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                requestLayout();
5959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static final Matrix.ScaleToFit[] sS2FArray = {
6009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Matrix.ScaleToFit.FILL,
6019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Matrix.ScaleToFit.START,
6029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Matrix.ScaleToFit.CENTER,
6039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Matrix.ScaleToFit.END
6049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    };
6059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static Matrix.ScaleToFit scaleTypeToScaleToFit(ScaleType st)  {
6079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // ScaleToFit enum to their corresponding Matrix.ScaleToFit values
6089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return sS2FArray[st.nativeInt - 1];
6099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
6129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
6139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        resolveUri();
6149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int w;
6159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int h;
6169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Desired aspect ratio of the view's contents (not including padding)
6189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        float desiredAspect = 0.0f;
6199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // We are allowed to change the view's width
6219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        boolean resizeWidth = false;
6229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // We are allowed to change the view's height
6249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        boolean resizeHeight = false;
6259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDrawable == null) {
6279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // If no drawable, its intrinsic size is 0.
6289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDrawableWidth = -1;
6299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDrawableHeight = -1;
6309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            w = h = 0;
6319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
6329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            w = mDrawableWidth;
6339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            h = mDrawableHeight;
6349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (w <= 0) w = 1;
6359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (h <= 0) h = 1;
6369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // We are supposed to adjust view bounds to match the aspect
6389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // ratio of our drawable. See if that is possible.
6399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mAdjustViewBounds) {
6409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                int widthSpecMode = MeasureSpec.getMode(widthMeasureSpec);
6429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                int heightSpecMode = MeasureSpec.getMode(heightMeasureSpec);
6439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                resizeWidth = widthSpecMode != MeasureSpec.EXACTLY;
6459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                resizeHeight = heightSpecMode != MeasureSpec.EXACTLY;
6469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                desiredAspect = (float)w/(float)h;
6489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
6499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int pleft = mPaddingLeft;
6529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int pright = mPaddingRight;
6539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int ptop = mPaddingTop;
6549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int pbottom = mPaddingBottom;
6559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int widthSize;
6579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int heightSize;
6589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (resizeWidth || resizeHeight) {
6609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            /* If we get here, it means we want to resize to match the
6619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                drawables aspect ratio, and we have the freedom to change at
6629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                least one dimension.
6639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            */
6649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Get the max possible width given our constraints
6669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            widthSize = resolveAdjustedSize(w + pleft + pright,
6679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                                 mMaxWidth, widthMeasureSpec);
6689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Get the max possible height given our constraints
6709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            heightSize = resolveAdjustedSize(h + ptop + pbottom,
6719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                                mMaxHeight, heightMeasureSpec);
6729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (desiredAspect != 0.0f) {
6749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // See what our actual aspect ratio is
6759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                float actualAspect = (float)(widthSize - pleft - pright) /
6769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                        (heightSize - ptop - pbottom);
6779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (Math.abs(actualAspect - desiredAspect) > 0.0000001) {
6799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    boolean done = false;
6819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // Try adjusting width to be proportional to height
6839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if (resizeWidth) {
6849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        int newWidth = (int)(desiredAspect *
6859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                            (heightSize - ptop - pbottom))
6869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                            + pleft + pright;
6879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        if (newWidth <= widthSize) {
6889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            widthSize = newWidth;
6899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            done = true;
6909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        }
6919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
6929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // Try adjusting height to be proportional to width
6949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if (!done && resizeHeight) {
6959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        int newHeight = (int)((widthSize - pleft - pright)
6969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                            / desiredAspect) + ptop + pbottom;
6979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        if (newHeight <= heightSize) {
6989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            heightSize = newHeight;
6999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        }
7009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
7019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
7029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
7039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
7049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            /* We are either don't want to preserve the drawables aspect ratio,
7059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project               or we are not allowed to change view dimensions. Just measure in
7069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project               the normal way.
7079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            */
7089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            w += pleft + pright;
7099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            h += ptop + pbottom;
7109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            w = Math.max(w, getSuggestedMinimumWidth());
7129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            h = Math.max(h, getSuggestedMinimumHeight());
7139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            widthSize = resolveSize(w, widthMeasureSpec);
7159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            heightSize = resolveSize(h, heightMeasureSpec);
7169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setMeasuredDimension(widthSize, heightSize);
7199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int resolveAdjustedSize(int desiredSize, int maxSize,
7229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                   int measureSpec) {
7239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int result = desiredSize;
7249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int specMode = MeasureSpec.getMode(measureSpec);
7259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int specSize =  MeasureSpec.getSize(measureSpec);
7269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (specMode) {
7279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case MeasureSpec.UNSPECIFIED:
7289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                /* Parent says we can be as big as we want. Just don't be larger
7299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                   than max size imposed on ourselves.
7309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                */
7319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                result = Math.min(desiredSize, maxSize);
7329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                break;
7339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case MeasureSpec.AT_MOST:
7349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // Parent says we can be as big as we want, up to specSize.
7359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // Don't be larger than specSize, and don't be larger than
7369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // the max size imposed on ourselves.
7379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                result = Math.min(Math.min(desiredSize, specSize), maxSize);
7389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                break;
7399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case MeasureSpec.EXACTLY:
7409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // No choice. Do what we are told.
7419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                result = specSize;
7429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                break;
7439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return result;
7459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
7489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected boolean setFrame(int l, int t, int r, int b) {
7499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        boolean changed = super.setFrame(l, t, r, b);
7509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mHaveFrame = true;
7519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        configureBounds();
7529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return changed;
7539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void configureBounds() {
7569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDrawable == null || !mHaveFrame) {
7579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return;
7589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int dwidth = mDrawableWidth;
7619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int dheight = mDrawableHeight;
7629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int vwidth = getWidth() - mPaddingLeft - mPaddingRight;
7649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int vheight = getHeight() - mPaddingTop - mPaddingBottom;
7659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        boolean fits = (dwidth < 0 || vwidth == dwidth) &&
7679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                       (dheight < 0 || vheight == dheight);
7689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (dwidth <= 0 || dheight <= 0 || ScaleType.FIT_XY == mScaleType) {
7709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            /* If the drawable has no intrinsic size, or we're told to
7719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                scaletofit, then we just fill our entire view.
7729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            */
7739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDrawable.setBounds(0, 0, vwidth, vheight);
7749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDrawMatrix = null;
7759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
7769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // We need to do the scaling ourself, so have the drawable
7779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // use its native size.
7789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDrawable.setBounds(0, 0, dwidth, dheight);
7799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (ScaleType.MATRIX == mScaleType) {
7819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // Use the specified matrix as-is.
7829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (mMatrix.isIdentity()) {
7839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mDrawMatrix = null;
7849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                } else {
7859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    mDrawMatrix = mMatrix;
7869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
7879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else if (fits) {
7889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // The bitmap fits exactly, no transform needed.
7899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDrawMatrix = null;
7909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else if (ScaleType.CENTER == mScaleType) {
7919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // Center bitmap in view, no scaling.
7929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDrawMatrix = mMatrix;
793b4938201bad1050a5e3821d60ea43e71861cc733Romain Guy                mDrawMatrix.setTranslate((int) ((vwidth - dwidth) * 0.5f + 0.5f),
794b4938201bad1050a5e3821d60ea43e71861cc733Romain Guy                                         (int) ((vheight - dheight) * 0.5f + 0.5f));
7959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else if (ScaleType.CENTER_CROP == mScaleType) {
7969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDrawMatrix = mMatrix;
7979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                float scale;
7999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                float dx = 0, dy = 0;
8009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (dwidth * vheight > vwidth * dheight) {
8029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    scale = (float) vheight / (float) dheight;
8039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    dx = (vwidth - dwidth * scale) * 0.5f;
8049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                } else {
8059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    scale = (float) vwidth / (float) dwidth;
8069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    dy = (vheight - dheight * scale) * 0.5f;
8079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
8089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDrawMatrix.setScale(scale, scale);
810b4938201bad1050a5e3821d60ea43e71861cc733Romain Guy                mDrawMatrix.postTranslate((int) (dx + 0.5f), (int) (dy + 0.5f));
8119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else if (ScaleType.CENTER_INSIDE == mScaleType) {
8129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDrawMatrix = mMatrix;
8139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                float scale;
8149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                float dx;
8159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                float dy;
8169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (dwidth <= vwidth && dheight <= vheight) {
8189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    scale = 1.0f;
8199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                } else {
8209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    scale = Math.min((float) vwidth / (float) dwidth,
8219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                            (float) vheight / (float) dheight);
8229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
8239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
824b4938201bad1050a5e3821d60ea43e71861cc733Romain Guy                dx = (int) ((vwidth - dwidth * scale) * 0.5f + 0.5f);
825b4938201bad1050a5e3821d60ea43e71861cc733Romain Guy                dy = (int) ((vheight - dheight * scale) * 0.5f + 0.5f);
8269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDrawMatrix.setScale(scale, scale);
8289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDrawMatrix.postTranslate(dx, dy);
8299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else {
8309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // Generate the required transform.
8319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mTempSrc.set(0, 0, dwidth, dheight);
8329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mTempDst.set(0, 0, vwidth, vheight);
8339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDrawMatrix = mMatrix;
8359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDrawMatrix.setRectToRect(mTempSrc, mTempDst,
8369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                          scaleTypeToScaleToFit(mScaleType));
8379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
8389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
8429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected void drawableStateChanged() {
8439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super.drawableStateChanged();
8449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Drawable d = mDrawable;
8459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (d != null && d.isStateful()) {
8469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            d.setState(getDrawableState());
8479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
8519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected void onDraw(Canvas canvas) {
8529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super.onDraw(canvas);
8539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDrawable == null) {
8559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return; // couldn't resolve the URI
8569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDrawableWidth == 0 || mDrawableHeight == 0) {
8599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return;     // nothing to draw (empty bounds)
8609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mDrawMatrix == null && mPaddingTop == 0 && mPaddingLeft == 0) {
8639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDrawable.draw(canvas);
8649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
8659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int saveCount = canvas.getSaveCount();
8669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            canvas.save();
8679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mCropToPadding) {
8699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                final int scrollX = mScrollX;
8709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                final int scrollY = mScrollY;
8719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                canvas.clipRect(scrollX + mPaddingLeft, scrollY + mPaddingTop,
8729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        scrollX + mRight - mLeft - mPaddingRight,
8739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        scrollY + mBottom - mTop - mPaddingBottom);
8749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
8759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            canvas.translate(mPaddingLeft, mPaddingTop);
8779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (mDrawMatrix != null) {
8799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                canvas.concat(mDrawMatrix);
8809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
8819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDrawable.draw(canvas);
8829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            canvas.restoreToCount(saveCount);
8839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
886fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato    /**
887fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * <p>Return the offset of the widget's text baseline from the widget's top
888fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * boundary. </p>
889fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     *
890fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * @return the offset of the baseline within the widget's bounds or -1
891fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     *         if baseline alignment is not supported.
892fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     */
8939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
894fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato    @ViewDebug.ExportedProperty(category = "layout")
8959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public int getBaseline() {
896fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato        if (mBaselineAlignBottom) {
897fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato            return getMeasuredHeight();
898fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato        } else {
899fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato            return mBaseline;
900fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato        }
901fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato    }
902fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato
903fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato    /**
904fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * <p>Set the offset of the widget's text baseline from the widget's top
905fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * boundary.  This value is overridden by the {@link #setBaselineAlignBottom}
906fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * property.</p>
907fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     *
908fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * @param baseline The baseline to use, or -1 if none is to be provided.
909fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     *
910fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * @see #setBaseline
911fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * @attr ref android.R.styleable#ImageView_baseline
912fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     */
913fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato    public void setBaseline(int baseline) {
914fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato        if (mBaseline != baseline) {
915fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato            mBaseline = baseline;
916fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato            requestLayout();
917fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato        }
918fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato    }
919fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato
920fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato    /**
921fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * Set whether to set the baseline of this view to the bottom of the view.
922fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * Setting this value overrides any calls to setBaseline.
923fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     *
924fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * @param aligned If true, the image view will be baseline aligned with
925fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     *      based on its bottom edge.
926fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     *
927fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * @attr ref android.R.styleable#ImageView_baselineAlignBottom
928fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     */
929fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato    public void setBaselineAlignBottom(boolean aligned) {
930fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato        if (mBaselineAlignBottom != aligned) {
931fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato            mBaselineAlignBottom = aligned;
932fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato            requestLayout();
933fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato        }
934fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato    }
935fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato
936fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato    /**
937fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * Return whether this view's baseline will be considered the bottom of the view.
938fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     *
939fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     * @see #setBaselineAlignBottom(boolean)
940fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato     */
941fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato    public boolean getBaselineAlignBottom() {
942fd52b18d9bf3cd62c7a07058536e9f97db65beeaJoe Onorato        return mBaselineAlignBottom;
9439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
94475986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov
9459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
9469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Set a tinting option for the image.
9479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
9489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param color Color tint to apply.
9499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param mode How to apply the color.  The standard mode is
9509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * {@link PorterDuff.Mode#SRC_ATOP}
9519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
9529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#ImageView_tint
9539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
9549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public final void setColorFilter(int color, PorterDuff.Mode mode) {
9559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setColorFilter(new PorterDuffColorFilter(color, mode));
9569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9582b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey    /**
9592b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey     * Set a tinting option for the image. Assumes
9602b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey     * {@link PorterDuff.Mode#SRC_ATOP} blending mode.
9612b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey     *
9622b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey     * @param color Color tint to apply.
9632b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey     * @attr ref android.R.styleable#ImageView_tint
9642b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey     */
9652b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey    @RemotableViewMethod
9662b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey    public final void setColorFilter(int color) {
9672b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey        setColorFilter(color, PorterDuff.Mode.SRC_ATOP);
9682b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey    }
9692b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey
9709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public final void clearColorFilter() {
9719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setColorFilter(null);
9729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
9759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Apply an arbitrary colorfilter to the image.
9769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
9779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param cf the colorfilter to apply (may be null)
9789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
9799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setColorFilter(ColorFilter cf) {
9809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mColorFilter != cf) {
9819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mColorFilter = cf;
9822b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey            mColorMod = true;
9839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            applyColorMod();
9849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            invalidate();
9859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
98775986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov
9882b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey    @RemotableViewMethod
9899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setAlpha(int alpha) {
9909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        alpha &= 0xFF;          // keep it legal
9919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mAlpha != alpha) {
9929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mAlpha = alpha;
9932b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey            mColorMod = true;
9949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            applyColorMod();
9959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            invalidate();
9969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void applyColorMod() {
10002b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey        // Only mutate and apply when modifications have occurred. This should
10012b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey        // not reset the mColorMod flag, since these filters need to be
10022b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey        // re-applied if the Drawable is changed.
10032b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey        if (mDrawable != null && mColorMod) {
10042b95c2413838c2e2b127ebab8fb4fead7d52e460Jeff Sharkey            mDrawable = mDrawable.mutate();
10059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDrawable.setColorFilter(mColorFilter);
10069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDrawable.setAlpha(mAlpha * mViewAlphaScale >> 8);
10079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1010