1、重写TextView的onDraw方法

article/2025/9/10 19:53:31

Android重写系统TextView

 

Git源码地址:

https://github.com/AndroidAppWidgetDemo/Android_Widget_CustomTextView

 

一、目的

重写Android系统TextView,是为了解决系统TextView中文换行所带来的参差不齐的锯齿效果。如图1.1为系统TextView的绘制结果,图1.2为期望中的TextView绘制结果。

图1.1 系统TextView的绘制结果

图1.2 期望TextView的绘制结果(重写后的效果)

系统TextView中文换行的大致原理为:

文本绘制绘制到某一行的行尾,并且剩余空间不足一个字符的空间时,将当前要绘制的字符放在下一行进行绘制(因此上一行结尾与该行的右边界有一小段空白)。

文本绘制到某一行的行尾,并且最后一个字符为标点符号时,那么将标点符号连同标点符号的上一个字符放在下一行进行绘制(因此上一行结尾与该行的右边界有稍微大的一段空白)。

所以这种绘制方式,会使TextView右侧的显示文本参差不齐。

二、重写TextView算法的原理

(1)、如何绘制:

对此TextView中的所有字符串使用“canvas.drawText(str, x,y,mPaint)”方法进行一个字、一个字的绘制。

(2)、目标:

主要的目标是解决标点带来的换行参差不齐的问题。

(3)、标点的说明:

这里要说明自己命名的几种标点名称:

单独出现的标点:“,” “。” “.”等

左侧标点:“《”“<”“{”等

右侧标点:“》”“>”“}”等

(4)、最初思考要处理的情况和解决方式:

 

对一行结尾的字符或者下一行开始的字符进行判断。根据标点的不同,分别进行字体间距的拉伸和压缩。

 

字间距压缩的情况:

正常绘制时(没有字间距的变化),如果一行绘制结束,但是在下一行的开始时,“单独出现的标点符号”被绘制到了该行的开头。即,“,”或者“。”出现在了行首的位置。

这种情况下,将上一行每两个字符之间的字间距进行压缩,使其刚好可以将本行首的标点放在上一行的行首。如图2.1所示。

图2.1 字间距进行压缩

 

字间距的拉伸的情况:

这种情况下即,“《”、“<”等标点符号出现在一行的行尾位置。那么将该行字符的字间距进行拉伸,使这些标点出现在下一行的行首。如图2.2所示。

图2.2 字间距进行拉伸

 

(5)、最终的解决方案:

后来发现,只是简单的对“行尾字符”、“下一行的开始字符”进行标点的判断还不够。有的时候,几个标点会连续同时出现。下面举例,如图2.3所示。

图2.3 字间距进行拉伸

因此在进行标点判断时,要对结尾的三个字符进行判断。即,一行的结尾字符、下一行的开始字符、下一行的第二个字符。如图2.4所示。

图2.4 要进行标点判断的三个字符位置

(6)、实现:

整个算法的实现,就是判断(5)中所介绍的三个字符与标点的对应情况。这种判断方式的实现,如图2.5所示。

图2.5标点判断的算法实现

三、简单代码的讲解

因为要解决的问题为文本的绘制问题,所以这里只重写了onDraw方法。onDraw方法中:

首先,通过this.getText().toString()方法,获取要绘制的文本字符串;

其次,对字符串进行第一次循环判断,第一次判断是将字符串分行(即获取整个文本中,每一行文本开始的Index和结束Index、该行字体间距拉伸了多少和压缩了多少);

再次,对字符串进行第二次循环,这次循环要做的工作就是根据第一次循环获取的信息进行一个字符一个字符的绘制了。

 

四、引入的问题

由于自定义TextView中的文本全部都是由“canvas.drawText(str, x,y,mPaint)”方法,一个字符一个字符的进行绘制的。所以当文本数量较大时,对与系统资源的占用较为严重,会造成界面中其他事件的反映较慢。

五、编码是否有影响??

 

(1)、代码中对于中文标点的判断:

/*

 * Chinese punctuation

 */

public static boolean isPunctuation(char c) {

       Character.UnicodeBlock ub = Character.UnicodeBlock.of(c);

       if (ub == Character.UnicodeBlock.GENERAL_PUNCTUATION

                     || ub == Character.UnicodeBlock.CJK_SYMBOLS_AND_PUNCTUATION

                     || ub == Character.UnicodeBlock.HALFWIDTH_AND_FULLWIDTH_FORMS) {

              return true;

       }

       return false;

}

 

对于中文标点的判断来自于网络

 

(2)、其他英文标点的判断:

 

/*

 * English punctuation

 */

