1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 package com.android.launcher3.allapps;
17 
18 import com.android.launcher3.util.Thunk;
19 
20 import java.util.HashSet;
21 import java.util.List;
22 
23 import androidx.recyclerview.widget.RecyclerView;
24 
25 public class AllAppsFastScrollHelper implements AllAppsGridAdapter.BindViewCallback {
26 
27     private static final int INITIAL_TOUCH_SETTLING_DURATION = 100;
28     private static final int REPEAT_TOUCH_SETTLING_DURATION = 200;
29 
30     private AllAppsRecyclerView mRv;
31     private AlphabeticalAppsList mApps;
32 
33     // Keeps track of the current and targeted fast scroll section (the section to scroll to after
34     // the initial delay)
35     int mTargetFastScrollPosition = -1;
36     @Thunk String mCurrentFastScrollSection;
37     @Thunk String mTargetFastScrollSection;
38 
39     // The settled states affect the delay before the fast scroll animation is applied
40     private boolean mHasFastScrollTouchSettled;
41     private boolean mHasFastScrollTouchSettledAtLeastOnce;
42 
43     // Set of all views animated during fast scroll.  We keep track of these ourselves since there
44     // is no way to reset a view once it gets scrapped or recycled without other hacks
45     private HashSet<RecyclerView.ViewHolder> mTrackedFastScrollViews = new HashSet<>();
46 
47     // Smooth fast-scroll animation frames
48     @Thunk int mFastScrollFrameIndex;
49     @Thunk final int[] mFastScrollFrames = new int[10];
50 
51     /**
52      * This runnable runs a single frame of the smooth scroll animation and posts the next frame
53      * if necessary.
54      */
55     @Thunk Runnable mSmoothSnapNextFrameRunnable = new Runnable() {
56         @Override
57         public void run() {
58             if (mFastScrollFrameIndex < mFastScrollFrames.length) {
59                 mRv.scrollBy(0, mFastScrollFrames[mFastScrollFrameIndex]);
60                 mFastScrollFrameIndex++;
61                 mRv.postOnAnimation(mSmoothSnapNextFrameRunnable);
62             }
63         }
64     };
65 
66     /**
67      * This runnable updates the current fast scroll section to the target fastscroll section.
68      */
69     Runnable mFastScrollToTargetSectionRunnable = new Runnable() {
70         @Override
71         public void run() {
72             // Update to the target section
73             mCurrentFastScrollSection = mTargetFastScrollSection;
74             mHasFastScrollTouchSettled = true;
75             mHasFastScrollTouchSettledAtLeastOnce = true;
76             updateTrackedViewsFastScrollFocusState();
77         }
78     };
79 
AllAppsFastScrollHelper(AllAppsRecyclerView rv, AlphabeticalAppsList apps)80     public AllAppsFastScrollHelper(AllAppsRecyclerView rv, AlphabeticalAppsList apps) {
81         mRv = rv;
82         mApps = apps;
83     }
84 
onSetAdapter(AllAppsGridAdapter adapter)85     public void onSetAdapter(AllAppsGridAdapter adapter) {
86         adapter.setBindViewCallback(this);
87     }
88 
89     /**
90      * Smooth scrolls the recycler view to the given section.
91      *
92      * @return whether the fastscroller can scroll to the new section.
93      */
smoothScrollToSection(int scrollY, int availableScrollHeight, AlphabeticalAppsList.FastScrollSectionInfo info)94     public boolean smoothScrollToSection(int scrollY, int availableScrollHeight,
95             AlphabeticalAppsList.FastScrollSectionInfo info) {
96         if (mTargetFastScrollPosition != info.fastScrollToItem.position) {
97             mTargetFastScrollPosition = info.fastScrollToItem.position;
98             smoothSnapToPosition(scrollY, availableScrollHeight, info);
99             return true;
100         }
101         return false;
102     }
103 
104     /**
105      * Smoothly snaps to a given position.  We do this manually by calculating the keyframes
106      * ourselves and animating the scroll on the recycler view.
107      */
smoothSnapToPosition(int scrollY, int availableScrollHeight, AlphabeticalAppsList.FastScrollSectionInfo info)108     private void smoothSnapToPosition(int scrollY, int availableScrollHeight,
109             AlphabeticalAppsList.FastScrollSectionInfo info) {
110         mRv.removeCallbacks(mSmoothSnapNextFrameRunnable);
111         mRv.removeCallbacks(mFastScrollToTargetSectionRunnable);
112 
113         trackAllChildViews();
114         if (mHasFastScrollTouchSettled) {
115             // In this case, the user has already settled once (and the fast scroll state has
116             // animated) and they are just fine-tuning their section from the last section, so
117             // we should make it feel fast and update immediately.
118             mCurrentFastScrollSection = info.sectionName;
119             mTargetFastScrollSection = null;
120             updateTrackedViewsFastScrollFocusState();
121         } else {
122             // Otherwise, the user has scrubbed really far, and we don't want to distract the user
123             // with the flashing fast scroll state change animation in addition to the fast scroll
124             // section popup, so reset the views to normal, and wait for the touch to settle again
125             // before animating the fast scroll state.
126             mCurrentFastScrollSection = null;
127             mTargetFastScrollSection = info.sectionName;
128             mHasFastScrollTouchSettled = false;
129             updateTrackedViewsFastScrollFocusState();
130 
131             // Delay scrolling to a new section until after some duration.  If the user has been
132             // scrubbing a while and makes multiple big jumps, then reduce the time needed for the
133             // fast scroll to settle so it doesn't feel so long.
134             mRv.postDelayed(mFastScrollToTargetSectionRunnable,
135                     mHasFastScrollTouchSettledAtLeastOnce ?
136                             REPEAT_TOUCH_SETTLING_DURATION :
137                             INITIAL_TOUCH_SETTLING_DURATION);
138         }
139 
140         // Calculate the full animation from the current scroll position to the final scroll
141         // position, and then run the animation for the duration.  If we are scrolling to the
142         // first fast scroll section, then just scroll to the top of the list itself.
143         List<AlphabeticalAppsList.FastScrollSectionInfo> fastScrollSections =
144                 mApps.getFastScrollerSections();
145         int newPosition = info.fastScrollToItem.position;
146         int newScrollY = fastScrollSections.size() > 0 && fastScrollSections.get(0) == info
147                         ? 0
148                         : Math.min(availableScrollHeight, mRv.getCurrentScrollY(newPosition, 0));
149         int numFrames = mFastScrollFrames.length;
150         int deltaY = newScrollY - scrollY;
151         float ySign = Math.signum(deltaY);
152         int step = (int) (ySign * Math.ceil((float) Math.abs(deltaY) / numFrames));
153         for (int i = 0; i < numFrames; i++) {
154             // TODO(winsonc): We can interpolate this as well.
155             mFastScrollFrames[i] = (int) (ySign * Math.min(Math.abs(step), Math.abs(deltaY)));
156             deltaY -= step;
157         }
158         mFastScrollFrameIndex = 0;
159         mRv.postOnAnimation(mSmoothSnapNextFrameRunnable);
160     }
161 
onFastScrollCompleted()162     public void onFastScrollCompleted() {
163         // TODO(winsonc): Handle the case when the user scrolls and releases before the animation
164         //                runs
165 
166         // Stop animating the fast scroll position and state
167         mRv.removeCallbacks(mSmoothSnapNextFrameRunnable);
168         mRv.removeCallbacks(mFastScrollToTargetSectionRunnable);
169 
170         // Reset the tracking variables
171         mHasFastScrollTouchSettled = false;
172         mHasFastScrollTouchSettledAtLeastOnce = false;
173         mCurrentFastScrollSection = null;
174         mTargetFastScrollSection = null;
175         mTargetFastScrollPosition = -1;
176 
177         updateTrackedViewsFastScrollFocusState();
178         mTrackedFastScrollViews.clear();
179     }
180 
181     @Override
onBindView(AllAppsGridAdapter.ViewHolder holder)182     public void onBindView(AllAppsGridAdapter.ViewHolder holder) {
183         // Update newly bound views to the current fast scroll state if we are fast scrolling
184         if (mCurrentFastScrollSection != null || mTargetFastScrollSection != null) {
185             mTrackedFastScrollViews.add(holder);
186         }
187     }
188 
189     /**
190      * Starts tracking all the recycler view's children which are FastScrollFocusableViews.
191      */
trackAllChildViews()192     private void trackAllChildViews() {
193         int childCount = mRv.getChildCount();
194         for (int i = 0; i < childCount; i++) {
195             RecyclerView.ViewHolder viewHolder = mRv.getChildViewHolder(mRv.getChildAt(i));
196             if (viewHolder != null) {
197                 mTrackedFastScrollViews.add(viewHolder);
198             }
199         }
200     }
201 
202     /**
203      * Updates the fast scroll focus on all the children.
204      */
updateTrackedViewsFastScrollFocusState()205     private void updateTrackedViewsFastScrollFocusState() {
206         for (RecyclerView.ViewHolder viewHolder : mTrackedFastScrollViews) {
207             int pos = viewHolder.getAdapterPosition();
208             boolean isActive = false;
209             if (mCurrentFastScrollSection != null
210                     && pos > RecyclerView.NO_POSITION
211                     && pos < mApps.getAdapterItems().size()) {
212                 AlphabeticalAppsList.AdapterItem item = mApps.getAdapterItems().get(pos);
213                 isActive = item != null &&
214                         mCurrentFastScrollSection.equals(item.sectionName) &&
215                         item.position == mTargetFastScrollPosition;
216             }
217             viewHolder.itemView.setActivated(isActive);
218         }
219     }
220 }
221