热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

AndroidImageSpan图文混排

本文示例项目地址对于Android开发者,部分场景下需要实现图文混排的排版方式(即在一个UI控件中同时显示图片与文字),较为

本文示例项目地址

对于 Android 开发者,部分场景下需要实现图文混排的排版方式(即在一个UI 控件中同时显示图片与文字),较为常见的场景如社交类 App 中对文字与表情的展示。

实现图文混排,主要有以下方法:

  1. 使用 WebView 加载 HTML
  2. 使用 Html.fromHtml(String source, int flags)获取 Spanned 对象后,通过 TextView 展示
  3. 使用 ImageSpan 展示图片

实际上方法 2 与方法 3 是相似的,而方法 3 对于不熟悉 HTML 的 Android 开发者更为友好并且提供了更高的自由度。本文主要分析方法 3 的基本使用与自定义绘制。

基本使用

ImageSpanDynamicDrawableSpan 的直接子类,开发者通过 SpannableString/SpannableStringBuildersetSpan() 方法将字符串的指定部分设置为由 ImageSpan 构造方法传入的图片。ImageSpan 的基本使用可以参考如下代码,完整代码可以查看示例项目的 BasicImageSpanActivity 类。

Drawable fuDrawable = getResources().getDrawable(R.drawable.image_fu);
fuDrawable.setBounds(0, 0, textView.getLineHeight(),textView.getLineHeight());
ImageSpan imageSpan = new ImageSpan(fuDrawable);
spannableString.setSpan(imageSpan, start, end, Spanned.SPAN_INCLUSIVE_INCLUSIVE);
textView.setText(spannableString);

对齐方式

图片与文字的位置关系需要通过设置 ImageSpan 的对齐方式来实现。通过在 ImageSpan 的构造方法中传入 verticalAlignment 参数可以实现对图片与文字的纵向对齐方式的设置。ImageSpan 为开发者提供了两种对齐方式:

  1. ALIGN_BOTTOM:图片底部与所在行底部对齐。
  2. ALIGN_BASELINE:图片底部与文字基线对齐。文字基线(baseline)是字体排印学的概念之一,其具体含义可以参考 维基百科的基线词条。同时,开发者可以将其简单的理解为文字的“重心”位置。

自定义绘制

对于图文混排,不少场景需要图片与文字居中对齐(即图片的中线与文字的中线重合),而 ImageSpan 并未提供这样的对齐方式。开发者可以通过重写(Override) ImageSpan 父类 DynamicDrawableSpandraw 方法来实现自己的绘制逻辑,从而实现图片与文字的居中对齐。DynamicDrawableSpandraw 方法的实现如下:

@Overridepublic void draw(Canvas canvas, CharSequence text,int start, int end, float x, int top, int y, int bottom, Paint paint) {Drawable b = getCachedDrawable();canvas.save();int transY = bottom - b.getBounds().bottom;if (mVerticalAlignment == ALIGN_BASELINE) {transY -= paint.getFontMetricsInt().descent;}canvas.translate(x, transY);b.draw(canvas);canvas.restore();}

方法中 bottom 为图片所在行底部坐标(以 TextView 左上角为原点),DynamicDrawableSpan 根据 mVerticalAlignment 的值,使用 canvastranslate 方法,将画布移动对应的距离(ALIGN_BOTTOM 移动 bottomdrawable 底部差值的距离,ALIGN_BASELINE 在前者的基础上减去 descent 值,descent 为基线到行底部的距离),来实现对应的对齐方式。开发者可以参考 DynamicDrawSpan 的实现方式,来实现居中对齐乃至更多的绘制逻辑。居中对齐的实现代码可以参考如下代码,完整代码可以参考示例项目中 CustomImageSpandraw 方法。

@Overridepublic void draw(Canvas canvas, CharSequence text, int start, int end,float x, int top, int y, int bottom, Paint paint) {Drawable b = getDrawable();Paint.FontMetricsInt fm = paint.getFontMetricsInt();int transY = (y + fm.descent + y + fm.ascent) / 2 - (b.getBounds().bottom + b.getBounds().top) / 2;canvas.save();canvas.translate(x, transY);b.draw(canvas);canvas.restore();}

代码中 y 为基线的纵坐标(以 TextView 左上角为坐标原点),fm.descent 为基线至文字底部的距离(为正值),fm.ascent 为基线至文字顶部的距离(为负值),y + fm.descent 的值为文字底部纵坐标,y + fm.ascent 的值为文字顶部纵坐标,二者相加除以2则得到了文字纵向中点的纵坐标,(b.getBounds().bottom + b.getBounds().top) / 2 则为 drawable 绘制区域纵向中点的纵坐标,而二者的差值即为实现居中对齐画布的纵向偏移。