public static boolean isHalfPunctuation(char c) {

       int count = (int) c;

 

       if (count >= 33 && count <= 47) {

              // !~/

              return true;

       } else if (count >= 58 && count <= 64) {

              // :~@

              return true;

       } else if (count >= 91 && count <= 96) {

              // [~

              return true;

       } else if (count >= 123 && count <= 126) {

              // {~~

              return true;

       }

       return false;

}

这里全部的英文标点是根据“ASCII编码表” 确定的,在不同的编码环境下,这些标点所对应的十进制数值是不变的。

 

(3)、成对出现标点的判断:

 

/*

 * the left half of the punctuation . For example:" ( < [ { "

 */

public static boolean isLeftPunctuation(char c) {

       int count = (int) c;

       if (count == 8220 || count == 12298 || count == 65288 || count == 12304

                     || count == 40 || count == 60 || count == 91 || count == 123) {

              return true;

       }

       return false;

}

/*

 * the right half of the punctuation . For example:" ) > ] } "

 */

public static boolean isRightPunctuation(char c) {

       int count = (int) c;

       if (count == 8221 || count == 12299 || count == 65289 || count == 12305

                     || count == 41 || count == 62 || count == 93 || count == 125) {

              return true;

       }

       return false;

}

这些成对出现的标点中,编码在0~255之间的是根据“ASCII编码表” 确定;其他的则为相应标点,转化为相应的十进制数值来确定。

 

当编码发生变化时,对于这些“相应标点,转化为相应十进制数值来确定的标点”是否会有影响?????????


http://chatgpt.dhexx.cn/article/HL1S0Jfd.shtml

相关文章

getX()、getRawX()、getTranslationX()、gettranslationX()...

关于屏幕上获取这些位置的情况应该分为两种&#xff0c;一种是在没有点击屏幕时&#xff0c;另外一种是有点击 我们来分别看看这几个X在图形上的直观表示和解释 有点击时的getX&#xff08;&#xff09; getY() getX()表示触摸点距离view本身左边界的距离getY()表示触摸点距离…

Direct2D函数DWrite

Win8 DirectX 11.1 中 Direct3D 和 Direct2D &#xff0c; DirectWrite 的交互比11方便多了&#xff0c;不用创建什么DX10.1的device&#xff0c;不用搞什么设备同步&#xff0c;直接往同一个DXGI Resource里写就是了。 这大概得益于Win 6.2 底层架构的进一步融合&#xff0c;…

Pthreads on Microsoft Windows

Posix Threads API (pthreads) 是在并行编程中用到的非常普通的API&#xff0c;这套API包括许多非常基础的同步方法&#xff0c;方便我们编写高效的多线程程序。然而&#xff0c;Microsoft Windows 并不包含这样的接口。幸运的是&#xff0c;这里有一个开源的Windows平台上的 P…

Text-Instance Graph: Exploring the Relational Semantics for Text-based Visual Question Answering

Text-Instance Graph: Exploring the Relational Semantics for Text-based Visual Question Answering 文章目录 Text-Instance Graph: Exploring the Relational Semantics for Text-based Visual Question AnsweringAbstract1. Introduction2. Related works2.1. Visual Que…

绘制系列(五)-DrawText()详解

绘制系列&#xff08;五&#xff09;-DrawText()详解 文字基础知识 1、setTextAlign() 设置文字对齐方式&#xff08;起始点相对于文字&#xff09; canvas.drawText(“文字”&#xff0c;起始点&#xff0c;paint) 比如&#xff1a; paint.setTextAlign(Paint.Align.RIGH…

Windows API一日一练(24)DrawText函数-绘制矩形内的文字

主目录传送门 https://blog.csdn.net/zb774095236/article/details/108291755 本次学习的源码模板下载&#xff1a; 1.百度云&#xff1a;https://pan.baidu.com/s/1B0YuPWgSxvLTHk59STzQQQ 提取码&#xff1a;1ins 学习目标&#xff1a; 在Win32API-13-14-窗口关闭和窗口…

wpf-绘制文字:DrawText和DrawGlyphRun

先上效果&#xff1a;上面的Hello,world是DrawGlyphRun绘制的&#xff0c;下面的hello是DrawText绘制的。绘制的结果都是不能复制的。 前台 <local:CanvasCustom x:Name"myCanvas"/>后台 class CanvasCustom : Canvas {protected override void OnRender(Dr…

android 文字drawable,TextDrawable

TextDrawable 是一个可以将纯文本生成Drawable的类库&#xff0c;你可以将这个Drawable设置给ImageView&#xff0c;从而可将文字转为图片。这在某些场合是很有用的。 效果图&#xff1a; 如何使用你可在xml中创建个ImageView android:layout_width"60dp" android:la…

