目标

前段时间打造了一款简单易用功能全面的图片上传组件,现在就来将上传的图片以图片集的形式展现到App上。出于用户体验考虑,加载新图片采用[无限]滚动模式,Android平台上我们优选RecyclerView组件。

显示图片,用的自然是ImageView,然而它并不支持直接加载网络图片,需要先通过其它网络组件(如HttpURLConnectionokhttp3等)将图片获取到本地,得到BitMap数据,然后通过setImageBitmap()加载。

ImageView也有setImageURI(Uri uri)方法,这里uri的命名容易让人产生错觉,其实只能是本地文件路径。

所幸,一些开源组件封装了繁琐的网络操作和缓存策略,提供了易用的API。这里我选择了Glide

实现

加载更多

项布局

有两个,一个用于列表中各个图片显示,一个显示加载更多/已全部加载放置在列表最末提示用户。

  1. <!--图片-->
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. android:orientation="vertical">
  6. <ImageView
  7. android:id="@+id/thumbnail_view"
  8. android:layout_width="match_parent"
  9. android:layout_height="match_parent"
  10. android:scaleType="centerCrop"/>
  11. </LinearLayout>
  1. <!--loadmore-->
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="wrap_content"
  5. android:gravity="center">
  6. <TextView
  7. android:id="@+id/tv_load_more"
  8. android:layout_width="wrap_content"
  9. android:layout_height="wrap_content"
  10. android:text="正在加载更多" />
  11. </LinearLayout>

RecyclerView.Adapter

RecyclerView的设计模式网上资料很多,此处不再赘述。先实现RecyclerView.Adapter

  1. class ThumbnailListAdapter(
  2. private val thumbnails: List<Thumbnail>,
  3. private val totalCount: Long,
  4. private val context: Context
  5. ) :
  6. RecyclerView.Adapter<ThumbnailListAdapter.ThumbnailViewHolder>() {
  7. // 调用若干次
  8. override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ThumbnailViewHolder {
  9. // viewType就是通过getItemViewType得到的
  10. val itemView = LayoutInflater.from(context).inflate(viewType, parent, false)
  11. return ThumbnailViewHolder(itemView)
  12. }
  13. // 搞分页/瀑布加载的同学不要把这个和数据库的总数量搞混,这里的itemCount表示现在内存中数据量
  14. // 我们可以[从后端]获取新数据添加到数据集,以实现loadmore功能
  15. override fun getItemCount(): Int {
  16. return if (thumbnails.isNotEmpty())
  17. thumbnails.size + 1 // +1 是因为除了thumbnails数据集之外,还有个写死的loadmore项
  18. else
  19. 0
  20. }
  21. // R.layout.xxx 是Int类型,可以直接返回
  22. override fun getItemViewType(position: Int): Int {
  23. return if (position < thumbnails.size)
  24. R.layout.list_thumbnail_image // 正常图片显示
  25. else
  26. R.layout.list_loadmore_footer // 末尾loadmore
  27. }
  28. // 有屏幕外item进入屏幕时就会调用
  29. override fun onBindViewHolder(holder: ThumbnailViewHolder, position: Int) {
  30. if (position < thumbnails.size) {
  31. Glide.with(context)
  32. .load(thumbnails[position].uri)
  33. .into(holder.itemView.thumbnail_view)
  34. } else {
  35. if (thumbnails.size >= totalCount)
  36. holder.itemView.tv_load_more.text = "全部加载完毕"
  37. }
  38. }
  39. // 必须这么继承一下
  40. class ThumbnailViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView)
  41. }

滚动监听

为RecyclerView添加滚动监听,在合适的时候加载新数据到数据集中。

  1. recyclerview.addOnScrollListener(object : RecyclerView.OnScrollListener() {
  2. override fun onScrollStateChanged(recyclerView: RecyclerView, newState: Int) {
  3. super.onScrollStateChanged(recyclerView, newState)
  4. // 已经在加载则跳过
  5. if (!_thumbnailsLoading) {
  6. // 找到最后可见项的索引
  7. val lastPos = layoutManager.findLastVisibleItemPosition()
  8. val sum = adapter.itemCount
  9. // 当快接近末尾项时(这里差额10,表示再显示10个item就没数据了)获取新数据
  10. if (newState == RecyclerView.SCROLL_STATE_IDLE && sum - lastPos <= 10) {
  11. vm.thumbnails.addAll(vm.getMoreAlbumCovers()) // 加载新数据到数据集中
  12. _thumbnailsLoading = true
  13. }
  14. }
  15. }
  16. })

