[Exclusive for lazy people] Universal RecylerAdapter, built-in XRecyclerView, compatible with pull-up and pull-down and animation, high reuse, universal for all pages, support empty pages

[Exclusive for lazy people] Universal RecylerAdapter, built-in XRecyclerView, compatible with pull-up and pull-down and animation, high reuse, universal for all pages, support empty pages

Hello everyone, Guo Laoshi is here again. I used to read articles, but now I finally understand how hard it is to write. As a taciturn programmer, I deeply admire the predecessors who have written countless articles ((/- -)/).

I believe everyone has heard of RecylerView (are you sure you haven't heard of it = =), in the era when ListView was rampant, RecyclerView came into our sight with mixed reviews. At that time, I just started a new project and took it to practice. The following is an introduction to the usage process. It is recommended to play with the demo. ( ・᷄-・᷅ )

Normally, for each different list, we often need to implement different Adapters to handle the corresponding logic, which leads to a lot of duplicate code. Driven by the motivation of optimizing code (laziness), I implemented a universal Adapter. The Holder mentioned later can be understood as an Item in the list, belonging to its logic processing class, and each type of Item has a Holder.

With only one Adapter, you can implement various types of lists, which are compatible with different types of Items in one list. All you need to do is maintain your Holder (similar to an Item in a List) and Model, without having to worry about anything else. This allows for high reuse and multi-style logic, as well as support for custom animations and multiple pull-up and pull-down implementations. You don't need to write any Adapter code (^o^)/.

1. CommonRecyclerManager: bind layoutId and your Holder class name.

This management class is used to bind Holder and R.layout.xxx, so that CommonRecyclerAdapter can use it to find the corresponding Holder and create it through the layoutId of the data Model.

  1. //Associate the layout ID and holder type commonRecyclerManager.addType(TextHolder.ID, TextHolder.class.getName());

2. RecyclerBaseHolder: Inherit this Holder to meet your needs.

RecyclerBaseHolder is the base class of all Holders. It inherits RecyclerView.ViewHolder and defines two methods, so you just inherit it. Find the control when createView, and read the data to fill the screen in onBind. This is where your dream comes true!

  1. //The implemented hodler inherits RecyclerBaseHolder and overloads the following method to meet your needs
  2. public class TextHolder extends RecyclerBaseHolder {
  3. //Layout id, I usually write the ids that this Holder needs to handle here for easy management
  4. public final static   int ID = R.layout.text_item;
  5. @BindView(R.id.item_text)
  6. TextView itemText;
  7.  
  8. public TextHolder(Context context, View v) {
  9. super(context, v);
  10. }
  11.  
  12. /iew created
  13. @Override
  14. public void createView( View v) {
  15. ButterKnife.bind(this, v);
  16. }
  17.  
  18. /iew has been created, with more data processing logic
  19. @Override
  20. public void onBind(RecyclerBaseModel model, int position) {
  21. //Convert to your model
  22. TextModel textModel = (TextModel) (model);
  23. itemText.setText(textModel.getText());
  24. }
  25.  
  26. //No need to write
  27. @Override
  28. public AnimatorSet getAnimator( View   view ) {
  29. // Implement your animation
  30. return   null ;
  31. }
  32. }

3. CommonRecyclerAdapter: Common adapter

You only need to pass in the data List and CommonRecyclerManager, and it will automatically generate the corresponding Holder in the RecyclerView according to the order of the Model and the layoutId of the data. Other functions only require simple configuration.

  1. //Create with data and manager
  2.  
  3. adapteradapter = new CommonRecyclerAdapter(getActivity(), commonRecyclerManager, datas);
  4.  
  5.  
  6. //Support needs to load more
  7.  
  8. adapter.setNeedLoadMore( true );
  9.  
  10.  
  11. //Support empty data display empty page
  12.  
  13. adapter.setShowNoData( true );
  14.  
  15.  
  16. //Set item display animation support to open
  17.  
  18. adapter.setNeedAnimation( true );

4. RecyclerBaseModel: The accumulation of data models must inherit it and never abandon it.

The purpose of inheriting it is that, because the entire Adapter is based on it as the base class, you need to inherit it. Ultimately, you need the layout ID corresponding to this Model so that it can find its own Holder.

  1. //Inherit RecyclerBaseModel to implement the data type you need public class TextModel extends RecyclerBaseModel {
  2. private String text = "" ;
  3.  
  4. public String getText() { return text;
  5. }
  6.  
  7. public void setText(String text) { this.text = text;
  8. }
  9. }

