Android 实现仿微博列表九宫图

先看效果图

  1. IMG_2494.PNG
  2. IMG_2495.PNG

需求

  1. 类似微博九宫图一样,单张图片显示比例(此处具体先省略)
  2. 2张图和4张图片显示 格子列数为4宫格

需要的库

com.github.bumptech.glide:glide:4.9.0
com.github.CymChad:BaseRecyclerViewAdapterHelper:2.9.44

大概思路

  1. 数据请求 省略(我用的酷安的数据热门评论接口)
  2. 根据评论中图片数据设置图片RecyclerViewSpanCount,每一条评论都含有一个展示图片的RecycleView,再设置ItemDecoration
  3. 对不同的图片个数的ViewHolder设置不同的type

上代码

CmsStyleCommentNormalViewHolder:类中

if (!TextUtils.isEmpty(dateItem.pic) && dateItem.picArr != null) {
                val nineImageList = arrayListOf<NineImageBean>()
                var spanCount = 0
                dateItem.picArr?.let { it1 ->
                    spanCount = if (it1.size == 1) {
                        1
                    } else if (it1.size == 2 || it1.size == 4) {
                        2
                    } else {
                        3
                    }
                    it1.forEachIndexed { index, s ->
                        nineImageList.add(NineImageBean().apply {
                            this.isGif = StringUtils.isGifUrl(s)
                            this.url = s
                            this.positionAdapter = index
                            if (spanCount == 1) {
                                this.type = NineImageBean.LargeImageType
                            } else {
                                this.type = NineImageBean.SmallImageType
                            }
                        })
                    }
                }
                val cmsChildImageNineAdapter: CmsChildImageNineAdapter
                val adapterTag = nineImageRv.tag
                if (adapterTag is CmsChildImageNineAdapter) {
                    cmsChildImageNineAdapter = adapterTag
                } else {
                    cmsChildImageNineAdapter = CmsChildImageNineAdapter(ArrayList())
                    nineImageRv.apply {
                        cmsChildImageNineAdapter.bindToRecyclerView(this)
                        this.isNestedScrollingEnabled = false
                        this.setHasFixedSize(true)
                        this.tag = cmsChildImageNineAdapter
                    }
                    cmsChildImageNineAdapter.setOnItemClickListener { adapter, _, position ->
                        val pictureBeanList = ArrayList<PictureBean>()
                        for (item in adapter.data) {
                            if (item is NineImageBean) {
                                pictureBeanList.add(PictureBean().apply {
                                    this.originalUrl = item.url
                                })
                            }
                        }
                        if (pictureBeanList.isNotEmpty()) {
                            LaunchNormalUtils.startPictureActivity(mContext, pictureBeanList, position)
                        }
                    }
                }
                nineImageRv.apply {
                    this.layoutParams.width = if (spanCount == 4 || spanCount == 2) {
                        singleImageWidth.toInt() * 2
                    } else if (spanCount == 1) {
                        ViewGroup.LayoutParams.WRAP_CONTENT
                    } else {
                        ViewGroup.LayoutParams.MATCH_PARENT
                    }
                    for (i in 0 until this.itemDecorationCount) {
                        this.removeItemDecorationAt(i)
                    }
                    this.layoutManager = GridLayoutManager(mContext, spanCount)
                    this.addItemDecoration(RecyclerViewItemDividerHelper.getNineImageAdapterDividerItem(mContext, spanCount, this, 16f))
                }
                cmsChildImageNineAdapter.setNewData(nineImageList)
                nineImageRv.visibility = View.VISIBLE
            } else {
                nineImageRv.visibility = View.GONE
            }
        }
  • 上面代码位于一个评论ViewHolder类
  • CmsChildImageNineAdapter:
class CmsChildImageNineAdapter(data: MutableList<NineImageBean>?): BaseMultiItemQuickAdapter<NineImageBean, BaseViewHolder>(data) {
    init {
        addItemType(NineImageBean.emptyType, R.layout.item_multi_not_found)
        addItemType(NineImageBean.SmallImageType, R.layout.item_multi_nine_image_small)
        addItemType(NineImageBean.LargeImageType, R.layout.item_multi_nine_image_large)
    }

    override fun convert(helper: BaseViewHolder, item: NineImageBean) {
        var associatedObject = helper.associatedObject
        when (item.itemType) {
            NineImageBean.SmallImageType -> {
                if (associatedObject == null) {
                    associatedObject = NineSmallViewHolder(mContext, helper)
                    helper.associatedObject = associatedObject
                }
                if (associatedObject is NineSmallViewHolder) {
                    associatedObject.updateView(item)
                }
            }
            NineImageBean.LargeImageType -> {
                if (associatedObject == null) {
                    associatedObject = NineLargeViewHolder(mContext, helper)
                    helper.associatedObject = associatedObject
                }
                if (associatedObject is NineLargeViewHolder) {
                    associatedObject.updateView(item)
                }
            }
        }
    }
  • NineSmallViewHolder
class NineSmallViewHolder(val mContext: Context, val baseViewHolder: BaseViewHolder) : IBaseViewMultiHolder<NineImageBean>(baseViewHolder.itemView) {
    private val smallImageIv: CustomGifImageView = itemView.findViewById(R.id.small_image_iv)
    private val smallGifFlagRtv: RoundTextView = itemView.findViewById(R.id.small_gif_flag_rtv)
    private var gifHandler: GifHandler? = null

