/expandable-recycler-view

Expandable Recycler View

Primary LanguageJava

Expandable RecyclerView

Build Status

Android Arsenal

About

The Expandable RecyclerView is a library written to allow for an expanded view to be attached to each ViewHolder. To allow for full functionality of a normal RecyclerView in both the parent ViewHolder and the expanded child ViewHolder, the recyclerview has been modified to use two types of ViewHolders, a child and a parent with the ability to customize each separately.

##Project Setup Gradle

Simply add this to your app's build.gradle:

compile 'com.bignerdranch.android:expandablerecyclerview:1.0.3'

You can also clone the project and add it as a module to your project.

Clone with:

git clone git@github.com:bignerdranch/expandable-recycler-view.git

Then, navigate inside the directory, clean and build from Android Studio or with

./gradlew clean app:assemble

Now run the sample app on any device or emulator/simulator to view the basic functionality of the list. Code for the sample is located under /app/src/main. All library code is located under /expandablerecyclerview/src/main. The CriminalIntent sample is located under /criminalintentsample/src/main.

Overview

Expandable RecyclerView can be used with any stock Android RecyclerView.

What you need to implement:

  • A custom adapter that extends ExpandableRecyclerAdapter
  • Two custom ViewHolders: a parent ViewHolder that extends ParentViewHolder and a child ViewHolder that extends ChildViewHolder
  • The list of objects you wish to display in your RecyclerView must extend ParentObject.
    • It is best practice to separate your child data into its own Object, although it is not required.
  • A parent layout and a child layout

Tutorial

I have written an in-depth tutorial but it has yet to be published yet. The link will be posted here when available.

Javadocs for the library and sample are available here.

Usage

First, create a stock RecyclerView in your layout file and inflate it in your activity/fragment as you would usually do.

Next, create an adapter class that extends ExpandableRecyclerAdapter. Implement the required methods.

Then, create two ViewHolders and their respective layouts. One ViewHolder must extend ParentViewHolder and the other must extend ChildViewHolder. Create their respective views and create variables in these ViewHolders to access those views.

Next, the Object that contains the data to be displayed in your RecyclerView must extend ParentObject

In onCreate or onCreateView of your activity or fragment, create and attach your custom expandable adapter like so:

RecyclerView mRecyclerView = (RecyclerView) findViewById(YOUR RECYCLERVIEW ID);
MyExpandable mExpandableAdapter = new MyExpandableAdapter(getActivity(), YOUR ParentObject LIST);
mRecyclerView.setAdapter(mExpandableAdapter);
mRecyclerView.setLayoutManager(new LinearLayoutManager(this));

Inside your ExpandableRecyclerAdapter, you can create and bind your Parent and Child ViewHolders just as you would create and bind ViewHolders in a normal RecyclerView.

Then, define a parent object and implement ParentObject. It is also best practice to create a separate child object to store any data that you need to display in the child view, but it is not required. When you implement ParentObject, you need to create an instance variable, List, to store all the children of the parent object in. The list must be type casted to Object. If it is null or empty, no child will be shown.

public class MyCustomParentObject implements ParentObject {
    private List<Object> mChildObjectList;
    
    /**
     * Your constructors, variables, data and methods for your Object go here
     */
    
    @Override
    public List<Object> getChildObject() {
        /**
         * You can either return a newly created list of children here or attach them later
         */
     
        return mChildObjectList;
    }
    
    @Override
    public void setChildObject(List<Object> childObjectList) {
        mChildObjectList = childObjectList;
    }
}

When generating the list of parent objects, you should attach all children to them there. If the children share data with your ParentObject, you can simply create a list of children in the constructor for your parent object or in the getter method for the list.

Extras

You can define a custom button, image or view to trigger the expansion rather than clicking the whole item (default). To do this, in your activity or fragment, call myCustomExpandingAdapter.setCustomClickableView(Your Custom View ID) and pass in the id.

If you do set a custom clickable view, you can also set an animation for the view to rotate 180 degrees when expanding and collapsing. This is useful primarily with arrows which signifies for the user to click it to change the expansion. By default the rotation is off. You can enable rotation by calling myCustomExpandingAdapter.setRotation(long durationInMS) in the constructor, below where you called setCustomClickableView(). When setting the rotation, you must pass in a duration in Milliseconds. If you'd like to use the default rotation duration rather than defining your own, call myCustomExpandingAdapter.setParentClickableViewAnimationDefaultDuration() instead. The default rotation duration is 200 ms.

After implementing these, in the activity or fragment that is holding your RecyclerView, simply set the adapter to your custom adapter, and set the layout manager to a new LinearLayoutManager. An example is here:

MyCustomExpandingAdapter myCustomExpandingAdapter = new MyCustomExpandingAdapter(this, objectList);

// Optional animation configuration goes here

mRecyclerView.setAdapter(myCustomExpandingAdapter);
mRecyclerView.setLayoutManager(new LinearLayoutManager(this));

Listening for Expansion and Collapsing

You can listen for expansion and collapsing events by implementing ExpandCollapseListener in the activity or fragment hosting your RecyclerView. Two methods will be added, onRecyclerViewItemExpanded(int position) and onRecyclerViewItemCollapsed(int position). This will allow you to listen for expansion and collapsing of ParentObjects. The position passed into these methods is the position of the item in the ParentObject list. Any expanded children before the item are not included in that position integer.

As an example, let's say we implemented ExpandCollapseListener in an activity. For this to work, after creating the adapter and before setting the RecyclerView's adapter, we must call addExpandCollapseListener(ExpandCollapseListener yourExpandCollapseListener on the adapter:

 MyCustomExpandingAdapter myCustomExpandingAdapter = new MyCustomExpandingAdapter(this, objectList);
 myCustomExpandingAdapter.addExpandCollapseListener(this);
 mRecyclerView.setAdapter(myCustomExpandingAdapter);

Saving Expanded States onResume() or on Rotation

To save expanded/collapsed states, inside onSaveInstanceState(Bundle outState) of your activity or fragment, call myCustomExpandingAdapter.onSaveInstanceState(outState). In onRestoreInstanceState(Bundle savedInstanceState(Bundle savedInstanceState), call myCustomExpandingAdapter.onRestoreInstanceState(savedInstanceState). Here is an example of how to override in your activity or fragment:

 @Override
 protected void onSaveInstanceState(Bundle outState) {
     super.onSaveInstanceState(outState);
     outState = myCustomExpandingAdapter.onSaveInstanceState(outState);
 }

 @Override
 protected void onRestoreInstanceState(Bundle savedInstanceState) {
     super.onRestoreInstanceState(savedInstanceState);
     mExpandableAdapter.onRestoreInstanceState(savedInstanceState);
 }

You can also check out the two sample applications for a full working demo.

License

  The MIT License
  
  Copyright (c) 2015 Big Nerd Ranch
  
  Permission is hereby granted, free of charge, to any person obtaining a copy
  of this software and associated documentation files (the "Software"), to deal
  in the Software without restriction, including without limitation the rights
  to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  copies of the Software, and to permit persons to whom the Software is
  furnished to do so, subject to the following conditions:
  
  The above copyright notice and this permission notice shall be included in
  all copies or substantial portions of the Software.
  
  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  THE SOFTWARE.