drawText用法总结

drawText 经常使用Canvas的draw***方法去绘制一些图像图形&#xff0c;绘制的坐标是从Canvas左上角开始计算的&#xff0c;如果想要把一个图像放到某个位置&#xff0c;直接drawBitmap传递图片左上角的坐标就行了。那drawText就不一样&#xff0c;如果你传递进去字符串&#x…

drawText详解

Canvas 作为绘制文本时&#xff0c;是以基线为基准绘制的&#xff0c;不是左上角 FontMetrics对象 它以四个基本坐标为基准&#xff0c;分别为&#xff1a; ・FontMetrics.top ・FontMetrics.ascent ・FontMetrics.descent ・FontMetrics.bottom ascent和top都为负数&#xff…

B. The Monster and the Squirrel

B. The Monster and the Squirrel Ari the monster always wakes up very early with the first ray of the sun and the first thing she does is feeding her squirrel. Ari draws a regular convex polygon on the floor and numbers it’s vertices 1, 2, …, n in cloc…

Springboot整合squirrel-foundation状态机

目录 一. 快速入门 1 . maven 2 . 快速开始 3 . Fluent Api 4 . 状态机四要素 5 . 状态机构建器 6 . 状态机转换操作(代码配置方式) 7 . 状态机转换操作(注解声明方式) 8 . 上下文不敏感状态机 二 : 使用注意事项 P1 : 异常 : StateMachineBuilderImpl cannot fi…

Hive的客户端界面工具–SQuirrel SQL Client--详细安装以及连接Hive过程

SQuirrel SQL Client是一款支持Hive的可视化工具&#xff0c;是市面上少数支持Hive中比较好用的&#xff0c;看下如何安装使用吧&#xff0c;下面是非常详细的安装过程。 1.下载客户端 SQuirrel SQL Client的官网及下载地址为&#xff1a;http://squirrel-sql.sourceforge.ne…

Linux安装SQuirreL SQL Client

环境和准备 操作系统&#xff1a; Ubuntu 20.04SQuirreL&#xff1a; squirrel-sql-snapshot-20220326_1238-standard.jarDb2 driver&#xff1a; db2jcc4.jardb2jcc_license_cu.jar MySQL driver&#xff1a; mysql-connector-java-8.0.27.jar 下载和安装 首先下载SQuirreL…

智能优化算法之松鼠算法(Squirrel search algorithm)

文章目录 背景Squirrel search algorithm(SSA)SSARandom initialization&#xff08;随机初始化&#xff09;Fitness evaluation&#xff08;适应值评价&#xff09;Sorting, declaration and random selection&#xff08;排序、声明和随机选择&#xff09;Generate new locat…

electron打包遇到 Making for target: squirrel - On platform: win32 - For arch: x64错误

上面横线处是出现错误的位置。报错的原因如下&#xff1a; 1、package.json的“author”和“description”在打包时是必填内容&#xff0c;随便填些内容即可打包成功。 2、和项目的绝对路径有关&#xff0c;项目的绝对路径不能出现中文&#xff0c;否则在红线处会报错。

WPF 使用Squirrel自动更新应用

前言 本文简单的介绍了如何使用 Squirrel 来为 WPF 客户端 进行自动检查更新。 Squirrel git 地址 &#xff1a;http:// https://github.com/Squirrel/Squirrel.Windows 本文使用了 Visual Studio 2022 进行演示讲解。 参考英文博客&#xff1a; https://intellitect.com/d…

SQuirrel SQL Client的安装

如果您的工作要求您在一天之中连接许多不同的数据库 &#xff08;oracle、DB2、mysql、postgresql、Sql Server等等&#xff09;&#xff0c;或者你经常需要在多个不同种类的数据库之间进行数导入导出。那么SQuirreL SQL Client 将会是比较理想的数据库客户端链接工具。 简单介…

数据库管理工具——SQuirreL SQL Client使用入门

如果您的工作要求您在一天之中连接许多不同的数据库 &#xff08;oracle、DB2、mysql、postgresql、Sql Server等等&#xff09;&#xff0c;或者你经常需要在多个不同种类的数据库之间进行数导入导出。那么SQuirreL SQL Client 将会是比较理想的数据库客户端链接工具。 简单介…

SQuirrel连接hive配置

1. 简介 最近由于大数据部门相关同事离职&#xff0c;不得不研究一下大数据相关组件&#xff0c;今天成功安装配置Hive&#xff0c;简单记录&#xff0c;一是为了加深印象&#xff0c;二是为以后备用&#xff0c;三是为大家提供参考&#xff0c;避免少踩坑。 在Hive的官网上…