优雅地刷新RecyclerView

RafaelBroad 8年前
   <h2>前言</h2>    <p>还是那句话,RecyclerView的viewType增多,逻辑变复杂,几个月后,你确定还能理清思路吗?假设我们服务端是多个接口返回数据,你确定能正确刷新相应type吗?想一个RecyclerView高效快捷管理整个界面吗?你还在使用notifyDataSetChanged无脑刷新吗?你想单个viewType在loadingView,dataView,errorView自如切换吗?如果你迟疑了,那你不妨试试本库。</p>    <h2>特点</h2>    <ul>     <li>与Adapter为组合关系,可配合大多数Adapter</li>     <li>一行代码刷新相应viewType</li>     <li>支持非死book的shimmer加载效果</li>     <li>支持粘性头</li>     <li>支持异步刷新,可扩展(如配合RxAndroid)</li>     <li>支持加载相应type错误页面</li>     <li>支持高频率刷新(流畅,异步执行)</li>    </ul>    <h2>效果</h2>    <h3>线性排布</h3>    <p><img src="https://simg.open-open.com/show/c3a66ecc775fbbb380d7c59f501460a5.gif"></p>    <h3>方格排布</h3>    <p><img src="https://simg.open-open.com/show/e88c7f43930c46f39a9abb8e4fb252c7.gif"></p>    <h3>关键字高亮</h3>    <p><img src="https://simg.open-open.com/show/a0480696ccbef66f4a7f7336343c3cc0.gif"></p>    <h3>刷新错误页面</h3>    <p><img src="https://simg.open-open.com/show/19cce7be04e5811cdde0f496f808ec69.gif"></p>    <h3>高频率刷新</h3>    <p><img src="https://simg.open-open.com/show/7ffe42cceec3b3c113c8bfa14fc30861.gif"></p>    <h2>更新内容</h2>    <ul>     <li>支持刷新type错误页面(可自定义)</li>     <li>支持同时刷新多个type(异步,高频率)</li>     <li>链式注册资源</li>     <li>支持刷新单个数据</li>     <li>提供helper的清除单个type,清除整个界面api</li>     <li>提供getHeaderId的默认实现抽象类</li>    </ul>    <h3>支持刷新type错误页面(可自定义)</h3>    <pre>  <code class="language-java">public void notifyMoudleErrorChanged(ErrorEntity errorData, int type);    public void notifyMoudleErrorChanged(int type);</code></pre>    <p>一行代码搞定,前者提供实体类是考虑有些用户需要根据实体数据属性去更新,因此错误页面的layoutId是用户提供的。</p>    <h3>支持同时刷新多个type(异步,高频率)</h3>    <pre>  <code class="language-java">//刷新队列,支持高频率刷新  private Queue<HandleBase<T>> mRefreshQueue;</code></pre>    <p>这里采用的是队列的形式管理刷新,提供清空队列的Api。</p>    <h3>链式注册资源</h3>    <pre>  <code class="language-java">registerMoudle(@IntRange(from = 0, to = 999) int type)                  .level(@IntRange(from = 0) int level)                  .layoutResId(@LayoutRes int layoutResId)                  .headerResId(@LayoutRes int headerResId)                  .loading()                  .loadingLayoutResId(@LayoutRes int loadingLayoutResId)                  .loadingHeaderResId(@LayoutRes int loadingHeaderResId)                  .error()                  .errorLayoutResId(@LayoutRes int errorLayoutResId)                  .register();</code></pre>    <p>由于参数越来越多,这里采用了较为流行的链式注册,内部通过ResourcesManager管理所有资源。</p>    <p>注:原来的注册方式已设置为过时,请及时更新,不出2个版本将移除。</p>    <h3>支持刷新单个数据</h3>    <pre>  <code class="language-java">public void notifyMoudleDataAndHeaderChanged(T data, T header, int type)</code></pre>    <p>可能某个type只有一个实体数据管理着整个type</p>    <pre>  <code class="language-java">public void notifyMoudleDataAndHeaderChanged(List<? extends T> data, T header, int type)</code></pre>    <p>可传T的子类集合</p>    <h3>提供helper的清除单个type,清除整个界面api</h3>    <pre>  <code class="language-java">/**   * 清除单个type数据   *   * @param type 数据类型   */  public void clearMoudle(int type);    /**   * 清除所有数据   */  public void clear();</code></pre>    <h3>提供getHeaderId的默认实现抽象类</h3>    <p>假设你不实现粘性头而强制要写getHeaderId确实挺讨厌的,由于现在android还不是很好地兼容java8,因此暂时提供DefaultMultiHeaderEntity。</p>    <h3>使用注意点</h3>    <p>type 取值范围</p>    <ul>     <li>数据类型 [0,1000)</li>     <li>头类型 [-1000,0)</li>     <li>shimmer数据类型 [-2000,-1000)</li>     <li>shimmer头类型 [-3000,-2000)</li>     <li>error类型 [-4000,-3000)</li>    </ul>    <p>常量差值</p>    <pre>  <code class="language-java">//头类型差值  public static final int HEADER_TYPE_DIFFER = 1000;  //shimmer数据类型差值  public static final int SHIMMER_DATA_TYPE_DIFFER = 2000;  //shimmer头类型差值  public static final int SHIMMER_HEADER_TYPE_DIFFER = 3000;  //错误类型差值  public static final int ERROR_TYPE_DIFFER = 4000;</code></pre>    <h2>结束</h2>    <p>库多多少少也更新几个版本了,你的意见,你的建议,你的star,你的分享,一直是我前进的动力。还有一点要说的就是现在关于LayoutManager,RecyclerView,Adapter的流派很多,我们更关注于数据的优雅刷新。</p>    <p>如果对本库还不是很了解的同学可以到我的github查看更多版本进行了解。</p>    <h2>gradle依赖</h2>    <pre>  <code class="language-java">compile 'com.crazysunj:multitypeadapter:1.3.0'</code></pre>    <p> </p>