为验证代码的效果,笔者使用一个高16、宽48的黑色(#000000)色块作为传入 ImageSpandrawable,展示了在不同对齐方式下图文混排的效果,该示例的完整代码可以参考示例项目的 CustomImageSpanActivity 类,代码的运行效果如下图:

为了方便读者对比不同对齐方式,截图中开启了开发者选项中的显示布局边界

由于开发者可以在 DynamicDrawableSpandraw 方法中实现自己的绘制逻辑,使用该方案来实现图文混排给予了开发者极大的自由度,开发者可以对与文字混排的图片进行更为细致的排版,本文仅以较为常见居中对齐作为示例,相信读者可以实现更多更好的图文混排绘制逻辑。

转:https://juejin.im/post/5a5f5e095188253dc33203b4



推荐阅读
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 自动轮播,反转播放的ViewPagerAdapter的使用方法和效果展示
    本文介绍了如何使用自动轮播、反转播放的ViewPagerAdapter,并展示了其效果。该ViewPagerAdapter支持无限循环、触摸暂停、切换缩放等功能。同时提供了使用GIF.gif的示例和github地址。通过LoopFragmentPagerAdapter类的getActualCount、getActualItem和getActualPagerTitle方法可以实现自定义的循环效果和标题展示。 ... [详细]
  • 在Docker中,将主机目录挂载到容器中作为volume使用时,常常会遇到文件权限问题。这是因为容器内外的UID不同所导致的。本文介绍了解决这个问题的方法,包括使用gosu和suexec工具以及在Dockerfile中配置volume的权限。通过这些方法,可以避免在使用Docker时出现无写权限的情况。 ... [详细]
  • 生成式对抗网络模型综述摘要生成式对抗网络模型(GAN)是基于深度学习的一种强大的生成模型,可以应用于计算机视觉、自然语言处理、半监督学习等重要领域。生成式对抗网络 ... [详细]
  • Spring源码解密之默认标签的解析方式分析
    本文分析了Spring源码解密中默认标签的解析方式。通过对命名空间的判断,区分默认命名空间和自定义命名空间,并采用不同的解析方式。其中,bean标签的解析最为复杂和重要。 ... [详细]
  • 本文介绍了设计师伊振华受邀参与沈阳市智慧城市运行管理中心项目的整体设计,并以数字赋能和创新驱动高质量发展的理念,建设了集成、智慧、高效的一体化城市综合管理平台,促进了城市的数字化转型。该中心被称为当代城市的智能心脏,为沈阳市的智慧城市建设做出了重要贡献。 ... [详细]
  • CSS3选择器的使用方法详解,提高Web开发效率和精准度
    本文详细介绍了CSS3新增的选择器方法,包括属性选择器的使用。通过CSS3选择器,可以提高Web开发的效率和精准度,使得查找元素更加方便和快捷。同时,本文还对属性选择器的各种用法进行了详细解释,并给出了相应的代码示例。通过学习本文,读者可以更好地掌握CSS3选择器的使用方法,提升自己的Web开发能力。 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • 使用Ubuntu中的Python获取浏览器历史记录原文: ... [详细]
  • sklearn数据集库中的常用数据集类型介绍
    本文介绍了sklearn数据集库中常用的数据集类型,包括玩具数据集和样本生成器。其中详细介绍了波士顿房价数据集,包含了波士顿506处房屋的13种不同特征以及房屋价格,适用于回归任务。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • 【MicroServices】【Arduino】装修甲醛检测,ArduinoDart甲醛、PM2.5、温湿度、光照传感器等,数据记录于SD卡,Python数据显示,UI5前台,微服务后台……
    这篇文章介绍了一个基于Arduino的装修甲醛检测项目,使用了ArduinoDart甲醛、PM2.5、温湿度、光照传感器等硬件,并将数据记录于SD卡,使用Python进行数据显示,使用UI5进行前台设计,使用微服务进行后台开发。该项目还在不断更新中,有兴趣的可以关注作者的博客和GitHub。 ... [详细]
  • CF:3D City Model(小思维)问题解析和代码实现
    本文通过解析CF:3D City Model问题,介绍了问题的背景和要求,并给出了相应的代码实现。该问题涉及到在一个矩形的网格上建造城市的情景,每个网格单元可以作为建筑的基础,建筑由多个立方体叠加而成。文章详细讲解了问题的解决思路,并给出了相应的代码实现供读者参考。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
author-avatar
yema狂想曲
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有