竖排文本控件

2024-03-06 08:08
文章标签 控件 文本 竖排

本文主要是介绍竖排文本控件,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

源码:https://github.com/lybeat/PlumbTextView

PlumbTextView

PlumbTextView是一个竖排列的文本控件。你可以很容易使用它定义多种竖排文本风格。

Feature

  1. 将文本竖直排列;

  2. 可以设置文本的列距和字距;

  3. 可以添加一个正则表达式去分割文本。PlumbTextView会在正则表达式所包含的字符处换列,并且其中的字符不会显示在PlumbTextView中;

  4. 可以在每一列文本的坐标添加一跟竖线;

  5. 可以为文本设置字体风格和第三方字体。

Screenshot

plumb_textview.gif

Gradle Dependency

 
  1. compile 'cc.sayaki.widget:plumb-textview:1.0.1'

Usage

你可以很容易的使用PlumbTextView,就像使用TextView那样。只需要在xml或者Java代码中设置你想要的属性效果就行了。

 
  1. <cc.sayaki.widget.PlumbTextView
  2.    android:layout_width="wrap_content"
  3.    android:layout_height="wrap_content"
  4.    android:paddingBottom="30dp"
  5.    android:paddingTop="30dp"
  6.    sayaki:columnSpacing="20dp"
  7.    sayaki:leftLine="true"
  8.    sayaki:leftLineColor="@color/colorAccent"
  9.    sayaki:leftLinePadding="4dp"
  10.    sayaki:letterSpacing="6dp"
  11.    sayaki:regex="[,。?!]"
  12.    sayaki:text="@string/text"
  13.    sayaki:textColor="@color/colorAccent"
  14.    sayaki:textSize="16sp"
  15.    sayaki:textStyle="bold|italic" />

Thinking

接下来讲讲PlumbTextView的实现思路。
思路很简单,主要就是在绘制完一个字符之后,将下次绘制的位置垂直向下移动一个字符的高度(包括字符本身的高度和两个字符之间的字距)。只是垂直绘制文字很简单,不过处理换列是一个比较麻烦的地方。

PlumbTextView有两种换列情况:

  1. 单列文本超过了PlumbTextView可以显示的内容高度;

  2. 提供了正则表达式,PlumbTextView会在正则表达式中所包含的字符处换列。

首先测量PlumbTextView自身的高度

 
  1. if (heightMode == MeasureSpec.EXACTLY) {
  2.     height = heightSize;
  3. } else {
  4.     if (!TextUtils.isEmpty(regex)) {
  5.         height = 0;
  6.         String[] texts = text.toString().split(regex);
  7.         for (String s : texts) {
  8.             height = Math.max(height, getDesiredHeight(s));
  9.         }
  10.         height += letterSpacing;
  11.     } else {
  12.         height = getDesiredHeight(text.toString());
  13.     }
  14.     if (height > heightSize) {
  15.         height = heightSize;
  16.     }
  17. }

根据换列规则拆分文本

 
  1. if (!TextUtils.isEmpty(regex)) {
  2.     String[] texts = text.toString().split(regex);
  3.     for (String s : texts) {
  4.         getFormatTexts(s);
  5.     }
  6. } else {
  7.     getFormatTexts(text.toString());
  8. }
  9.  
  10. // 获取拆分后的文本
  11. private void getFormatTexts(String s) {
  12.     int contentHeight = height - getPaddingTop() - getPaddingBottom();
  13.     if (getDesiredHeight(s) > contentHeight) {
  14.         int count = contentHeight / charHeight;
  15.         int i = 0;
  16.         // 有的文本拆分过后可能仍然大于控件可显示的高度,需要再拆分
  17.         for (; i < getDesiredHeight(s) / contentHeight; i++) {
  18.             formatTexts.add(s.substring(* count, (+ 1) * count));
  19.         }
  20.         // 最后一列文本不满一列
  21.         if (getDesiredHeight(s) % contentHeight != 0) {
  22.             formatTexts.add(s.substring(* count, s.length()));
  23.         }
  24.     } else {
  25.         formatTexts.add(s);
  26.     }
  27. }

测量PlumbTextView的宽度

 
  1. if (widthMode == MeasureSpec.EXACTLY) {
  2.     width = widthSize;
  3. } else {
  4.     if (!TextUtils.isEmpty(regex)) {
  5.         width = columnWidth * formatTexts.size();
  6.     } else {
  7.         width = columnWidth * (getDesiredHeight(text.toString())
  8.                 / (height - getPaddingTop() - getPaddingBottom()) + 1);
  9.     }
  10.     if (width > widthSize) {
  11.         width = widthSize;
  12.     }
  13. }

上述操作均在onMeasure方法中,因此为了避免多次测量造成拆分后的文本重复,在每次拆分之前先清空formatTexts。