不要将上面预加载数据和Glide的预加载图片混淆起来,拿到数据,和通过数据中的uri获取图片并下载,这是两个步骤。Glide专门针对RecyclerView提供了预加载方案,是为了减少滑动时图片还未从网络请求导致的等待加载情况,目前只支持LinearLayoutManager或其子类布局

布局

StaggeredGridLayoutManager

按列瀑布流显示图片。简单地将RecyclerView的layoutManager设为StaggeredGridLayoutManager实例即可,注意StaggeredGridLayoutManager目前还是beta版。

  1. val sgLayoutManager =
  2. StaggeredGridLayoutManager(3, StaggeredGridLayoutManager.VERTICAL)
  3. recyclerview.layoutManager = sgLayoutManager

使用StaggeredGridLayoutManager会发现上下滑动过程中,经常发生图片块重排。根据网上说法,这是因为复用的ViewHolder和该ViewHolder要加载的图片,它们的尺寸不一致导致。比如某个ViewHolder之前加载的图片高度为60,之后被回收,但是尺寸信息仍然保留着,后来被一张高度80的图片复用,由于StaggeredGridLayoutManager是根据ViewHolder的尺寸排序布局,尺寸的变化导致发生多次排序。解决方法是在ViewHolder绑定数据时(在RecyclerView.Adapter.onBindViewHolder()中),就事先设置好本次布局的最终尺寸,如下:

  1. override fun onBindViewHolder(holder: ThumbnailViewHolder, position: Int) {
  2. val layoutParams =
  3. holder.itemView.thumbnail_view.layoutParams as LinearLayout.LayoutParams
  4. //手动设置ViewHolder高度
  5. layoutParams.height = thumbnails[position].height
  6. Glide.with(context).load(thumbnails[position].uri)
  7. .into(holder.itemView.thumbnail_view)
  8. }

当由下滑回到最顶部时,经常会出现顶部(第一行)的图片相互重排。仔细观察,这是因为第一行初次布局时是按顺序排列而非按空缺插入,往回滑时则是按空缺(哪里最空最先排哪里),这导致顺序可能与初次排序不一致。不过还好,最终仍会按照图片尺寸各自归位。而且这种情况只会出现在第一次由下滑回到顶部时。

GreedoLayoutManager

StaggeredGridLayoutManager一共有3k多行代码,又是beta版。代码洁癖的我把目光投向了GreedoLayoutManager,它是500px开源的一个LayoutManager,能在保持图片宽高比例的前提下将多张图片拼接到一行显示,原理很简单,看下面动图:

替换LayoutManager也相当简单,重新设置下RecyclerView的layoutManager即可。

  1. val layoutManager =
  2. GreedoLayoutManager(adapter).also { it.setMaxRowHeight(resources.displayMetrics.heightPixels / 3) }
  3. recyclerview.layoutManager = layoutManager

GreedoLayoutManager在布局之前需要知道item的宽高比例,只要让Adapter实现SizeCalculatorDelegate接口即可

  1. override fun aspectRatioForIndex(index: Int): Double {
  2. val thumbnail = thumbnails[index]
  3. return thumbnail.width / thumbnail.height.toDouble()
  4. }

运行界面显示:

可以看到每张图片都比预期大很多,只能看到一小部分。经研究发现,上面定义的图片展示项的布局(LinearLayout内嵌ImageView),最终呈现后,LinearLayout的尺寸是每个网格的尺寸,而内嵌的ImageView则超出了LinearLayout,似乎其最终尺寸是MeasuredSize——我们在onCreateViewHolder时使用了LayoutInflater.from(context).inflate(viewType, parent, false),这里的parent是RecyclerView,而在布局xml中宽高都设置为match_parent,因此其中ImageView的MeasuredSize同RecyclerView的宽高——然而ImageView最终尺寸应该同样适配网格尺寸才对。