    private inner class ParamTag {
        var isFirstFrame = true
        var isAlwaysPlayerGif = true
    }

    override fun updateView(dateItem: NineImageBean) {
        super.updateView(dateItem)
        smallImageIv.apply {
            if (dateItem.isGif && !this.isGifRunning()) {
                smallGifFlagRtv.visibility = View.VISIBLE
            } else {
                smallGifFlagRtv.visibility = View.GONE
            }
            if (!dateItem.isGif) {
                loadNormalImage(dateItem)
            } else {
                if (!this.isGifRunning()) {
                    loadGifImage(ParamTag().apply {
                        this.isFirstFrame = true
                    })
                }
            }
        }
    }

 private fun loadNormalImage(dateItem: NineImageBean) {
        ImageLoader.Builder(mContext)
                .getRequestManager()
                .asBitmap()
                .load(dateItem.url)
                .apply(ImageLoader.defaultRequestOptions(R.color.placeholder_color))
                .into(smallImageIv)
    }
  • 我粘贴的事项目中代码,我也懒得删除了,所以CustomGifImageView 就用imageview就行,也不用判断GIF加载什么的,就gilde直接加载就OK
  • NineLargeViewHolder
class NineLargeViewHolder(val mContext: Context, val baseViewHolder: BaseViewHolder) : IBaseViewMultiHolder<NineImageBean>(baseViewHolder.itemView) {
    private val largeIv: MatrixScaleImageView = itemView.findViewById(R.id.large_iv)
    private val largeFlagRtv: RoundTextView = itemView.findViewById(R.id.large_flag_rtv)
    private val gifFlagRtv: RoundTextView = itemView.findViewById(R.id.gif_flag_rtv)

    override fun updateView(dateItem: NineImageBean) {
        super.updateView(dateItem)
        largeFlagRtv.visibility = View.GONE
        gifFlagRtv.visibility = View.GONE
        dateItem.picImageSize?.apply {
            val matrixScaleHelper = MatrixScaleHelper(this[0].toFloat(), this[1].toFloat())
            largeIv.apply {
                gifFlagRtv.visibility = if (dateItem.isGif && !this.isGifRunning()) {
                    View.VISIBLE
                } else {
                    View.GONE
                }
                largeFlagRtv.visibility = if (matrixScaleHelper.isLongBitmap && !dateItem.isGif) {
                    View.VISIBLE
                } else {
                    View.GONE
                }
                if (!dateItem.isGif) {
                    loadNormalImage()
                } else {
                    if (!this.isGifRunning()) {
                        loadGifImage(true)
                    }
                }
            }
        }
    }

    override fun changeTheme(theme: Theme) {
        super.changeTheme(theme)
        largeFlagRtv.setBackgroundColor(ColorUtils.getThemeColor(mContext, R.attr.colorAccent))
        gifFlagRtv.setBackgroundColor(ColorUtils.getThemeColor(mContext, R.attr.colorAccent))
    }

    fun startPlayerGif() {
        val nineImageBean = getItemTagDate() as? NineImageBean ?: return
        if (nineImageBean.isGif) {
            loadGifImage(false)
        }
    }

    fun stopPlayerGif() {
        val nineImageBean = getItemTagDate() as? NineImageBean ?: return
        if (nineImageBean.isGif) {
            loadGifImage(true)
        }
    }

    private fun loadNormalImage() {
        val nineImageBean = getItemTagDate()
        if (nineImageBean is NineImageBean) {
            val picImageSize = nineImageBean.picImageSize ?: return
            val matrixScaleHelper = MatrixScaleHelper(picImageSize[0].toFloat(), picImageSize[1].toFloat())
            ImageLoader.Builder(mContext, nineImageBean.url)
                    .setTransition(ImageLoader.defaultTransitionOptions())
                    .setRequestOptions(ImageLoader.defaultRequestOptions(R.color.placeholder_color)
                            .transform(matrixScaleHelper.getCropTransformation(largeIv)))
                    .build(largeIv)
        }
    }
  • GIF 代码我就不粘贴了,其中一部分,这个viewHolder比较特殊,一问微博中单个图片比较特殊,是有比例显示的,大长图显示上半部分,普通图片等比例缩放,MatrixScaleHelper类就是一个等比缩放类
  • 粘贴的是部分代码和思路,全部代码整理起来比较麻烦,实现效果比较多
  • 后面会陆续贴出来实现微博九宫图 GIF图片依次播放,还有单张图片等比缩放,以及换肤(不重启应用)过程中RecycleView的处理

最后看一样我实现的效果图吧

Screenshot_20190225-163519.png
Screenshot_20190225-163534.png
Screenshot_20190225-163557.png
Screenshot_20190225-163649.png
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,440评论 5 467
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,814评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,427评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,710评论 1 270
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,625评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,014评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,511评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,162评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,311评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,262评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,278评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,989评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,583评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,664评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,904评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,274评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,856评论 2 339

推荐阅读更多精彩内容