To sum up,

1. Implement your Holder and inherit RecyclerBaseHolder. This is where you implement your needs, which is equivalent to the logic of Item.

2. Let your data model inherit RecyclerBaseModel and set the Model's LayoutId (very important), so that the model will find the Holder corresponding to the LayoutId through CommonRecyclerManager and generate it.

3. You need a CommonRecyclerManager to bind your LayoutId and the Holder class name that handles this layout.

4. Generate CommonRecyclerAdapter through CommonRecyclerManager and Model's data list.

5. Give the Adapter to the Recycler.

Does the logic look a bit complicated? In fact, it is the LayoutId of the model. CommonRecyclerManager associates the Holder that handles it. In this way, we only need to set different LayoutId models according to the data in the data List, and the Adapter will automatically match the corresponding Holder.

In this way, you only need to implement the Holder and assemble the Model, and any list can be used without writing the Adapter logic. According to the order of the model, the Adapter automatically generates the corresponding Holder, and the same Holder can be bound to different LayoutIds. In the future, you only need to maintain and be compatible with your Holder, and use your holder logic in all lists. Isn't the code much cleaner in an instant?

Pull down to refresh and pull up to load more

For ordinary lists, you can just use the system's SwipeRefreshLayout, which is simple and easy to use. To pull down to load more, just add the following method and easily pull up and down to refresh <( ̄︶ ̄), which is simple and crude. Just remember to add a lock to avoid repeated entry.

  1. // Turn on support for loading more adapter.setNeedLoadMore( true );
  2.  
  3. recycler.addOnScrollListener(new LoadMoreScrollListener() { @Override
  4. public void onLoadMore() { //Note locking
  5. if (!isLoadMore) {
  6. isLoadMore = true ;
  7. recycler.postDelayed(new Runnable() { @Override
  8. public void run() {
  9. isLoadMore = false ;
  10. loadMore();
  11. }
  12.  
  13. }, 2000);
  14. }
  15. } //Which item is currently visible?
  16. @Override
  17. public void onScrolled( int firstPosition) {
  18. }
  19. });

Other Configuration

You can also configure whether to display animation effects, configure the color of the pull-up loading, single click and long press, etc., see below.

  1. //Support empty data display empty page
  2.  
  3. adapter.setShowNoData( true );
  4.  
  5.  
  6. //Display empty data model, do not set the default empty page
  7.  
  8. adapter.setNoDataModel(noDataModel);
  9.  
  10.  
  11. //Display the layout of the empty data page, do not set the display default, the layout id needs to be associated with the hodler through CommonRecyclerManager
  12.  
  13. adapter.setNoDataLayoutId(noDataLayoutId);
  14.  
  15.  
  16. //Set animation support to open
  17.  
  18. adapter.setNeedAnimation( true );
  19.  
  20.  
  21. //Add click
  22.  
  23. adapter.setOnItemClickListener();
  24.  
  25. XRecyclerView compatible support

XRecyclerView is added and modified here. XRecyclerView has a built-in internal Adapter, which supports adding headers, built-in pull-down controls, and fully supports CommonRecyclerAdapter after some adjustments.

No need to listen to sliding, no need for SwipeRefreshLayout, easy to add refresh and load more. It also supports dynamic configuration, various styles of pull-up and pull-down support, specifically under ProgressStyle, there are multiple types of support configuration, which solves the problem that Adapter is not compatible with waterfall flow pull-up.

