| package android.support.v7.internal.widget; |
| |
| /* |
| * Copyright (C) 2013 Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| |
| import android.content.Context; |
| import android.content.res.TypedArray; |
| import android.graphics.Bitmap; |
| import android.graphics.BitmapShader; |
| import android.graphics.Canvas; |
| import android.graphics.Rect; |
| import android.graphics.Shader; |
| import android.graphics.drawable.Animatable; |
| import android.graphics.drawable.AnimationDrawable; |
| import android.graphics.drawable.BitmapDrawable; |
| import android.graphics.drawable.ClipDrawable; |
| import android.graphics.drawable.Drawable; |
| import android.graphics.drawable.LayerDrawable; |
| import android.graphics.drawable.ShapeDrawable; |
| import android.graphics.drawable.shapes.RoundRectShape; |
| import android.graphics.drawable.shapes.Shape; |
| import android.os.Build; |
| import android.os.Parcel; |
| import android.os.Parcelable; |
| import android.os.SystemClock; |
| import android.util.AttributeSet; |
| import android.view.Gravity; |
| import android.view.View; |
| import android.view.animation.AlphaAnimation; |
| import android.view.animation.Animation; |
| import android.view.animation.AnimationUtils; |
| import android.view.animation.Interpolator; |
| import android.view.animation.LinearInterpolator; |
| import android.view.animation.Transformation; |
| |
| /** |
| * @hide |
| */ |
| public class ProgressBarCompat extends View { |
| |
| private static final int MAX_LEVEL = 10000; |
| private static final int ANIMATION_RESOLUTION = 200; |
| |
| /** |
| * android.R.styleable.ProgressBar is internalised, so we need to create it ourselves. |
| */ |
| private static final int[] android_R_styleable_ProgressBar = new int[]{ |
| android.R.attr.max, |
| android.R.attr.progress, |
| android.R.attr.secondaryProgress, |
| android.R.attr.indeterminate, |
| android.R.attr.indeterminateOnly, |
| android.R.attr.indeterminateDrawable, |
| android.R.attr.progressDrawable, |
| android.R.attr.indeterminateDuration, |
| android.R.attr.indeterminateBehavior, |
| android.R.attr.minWidth, |
| android.R.attr.maxWidth, |
| android.R.attr.minHeight, |
| android.R.attr.maxHeight, |
| android.R.attr.interpolator, |
| }; |
| |
| int mMinWidth; |
| int mMaxWidth; |
| int mMinHeight; |
| int mMaxHeight; |
| |
| private int mProgress; |
| private int mSecondaryProgress; |
| private int mMax; |
| |
| private int mBehavior; |
| private int mDuration; |
| private boolean mIndeterminate; |
| private boolean mOnlyIndeterminate; |
| private Transformation mTransformation; |
| private AlphaAnimation mAnimation; |
| private Drawable mIndeterminateDrawable; |
| private Drawable mProgressDrawable; |
| private Drawable mCurrentDrawable; |
| Bitmap mSampleTile; |
| private boolean mNoInvalidate; |
| private Interpolator mInterpolator; |
| private RefreshProgressRunnable mRefreshProgressRunnable; |
| private long mUiThreadId; |
| private boolean mShouldStartAnimationDrawable; |
| private long mLastDrawTime; |
| |
| private boolean mInDrawing; |
| |
| /** |
| * @hide |
| */ |
| public ProgressBarCompat(Context context, AttributeSet attrs, int defStyle, int styleRes) { |
| super(context, attrs, defStyle); |
| mUiThreadId = Thread.currentThread().getId(); |
| initProgressBar(); |
| |
| TypedArray a = context.obtainStyledAttributes(attrs, android_R_styleable_ProgressBar, |
| defStyle, styleRes); |
| |
| mNoInvalidate = true; |
| |
| setMax(a.getInt(0, mMax)); |
| setProgress(a.getInt(1, mProgress)); |
| setSecondaryProgress(a.getInt(2, mSecondaryProgress)); |
| |
| final boolean indeterminate = a.getBoolean(3, mIndeterminate); |
| mOnlyIndeterminate = a.getBoolean(4, mOnlyIndeterminate); |
| |
| Drawable drawable = a.getDrawable(5); |
| if (drawable != null) { |
| drawable = tileifyIndeterminate(drawable); |
| setIndeterminateDrawable(drawable); |
| } |
| |
| drawable = a.getDrawable(6); |
| if (drawable != null) { |
| drawable = tileify(drawable, false); |
| // Calling this method can set mMaxHeight, make sure the corresponding |
| // XML attribute for mMaxHeight is read after calling this method |
| setProgressDrawable(drawable); |
| } |
| |
| mDuration = a.getInt(7, mDuration); |
| mBehavior = a.getInt(8, mBehavior); |
| mMinWidth = a.getDimensionPixelSize(9, mMinWidth); |
| mMaxWidth = a.getDimensionPixelSize(10, mMaxWidth); |
| mMinHeight = a.getDimensionPixelSize(11, mMinHeight); |
| mMaxHeight = a.getDimensionPixelSize(12, mMaxHeight); |
| |
| final int resID = a.getResourceId(13, android.R.anim.linear_interpolator); |
| if (resID > 0) { |
| setInterpolator(context, resID); |
| } |
| |
| a.recycle(); |
| |
| mNoInvalidate = false; |
| setIndeterminate(mOnlyIndeterminate || indeterminate); |
| } |
| |
| /** |
| * Converts a drawable to a tiled version of itself. It will recursively |
| * traverse layer and state list drawables. |
| */ |
| private Drawable tileify(Drawable drawable, boolean clip) { |
| |
| if (drawable instanceof LayerDrawable) { |
| LayerDrawable background = (LayerDrawable) drawable; |
| final int N = background.getNumberOfLayers(); |
| Drawable[] outDrawables = new Drawable[N]; |
| |
| for (int i = 0; i < N; i++) { |
| int id = background.getId(i); |
| outDrawables[i] = tileify(background.getDrawable(i), |
| (id == android.R.id.progress || id == android.R.id.secondaryProgress)); |
| } |
| |
| LayerDrawable newBg = new LayerDrawable(outDrawables); |
| |
| for (int i = 0; i < N; i++) { |
| newBg.setId(i, background.getId(i)); |
| } |
| |
| return newBg; |
| |
| } else if (drawable instanceof BitmapDrawable) { |
| final Bitmap tileBitmap = ((BitmapDrawable) drawable).getBitmap(); |
| if (mSampleTile == null) { |
| mSampleTile = tileBitmap; |
| } |
| |
| final ShapeDrawable shapeDrawable = new ShapeDrawable(getDrawableShape()); |
| |
| final BitmapShader bitmapShader = new BitmapShader(tileBitmap, |
| Shader.TileMode.REPEAT, Shader.TileMode.CLAMP); |
| shapeDrawable.getPaint().setShader(bitmapShader); |
| |
| return (clip) ? new ClipDrawable(shapeDrawable, Gravity.LEFT, |
| ClipDrawable.HORIZONTAL) : shapeDrawable; |
| } |
| |
| return drawable; |
| } |
| |
| Shape getDrawableShape() { |
| final float[] roundedCorners = new float[] { 5, 5, 5, 5, 5, 5, 5, 5 }; |
| return new RoundRectShape(roundedCorners, null, null); |
| } |
| |
| /** |
| * Convert a AnimationDrawable for use as a barberpole animation. |
| * Each frame of the animation is wrapped in a ClipDrawable and |
| * given a tiling BitmapShader. |
| */ |
| private Drawable tileifyIndeterminate(Drawable drawable) { |
| if (drawable instanceof AnimationDrawable) { |
| AnimationDrawable background = (AnimationDrawable) drawable; |
| final int N = background.getNumberOfFrames(); |
| AnimationDrawable newBg = new AnimationDrawable(); |
| newBg.setOneShot(background.isOneShot()); |
| |
| for (int i = 0; i < N; i++) { |
| Drawable frame = tileify(background.getFrame(i), true); |
| frame.setLevel(10000); |
| newBg.addFrame(frame, background.getDuration(i)); |
| } |
| newBg.setLevel(10000); |
| drawable = newBg; |
| } |
| return drawable; |
| } |
| |
| /** |
| * <p> |
| * Initialize the progress bar's default values: |
| * </p> |
| * <ul> |
| * <li>progress = 0</li> |
| * <li>max = 100</li> |
| * <li>animation duration = 4000 ms</li> |
| * <li>indeterminate = false</li> |
| * <li>behavior = repeat</li> |
| * </ul> |
| */ |
| private void initProgressBar() { |
| mMax = 100; |
| mProgress = 0; |
| mSecondaryProgress = 0; |
| mIndeterminate = false; |
| mOnlyIndeterminate = false; |
| mDuration = 4000; |
| mBehavior = AlphaAnimation.RESTART; |
| mMinWidth = 24; |
| mMaxWidth = 48; |
| mMinHeight = 24; |
| mMaxHeight = 48; |
| } |
| |
| /** |
| * <p>Indicate whether this progress bar is in indeterminate mode.</p> |
| * |
| * @return true if the progress bar is in indeterminate mode |
| */ |
| public synchronized boolean isIndeterminate() { |
| return mIndeterminate; |
| } |
| |
| /** |
| * <p>Change the indeterminate mode for this progress bar. In indeterminate |
| * mode, the progress is ignored and the progress bar shows an infinite |
| * animation instead.</p> |
| * |
| * If this progress bar's style only supports indeterminate mode (such as the circular |
| * progress bars), then this will be ignored. |
| * |
| * @param indeterminate true to enable the indeterminate mode |
| */ |
| public synchronized void setIndeterminate(boolean indeterminate) { |
| if ((!mOnlyIndeterminate || !mIndeterminate) && indeterminate != mIndeterminate) { |
| mIndeterminate = indeterminate; |
| |
| if (indeterminate) { |
| // swap between indeterminate and regular backgrounds |
| mCurrentDrawable = mIndeterminateDrawable; |
| startAnimation(); |
| } else { |
| mCurrentDrawable = mProgressDrawable; |
| stopAnimation(); |
| } |
| } |
| } |
| |
| /** |
| * <p>Get the drawable used to draw the progress bar in |
| * indeterminate mode.</p> |
| * |
| * @return a {@link android.graphics.drawable.Drawable} instance |
| * |
| * @see #setIndeterminateDrawable(android.graphics.drawable.Drawable) |
| * @see #setIndeterminate(boolean) |
| */ |
| public Drawable getIndeterminateDrawable() { |
| return mIndeterminateDrawable; |
| } |
| |
| /** |
| * <p>Define the drawable used to draw the progress bar in |
| * indeterminate mode.</p> |
| * |
| * @param d the new drawable |
| * |
| * @see #getIndeterminateDrawable() |
| * @see #setIndeterminate(boolean) |
| */ |
| public void setIndeterminateDrawable(Drawable d) { |
| if (d != null) { |
| d.setCallback(this); |
| } |
| mIndeterminateDrawable = d; |
| if (mIndeterminate) { |
| mCurrentDrawable = d; |
| postInvalidate(); |
| } |
| } |
| |
| /** |
| * <p>Get the drawable used to draw the progress bar in |
| * progress mode.</p> |
| * |
| * @return a {@link android.graphics.drawable.Drawable} instance |
| * |
| * @see #setProgressDrawable(android.graphics.drawable.Drawable) |
| * @see #setIndeterminate(boolean) |
| */ |
| public Drawable getProgressDrawable() { |
| return mProgressDrawable; |
| } |
| |
| /** |
| * <p>Define the drawable used to draw the progress bar in |
| * progress mode.</p> |
| * |
| * @param d the new drawable |
| * |
| * @see #getProgressDrawable() |
| * @see #setIndeterminate(boolean) |
| */ |
| public void setProgressDrawable(Drawable d) { |
| boolean needUpdate; |
| if (mProgressDrawable != null && d != mProgressDrawable) { |
| mProgressDrawable.setCallback(null); |
| needUpdate = true; |
| } else { |
| needUpdate = false; |
| } |
| |
| if (d != null) { |
| d.setCallback(this); |
| |
| // Make sure the android_R_styleable_ProgressBar is always tall enough |
| int drawableHeight = d.getMinimumHeight(); |
| if (mMaxHeight < drawableHeight) { |
| mMaxHeight = drawableHeight; |
| requestLayout(); |
| } |
| } |
| mProgressDrawable = d; |
| if (!mIndeterminate) { |
| mCurrentDrawable = d; |
| postInvalidate(); |
| } |
| |
| if (needUpdate) { |
| updateDrawableBounds(getWidth(), getHeight()); |
| updateDrawableState(); |
| doRefreshProgress(android.R.id.progress, mProgress, false, false); |
| doRefreshProgress(android.R.id.secondaryProgress, mSecondaryProgress, false, false); |
| } |
| } |
| |
| @Override |
| protected boolean verifyDrawable(Drawable who) { |
| return who == mProgressDrawable || who == mIndeterminateDrawable |
| || super.verifyDrawable(who); |
| } |
| |
| @Override |
| public void postInvalidate() { |
| if (!mNoInvalidate) { |
| super.postInvalidate(); |
| } |
| } |
| |
| private class RefreshProgressRunnable implements Runnable { |
| |
| private int mId; |
| private int mProgress; |
| private boolean mFromUser; |
| |
| RefreshProgressRunnable(int id, int progress, boolean fromUser) { |
| mId = id; |
| mProgress = progress; |
| mFromUser = fromUser; |
| } |
| |
| public void run() { |
| doRefreshProgress(mId, mProgress, mFromUser, true); |
| // Put ourselves back in the cache when we are done |
| mRefreshProgressRunnable = this; |
| } |
| |
| public void setup(int id, int progress, boolean fromUser) { |
| mId = id; |
| mProgress = progress; |
| mFromUser = fromUser; |
| } |
| |
| } |
| |
| private synchronized void doRefreshProgress(int id, int progress, boolean fromUser, |
| boolean callBackToApp) { |
| float scale = mMax > 0 ? (float) progress / (float) mMax : 0; |
| final Drawable d = mCurrentDrawable; |
| if (d != null) { |
| Drawable progressDrawable = null; |
| |
| if (d instanceof LayerDrawable) { |
| progressDrawable = ((LayerDrawable) d).findDrawableByLayerId(id); |
| } |
| |
| final int level = (int) (scale * MAX_LEVEL); |
| (progressDrawable != null ? progressDrawable : d).setLevel(level); |
| } else { |
| invalidate(); |
| } |
| } |
| |
| private synchronized void refreshProgress(int id, int progress, boolean fromUser) { |
| if (mUiThreadId == Thread.currentThread().getId()) { |
| doRefreshProgress(id, progress, fromUser, true); |
| } else { |
| RefreshProgressRunnable r; |
| if (mRefreshProgressRunnable != null) { |
| // Use cached RefreshProgressRunnable if available |
| r = mRefreshProgressRunnable; |
| // Uncache it |
| mRefreshProgressRunnable = null; |
| r.setup(id, progress, fromUser); |
| } else { |
| // Make a new one |
| r = new RefreshProgressRunnable(id, progress, fromUser); |
| } |
| post(r); |
| } |
| } |
| |
| /** |
| * <p>Set the current progress to the specified value. Does not do anything |
| * if the progress bar is in indeterminate mode.</p> |
| * |
| * @param progress the new progress, between 0 and {@link #getMax()} |
| * |
| * @see #setIndeterminate(boolean) |
| * @see #isIndeterminate() |
| * @see #getProgress() |
| * @see #incrementProgressBy(int) |
| */ |
| public synchronized void setProgress(int progress) { |
| setProgress(progress, false); |
| } |
| |
| synchronized void setProgress(int progress, boolean fromUser) { |
| if (mIndeterminate) { |
| return; |
| } |
| |
| if (progress < 0) { |
| progress = 0; |
| } |
| |
| if (progress > mMax) { |
| progress = mMax; |
| } |
| |
| if (progress != mProgress) { |
| mProgress = progress; |
| refreshProgress(android.R.id.progress, mProgress, fromUser); |
| } |
| } |
| |
| /** |
| * <p> |
| * Set the current secondary progress to the specified value. Does not do |
| * anything if the progress bar is in indeterminate mode. |
| * </p> |
| * |
| * @param secondaryProgress the new secondary progress, between 0 and {@link #getMax()} |
| * @see #setIndeterminate(boolean) |
| * @see #isIndeterminate() |
| * @see #getSecondaryProgress() |
| * @see #incrementSecondaryProgressBy(int) |
| */ |
| public synchronized void setSecondaryProgress(int secondaryProgress) { |
| if (mIndeterminate) { |
| return; |
| } |
| |
| if (secondaryProgress < 0) { |
| secondaryProgress = 0; |
| } |
| |
| if (secondaryProgress > mMax) { |
| secondaryProgress = mMax; |
| } |
| |
| if (secondaryProgress != mSecondaryProgress) { |
| mSecondaryProgress = secondaryProgress; |
| refreshProgress(android.R.id.secondaryProgress, mSecondaryProgress, false); |
| } |
| } |
| |
| /** |
| * <p>Get the progress bar's current level of progress. Return 0 when the |
| * progress bar is in indeterminate mode.</p> |
| * |
| * @return the current progress, between 0 and {@link #getMax()} |
| * |
| * @see #setIndeterminate(boolean) |
| * @see #isIndeterminate() |
| * @see #setProgress(int) |
| * @see #setMax(int) |
| * @see #getMax() |
| */ |
| public synchronized int getProgress() { |
| return mIndeterminate ? 0 : mProgress; |
| } |
| |
| /** |
| * <p>Get the progress bar's current level of secondary progress. Return 0 when the |
| * progress bar is in indeterminate mode.</p> |
| * |
| * @return the current secondary progress, between 0 and {@link #getMax()} |
| * |
| * @see #setIndeterminate(boolean) |
| * @see #isIndeterminate() |
| * @see #setSecondaryProgress(int) |
| * @see #setMax(int) |
| * @see #getMax() |
| */ |
| public synchronized int getSecondaryProgress() { |
| return mIndeterminate ? 0 : mSecondaryProgress; |
| } |
| |
| /** |
| * <p>Return the upper limit of this progress bar's range.</p> |
| * |
| * @return a positive integer |
| * |
| * @see #setMax(int) |
| * @see #getProgress() |
| * @see #getSecondaryProgress() |
| */ |
| public synchronized int getMax() { |
| return mMax; |
| } |
| |
| /** |
| * <p>Set the range of the progress bar to 0...<tt>max</tt>.</p> |
| * |
| * @param max the upper range of this progress bar |
| * |
| * @see #getMax() |
| * @see #setProgress(int) |
| * @see #setSecondaryProgress(int) |
| */ |
| public synchronized void setMax(int max) { |
| if (max < 0) { |
| max = 0; |
| } |
| if (max != mMax) { |
| mMax = max; |
| postInvalidate(); |
| |
| if (mProgress > max) { |
| mProgress = max; |
| } |
| refreshProgress(android.R.id.progress, mProgress, false); |
| } |
| } |
| |
| /** |
| * <p>Increase the progress bar's progress by the specified amount.</p> |
| * |
| * @param diff the amount by which the progress must be increased |
| * |
| * @see #setProgress(int) |
| */ |
| public synchronized final void incrementProgressBy(int diff) { |
| setProgress(mProgress + diff); |
| } |
| |
| /** |
| * <p>Increase the progress bar's secondary progress by the specified amount.</p> |
| * |
| * @param diff the amount by which the secondary progress must be increased |
| * |
| * @see #setSecondaryProgress(int) |
| */ |
| public synchronized final void incrementSecondaryProgressBy(int diff) { |
| setSecondaryProgress(mSecondaryProgress + diff); |
| } |
| |
| /** |
| * <p>Start the indeterminate progress animation.</p> |
| */ |
| void startAnimation() { |
| if (getVisibility() != VISIBLE) { |
| return; |
| } |
| |
| if (mIndeterminateDrawable instanceof Animatable) { |
| mShouldStartAnimationDrawable = true; |
| mAnimation = null; |
| } else { |
| if (mInterpolator == null) { |
| mInterpolator = new LinearInterpolator(); |
| } |
| |
| mTransformation = new Transformation(); |
| mAnimation = new AlphaAnimation(0.0f, 1.0f); |
| mAnimation.setRepeatMode(mBehavior); |
| mAnimation.setRepeatCount(Animation.INFINITE); |
| mAnimation.setDuration(mDuration); |
| mAnimation.setInterpolator(mInterpolator); |
| mAnimation.setStartTime(Animation.START_ON_FIRST_FRAME); |
| } |
| postInvalidate(); |
| } |
| |
| /** |
| * <p>Stop the indeterminate progress animation.</p> |
| */ |
| void stopAnimation() { |
| mAnimation = null; |
| mTransformation = null; |
| if (mIndeterminateDrawable instanceof Animatable) { |
| ((Animatable) mIndeterminateDrawable).stop(); |
| mShouldStartAnimationDrawable = false; |
| } |
| postInvalidate(); |
| } |
| |
| /** |
| * Sets the acceleration curve for the indeterminate animation. |
| * The interpolator is loaded as a resource from the specified context. |
| * |
| * @param context The application environment |
| * @param resID The resource identifier of the interpolator to load |
| */ |
| public void setInterpolator(Context context, int resID) { |
| setInterpolator(AnimationUtils.loadInterpolator(context, resID)); |
| } |
| |
| /** |
| * Sets the acceleration curve for the indeterminate animation. |
| * Defaults to a linear interpolation. |
| * |
| * @param interpolator The interpolator which defines the acceleration curve |
| */ |
| public void setInterpolator(Interpolator interpolator) { |
| mInterpolator = interpolator; |
| } |
| |
| /** |
| * Gets the acceleration curve type for the indeterminate animation. |
| * |
| * @return the {@link Interpolator} associated to this animation |
| */ |
| public Interpolator getInterpolator() { |
| return mInterpolator; |
| } |
| |
| @Override |
| public void setVisibility(int v) { |
| if (getVisibility() != v) { |
| super.setVisibility(v); |
| |
| if (mIndeterminate) { |
| // let's be nice with the UI thread |
| if (v == GONE || v == INVISIBLE) { |
| stopAnimation(); |
| } else { |
| startAnimation(); |
| } |
| } |
| } |
| } |
| |
| @Override |
| protected void onVisibilityChanged(View changedView, int visibility) { |
| if (Build.VERSION.SDK_INT >= 8) { |
| super.onVisibilityChanged(changedView, visibility); |
| } |
| |
| if (mIndeterminate) { |
| // let's be nice with the UI thread |
| if (visibility == GONE || visibility == INVISIBLE) { |
| stopAnimation(); |
| } else { |
| startAnimation(); |
| } |
| } |
| } |
| |
| @Override |
| public void invalidateDrawable(Drawable dr) { |
| if (!mInDrawing) { |
| if (verifyDrawable(dr)) { |
| final Rect dirty = dr.getBounds(); |
| final int scrollX = getScrollX() + getPaddingLeft(); |
| final int scrollY = getScrollY() + getPaddingTop(); |
| |
| invalidate(dirty.left + scrollX, dirty.top + scrollY, |
| dirty.right + scrollX, dirty.bottom + scrollY); |
| } else { |
| super.invalidateDrawable(dr); |
| } |
| } |
| } |
| |
| @Override |
| protected void onSizeChanged(int w, int h, int oldw, int oldh) { |
| updateDrawableBounds(w, h); |
| } |
| |
| private void updateDrawableBounds(int w, int h) { |
| // onDraw will translate the canvas so we draw starting at 0,0 |
| int right = w - getPaddingRight() - getPaddingLeft(); |
| int bottom = h - getPaddingBottom() - getPaddingTop(); |
| int top = 0; |
| int left = 0; |
| |
| if (mIndeterminateDrawable != null) { |
| // Aspect ratio logic does not apply to AnimationDrawables |
| if (mOnlyIndeterminate && !(mIndeterminateDrawable instanceof AnimationDrawable)) { |
| // Maintain aspect ratio. Certain kinds of animated drawables |
| // get very confused otherwise. |
| final int intrinsicWidth = mIndeterminateDrawable.getIntrinsicWidth(); |
| final int intrinsicHeight = mIndeterminateDrawable.getIntrinsicHeight(); |
| final float intrinsicAspect = (float) intrinsicWidth / intrinsicHeight; |
| final float boundAspect = (float) w / h; |
| if (intrinsicAspect != boundAspect) { |
| if (boundAspect > intrinsicAspect) { |
| // New width is larger. Make it smaller to match height. |
| final int width = (int) (h * intrinsicAspect); |
| left = (w - width) / 2; |
| right = left + width; |
| } else { |
| // New height is larger. Make it smaller to match width. |
| final int height = (int) (w * (1 / intrinsicAspect)); |
| top = (h - height) / 2; |
| bottom = top + height; |
| } |
| } |
| } |
| mIndeterminateDrawable.setBounds(left, top, right, bottom); |
| } |
| |
| if (mProgressDrawable != null) { |
| mProgressDrawable.setBounds(0, 0, right, bottom); |
| } |
| } |
| |
| @Override |
| protected synchronized void onDraw(Canvas canvas) { |
| super.onDraw(canvas); |
| |
| Drawable d = mCurrentDrawable; |
| if (d != null) { |
| // Translate canvas so a indeterminate circular progress bar with padding |
| // rotates properly in its animation |
| canvas.save(); |
| canvas.translate(getPaddingLeft(), getPaddingTop()); |
| long time = getDrawingTime(); |
| if (mAnimation != null) { |
| mAnimation.getTransformation(time, mTransformation); |
| float scale = mTransformation.getAlpha(); |
| try { |
| mInDrawing = true; |
| d.setLevel((int) (scale * MAX_LEVEL)); |
| } finally { |
| mInDrawing = false; |
| } |
| if (SystemClock.uptimeMillis() - mLastDrawTime >= ANIMATION_RESOLUTION) { |
| mLastDrawTime = SystemClock.uptimeMillis(); |
| postInvalidateDelayed(ANIMATION_RESOLUTION); |
| } |
| } |
| d.draw(canvas); |
| canvas.restore(); |
| if (mShouldStartAnimationDrawable && d instanceof Animatable) { |
| ((Animatable) d).start(); |
| mShouldStartAnimationDrawable = false; |
| } |
| } |
| } |
| |
| @Override |
| protected synchronized void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { |
| Drawable d = mCurrentDrawable; |
| |
| int dw = 0; |
| int dh = 0; |
| if (d != null) { |
| dw = Math.max(mMinWidth, Math.min(mMaxWidth, d.getIntrinsicWidth())); |
| dh = Math.max(mMinHeight, Math.min(mMaxHeight, d.getIntrinsicHeight())); |
| } |
| updateDrawableState(); |
| dw += getPaddingLeft() + getPaddingRight(); |
| dh += getPaddingTop() + getPaddingBottom(); |
| |
| setMeasuredDimension(resolveSize(dw, widthMeasureSpec), |
| resolveSize(dh, heightMeasureSpec)); |
| } |
| |
| @Override |
| protected void drawableStateChanged() { |
| super.drawableStateChanged(); |
| updateDrawableState(); |
| } |
| |
| private void updateDrawableState() { |
| int[] state = getDrawableState(); |
| |
| if (mProgressDrawable != null && mProgressDrawable.isStateful()) { |
| mProgressDrawable.setState(state); |
| } |
| |
| if (mIndeterminateDrawable != null && mIndeterminateDrawable.isStateful()) { |
| mIndeterminateDrawable.setState(state); |
| } |
| } |
| |
| static class SavedState extends BaseSavedState { |
| int progress; |
| int secondaryProgress; |
| |
| /** |
| * Constructor called from {@link ProgressBarCompat#onSaveInstanceState()} |
| */ |
| SavedState(Parcelable superState) { |
| super(superState); |
| } |
| |
| /** |
| * Constructor called from {@link #CREATOR} |
| */ |
| private SavedState(Parcel in) { |
| super(in); |
| progress = in.readInt(); |
| secondaryProgress = in.readInt(); |
| } |
| |
| @Override |
| public void writeToParcel(Parcel out, int flags) { |
| super.writeToParcel(out, flags); |
| out.writeInt(progress); |
| out.writeInt(secondaryProgress); |
| } |
| |
| public static final Parcelable.Creator<SavedState> CREATOR |
| = new Parcelable.Creator<SavedState>() { |
| public SavedState createFromParcel(Parcel in) { |
| return new SavedState(in); |
| } |
| |
| public SavedState[] newArray(int size) { |
| return new SavedState[size]; |
| } |
| }; |
| } |
| |
| @Override |
| public Parcelable onSaveInstanceState() { |
| // Force our ancestor class to save its state |
| Parcelable superState = super.onSaveInstanceState(); |
| SavedState ss = new SavedState(superState); |
| |
| ss.progress = mProgress; |
| ss.secondaryProgress = mSecondaryProgress; |
| |
| return ss; |
| } |
| |
| @Override |
| public void onRestoreInstanceState(Parcelable state) { |
| SavedState ss = (SavedState) state; |
| super.onRestoreInstanceState(ss.getSuperState()); |
| |
| setProgress(ss.progress); |
| setSecondaryProgress(ss.secondaryProgress); |
| } |
| |
| @Override |
| protected void onAttachedToWindow() { |
| super.onAttachedToWindow(); |
| if (mIndeterminate) { |
| startAnimation(); |
| } |
| } |
| |
| @Override |
| protected void onDetachedFromWindow() { |
| if (mIndeterminate) { |
| stopAnimation(); |
| } |
| if(mRefreshProgressRunnable != null) { |
| removeCallbacks(mRefreshProgressRunnable); |
| } |
| |
| // This should come after stopAnimation(), otherwise an invalidate message remains in the |
| // queue, which can prevent the entire view hierarchy from being GC'ed during a rotation |
| super.onDetachedFromWindow(); |
| } |
| |
| } |