以width为例:

  1. 期望:ImageView.width == LinearLayout.width == 网格.width
  2. 实际:ImageView.width == ImageView.measuredWith == RecyclerView.width

我们看到每个框格其实是ImageView被截取的左上角那部分。

经过一番搜索,网上各种对getWidthgetMeasuredWidth区别的阐述,并没有解决我的困惑,直到这篇从源码的角度分析,getWidth() 与 getMeasuredWidth() 的不同之处让我知道,其实Android系统并没有对width下定义,自定义布局时可随意设置子项大小,是否超出屏幕也没有限制。在我们这个场景下,估计GreedoLayoutMananger在处理了最外层控件(这里是LinearLayout)的width后,并没有递归处理内部控件的width,从而导致了这个bug。

既然如此,那么就不要外围的LinearLayout,直接使用ImageView,反倒省了一点开销。

  1. override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ThumbnailViewHolder {
  2. return if (viewType == 0) {
  3. val imageView = ImageView(parent.context).apply {
  4. scaleType = ImageView.ScaleType.CENTER_CROP
  5. layoutParams = ViewGroup.LayoutParams(
  6. ViewGroup.LayoutParams.MATCH_PARENT,
  7. ViewGroup.LayoutParams.MATCH_PARENT
  8. )
  9. }
  10. ThumbnailViewHolder(imageView)
  11. } else {
  12. val itemView = LayoutInflater.from(context).inflate(viewType, parent, false)
  13. ThumbnailViewHolder(itemView)
  14. }
  15. }

当然也有ViewHolder重用导致的显示问题,图片只显示一部分,且是按ViewHolder重用前的宽高比例显示,如下:

懒得深究,使用Glide官方文档建议的waitForLayout()并没有用,override(width, height)提前告知图片尺寸解决。

  1. Glide.with(context)
  2. .load(thumbnails[position].uri)
  3. .override(thumbnails[position].width, thumbnails[position].height)
  4. .into(holder.itemView as ImageView)
  5. // .waitForLayout() //并没有用

下拉刷新

使用SwipeRefreshLayout,easy,按过不表。最后成品如下

其它

一般常用detachAndScrapView,RecyclerView会自动帮我们处理后续重用View[Holder]的逻辑。然而在某些场景下(如只是重排当前显示的Views而不是移除),我们可以使用更轻量级的detachView(detach之后view就不在界面上显示了),不过要记得在下次布局之前手动调用attachView(位置的话,detach之前在哪,attach后就在哪)或removeDetachedView/recycleView

注意detach之后,RecyclerView.getChildCount()就相应减少。

真正把 view layout到界面上的是RecyclerView的layoutDecorated方法。

