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 
17 package com.google.android.setupdesign.view;
18 
19 import android.content.Context;
20 import android.content.res.TypedArray;
21 import android.os.Build;
22 import androidx.recyclerview.widget.RecyclerView;
23 import android.util.AttributeSet;
24 import android.view.LayoutInflater;
25 import android.view.View;
26 import android.view.ViewGroup;
27 import android.view.accessibility.AccessibilityEvent;
28 import android.widget.FrameLayout;
29 import com.google.android.setupdesign.DividerItemDecoration;
30 import com.google.android.setupdesign.R;
31 
32 /**
33  * A RecyclerView that can display a header item at the start of the list. The header can be set by
34  * {@code app:sudHeader} in XML. Note that the header will not be inflated until a layout manager is
35  * set.
36  */
37 public class HeaderRecyclerView extends RecyclerView {
38 
39   private static class HeaderViewHolder extends ViewHolder
40       implements DividerItemDecoration.DividedViewHolder {
41 
HeaderViewHolder(View itemView)42     HeaderViewHolder(View itemView) {
43       super(itemView);
44     }
45 
46     @Override
isDividerAllowedAbove()47     public boolean isDividerAllowedAbove() {
48       return false;
49     }
50 
51     @Override
isDividerAllowedBelow()52     public boolean isDividerAllowedBelow() {
53       return false;
54     }
55   }
56 
57   /**
58    * An adapter that can optionally add one header item to the RecyclerView.
59    *
60    * @param <CVH> Type of the content view holder. i.e. view holder type of the wrapped adapter.
61    */
62   public static class HeaderAdapter<CVH extends ViewHolder>
63       extends RecyclerView.Adapter<ViewHolder> {
64 
65     private static final int HEADER_VIEW_TYPE = Integer.MAX_VALUE;
66 
67     private final RecyclerView.Adapter<CVH> adapter;
68     private View header;
69 
70     private final AdapterDataObserver observer =
71         new AdapterDataObserver() {
72 
73           @Override
74           public void onChanged() {
75             notifyDataSetChanged();
76           }
77 
78           @Override
79           public void onItemRangeChanged(int positionStart, int itemCount) {
80             if (header != null) {
81               positionStart++;
82             }
83             notifyItemRangeChanged(positionStart, itemCount);
84           }
85 
86           @Override
87           public void onItemRangeInserted(int positionStart, int itemCount) {
88             if (header != null) {
89               positionStart++;
90             }
91             notifyItemRangeInserted(positionStart, itemCount);
92           }
93 
94           @Override
95           public void onItemRangeMoved(int fromPosition, int toPosition, int itemCount) {
96             if (header != null) {
97               fromPosition++;
98               toPosition++;
99             }
100             // Why is there no notifyItemRangeMoved?
101             for (int i = 0; i < itemCount; i++) {
102               notifyItemMoved(fromPosition + i, toPosition + i);
103             }
104           }
105 
106           @Override
107           public void onItemRangeRemoved(int positionStart, int itemCount) {
108             if (header != null) {
109               positionStart++;
110             }
111             notifyItemRangeRemoved(positionStart, itemCount);
112           }
113         };
114 
HeaderAdapter(RecyclerView.Adapter<CVH> adapter)115     public HeaderAdapter(RecyclerView.Adapter<CVH> adapter) {
116       this.adapter = adapter;
117       this.adapter.registerAdapterDataObserver(observer);
118       setHasStableIds(this.adapter.hasStableIds());
119     }
120 
121     @Override
onCreateViewHolder(ViewGroup parent, int viewType)122     public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
123       // Returning the same view (header) results in crash ".. but view is not a real child."
124       // The framework creates more than one instance of header because of "disappear"
125       // animations applied on the header and this necessitates creation of another header
126       // view to use after the animation. We work around this restriction by returning an
127       // empty FrameLayout to which the header is attached using #onBindViewHolder method.
128       if (viewType == HEADER_VIEW_TYPE) {
129         FrameLayout frameLayout = new FrameLayout(parent.getContext());
130         FrameLayout.LayoutParams params =
131             new FrameLayout.LayoutParams(
132                 FrameLayout.LayoutParams.MATCH_PARENT, FrameLayout.LayoutParams.WRAP_CONTENT);
133         frameLayout.setLayoutParams(params);
134         return new HeaderViewHolder(frameLayout);
135       } else {
136         return adapter.onCreateViewHolder(parent, viewType);
137       }
138     }
139 
140     @Override
141     @SuppressWarnings("unchecked") // Non-header position always return type CVH
onBindViewHolder(ViewHolder holder, int position)142     public void onBindViewHolder(ViewHolder holder, int position) {
143       if (header != null) {
144         position--;
145       }
146 
147       if (holder instanceof HeaderViewHolder) {
148         if (header == null) {
149           throw new IllegalStateException("HeaderViewHolder cannot find mHeader");
150         }
151         if (header.getParent() != null) {
152           ((ViewGroup) header.getParent()).removeView(header);
153         }
154         FrameLayout mHeaderParent = (FrameLayout) holder.itemView;
155         mHeaderParent.addView(header);
156       } else {
157         adapter.onBindViewHolder((CVH) holder, position);
158       }
159     }
160 
161     @Override
getItemViewType(int position)162     public int getItemViewType(int position) {
163       if (header != null) {
164         position--;
165       }
166       if (position < 0) {
167         return HEADER_VIEW_TYPE;
168       }
169       return adapter.getItemViewType(position);
170     }
171 
172     @Override
getItemCount()173     public int getItemCount() {
174       int count = adapter.getItemCount();
175       if (header != null) {
176         count++;
177       }
178       return count;
179     }
180 
181     @Override
getItemId(int position)182     public long getItemId(int position) {
183       if (header != null) {
184         position--;
185       }
186       if (position < 0) {
187         return Long.MAX_VALUE;
188       }
189       return adapter.getItemId(position);
190     }
191 
setHeader(View header)192     public void setHeader(View header) {
193       this.header = header;
194     }
195 
getWrappedAdapter()196     public RecyclerView.Adapter<CVH> getWrappedAdapter() {
197       return adapter;
198     }
199   }
200 
201   private View header;
202   private int headerRes;
203 
HeaderRecyclerView(Context context)204   public HeaderRecyclerView(Context context) {
205     super(context);
206     init(null, 0);
207   }
208 
HeaderRecyclerView(Context context, AttributeSet attrs)209   public HeaderRecyclerView(Context context, AttributeSet attrs) {
210     super(context, attrs);
211     init(attrs, 0);
212   }
213 
HeaderRecyclerView(Context context, AttributeSet attrs, int defStyleAttr)214   public HeaderRecyclerView(Context context, AttributeSet attrs, int defStyleAttr) {
215     super(context, attrs, defStyleAttr);
216     init(attrs, defStyleAttr);
217   }
218 
init(AttributeSet attrs, int defStyleAttr)219   private void init(AttributeSet attrs, int defStyleAttr) {
220     final TypedArray a =
221         getContext()
222             .obtainStyledAttributes(attrs, R.styleable.SudHeaderRecyclerView, defStyleAttr, 0);
223     headerRes = a.getResourceId(R.styleable.SudHeaderRecyclerView_sudHeader, 0);
224     a.recycle();
225   }
226 
227   @Override
onInitializeAccessibilityEvent(AccessibilityEvent event)228   public void onInitializeAccessibilityEvent(AccessibilityEvent event) {
229     super.onInitializeAccessibilityEvent(event);
230 
231     // Decoration-only headers should not count as an item for accessibility, adjust the
232     // accessibility event to account for that.
233     final int numberOfHeaders = header != null ? 1 : 0;
234     event.setItemCount(event.getItemCount() - numberOfHeaders);
235     event.setFromIndex(Math.max(event.getFromIndex() - numberOfHeaders, 0));
236     if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
237       event.setToIndex(Math.max(event.getToIndex() - numberOfHeaders, 0));
238     }
239   }
240 
241   /** Gets the header view of this RecyclerView, or {@code null} if there are no headers. */
getHeader()242   public View getHeader() {
243     return header;
244   }
245 
246   /**
247    * Set the view to use as the header of this recycler view. Note: This must be called before
248    * setAdapter.
249    */
setHeader(View header)250   public void setHeader(View header) {
251     this.header = header;
252   }
253 
254   @Override
setLayoutManager(LayoutManager layout)255   public void setLayoutManager(LayoutManager layout) {
256     super.setLayoutManager(layout);
257     if (layout != null && header == null && headerRes != 0) {
258       // Inflating a child view requires the layout manager to be set. Check here to see if
259       // any header item is specified in XML and inflate them.
260       final LayoutInflater inflater = LayoutInflater.from(getContext());
261       header = inflater.inflate(headerRes, this, false);
262     }
263   }
264 
265   @Override
266   @SuppressWarnings("rawtypes,unchecked") // RecyclerView.setAdapter uses raw type :(
setAdapter(Adapter adapter)267   public void setAdapter(Adapter adapter) {
268     if (header != null && adapter != null) {
269       final HeaderAdapter headerAdapter = new HeaderAdapter(adapter);
270       headerAdapter.setHeader(header);
271       adapter = headerAdapter;
272     }
273     super.setAdapter(adapter);
274   }
275 }
276