1 /*
2  * Copyright (C) 2016 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 
17 package com.android.documentsui;
18 
19 import android.content.Context;
20 import android.util.AttributeSet;
21 import android.view.GestureDetector;
22 import android.view.KeyEvent;
23 import android.view.LayoutInflater;
24 import android.view.MotionEvent;
25 import android.view.View;
26 import android.view.ViewGroup;
27 
28 import androidx.recyclerview.widget.LinearLayoutManager;
29 import androidx.recyclerview.widget.RecyclerView;
30 
31 import com.android.documentsui.NavigationViewManager.Breadcrumb;
32 import com.android.documentsui.NavigationViewManager.Environment;
33 import com.android.documentsui.dirlist.AccessibilityEventRouter;
34 
35 import java.util.function.Consumer;
36 import java.util.function.IntConsumer;
37 
38 /**
39  * Horizontal breadcrumb
40  */
41 public final class HorizontalBreadcrumb extends RecyclerView implements Breadcrumb {
42 
43     private static final int USER_NO_SCROLL_OFFSET_THRESHOLD = 5;
44 
45     private LinearLayoutManager mLayoutManager;
46     private BreadcrumbAdapter mAdapter;
47     private IntConsumer mClickListener;
48 
HorizontalBreadcrumb(Context context, AttributeSet attrs, int defStyleAttr)49     public HorizontalBreadcrumb(Context context, AttributeSet attrs, int defStyleAttr) {
50         super(context, attrs, defStyleAttr);
51     }
52 
HorizontalBreadcrumb(Context context, AttributeSet attrs)53     public HorizontalBreadcrumb(Context context, AttributeSet attrs) {
54         super(context, attrs);
55     }
56 
HorizontalBreadcrumb(Context context)57     public HorizontalBreadcrumb(Context context) {
58         super(context);
59     }
60 
61     @Override
setup(Environment env, com.android.documentsui.base.State state, IntConsumer listener)62     public void setup(Environment env,
63             com.android.documentsui.base.State state,
64             IntConsumer listener) {
65 
66         mClickListener = listener;
67         mLayoutManager = new HorizontalBreadcrumbLinearLayoutManager(
68                 getContext(), LinearLayoutManager.HORIZONTAL, false);
69         mAdapter = new BreadcrumbAdapter(state, env, this::onKey);
70         // Since we are using GestureDetector to detect click events, a11y services don't know which
71         // views are clickable because we aren't using View.OnClickListener. Thus, we need to use a
72         // custom accessibility delegate to route click events correctly.
73         // See AccessibilityClickEventRouter for more details on how we are routing these a11y
74         // events.
75         setAccessibilityDelegateCompat(
76                 new AccessibilityEventRouter(this,
77                         (View child) -> onAccessibilityClick(child), null));
78 
79         setLayoutManager(mLayoutManager);
80         addOnItemTouchListener(new ClickListener(getContext(), this::onSingleTapUp));
81     }
82 
83     @Override
show(boolean visibility)84     public void show(boolean visibility) {
85         if (visibility) {
86             setVisibility(VISIBLE);
87             boolean shouldScroll = !hasUserDefineScrollOffset();
88             if (getAdapter() == null) {
89                 setAdapter(mAdapter);
90             } else {
91                 int currentItemCount = mAdapter.getItemCount();
92                 int lastItemCount = mAdapter.getLastItemSize();
93                 if (currentItemCount > lastItemCount) {
94                     mAdapter.notifyItemRangeInserted(lastItemCount,
95                             currentItemCount - lastItemCount);
96                     mAdapter.notifyItemChanged(lastItemCount - 1);
97                 } else if (currentItemCount < lastItemCount) {
98                     mAdapter.notifyItemRangeRemoved(currentItemCount,
99                             lastItemCount - currentItemCount);
100                     mAdapter.notifyItemChanged(currentItemCount - 1);
101                 } else {
102                     mAdapter.notifyItemChanged(currentItemCount - 1);
103                 }
104             }
105             if (shouldScroll) {
106                 mLayoutManager.scrollToPosition(mAdapter.getItemCount() - 1);
107             }
108         } else {
109             setVisibility(GONE);
110             setAdapter(null);
111         }
112         mAdapter.updateLastItemSize();
113     }
114 
hasUserDefineScrollOffset()115     private boolean hasUserDefineScrollOffset() {
116         final int maxOffset = computeHorizontalScrollRange() - computeHorizontalScrollExtent();
117         return (maxOffset - computeHorizontalScrollOffset() > USER_NO_SCROLL_OFFSET_THRESHOLD);
118     }
119 
onAccessibilityClick(View child)120     private boolean onAccessibilityClick(View child) {
121         int pos = getChildAdapterPosition(child);
122         if (pos != getAdapter().getItemCount() - 1) {
123             mClickListener.accept(pos);
124             return true;
125         }
126         return false;
127     }
128 
onKey(View v, int keyCode, KeyEvent event)129     private boolean onKey(View v, int keyCode, KeyEvent event) {
130         switch (keyCode) {
131             case KeyEvent.KEYCODE_ENTER:
132                 return onAccessibilityClick(v);
133             default:
134                 return false;
135         }
136     }
137 
138     @Override
postUpdate()139     public void postUpdate() {
140     }
141 
onSingleTapUp(MotionEvent e)142     private void onSingleTapUp(MotionEvent e) {
143         View itemView = findChildViewUnder(e.getX(), e.getY());
144         int pos = getChildAdapterPosition(itemView);
145         if (pos != mAdapter.getItemCount() - 1 && pos != -1) {
146             mClickListener.accept(pos);
147         }
148     }
149 
150     private static final class BreadcrumbAdapter
151             extends RecyclerView.Adapter<BreadcrumbHolder> {
152 
153         private final Environment mEnv;
154         private final com.android.documentsui.base.State mState;
155         private final View.OnKeyListener mClickListener;
156         // We keep the old item size so the breadcrumb will only re-render views that are necessary
157         private int mLastItemSize;
158 
BreadcrumbAdapter(com.android.documentsui.base.State state, Environment env, View.OnKeyListener clickListener)159         public BreadcrumbAdapter(com.android.documentsui.base.State state,
160                 Environment env,
161                 View.OnKeyListener clickListener) {
162             mState = state;
163             mEnv = env;
164             mClickListener = clickListener;
165             mLastItemSize = getItemCount();
166         }
167 
168         @Override
onCreateViewHolder(ViewGroup parent, int viewType)169         public BreadcrumbHolder onCreateViewHolder(ViewGroup parent, int viewType) {
170             View v = LayoutInflater.from(parent.getContext())
171                     .inflate(R.layout.navigation_breadcrumb_item, null);
172             return new BreadcrumbHolder(v);
173         }
174 
175         @Override
onBindViewHolder(BreadcrumbHolder holder, int position)176         public void onBindViewHolder(BreadcrumbHolder holder, int position) {
177             final int padding = (int) holder.itemView.getResources()
178                     .getDimension(R.dimen.breadcrumb_item_padding);
179             final boolean isFirst = position == 0;
180             // Note that when isFirst is true, there might not be a DocumentInfo on the stack as it
181             // could be an error state screen accessible from the root info.
182             final boolean isLast = position == getItemCount() - 1;
183 
184             holder.mTitle.setText(
185                     isFirst ? mEnv.getCurrentRoot().title : mState.stack.get(position).displayName);
186             holder.mTitle.setEnabled(isLast);
187             holder.mTitle.setPadding(isFirst ? padding * 3 : padding,
188                     padding, isLast ? padding * 2 : padding, padding);
189             holder.mArrow.setVisibility(isLast ? View.GONE : View.VISIBLE);
190 
191             holder.itemView.setOnKeyListener(mClickListener);
192             holder.setLast(isLast);
193         }
194 
195         @Override
getItemCount()196         public int getItemCount() {
197             // Don't show recents in the breadcrumb.
198             if (mState.stack.isRecents()) {
199                 return 0;
200             }
201             // Continue showing the root title in the breadcrumb for cross-profile error screens.
202             if (mState.supportsCrossProfile()
203                     && mState.stack.size() == 0
204                     && mState.stack.getRoot() != null
205                     && mState.stack.getRoot().supportsCrossProfile()) {
206                 return 1;
207             }
208             return mState.stack.size();
209         }
210 
getLastItemSize()211         public int getLastItemSize() {
212             return mLastItemSize;
213         }
214 
updateLastItemSize()215         public void updateLastItemSize() {
216             mLastItemSize = getItemCount();
217         }
218     }
219 
220     private static final class ClickListener extends GestureDetector
221             implements OnItemTouchListener {
222 
ClickListener(Context context, Consumer<MotionEvent> listener)223         public ClickListener(Context context, Consumer<MotionEvent> listener) {
224             super(context, new SimpleOnGestureListener() {
225                 @Override
226                 public boolean onSingleTapUp(MotionEvent e) {
227                     listener.accept(e);
228                     return true;
229                 }
230             });
231         }
232 
233         @Override
onInterceptTouchEvent(RecyclerView rv, MotionEvent e)234         public boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e) {
235             onTouchEvent(e);
236             return false;
237         }
238 
239         @Override
onTouchEvent(RecyclerView rv, MotionEvent e)240         public void onTouchEvent(RecyclerView rv, MotionEvent e) {
241             onTouchEvent(e);
242         }
243 
244         @Override
onRequestDisallowInterceptTouchEvent(boolean disallowIntercept)245         public void onRequestDisallowInterceptTouchEvent(boolean disallowIntercept) {
246         }
247     }
248 
249     private static class HorizontalBreadcrumbLinearLayoutManager extends LinearLayoutManager {
250 
251         /**
252          * Disable predictive animations. There is a bug in RecyclerView which causes views that
253          * are being reloaded to pull invalid view holders from the internal recycler stack if the
254          * adapter size has decreased since the ViewHolder was recycled.
255          */
256         @Override
supportsPredictiveItemAnimations()257         public boolean supportsPredictiveItemAnimations() {
258             return false;
259         }
260 
HorizontalBreadcrumbLinearLayoutManager( Context context, int orientation, boolean reverseLayout)261         HorizontalBreadcrumbLinearLayoutManager(
262                 Context context, int orientation, boolean reverseLayout) {
263             super(context, orientation, reverseLayout);
264         }
265     }
266 }
267