1 package com.android.systemui.qs;
2 
3 import android.content.Context;
4 import android.content.res.ColorStateList;
5 import android.content.res.TypedArray;
6 import android.graphics.drawable.AnimatedVectorDrawable;
7 import android.util.AttributeSet;
8 import android.util.Log;
9 import android.util.TypedValue;
10 import android.view.View;
11 import android.view.ViewGroup;
12 import android.widget.ImageView;
13 import com.android.systemui.R;
14 
15 import java.util.ArrayList;
16 
17 public class PageIndicator extends ViewGroup {
18 
19     private static final String TAG = "PageIndicator";
20     private static final boolean DEBUG = false;
21 
22     private static final long ANIMATION_DURATION = 250;
23 
24     // The size of a single dot in relation to the whole animation.
25     private static final float SINGLE_SCALE = .4f;
26 
27     private static final float MINOR_ALPHA = .3f;
28 
29     private final ArrayList<Integer> mQueuedPositions = new ArrayList<>();
30 
31     private final int mPageIndicatorWidth;
32     private final int mPageIndicatorHeight;
33     private final int mPageDotWidth;
34 
35     private int mPosition = -1;
36     private boolean mAnimating;
37 
PageIndicator(Context context, AttributeSet attrs)38     public PageIndicator(Context context, AttributeSet attrs) {
39         super(context, attrs);
40         mPageIndicatorWidth =
41                 (int) mContext.getResources().getDimension(R.dimen.qs_page_indicator_width);
42         mPageIndicatorHeight =
43                 (int) mContext.getResources().getDimension(R.dimen.qs_page_indicator_height);
44         mPageDotWidth = (int) (mPageIndicatorWidth * SINGLE_SCALE);
45     }
46 
setNumPages(int numPages)47     public void setNumPages(int numPages) {
48         setVisibility(numPages > 1 ? View.VISIBLE : View.INVISIBLE);
49         if (mAnimating) {
50             Log.w(TAG, "setNumPages during animation");
51         }
52         while (numPages < getChildCount()) {
53             removeViewAt(getChildCount() - 1);
54         }
55         TypedArray array = getContext().obtainStyledAttributes(
56                 new int[]{android.R.attr.colorForeground});
57         int color = array.getColor(0, 0);
58         array.recycle();
59         while (numPages > getChildCount()) {
60             ImageView v = new ImageView(mContext);
61             v.setImageResource(R.drawable.minor_a_b);
62             v.setImageTintList(ColorStateList.valueOf(color));
63             addView(v, new LayoutParams(mPageIndicatorWidth, mPageIndicatorHeight));
64         }
65         // Refresh state.
66         setIndex(mPosition >> 1);
67     }
68 
setLocation(float location)69     public void setLocation(float location) {
70         int index = (int) location;
71         setContentDescription(getContext().getString(R.string.accessibility_quick_settings_page,
72                 (index + 1), getChildCount()));
73         int position = index << 1 | ((location != index) ? 1 : 0);
74         if (DEBUG) Log.d(TAG, "setLocation " + location + " " + index + " " + position);
75 
76         int lastPosition = mPosition;
77         if (mQueuedPositions.size() != 0) {
78             lastPosition = mQueuedPositions.get(mQueuedPositions.size() - 1);
79         }
80         if (position == lastPosition) return;
81         if (mAnimating) {
82             if (DEBUG) Log.d(TAG, "Queueing transition to " + Integer.toHexString(position));
83             mQueuedPositions.add(position);
84             return;
85         }
86 
87         setPosition(position);
88     }
89 
setPosition(int position)90     private void setPosition(int position) {
91         if (isVisibleToUser() && Math.abs(mPosition - position) == 1) {
92             animate(mPosition, position);
93         } else {
94             if (DEBUG) Log.d(TAG, "Skipping animation " + isVisibleToUser() + " " + mPosition
95                     + " " + position);
96             setIndex(position >> 1);
97         }
98         mPosition = position;
99     }
100 
setIndex(int index)101     private void setIndex(int index) {
102         final int N = getChildCount();
103         for (int i = 0; i < N; i++) {
104             ImageView v = (ImageView) getChildAt(i);
105             // Clear out any animation positioning.
106             v.setTranslationX(0);
107             v.setImageResource(R.drawable.major_a_b);
108             v.setAlpha(getAlpha(i == index));
109         }
110     }
111 
animate(int from, int to)112     private void animate(int from, int to) {
113         if (DEBUG) Log.d(TAG, "Animating from " + Integer.toHexString(from) + " to "
114                 + Integer.toHexString(to));
115         int fromIndex = from >> 1;
116         int toIndex = to >> 1;
117 
118         // Set the position of everything, then we will manually control the two views involved
119         // in the animation.
120         setIndex(fromIndex);
121 
122         boolean fromTransition = (from & 1) != 0;
123         boolean isAState = fromTransition ? from > to : from < to;
124         int firstIndex = Math.min(fromIndex, toIndex);
125         int secondIndex = Math.max(fromIndex, toIndex);
126         if (secondIndex == firstIndex) {
127             secondIndex++;
128         }
129         ImageView first = (ImageView) getChildAt(firstIndex);
130         ImageView second = (ImageView) getChildAt(secondIndex);
131         if (first == null || second == null) {
132             // may happen during reInflation or other weird cases
133             return;
134         }
135         // Lay the two views on top of each other.
136         second.setTranslationX(first.getX() - second.getX());
137 
138         playAnimation(first, getTransition(fromTransition, isAState, false));
139         first.setAlpha(getAlpha(false));
140 
141         playAnimation(second, getTransition(fromTransition, isAState, true));
142         second.setAlpha(getAlpha(true));
143 
144         mAnimating = true;
145     }
146 
147     private float getAlpha(boolean isMajor) {
148         return isMajor ? 1 : MINOR_ALPHA;
149     }
150 
151     private void playAnimation(ImageView imageView, int res) {
152         final AnimatedVectorDrawable avd = (AnimatedVectorDrawable) getContext().getDrawable(res);
153         imageView.setImageDrawable(avd);
154         avd.forceAnimationOnUI();
155         avd.start();
156         // TODO: Figure out how to user an AVD animation callback instead, which doesn't
157         // seem to be working right now...
158         postDelayed(mAnimationDone, ANIMATION_DURATION);
159     }
160 
161     private int getTransition(boolean fromB, boolean isMajorAState, boolean isMajor) {
162         if (isMajor) {
163             if (fromB) {
164                 if (isMajorAState) {
165                     return R.drawable.major_b_a_animation;
166                 } else {
167                     return R.drawable.major_b_c_animation;
168                 }
169             } else {
170                 if (isMajorAState) {
171                     return R.drawable.major_a_b_animation;
172                 } else {
173                     return R.drawable.major_c_b_animation;
174                 }
175             }
176         } else {
177             if (fromB) {
178                 if (isMajorAState) {
179                     return R.drawable.minor_b_c_animation;
180                 } else {
181                     return R.drawable.minor_b_a_animation;
182                 }
183             } else {
184                 if (isMajorAState) {
185                     return R.drawable.minor_c_b_animation;
186                 } else {
187                     return R.drawable.minor_a_b_animation;
188                 }
189             }
190         }
191     }
192 
193     @Override
194     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
195         final int N = getChildCount();
196         if (N == 0) {
197             super.onMeasure(widthMeasureSpec, heightMeasureSpec);
198             return;
199         }
200         final int widthChildSpec = MeasureSpec.makeMeasureSpec(mPageIndicatorWidth,
201                 MeasureSpec.EXACTLY);
202         final int heightChildSpec = MeasureSpec.makeMeasureSpec(mPageIndicatorHeight,
203                 MeasureSpec.EXACTLY);
204         for (int i = 0; i < N; i++) {
205             getChildAt(i).measure(widthChildSpec, heightChildSpec);
206         }
207         int width = (mPageIndicatorWidth - mPageDotWidth) * (N - 1) + mPageDotWidth;
208         setMeasuredDimension(width, mPageIndicatorHeight);
209     }
210 
211     @Override
212     protected void onLayout(boolean changed, int l, int t, int r, int b) {
213         final int N = getChildCount();
214         if (N == 0) {
215             return;
216         }
217         for (int i = 0; i < N; i++) {
218             int left = (mPageIndicatorWidth - mPageDotWidth) * i;
219             getChildAt(i).layout(left, 0, mPageIndicatorWidth + left, mPageIndicatorHeight);
220         }
221     }
222 
223     private final Runnable mAnimationDone = new Runnable() {
224         @Override
225         public void run() {
226             if (DEBUG) Log.d(TAG, "onAnimationEnd - queued: " + mQueuedPositions.size());
227             mAnimating = false;
228             if (mQueuedPositions.size() != 0) {
229                 setPosition(mQueuedPositions.remove(0));
230             }
231         }
232     };
233 }
234