【从零开始撸一个App】RecyclerView的使用的更多相关文章

  1. 【从零开始撸一个App】Kotlin

    工欲善其事必先利其器.像我们从零开始撸一个App的话,选择最合适的语言是首要任务.如果你跟我一样对Java蹒跚的步态和僵硬的语法颇感无奈,那么Kotlin在很大程度上不会令你失望.虽然为了符合JVM规 ...

  2. 【从零开始撸一个App】PKCE

    一个成功的App背后肯定有一堆后端服务提供支撑,认证授权服务(Authentication and Authorization Service,以下称AAS)就是其中之一,它是约束App.保障资源安全 ...

  3. 【从零开始撸一个App】Dagger2

    Dagger2是一个IOC框架,一般用于Android平台,第一次接触的朋友,一定会被搞得晕头转向.它延续了Java平台Spring框架代码碎片化,注解满天飞的传统.尝试将各处代码片段串联起来,理清思 ...

  4. 【从零开始撸一个App】Fragment和导航中的使用

    Fragment简介 Fragment自从Android 3.0引入开始,它所承担的角色就是显而易见的.它之于Activity就如html片段之于页面,好处无需赘述. Fragment的生命周期和Ac ...

  5. Android(4)—Mono For Android 第一个App应用程序

    0.前言 年前就计划着写这篇博客,总结一下自己做的第一个App,却一直被新项目所累,今天抽空把它写完,记录并回顾一下相关知识点,也为刚学习Mono的同学提供佐证->C#也是开发Android的! ...

  6. 深入浅出React Native 3: 从零开始写一个Hello World

    这是深入浅出React Native的第三篇文章. 1. 环境配置 2. 我的第一个应用 将index.ios.js中的代码全部删掉,为什么要删掉呢?因为我们准备从零开始写一个应用~学习技术最好的方式 ...

  7. Django1.8教程——从零开始搭建一个完整django博客(一)

    第一个Django项目将是一个完整的博客网站.它和我们博客园使用的博客别无二致,一样有分类.标签.归档.查询等功能.如果你对Django感兴趣的话,这是一个绝好的机会.该教程将和你一起,从零开始,搭建 ...

  8. 从零开始写一个武侠冒险游戏-6-用GPU提升性能(1)

    从零开始写一个武侠冒险游戏-6-用GPU提升性能(1) ----把帧动画的实现放在GPU上 作者:FreeBlues 修订记录 2016.06.19 初稿完成. 2016.08.05 增加对 XCod ...

  9. 从零开始构建一个的asp.net Core 项目

    最近突发奇想,想从零开始构建一个Core的MVC项目,于是开始了构建过程. 首先我们添加一个空的CORE下的MVC项目,创建完成之后我们运行一下(Ctrl +F5).我们会在页面上看到"He ...

随机推荐

  1. 杭电OJ----1002A + B问题II(超大数计算问题)

    Problem Description I have a very simple problem for you. Given two integers A and B, your job is to ...

  2. RHEL8/CentOS8的网络IP配置

    rhel8与7的区别 在rhel7上,同时支持network.service和NetworkManager.service(简称NM).默认情况下,这2个服务都有开启,但许多人都会将NM禁用掉. 在r ...

  3. R语言学习笔记-Corrplot相关性分析

    示例图像 首先安装需要的包 install.packages("Corrplot") #安装Corrplot install.packages("RColorBrewer ...

  4. #3使用html+css+js制作网页 番外篇 使用python flask 框架 (I)

    #3使用html+css+js制作网页 番外篇 使用python flask 框架(I 第一部) 0. 本系列教程 1. 准备 a.python b. flask c. flask 环境安装 d. f ...

  5. CentOS7上安装jdk,mysql

    最近笔者的云服务器由于中毒,重装系统了... 所以就记录下所有服务的搭建过程吧 1.安装jdk 在oracle上下载linux系统的jdk,笔者这里使用的是1.8 https://www.oracle ...

  6. 通过DNSLOG回显验证漏洞

    通过DNSLOG回显验证漏洞 前言 实际渗透测试中,有些漏洞因为没有回显导致无法准确判断漏洞是否存在,可能导致渗透测试人员浪费大量精力在一个并不存在的漏洞上,因此为了验证一些无回显漏洞,可结合DNSl ...

  7. VsCode/Pycharm配合python env 使用

    前言 用惯了vscode,这几天试了一下pycharm,还是回来了. pycharm一个好处就是python env 环境支持的比较好, vscode虽然也支持但是要改一些东西 env的使用查看我的上 ...

  8. Unity 编辑器(移除missing)

    移除 Missing(Mono Script) ` private static void FindMissingReferences() { GameObject[] pAllObjects = ( ...

  9. LeetCode542 01矩阵

    给定一个由 0 和 1 组成的矩阵,找出每个元素到最近的 0 的距离. 两个相邻元素间的距离为 1 . 示例 1: 输入: 0 0 0 0 1 0 0 0 0 输出: 0 0 0 0 1 0 0 0 ...

  10. Vim 自动添加脚本头部信息

    每次写脚本还在为忘记添加头部信息啥的烦恼? 按照下面这么做,帮你减轻点烦恼. # 打开配置文件: vim /root/.vimrc # 添加如下信息: autocmd BufNewFile *.sh ...