The usage here is the same as that of ordinary RecyclerView. It supports cooperation with CommonRecyclerAdapter, and it also supports displaying empty pages and adding various headers. The only thing you need to pay attention to is that when adding the dividing line class addItemDecoration and clicking, you need to convert the added header and the absolute position of the refresh into a relative position. I once made myself cry here ╥﹏╥... Let's look at the code below.

  1. //Whether to block the pull-down
  2.  
  3. //xRecycler.setPullRefreshEnabled( false );
  4.  
  5. //Pull up to load more styles, you can also set pull down
  6.  
  7. xRecycler.setLoadingMoreProgressStyle(ProgressStyle.SysProgress);
  8.  
  9. //Set the manager to associate the layout with the holder class name. Different ids can manage one holder.
  10.  
  11. CommonRecyclerManager commonRecyclerManager = new CommonRecyclerManager();
  12.  
  13. commonRecyclerManager.addType(ImageHolder.ID, ImageHolder.class.getName());
  14.  
  15. commonRecyclerManager.addType(TextHolder.ID, TextHolder.class.getName());
  16.  
  17. commonRecyclerManager.addType(ClickHolder.ID, ClickHolder.class.getName());
  18.  
  19. // Initialize the general manager
  20.  
  21. commonRecyclerAdapter = new CommonRecyclerAdapter(getActivity(), commonRecyclerManager, dataList);
  22.  
  23. xRecycler.setAdapter(commonRecyclerAdapter);
  24.  
  25.  
  26. ImageView imageView = new ImageView(getActivity());
  27.  
  28. imageView.setImageResource(R.drawable.xxx1);
  29.  
  30. imageView.setScaleType(ImageView.ScaleType.CENTER_CROP);
  31.  
  32. imageView.setMinimumHeight(dip2px(getActivity(), 100));
  33.  
  34. //Add header
  35.  
  36. xRecycler.addHeaderView(imageView);
  37.  
  38. //It also supports setting empty local
  39.  
  40. //xRecycler.setEmptyView();
  41.  
  42. xRecycler.setLoadingListener(new XRecyclerView.LoadingListener() {
  43.  
  44. @Override
  45.  
  46. public void onRefresh() {
  47.  
  48. xRecycler.postDelayed(new Runnable() {
  49.  
  50. @Override
  51.  
  52. public void run() {
  53.  
  54. xRecycler.refreshComplete();
  55.  
  56. }
  57.  
  58. }, 2000);
  59.  
  60. }
  61.  
  62.  
  63. @Override
  64.  
  65. public void onLoadMore() {
  66.  
  67. xRecycler.postDelayed(new Runnable() {
  68.  
  69. @Override
  70.  
  71. public void run() {
  72.  
  73. loadMore();
  74.  
  75. }
  76.  
  77. }, 2000);
  78.  
  79. }
  80.  
  81. });
  82.  
  83.  
  84. commonRecyclerAdapter.setOnItemClickListener(new OnItemClickListener() {
  85.  
  86. @Override
  87.  
  88. public void onItemClick(Context context, int position) {
  89.  
  90. //You need to subtract the number of your header and refreshed views
  91.  
  92. Toast.makeText(getActivity(), "Clicked!! " + (position - 2), Toast.LENGTH_SHORT).show();
  93.  
  94. }
  95.  
  96. });

***

You should have known the general usage by now. The detailed internal implementation can be viewed through DEMO. The general logic is that CommonRecyclerManager associates layoutId and Holder class name, CommonRecyclerAdapter finds the Holder through the layoutId of the Model, and then creates the view with the layoutId, and passes the View, position, and model into the Holder to fill in the data. The layoutId is also *type id**, note:

When using it, remember to set setResLayoutId() for your model, which is the most easily forgotten.

<<:  Aite Tribe Stories (2): The Road to Transformation Caused by Chance

>>:  Android Permission Management Principles (including 6.0)

Recommend

APP developers, 4 major changes seen at Apple WWDC

Apple's annual Worldwide Developers Conferenc...

Is it useful to use dou+ if TikTok has no views? Is Douyin DOU+ useful?

When operating a Douyin account, there may actual...

50 million alliance: 0 fans bring over 10,000 GMV to the community

Community introduction: How to quickly get more t...

Competitive product analysis report: Douyu VS Huya

Live streaming is a carnival for a group of lonel...

How to get into Google

When I first joined Google about two years ago, I...

The most popular iOS automation testing tools in 2022

The year-over-year growth in iOS device sales has...

Three channels for App operation and promotion: online, offline and new media.

App operation is not easy, which depends on produ...

E-commerce live streaming practical process summary

The market is very good, but you need to be cauti...

Himalaya FM audio traffic promotion plan

Himalaya FM is a well-known audio sharing platfor...

Three steps to create a valuable corporate WeChat public account

When mini programs are very popular, talking abou...