1 /*
2  * Copyright (C) 2014 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.example.android.supportv7.widget;
17 
18 import com.example.android.supportv7.R;
19 
20 import android.animation.Animator;
21 import android.animation.ValueAnimator;
22 import android.annotation.TargetApi;
23 import android.app.Activity;
24 import android.content.Context;
25 import android.os.Build;
26 import android.os.Bundle;
27 import android.support.v4.util.ArrayMap;
28 import android.support.v4.view.MenuItemCompat;
29 import android.support.v4.view.ViewCompat;
30 import android.support.v4.view.ViewPropertyAnimatorListener;
31 import android.support.v7.widget.DefaultItemAnimator;
32 import android.support.v7.widget.RecyclerView;
33 import android.util.DisplayMetrics;
34 import android.util.TypedValue;
35 import android.view.Menu;
36 import android.view.MenuItem;
37 import android.view.View;
38 import android.view.ViewGroup;
39 import android.widget.CheckBox;
40 import android.widget.CompoundButton;
41 import android.widget.TextView;
42 
43 import java.util.ArrayList;
44 import java.util.List;
45 
46 public class AnimatedRecyclerView extends Activity {
47 
48     private static final int SCROLL_DISTANCE = 80; // dp
49 
50     private RecyclerView mRecyclerView;
51 
52     private int mNumItemsAdded = 0;
53     ArrayList<String> mItems = new ArrayList<String>();
54     MyAdapter mAdapter;
55 
56     boolean mAnimationsEnabled = true;
57     boolean mPredictiveAnimationsEnabled = true;
58     RecyclerView.ItemAnimator mCachedAnimator = null;
59     boolean mEnableInPlaceChange = true;
60 
61     @Override
onCreate(Bundle savedInstanceState)62     protected void onCreate(Bundle savedInstanceState) {
63         super.onCreate(savedInstanceState);
64         setContentView(R.layout.animated_recycler_view);
65 
66         ViewGroup container = (ViewGroup) findViewById(R.id.container);
67         mRecyclerView = new RecyclerView(this);
68         mCachedAnimator = createAnimator();
69         mCachedAnimator.setChangeDuration(2000);
70         mRecyclerView.setItemAnimator(mCachedAnimator);
71         mRecyclerView.setLayoutManager(new MyLayoutManager(this));
72         mRecyclerView.setHasFixedSize(true);
73         mRecyclerView.setLayoutParams(new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
74                 ViewGroup.LayoutParams.MATCH_PARENT));
75         for (int i = 0; i < 6; ++i) {
76             mItems.add("Item #" + i);
77         }
78         mAdapter = new MyAdapter(mItems);
79         mRecyclerView.setAdapter(mAdapter);
80         container.addView(mRecyclerView);
81 
82         CheckBox enableAnimations = (CheckBox) findViewById(R.id.enableAnimations);
83         enableAnimations.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
84             @Override
85             public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
86                 if (isChecked && mRecyclerView.getItemAnimator() == null) {
87                     mRecyclerView.setItemAnimator(mCachedAnimator);
88                 } else if (!isChecked && mRecyclerView.getItemAnimator() != null) {
89                     mRecyclerView.setItemAnimator(null);
90                 }
91                 mAnimationsEnabled = isChecked;
92             }
93         });
94 
95         CheckBox enablePredictiveAnimations =
96                 (CheckBox) findViewById(R.id.enablePredictiveAnimations);
97         enablePredictiveAnimations.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
98             @Override
99             public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
100                 mPredictiveAnimationsEnabled = isChecked;
101             }
102         });
103 
104         CheckBox enableInPlaceChange = (CheckBox) findViewById(R.id.enableInPlaceChange);
105         enableInPlaceChange.setChecked(mEnableInPlaceChange);
106         enableInPlaceChange.setOnCheckedChangeListener(
107                 new CompoundButton.OnCheckedChangeListener() {
108                     @Override
109                     public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
110                         mEnableInPlaceChange = isChecked;
111                     }
112                 });
113     }
114 
createAnimator()115     private RecyclerView.ItemAnimator createAnimator() {
116         return new DefaultItemAnimator() {
117             List<ItemChangeAnimator> mPendingChangeAnimations = new ArrayList<>();
118             ArrayMap<RecyclerView.ViewHolder, ItemChangeAnimator> mRunningAnimations
119                     = new ArrayMap<>();
120             ArrayMap<MyViewHolder, Long> mPendingSettleList = new ArrayMap<>();
121 
122             @Override
123             public void runPendingAnimations() {
124                 super.runPendingAnimations();
125                 for (ItemChangeAnimator anim : mPendingChangeAnimations) {
126                     anim.start();
127                     mRunningAnimations.put(anim.mViewHolder, anim);
128                 }
129                 mPendingChangeAnimations.clear();
130                 for (int i = mPendingSettleList.size() - 1; i >=0; i--) {
131                     final MyViewHolder vh = mPendingSettleList.keyAt(i);
132                     final long duration = mPendingSettleList.valueAt(i);
133                     ViewCompat.animate(vh.textView).translationX(0f).alpha(1f)
134                             .setDuration(duration).setListener(
135                             new ViewPropertyAnimatorListener() {
136                                 @Override
137                                 public void onAnimationStart(View view) {
138                                     dispatchAnimationStarted(vh);
139                                 }
140 
141                                 @Override
142                                 public void onAnimationEnd(View view) {
143                                     ViewCompat.setTranslationX(vh.textView, 0f);
144                                     ViewCompat.setAlpha(vh.textView, 1f);
145                                     dispatchAnimationFinished(vh);
146                                 }
147 
148                                 @Override
149                                 public void onAnimationCancel(View view) {
150 
151                                 }
152                             }).start();
153                 }
154                 mPendingSettleList.clear();
155             }
156 
157             @Override
158             public ItemHolderInfo recordPreLayoutInformation(RecyclerView.State state,
159                     RecyclerView.ViewHolder viewHolder,
160                     @AdapterChanges int changeFlags, List<Object> payloads) {
161                 MyItemInfo info = (MyItemInfo) super
162                         .recordPreLayoutInformation(state, viewHolder, changeFlags, payloads);
163                 info.text = ((MyViewHolder) viewHolder).textView.getText();
164                 return info;
165             }
166 
167             @Override
168             public ItemHolderInfo recordPostLayoutInformation(RecyclerView.State state,
169                     RecyclerView.ViewHolder viewHolder) {
170                 MyItemInfo info = (MyItemInfo) super.recordPostLayoutInformation(state, viewHolder);
171                 info.text = ((MyViewHolder) viewHolder).textView.getText();
172                 return info;
173             }
174 
175 
176             @Override
177             public boolean canReuseUpdatedViewHolder(RecyclerView.ViewHolder viewHolder) {
178                 return mEnableInPlaceChange;
179             }
180 
181             @Override
182             public void endAnimation(RecyclerView.ViewHolder item) {
183                 super.endAnimation(item);
184                 for (int i = mPendingChangeAnimations.size() - 1; i >= 0; i--) {
185                     ItemChangeAnimator anim = mPendingChangeAnimations.get(i);
186                     if (anim.mViewHolder == item) {
187                         mPendingChangeAnimations.remove(i);
188                         anim.setFraction(1f);
189                         dispatchChangeFinished(item, true);
190                     }
191                 }
192                 for (int i = mRunningAnimations.size() - 1; i >= 0; i--) {
193                     ItemChangeAnimator animator = mRunningAnimations.get(item);
194                     if (animator != null) {
195                         animator.end();
196                         mRunningAnimations.removeAt(i);
197                     }
198                 }
199                 for (int  i = mPendingSettleList.size() - 1; i >= 0; i--) {
200                     final MyViewHolder vh = mPendingSettleList.keyAt(i);
201                     if (vh == item) {
202                         mPendingSettleList.removeAt(i);
203                         dispatchChangeFinished(item, true);
204                     }
205                 }
206             }
207 
208             @Override
209             public boolean animateChange(RecyclerView.ViewHolder oldHolder,
210                     RecyclerView.ViewHolder newHolder, ItemHolderInfo preInfo,
211                     ItemHolderInfo postInfo) {
212                 if (Build.VERSION.SDK_INT < Build.VERSION_CODES.HONEYCOMB_MR1
213                         || oldHolder != newHolder) {
214                     return super.animateChange(oldHolder, newHolder, preInfo, postInfo);
215                 }
216                 return animateChangeApiHoneycombMr1(oldHolder, newHolder, preInfo, postInfo);
217             }
218 
219             @TargetApi(Build.VERSION_CODES.HONEYCOMB_MR1)
220             private boolean animateChangeApiHoneycombMr1(RecyclerView.ViewHolder oldHolder,
221                     RecyclerView.ViewHolder newHolder,
222                     ItemHolderInfo preInfo, ItemHolderInfo postInfo) {
223                 endAnimation(oldHolder);
224                 MyItemInfo pre = (MyItemInfo) preInfo;
225                 MyItemInfo post = (MyItemInfo) postInfo;
226                 MyViewHolder vh = (MyViewHolder) oldHolder;
227 
228                 CharSequence finalText = post.text;
229 
230                 if (pre.text.equals(post.text)) {
231                     // same content. Just translate back to 0
232                     final long duration = (long) (getChangeDuration()
233                             * (ViewCompat.getTranslationX(vh.textView) / vh.textView.getWidth()));
234                     mPendingSettleList.put(vh, duration);
235                     // we set it here because previous endAnimation would set it to other value.
236                     vh.textView.setText(finalText);
237                 } else {
238                     // different content, get out and come back.
239                     vh.textView.setText(pre.text);
240                     final ItemChangeAnimator anim = new ItemChangeAnimator(vh, finalText,
241                             getChangeDuration()) {
242                         @Override
243                         public void onAnimationEnd(Animator animation) {
244                             setFraction(1f);
245                             dispatchChangeFinished(mViewHolder, true);
246                         }
247 
248                         @Override
249                         public void onAnimationStart(Animator animation) {
250                             dispatchChangeStarting(mViewHolder, true);
251                         }
252                     };
253                     mPendingChangeAnimations.add(anim);
254                 }
255                 return true;
256             }
257 
258             @Override
259             public ItemHolderInfo obtainHolderInfo() {
260                 return new MyItemInfo();
261             }
262         };
263     }
264 
265 
266     @TargetApi(Build.VERSION_CODES.HONEYCOMB_MR1)
267     abstract private static class ItemChangeAnimator implements
268             ValueAnimator.AnimatorUpdateListener, Animator.AnimatorListener {
269         CharSequence mFinalText;
270         ValueAnimator mValueAnimator;
271         MyViewHolder mViewHolder;
272         final float mMaxX;
273         final float mStartRatio;
ItemChangeAnimator(MyViewHolder viewHolder, CharSequence finalText, long duration)274         public ItemChangeAnimator(MyViewHolder viewHolder, CharSequence finalText, long duration) {
275             mViewHolder = viewHolder;
276             mMaxX = mViewHolder.itemView.getWidth();
277             mStartRatio = ViewCompat.getTranslationX(mViewHolder.textView) / mMaxX;
278             mFinalText = finalText;
279             mValueAnimator = ValueAnimator.ofFloat(0f, 1f);
280             mValueAnimator.addUpdateListener(this);
281             mValueAnimator.addListener(this);
282             mValueAnimator.setDuration(duration);
283             mValueAnimator.setTarget(mViewHolder.itemView);
284         }
285 
setFraction(float fraction)286         void setFraction(float fraction) {
287             fraction = mStartRatio + (1f - mStartRatio) * fraction;
288             if (fraction < .5f) {
289                 ViewCompat.setTranslationX(mViewHolder.textView, fraction * mMaxX);
290                 ViewCompat.setAlpha(mViewHolder.textView, 1f - fraction);
291             } else {
292                 ViewCompat.setTranslationX(mViewHolder.textView, (1f - fraction) * mMaxX);
293                 ViewCompat.setAlpha(mViewHolder.textView, fraction);
294                 maybeSetFinalText();
295             }
296         }
297 
298         @Override
onAnimationUpdate(ValueAnimator valueAnimator)299         public void onAnimationUpdate(ValueAnimator valueAnimator) {
300             setFraction(valueAnimator.getAnimatedFraction());
301         }
302 
start()303         public void start() {
304             mValueAnimator.start();
305         }
306 
307         @Override
onAnimationEnd(Animator animation)308         public void onAnimationEnd(Animator animation) {
309             maybeSetFinalText();
310             ViewCompat.setAlpha(mViewHolder.textView, 1f);
311         }
312 
maybeSetFinalText()313         public void maybeSetFinalText() {
314             if (mFinalText != null) {
315                 mViewHolder.textView.setText(mFinalText);
316                 mFinalText = null;
317             }
318         }
319 
end()320         public void end() {
321             mValueAnimator.cancel();
322         }
323 
324         @Override
onAnimationStart(Animator animation)325         public void onAnimationStart(Animator animation) {
326         }
327 
328         @Override
onAnimationCancel(Animator animation)329         public void onAnimationCancel(Animator animation) {
330         }
331 
332         @Override
onAnimationRepeat(Animator animation)333         public void onAnimationRepeat(Animator animation) {
334         }
335     }
336 
337     private static class MyItemInfo extends DefaultItemAnimator.ItemHolderInfo {
338         CharSequence text;
339     }
340 
341     @Override
onCreateOptionsMenu(Menu menu)342     public boolean onCreateOptionsMenu(Menu menu) {
343         super.onCreateOptionsMenu(menu);
344         MenuItemCompat.setShowAsAction(menu.add("Layout"), MenuItemCompat.SHOW_AS_ACTION_IF_ROOM);
345         return true;
346     }
347 
348     @Override
onOptionsItemSelected(MenuItem item)349     public boolean onOptionsItemSelected(MenuItem item) {
350         mRecyclerView.requestLayout();
351         return super.onOptionsItemSelected(item);
352     }
353 
354     @SuppressWarnings("unused")
checkboxClicked(View view)355     public void checkboxClicked(View view) {
356         ViewGroup parent = (ViewGroup) view.getParent();
357         boolean selected = ((CheckBox) view).isChecked();
358         MyViewHolder holder = (MyViewHolder) mRecyclerView.getChildViewHolder(parent);
359         mAdapter.selectItem(holder, selected);
360     }
361 
362     @SuppressWarnings("unused")
itemClicked(View view)363     public void itemClicked(View view) {
364         ViewGroup parent = (ViewGroup) view;
365         MyViewHolder holder = (MyViewHolder) mRecyclerView.getChildViewHolder(parent);
366         final int position = holder.getAdapterPosition();
367         if (position == RecyclerView.NO_POSITION) {
368             return;
369         }
370         mAdapter.toggleExpanded(holder);
371         mAdapter.notifyItemChanged(position);
372     }
373 
deleteSelectedItems(View view)374     public void deleteSelectedItems(View view) {
375         int numItems = mItems.size();
376         if (numItems > 0) {
377             for (int i = numItems - 1; i >= 0; --i) {
378                 final String itemText = mItems.get(i);
379                 boolean selected = mAdapter.mSelected.get(itemText);
380                 if (selected) {
381                     removeAtPosition(i);
382                 }
383             }
384         }
385     }
386 
generateNewText()387     private String generateNewText() {
388         return "Added Item #" + mNumItemsAdded++;
389     }
390 
d1a2d3(View view)391     public void d1a2d3(View view) {
392         removeAtPosition(1);
393         addAtPosition(2, "Added Item #" + mNumItemsAdded++);
394         removeAtPosition(3);
395     }
396 
removeAtPosition(int position)397     private void removeAtPosition(int position) {
398         if(position < mItems.size()) {
399             mItems.remove(position);
400             mAdapter.notifyItemRemoved(position);
401         }
402     }
403 
addAtPosition(int position, String text)404     private void addAtPosition(int position, String text) {
405         if (position > mItems.size()) {
406             position = mItems.size();
407         }
408         mItems.add(position, text);
409         mAdapter.mSelected.put(text, Boolean.FALSE);
410         mAdapter.mExpanded.put(text, Boolean.FALSE);
411         mAdapter.notifyItemInserted(position);
412     }
413 
addDeleteItem(View view)414     public void addDeleteItem(View view) {
415         addItem(view);
416         deleteSelectedItems(view);
417     }
418 
deleteAddItem(View view)419     public void deleteAddItem(View view) {
420         deleteSelectedItems(view);
421         addItem(view);
422     }
423 
addItem(View view)424     public void addItem(View view) {
425         addAtPosition(3, "Added Item #" + mNumItemsAdded++);
426     }
427 
428     /**
429      * A basic ListView-style LayoutManager.
430      */
431     class MyLayoutManager extends RecyclerView.LayoutManager {
432         private static final String TAG = "MyLayoutManager";
433         private int mFirstPosition;
434         private final int mScrollDistance;
435 
MyLayoutManager(Context c)436         public MyLayoutManager(Context c) {
437             final DisplayMetrics dm = c.getResources().getDisplayMetrics();
438             mScrollDistance = (int) (SCROLL_DISTANCE * dm.density + 0.5f);
439         }
440 
441         @Override
supportsPredictiveItemAnimations()442         public boolean supportsPredictiveItemAnimations() {
443             return mPredictiveAnimationsEnabled;
444         }
445 
446         @Override
onLayoutChildren(RecyclerView.Recycler recycler, RecyclerView.State state)447         public void onLayoutChildren(RecyclerView.Recycler recycler, RecyclerView.State state) {
448             int parentBottom = getHeight() - getPaddingBottom();
449 
450             final View oldTopView = getChildCount() > 0 ? getChildAt(0) : null;
451             int oldTop = getPaddingTop();
452             if (oldTopView != null) {
453                 oldTop = Math.min(oldTopView.getTop(), oldTop);
454             }
455 
456             // Note that we add everything to the scrap, but we do not clean it up;
457             // that is handled by the RecyclerView after this method returns
458             detachAndScrapAttachedViews(recycler);
459 
460             int top = oldTop;
461             int bottom = top;
462             final int left = getPaddingLeft();
463             final int right = getWidth() - getPaddingRight();
464 
465             int count = state.getItemCount();
466             for (int i = 0; mFirstPosition + i < count && top < parentBottom; i++, top = bottom) {
467                 View v = recycler.getViewForPosition(mFirstPosition + i);
468 
469                 RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) v.getLayoutParams();
470                 addView(v);
471                 measureChild(v, 0, 0);
472                 bottom = top + v.getMeasuredHeight();
473                 v.layout(left, top, right, bottom);
474                 if (mPredictiveAnimationsEnabled && params.isItemRemoved()) {
475                     parentBottom += v.getHeight();
476                 }
477             }
478 
479             if (mAnimationsEnabled && mPredictiveAnimationsEnabled && !state.isPreLayout()) {
480                 // Now that we've run a full layout, figure out which views were not used
481                 // (cached in previousViews). For each of these views, position it where
482                 // it would go, according to its position relative to the visible
483                 // positions in the list. This information will be used by RecyclerView to
484                 // record post-layout positions of these items for the purposes of animating them
485                 // out of view
486 
487                 View lastVisibleView = getChildAt(getChildCount() - 1);
488                 if (lastVisibleView != null) {
489                     RecyclerView.LayoutParams lastParams =
490                             (RecyclerView.LayoutParams) lastVisibleView.getLayoutParams();
491                     int lastPosition = lastParams.getViewLayoutPosition();
492                     final List<RecyclerView.ViewHolder> previousViews = recycler.getScrapList();
493                     count = previousViews.size();
494                     for (int i = 0; i < count; ++i) {
495                         View view = previousViews.get(i).itemView;
496                         RecyclerView.LayoutParams params =
497                                 (RecyclerView.LayoutParams) view.getLayoutParams();
498                         if (params.isItemRemoved()) {
499                             continue;
500                         }
501                         int position = params.getViewLayoutPosition();
502                         int newTop;
503                         if (position < mFirstPosition) {
504                             newTop = view.getHeight() * (position - mFirstPosition);
505                         } else {
506                             newTop = lastVisibleView.getTop() + view.getHeight() *
507                                     (position - lastPosition);
508                         }
509                         view.offsetTopAndBottom(newTop - view.getTop());
510                     }
511                 }
512             }
513         }
514 
515         @Override
generateDefaultLayoutParams()516         public RecyclerView.LayoutParams generateDefaultLayoutParams() {
517             return new RecyclerView.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
518                     ViewGroup.LayoutParams.WRAP_CONTENT);
519         }
520 
521         @Override
canScrollVertically()522         public boolean canScrollVertically() {
523             return true;
524         }
525 
526         @Override
scrollVerticallyBy(int dy, RecyclerView.Recycler recycler, RecyclerView.State state)527         public int scrollVerticallyBy(int dy, RecyclerView.Recycler recycler,
528                 RecyclerView.State state) {
529             if (getChildCount() == 0) {
530                 return 0;
531             }
532 
533             int scrolled = 0;
534             final int left = getPaddingLeft();
535             final int right = getWidth() - getPaddingRight();
536             if (dy < 0) {
537                 while (scrolled > dy) {
538                     final View topView = getChildAt(0);
539                     final int hangingTop = Math.max(-topView.getTop(), 0);
540                     final int scrollBy = Math.min(scrolled - dy, hangingTop);
541                     scrolled -= scrollBy;
542                     offsetChildrenVertical(scrollBy);
543                     if (mFirstPosition > 0 && scrolled > dy) {
544                         mFirstPosition--;
545                         View v = recycler.getViewForPosition(mFirstPosition);
546                         addView(v, 0);
547                         measureChild(v, 0, 0);
548                         final int bottom = topView.getTop(); // TODO decorated top?
549                         final int top = bottom - v.getMeasuredHeight();
550                         v.layout(left, top, right, bottom);
551                     } else {
552                         break;
553                     }
554                 }
555             } else if (dy > 0) {
556                 final int parentHeight = getHeight();
557                 while (scrolled < dy) {
558                     final View bottomView = getChildAt(getChildCount() - 1);
559                     final int hangingBottom = Math.max(bottomView.getBottom() - parentHeight, 0);
560                     final int scrollBy = -Math.min(dy - scrolled, hangingBottom);
561                     scrolled -= scrollBy;
562                     offsetChildrenVertical(scrollBy);
563                     if (scrolled < dy && state.getItemCount() > mFirstPosition + getChildCount()) {
564                         View v = recycler.getViewForPosition(mFirstPosition + getChildCount());
565                         final int top = getChildAt(getChildCount() - 1).getBottom();
566                         addView(v);
567                         measureChild(v, 0, 0);
568                         final int bottom = top + v.getMeasuredHeight();
569                         v.layout(left, top, right, bottom);
570                     } else {
571                         break;
572                     }
573                 }
574             }
575             recycleViewsOutOfBounds(recycler);
576             return scrolled;
577         }
578 
579         @Override
onFocusSearchFailed(View focused, int direction, RecyclerView.Recycler recycler, RecyclerView.State state)580         public View onFocusSearchFailed(View focused, int direction,
581                 RecyclerView.Recycler recycler, RecyclerView.State state) {
582             final int oldCount = getChildCount();
583 
584             if (oldCount == 0) {
585                 return null;
586             }
587 
588             final int left = getPaddingLeft();
589             final int right = getWidth() - getPaddingRight();
590 
591             View toFocus = null;
592             int newViewsHeight = 0;
593             if (direction == View.FOCUS_UP || direction == View.FOCUS_BACKWARD) {
594                 while (mFirstPosition > 0 && newViewsHeight < mScrollDistance) {
595                     mFirstPosition--;
596                     View v = recycler.getViewForPosition(mFirstPosition);
597                     final int bottom = getChildAt(0).getTop(); // TODO decorated top?
598                     addView(v, 0);
599                     measureChild(v, 0, 0);
600                     final int top = bottom - v.getMeasuredHeight();
601                     v.layout(left, top, right, bottom);
602                     if (v.isFocusable()) {
603                         toFocus = v;
604                         break;
605                     }
606                 }
607             }
608             if (direction == View.FOCUS_DOWN || direction == View.FOCUS_FORWARD) {
609                 while (mFirstPosition + getChildCount() < state.getItemCount() &&
610                         newViewsHeight < mScrollDistance) {
611                     View v = recycler.getViewForPosition(mFirstPosition + getChildCount());
612                     final int top = getChildAt(getChildCount() - 1).getBottom();
613                     addView(v);
614                     measureChild(v, 0, 0);
615                     final int bottom = top + v.getMeasuredHeight();
616                     v.layout(left, top, right, bottom);
617                     if (v.isFocusable()) {
618                         toFocus = v;
619                         break;
620                     }
621                 }
622             }
623 
624             return toFocus;
625         }
626 
recycleViewsOutOfBounds(RecyclerView.Recycler recycler)627         public void recycleViewsOutOfBounds(RecyclerView.Recycler recycler) {
628             final int childCount = getChildCount();
629             final int parentWidth = getWidth();
630             final int parentHeight = getHeight();
631             boolean foundFirst = false;
632             int first = 0;
633             int last = 0;
634             for (int i = 0; i < childCount; i++) {
635                 final View v = getChildAt(i);
636                 if (v.hasFocus() || (v.getRight() >= 0 && v.getLeft() <= parentWidth &&
637                         v.getBottom() >= 0 && v.getTop() <= parentHeight)) {
638                     if (!foundFirst) {
639                         first = i;
640                         foundFirst = true;
641                     }
642                     last = i;
643                 }
644             }
645             for (int i = childCount - 1; i > last; i--) {
646                 removeAndRecycleViewAt(i, recycler);
647             }
648             for (int i = first - 1; i >= 0; i--) {
649                 removeAndRecycleViewAt(i, recycler);
650             }
651             if (getChildCount() == 0) {
652                 mFirstPosition = 0;
653             } else {
654                 mFirstPosition += first;
655             }
656         }
657 
658         @Override
onItemsAdded(RecyclerView recyclerView, int positionStart, int itemCount)659         public void onItemsAdded(RecyclerView recyclerView, int positionStart, int itemCount) {
660             if (positionStart < mFirstPosition) {
661                 mFirstPosition += itemCount;
662             }
663         }
664 
665         @Override
onItemsRemoved(RecyclerView recyclerView, int positionStart, int itemCount)666         public void onItemsRemoved(RecyclerView recyclerView, int positionStart, int itemCount) {
667             if (positionStart < mFirstPosition) {
668                 mFirstPosition -= itemCount;
669             }
670         }
671     }
672 
673     class MyAdapter extends RecyclerView.Adapter {
674         private int mBackground;
675         List<String> mData;
676         ArrayMap<String, Boolean> mSelected = new ArrayMap<String, Boolean>();
677         ArrayMap<String, Boolean> mExpanded = new ArrayMap<String, Boolean>();
678 
MyAdapter(List<String> data)679         public MyAdapter(List<String> data) {
680             TypedValue val = new TypedValue();
681             AnimatedRecyclerView.this.getTheme().resolveAttribute(
682                     R.attr.selectableItemBackground, val, true);
683             mBackground = val.resourceId;
684             mData = data;
685             for (String itemText : mData) {
686                 mSelected.put(itemText, Boolean.FALSE);
687                 mExpanded.put(itemText, Boolean.FALSE);
688             }
689         }
690 
691         @Override
onCreateViewHolder(ViewGroup parent, int viewType)692         public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
693             MyViewHolder h = new MyViewHolder(getLayoutInflater().inflate(R.layout.selectable_item,
694                     null));
695             h.textView.setMinimumHeight(128);
696             h.textView.setFocusable(true);
697             h.textView.setBackgroundResource(mBackground);
698             return h;
699         }
700 
701         @Override
onBindViewHolder(RecyclerView.ViewHolder holder, int position)702         public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
703             String itemText = mData.get(position);
704             MyViewHolder myViewHolder = (MyViewHolder) holder;
705             myViewHolder.boundText = itemText;
706             myViewHolder.textView.setText(itemText);
707             boolean selected = false;
708             if (mSelected.get(itemText) != null) {
709                 selected = mSelected.get(itemText);
710             }
711             myViewHolder.checkBox.setChecked(selected);
712             Boolean expanded = mExpanded.get(itemText);
713             if (Boolean.TRUE.equals(expanded)) {
714                 myViewHolder.textView.setText("More text for the expanded version");
715             } else {
716                 myViewHolder.textView.setText(itemText);
717             }
718         }
719 
720         @Override
getItemCount()721         public int getItemCount() {
722             return mData.size();
723         }
724 
selectItem(MyViewHolder holder, boolean selected)725         public void selectItem(MyViewHolder holder, boolean selected) {
726             mSelected.put(holder.boundText, selected);
727         }
728 
toggleExpanded(MyViewHolder holder)729         public void toggleExpanded(MyViewHolder holder) {
730             mExpanded.put(holder.boundText, !mExpanded.get(holder.boundText));
731         }
732     }
733 
734     static class MyViewHolder extends RecyclerView.ViewHolder {
735         public TextView textView;
736         public CheckBox checkBox;
737         public String boundText;
738 
MyViewHolder(View v)739         public MyViewHolder(View v) {
740             super(v);
741             textView = (TextView) v.findViewById(R.id.text);
742             checkBox = (CheckBox) v.findViewById(R.id.selected);
743         }
744 
745         @Override
toString()746         public String toString() {
747             return super.toString() + " \"" + textView.getText() + "\"";
748         }
749     }
750 }
751