Divide elements on groups in RecyclerView

AndroidMaterial DesignAndroid Recyclerview

Android Problem Overview


I need to divide elements in RecyclerView on groups with titles (like in the Inbox app on the picture below) so help me please to figure out what approach would be better for my case:

  1. I can use Heterogenous layouts for it but it is not so convenient to insert new elements in groups (because I need check if elements of the same group is already added or I need to add new divider). So in this case I'll wrap all operations with such data structure into a separate class.

  2. Theoretically I can wrap each group in its own RecyclerView with label is it a good idea?

Inbox app

Android Solutions


Solution 1 - Android

For example you can:

  1. Use a TreeMap<Date,List<Event>> for splitting elements by date. This will be a collection for keeping your business objects. Of course if you already have a similar structure you can keep it. It's just important to have something for easily building list of items for populating UI with right elements order.

  2. Define a dedicated abstract type for List items (e.g. ListItem) to wrap your business objects. Its implementation could be something like this:

     public abstract class ListItem {
    
         public static final int TYPE_HEADER = 0;
         public static final int TYPE_EVENT = 1;
    
         abstract public int getType();
     } 
    
  3. Define a class for each of your List element type (here I added just two types but you can use many as you need):

     public class HeaderItem extends ListItem {
    
         private Date date;
    
         // here getters and setters 
         // for title and so on, built
         // using date
    
         @Override
         public int getType() {
     	    return TYPE_HEADER;
         }
    
     }
    
     public class EventItem extends ListItem {
    
         private Event event;
    
         // here getters and setters 
         // for title and so on, built 
         // using event
    
         @Override
         public int getType() {
     	    return TYPE_EVENT;
         }
    
     }
    
  4. Create a List as follows (where mEventsMap is map build at point 1):

     List<ListItem> mItems;
     // ...
     mItems = new ArrayList<>();
     for (Date date : mEventsMap.keySet()) {
         HeaderItem header = new HeaderItem();
         header.setDate(date); 
         mItems.add(header);
         for (Event event : mEventsMap.get(date)) {
     	    EventItem item = new EventItem();
     	    item.setEvent(event);
     	    mItems.add(item);
         }
     }
    
  5. Define an adapter for your RecyclerView, working on List defined at point 4. Here what is important is to override getItemViewType method as follows:

     @Override
     public int getItemViewType(int position) {
         return mItems.get(position).getType();
     }
    

    Then you need to have two layouts and ViewHolder for header and event items. Adapter methods should take care of this accordingly:

     @Override
     public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
     	if (viewType == ListItem.TYPE_HEADER) {
         	View itemView = mLayoutInflater.inflate(R.layout.view_list_item_header, parent, false);
         	return new HeaderViewHolder(itemView);
     	} else {
         	View itemView = mLayoutInflater.inflate(R.layout.view_list_item_event, parent, false);
         	return new EventViewHolder(itemView);
     	}
     }
    
    
     @Override
     public void onBindViewHolder(final RecyclerView.ViewHolder viewHolder, final int position) {
         int type = getItemViewType(position);
         if (type == ListItem.TYPE_HEADER) {
             HeaderItem header = (HeaderItem) mItems.get(position);
             HeaderViewHolder holder = (HeaderViewHolder) viewHolder;
             // your logic here
         } else {            
         	EventItem event = (EventItem) mItems.get(position);
             EventViewHolder holder = (EventViewHolder) viewHolder;
             // your logic here
         }
     }
    

Here it is a repository on GitHub providing an implementation of the approach explained above.

Solution 2 - Android

You can try to use the library I've wrote to solve this problem in my project. Gradle dependency (needs jcenter repo included):

dependencies {
    //your other dependencies
    compile 'su.j2e:rv-joiner:1.0.3'//latest version by now
}

Then, in your situation, you can do smth like this:

//init your RecyclerView as usual
RecyclerView rv = (RecyclerView) findViewById(R.id.rv);
rv.setLayoutManager(new LinearLayoutManager(this));

//construct a joiner
RvJoiner rvJoiner = new RvJoiner();
rvJoiner.add(new JoinableLayout(R.layout.today));
YourAdapter todayAdapter = new YourAdapter();
rvJoiner.add(new JoinableAdapter(todayAdapter));
rvJoiner.add(new JoinableLayout(R.layout.yesterday));
YourAdapter yesterdayAdapter = new YourAdapter();
rvJoiner.add(new JoinableAdapter(yesterdayAdapter));

//set join adapter to your RecyclerView
rv.setAdapter(rvJoiner.getAdapter());

When you need to add item, add it to appropriate adapter, like:

if (timeIsToday) {
    todayAdapter.addItem(item);//or other func you've written
} else if (timeIsYesterday) {
    yesterdayAdapter.addItem(item);
}

If you need to add new group to recycler view dynamically, you can use this methods:

rvJoiner.add(new JoinableLayout(R.layout.tomorrow));
YourAdapter tomorrowAdapter = new YourAdapter();
rvJoiner.add(new JoinableAdapter(tomorrowAdapter));

You can check this link for more library description. I can't say that it's surely the best way to achieve you goal, but it helps me sometimes.

UPD:

I've found the way to do this without using external libraries. Use RecyclerView.ItemDecoration class. For example, to group items by 3 item in group you can do this:

recyclerView.addItemDecoration(new RecyclerView.ItemDecoration() {

		private int textSize = 50;
		private int groupSpacing = 100;
		private int itemsInGroup = 3;

		private Paint paint = new Paint();
		{
			paint.setTextSize(textSize);
		}

		@Override
		public void onDrawOver(Canvas c, RecyclerView parent, RecyclerView.State state) {
			for (int i = 0; i < parent.getChildCount(); i++) {
				View view = parent.getChildAt(i);
				int position = parent.getChildAdapterPosition(view);
				if (position % itemsInGroup == 0) {
					c.drawText("Group " + (position / itemsInGroup + 1), view.getLeft(),
							view.getTop() - groupSpacing / 2 + textSize / 3, paint);
				}
			}
		}

		@Override
		public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
			if (parent.getChildAdapterPosition(view) % itemsInGroup == 0) {
				outRect.set(0, groupSpacing, 0, 0);
			}
		}
	});

Hope it helps.

Solution 3 - Android

This question is from back in 2016, in the meantime (2020) there are different libraries available for grouping recycler views. The most popular ones:

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestionLeoView Question on Stackoverflow
Solution 1 - AndroidthetonriflesView Answer on Stackoverflow
Solution 2 - Androidj2esuView Answer on Stackoverflow
Solution 3 - AndroidbytesculptorView Answer on Stackoverflow