现在已经获得了按列拆分好的文本了,要绘制就变得简单了。

 
  1. @Override
  2. protected void onDraw(Canvas canvas) {
  3.     float x = width - getPaddingLeft() - getPaddingRight();
  4.     float y = getPaddingTop();
  5.     for (int i = 0; i < formatTexts.size(); i++) {
  6.         // 换列
  7.         x = i == 0 ? width - columnWidth + columnSpacing : x - columnWidth;
  8.         // 绘制每一列文本
  9.         for (int j = 0; j < formatTexts.get(i).length(); j++) {
  10.             // 向下移动绘制点
  11.             y = j == 0 ? charHeight - letterSpacing + getPaddingTop() : y + charHeight;
  12.             canvas.drawText(formatTexts.get(i), j, j + 1, x, y, textPaint);
  13.         }
  14.         if (leftLine) {
  15.             // 在每列文本之后绘制竖线
  16.             canvas.drawLine(- leftLinePadding, getPaddingTop(),
  17.                     x - leftLinePadding, y + letterSpacing, leftLinePaint);
  18.         }
  19.     }
  20. }

这篇关于竖排文本控件的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/779375

相关文章

Java实现将Markdown转换为纯文本

《Java实现将Markdown转换为纯文本》这篇文章主要为大家详细介绍了两种在Java中实现Markdown转纯文本的主流方法,文中的示例代码讲解详细,大家可以根据需求选择适合的方案... 目录方法一:使用正则表达式(轻量级方案)方法二:使用 Flexmark-Java 库(专业方案)1. 添加依赖(Ma

Linux使用cut进行文本提取的操作方法

《Linux使用cut进行文本提取的操作方法》Linux中的cut命令是一个命令行实用程序,用于从文件或标准输入中提取文本行的部分,本文给大家介绍了Linux使用cut进行文本提取的操作方法,文中有详... 目录简介基础语法常用选项范围选择示例用法-f:字段选择-d:分隔符-c:字符选择-b:字节选择--c

C#使用DeepSeek API实现自然语言处理,文本分类和情感分析

《C#使用DeepSeekAPI实现自然语言处理,文本分类和情感分析》在C#中使用DeepSeekAPI可以实现多种功能,例如自然语言处理、文本分类、情感分析等,本文主要为大家介绍了具体实现步骤,... 目录准备工作文本生成文本分类问答系统代码生成翻译功能文本摘要文本校对图像描述生成总结在C#中使用Deep

C#实现WinForm控件焦点的获取与失去

《C#实现WinForm控件焦点的获取与失去》在一个数据输入表单中,当用户从一个文本框切换到另一个文本框时,需要准确地判断焦点的转移,以便进行数据验证、提示信息显示等操作,本文将探讨Winform控件... 目录前言获取焦点改变TabIndex属性值调用Focus方法失去焦点总结最后前言在一个数据输入表单

通过C#获取PDF中指定文本或所有文本的字体信息

《通过C#获取PDF中指定文本或所有文本的字体信息》在设计和出版行业中,字体的选择和使用对最终作品的质量有着重要影响,然而,有时我们可能会遇到包含未知字体的PDF文件,这使得我们无法准确地复制或修改文... 目录引言C# 获取PDF中指定文本的字体信息C# 获取PDF文档中用到的所有字体信息引言在设计和出

Java操作xls替换文本或图片的功能实现

《Java操作xls替换文本或图片的功能实现》这篇文章主要给大家介绍了关于Java操作xls替换文本或图片功能实现的相关资料,文中通过示例代码讲解了文件上传、文件处理和Excel文件生成,需要的朋友可... 目录准备xls模板文件:template.xls准备需要替换的图片和数据功能实现包声明与导入类声明与

python解析HTML并提取span标签中的文本

《python解析HTML并提取span标签中的文本》在网页开发和数据抓取过程中,我们经常需要从HTML页面中提取信息,尤其是span元素中的文本,span标签是一个行内元素,通常用于包装一小段文本或... 目录一、安装相关依赖二、html 页面结构三、使用 BeautifulSoup javascript

lvgl8.3.6 控件垂直布局 label控件在image控件的下方显示

在使用 LVGL 8.3.6 创建一个垂直布局,其中 label 控件位于 image 控件下方,你可以使用 lv_obj_set_flex_flow 来设置布局为垂直,并确保 label 控件在 image 控件后添加。这里是如何步骤性地实现它的一个基本示例: 创建父容器:首先创建一个容器对象,该对象将作为布局的基础。设置容器为垂直布局:使用 lv_obj_set_flex_flow 设置容器

Level3 — PART 3 — 自然语言处理与文本分析

目录 自然语言处理概要 分词与词性标注 N-Gram 分词 分词及词性标注的难点 法则式分词法 全切分 FMM和BMM Bi-direction MM 优缺点 统计式分词法 N-Gram概率模型 HMM概率模型 词性标注(Part-of-Speech Tagging) HMM 文本挖掘概要 信息检索(Information Retrieval) 全文扫描 关键词

小程序button控件上下边框的显示和隐藏

问题 想使用button自带的loading图标功能,但又不需要button显示边框线 button控件有一条淡灰色的边框,在控件上了样式 border:none; 无法让button边框隐藏 代码如下: <button class="btn">.btn{border:none; /*一般使用这个就是可以去掉边框了*/} 解决方案 发现button控件有一个伪元素(::after