HarmonyOS角落里的知识:一杯冰美式的时间 -- 之打字机

news2024/11/24 11:11:32

一、前言

模拟编辑器或者模拟输入框中文字啪啦啪啦输入的效果,往往能够吸引人们的眼球,让用户的注意力聚焦在输入的内容上,本文将和大家探讨打字机效果的实现方式以及应用。Demo基于API12。

二、思路

拆分开来很简单,将字符串拆分,只需把要展示的文本进行切割,使用定时器不断追加文字即可。光标我们可以使用自带的TextArea来实现。

效果如下:

三、数据源

随便抄了一段文本:

     private targetTxt: string = `碧海青天夜夜心,闲云潭影日悠悠。
 花开堪折直须折,莫待无花空折枝。
 江南好,风景旧曾谙。日出江花红胜火,春来江水绿如蓝。能不忆江南?
 ​
 松风吹解带,山月照弹琴。
 遥望洞庭山水翠,白银盘里一青螺。
 人生若只如初见,何事秋风悲画扇?
 ​
 浮生若梦,为欢几何?
 长风破浪会有时,直挂云帆济沧海。
 岁月不居,时节如流。人生天地间,若白驹过隙,忽然而已。
 ​
 愿逐月华流照君,千里共婵娟。 `

分割:

 aboutToAppear(): void {
     this.targetTxtArray = this.targetTxt.split("");
 }

准备一个@State变量,用于显示UI

 @State currentTxt: string = ""

四、TextArea

使用TextArea要面对的就是输入框是有焦点和事件以及配套的键盘的,同时又需要光标。如下:

 TextArea({
     text: this.currentTxt
 })
     .width("auto")
     .height("auto")
     .animation({ duration: 200, curve: Curve.Smooth })
     .focusable(true)
     .defaultFocus(true)
     .enableKeyboardOnFocus(false)
     .fontColor("#fefae0")
     .caretColor("#d4a373")
     .backgroundColor("#ccd5ae")
     .hitTestBehavior(HitTestMode.None)
  • 使用focusable(true)defaultFocus(true)来获取焦点,达到显示光标的效果。
  • enableKeyboardOnFocus(false) 用于限制它弹出键盘
  • hitTestBehavior(HitTestMode.None)则屏蔽了所有的事件

我们得到了一个干净的,有光标的,无法操作的输入框~

五、setInterval

显然需要一个定时器来间隔添加文字。

 @State currentProgress: number = 0
 @State inputStepChar: number = 1
 private inputTxt(step: number) {
     if (this.intervalId != -1) {
         clearInterval(this.intervalId)
         this.intervalId = -1
     }
     this.intervalId = setInterval(() => {
          if (this.currentProgress >= this.targetTxtArray.length - 1) {
              clearInterval(this.intervalId)
              break
          } else {
              this.currentTxt += this.targetTxtArray[++ this.currentProgress]
          }
     }, step)
 }

因为可能被多次调用,我们需要存一个定时器ID,在后续的触发中将定时器清除。

step就是输入的间隔了,也就是输入速度。

每间隔一次,就在数组中取一个字符串,直到取完。

六、删除

因为有输入,那就可以有删除,也很简单。反过来就行了~

  private removeTxt(step: number) {
      if (this.intervalId != -1) {
          clearInterval(this.intervalId)
          this.intervalId = -1
      }
      this.intervalId = setInterval(() => {
          if (this.currentProgress <= 0) {
              clearInterval(this.intervalId)
          } else {
              this.currentProgress--
              this.currentTxt = this.currentTxt.substring(0, this.currentTxt.length - 1);
          }
      }, step)
  }

每间隔一次,就在currentTxt中移除最后一个字符,并currentProgress递减,直到删完。

七、删除、添加多个

默认是一个个增加,一个个删除。多个的话,我们直接点。使用一个For循环,将原有的逻辑套进去就好了。

 private inputTxt(step: number) {
     if (this.intervalId != -1) {
         clearInterval(this.intervalId)
         this.intervalId = -1
     }
     this.intervalId = setInterval(() => {
         for (let index = 0; index < this.inputStepChar; index++) {
             if (this.currentProgress >= this.targetTxtArray.length - 1) {
                 clearInterval(this.intervalId)
                 break
             } else {
                 this.currentTxt += this.targetTxtArray[++ this.currentProgress]
             }
         }
     }, step)
 }

inputStepChar就是每次改变的字符数了,想多少就多少。

八、最终代码

 let maxSpeed: number = 1000
 let minSpeed: number = 50
 let minStepChar: number = 1
 let maxStepChar: number = 10
 ​
 ​
 /**
  * @Des
  * @Author zyc
  * @Date 2024/5/30
  */
 @Component
 export struct TypeWriterComponent {
     private targetTxt: string = `碧海青天夜夜心,闲云潭影日悠悠。
 花开堪折直须折,莫待无花空折枝。
 江南好,风景旧曾谙。日出江花红胜火,春来江水绿如蓝。能不忆江南?
 ​
 松风吹解带,山月照弹琴。
 遥望洞庭山水翠,白银盘里一青螺。
 人生若只如初见,何事秋风悲画扇?
 ​
 浮生若梦,为欢几何?
 长风破浪会有时,直挂云帆济沧海。
 岁月不居,时节如流。人生天地间,若白驹过隙,忽然而已。
 ​
 愿逐月华流照君,千里共婵娟。 `
     private intervalId: number = -1
     private targetTxtArray: string[] = []
     private defInputSpeed: number = 200
     private defRemoveSpeed: number = 100
     @State inputStepChar: number = 1
     @State removeStepChar: number = 1
     @State currentProgress: number = 0
     @State currentTxt: string = ""
     @State inputSpeed: number = 0
     @State removeSpeed: number = 0
 ​
     aboutToAppear(): void {
         this.targetTxtArray = this.targetTxt.split("");
         this.defRemoveSpeed = Math.abs(this.defRemoveSpeed - maxSpeed) + minSpeed
         this.defInputSpeed = Math.abs(this.defInputSpeed - maxSpeed) + minSpeed
         this.removeSpeed = this.defRemoveSpeed
         this.inputSpeed = this.defInputSpeed
     }
 ​
     private inputTxt(step: number) {
         if (this.intervalId != -1) {
             clearInterval(this.intervalId)
             this.intervalId = -1
         }
         this.intervalId = setInterval(() => {z
             for (let index = 0; index < this.inputStepChar; index++) {
                 if (this.currentProgress >= this.targetTxtArray.length - 1) {
                     clearInterval(this.intervalId)
                     break
                 } else {
                     this.currentTxt += this.targetTxtArray[++ this.currentProgress]
                 }
             }
         }, step)
     }
 ​
     private removeTxt(step: number) {
         if (this.intervalId != -1) {
             clearInterval(this.intervalId)
             this.intervalId = -1
         }
         this.intervalId = setInterval(() => {
             for (let index = 0; index < this.removeStepChar; index++) {
                 if (this.currentProgress <= 0) {
                     clearInterval(this.intervalId)
                     break
                 } else {
                     this.currentProgress--
                     this.currentTxt = this.currentTxt.substring(0, this.currentTxt.length - 1);
                 }
             }
 ​
         }, step)
     }
 ​
     build() {
         Column({ space: 10 }) {
             TextArea({
                 text: this.currentTxt
             })
                 .width("auto")
                 .height("auto")
                 .animation({ duration: 200, curve: Curve.Smooth })
                 .focusable(true)
                 .defaultFocus(true)
                 .enableKeyboardOnFocus(false)
                 .fontColor("#fefae0")
                 .caretColor("#d4a373")
                 .backgroundColor("#ccd5ae")
                 .hitTestBehavior(HitTestMode.None)
 ​
             Blank()
             Row() {
                 Text("输入长度:").fontColor(Color.Black)
                 Slider({
                     style: SliderStyle.InSet,
                     value: 1,
                     step: 1,
                     min: minStepChar,
                     max: maxStepChar,
                 })
                     .layoutWeight(1)
                     .showSteps(true)
                     .stepSize(3)
                     .showTips(true, `${this.inputStepChar}`)
                     .selectedColor("#f07167")
                     .trackColor("#fdfcdc")
                     .stepColor("#fed9b7")
                     .onChange(value => {
                         this.inputStepChar = value
                         this.inputTxt(this.inputSpeed)
                     })
             }
 ​
             Row() {
                 Text("删除长度:").fontColor(Color.Black)
                 Slider({
                     style: SliderStyle.InSet,
                     value: 1,
                     step: 1,
                     min: minStepChar,
                     max: maxStepChar,
                 })
                     .layoutWeight(1)
                     .showSteps(true)
                     .stepSize(3)
                     .showTips(true, `${this.removeStepChar}`)
                     .selectedColor("#77bfa3")
                     .trackColor("#edeec9")
                     .stepColor("#bfd8bd")
                     .onChange(value => {
                         this.removeStepChar = value
                         this.removeTxt(this.inputSpeed)
                     })
             }
 ​
             Row() {
                 Text("输入速度:").fontColor(Color.Black)
                 Slider({
                     style: SliderStyle.InSet,
                     value: this.defInputSpeed,
                     step: 50,
                     min: minSpeed,
                     max: maxSpeed,
                 })
                     .layoutWeight(1)
                     .showSteps(true)
                     .stepSize(3)
                     .showTips(true, `${this.inputSpeed}`)
                     .selectedColor("#588157")
                     .trackColor("#dad7cd")
                     .stepColor("#a3b18a")
                     .onChange(value => {
                         this.inputSpeed = Math.abs(value - maxSpeed) + minSpeed
                         this.inputTxt(this.inputSpeed)
                     })
             }
 ​
             Row() {
                 Text("删除速度:").fontColor(Color.Black)
                 Slider({
                     style: SliderStyle.InSet,
                     value: this.defRemoveSpeed,
                     step: 50,
                     min: minSpeed,
                     max: maxSpeed,
                 })
                     .layoutWeight(1)
                     .showSteps(true)
                     .stepSize(3)
                     .showTips(true, `${this.removeSpeed}`)
                     .selectedColor("#ddb892")
                     .trackColor("#ede0d4")
                     .stepColor("#e6ccb2")
                     .onChange(value => {
                         this.removeSpeed = Math.abs(value - maxSpeed) + minSpeed
                         this.removeTxt(this.removeSpeed)
                     })
             }
 ​
             Row({ space: 30 }) {
                 Button("输出")
                     .onClick(() => {
                         this.inputTxt(this.inputSpeed)
                     })
                 Button("撤回")
                     .onClick(() => {
                         this.removeTxt(this.removeSpeed)
                     })
             }
         }
         .padding(horizontalBottom(20, 40))
         .size(matchSize)
     }
 }



最后

有很多小伙伴不知道学习哪些鸿蒙开发技术?不知道需要重点掌握哪些鸿蒙应用开发知识点?而且学习时频繁踩坑,最终浪费大量时间。所以有一份实用的鸿蒙(HarmonyOS NEXT)资料用来跟着学习是非常有必要的。 

鸿蒙HarmonyOS Next全套学习资料←点击领取!(安全链接,放心点击

这份鸿蒙(HarmonyOS NEXT)资料包含了鸿蒙开发必掌握的核心知识要点,内容包含了(ArkTS、ArkUI开发组件、Stage模型、多端部署、分布式应用开发、音频、视频、WebGL、OpenHarmony多媒体技术、Napi组件、OpenHarmony内核、Harmony南向开发、鸿蒙项目实战等等)鸿蒙(HarmonyOS NEXT)技术知识点。

希望这一份鸿蒙学习资料能够给大家带来帮助,有需要的小伙伴自行领取,限时开源,先到先得~无套路领取!!

鸿蒙(HarmonyOS NEXT)最新学习路线

有了路线图,怎么能没有学习资料呢,小编也准备了一份联合鸿蒙官方发布笔记整理收纳的一套系统性的鸿蒙(OpenHarmony )学习手册(共计1236页)与鸿蒙(OpenHarmony )开发入门教学视频,内容包含:ArkTS、ArkUI、Web开发、应用模型、资源分类…等知识点。

获取以上完整版高清学习路线,请点击→纯血版全套鸿蒙HarmonyOS学习资料

HarmonyOS Next 最新全套视频教程

《鸿蒙 (OpenHarmony)开发基础到实战手册》

OpenHarmony北向、南向开发环境搭建

《鸿蒙开发基础》

  • ArkTS语言
  • 安装DevEco Studio
  • 运用你的第一个ArkTS应用
  • ArkUI声明式UI开发
  • .……

《鸿蒙开发进阶》

  • Stage模型入门
  • 网络管理
  • 数据管理
  • 电话服务
  • 分布式应用开发
  • 通知与窗口管理
  • 多媒体技术
  • 安全技能
  • 任务管理
  • WebGL
  • 国际化开发
  • 应用测试
  • DFX面向未来设计
  • 鸿蒙系统移植和裁剪定制
  • ……

《鸿蒙进阶实战》

  • ArkTS实践
  • UIAbility应用
  • 网络案例
  • ……

大厂面试必问面试题

鸿蒙南向开发技术

鸿蒙APP开发必备

鸿蒙生态应用开发白皮书V2.0PDF

获取以上完整鸿蒙HarmonyOS学习资料,请点击→

纯血版全套鸿蒙HarmonyOS学习资料

总结
总的来说,华为鸿蒙不再兼容安卓,对中年程序员来说是一个挑战,也是一个机会。只有积极应对变化,不断学习和提升自己,他们才能在这个变革的时代中立于不败之地。 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1835716.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

高等数学笔记(三):导数

一、导数概念 1.1 导数的定义 1.1.1 函数在一点处的导数与导函数 1.1.2 单侧导数 1.2 导数的几何意义 1.3 函数可导性与连续性的关系 二、函数的求导法则 2.1 函数的和、差、积、商的求导法则 2.2 反函数的求导法则 2.3 复合函数的求导法则 2.4 基本求导法则与导数公式 三…

以太坊智能合约不能调用:一定注意智能合约地址,每次部署地址都会变化;nonce值 什么作用,是什么;在交易中调用智能合约添加附加信息

目录 以太坊智能合约不能调用 一定注意智能合约地址,每次部署地址都会变化 Transaction must include these fields: %r" % missing_keys 缺少nonce nonce值 什么作用,是什么 在交易中调用智能合约添加附加信息 1. 定义智能合约 2. 部署并调用智能合约 注意事项…

如何恢复iPhone iCloud云盘资料删除?给出建议

&#x1f3c6;本文收录于「Bug调优」专栏&#xff0c;主要记录项目实战过程中的Bug之前因后果及提供真实有效的解决方案&#xff0c;希望能够助你一臂之力&#xff0c;帮你早日登顶实现财富自由&#x1f680;&#xff1b;同时&#xff0c;欢迎大家关注&&收藏&&…

JAVAEE之网络原理(2)_传输控制协议(TCP)、概念、格式、确认应答及超时重传机制

前言 在上一节中&#xff0c;我们介绍了 UDP (用户数据报) 的相关知识&#xff0c;在这一节中我们将继续介绍传输层中另一种更为重要的协议。 一、什么是TCP协议&#xff1f; 1.1 TCP 基本概念 TCP协议全称&#xff1a;传输控制协议&#xff08;TCP&#xff0c;Transmission C…

redhat 7.8修改网卡名称,最佳实践

背景&#xff1a; 因业务需求&#xff0c;需要将新创建的redhat7.8服务器的网卡名称修改为ens160&#xff0c;目前服务器的网卡名称是ens192。 一、修改网卡配置信息&#xff1a; 查看当前网卡信息&#xff0c;并获取到网卡到Mac地址 [rootlocalhost ~]# ip addr 1: lo: <L…

【CT】LeetCode手撕—103. 二叉树的锯齿形层序遍历

目录 题目1- 思路2- 实现⭐103. 二叉树的锯齿形层序遍历——题解思路 2- ACM实现 题目 原题连接&#xff1a;103. 二叉树的锯齿形层序遍历 1- 思路 二叉树的层序遍历&#xff0c;遇到奇数时&#xff0c;利用 Collections.reverse() 翻转即可 2- 实现 ⭐103. 二叉树的锯齿形层…

QT修改界面图标及exe程序图标

目录 步骤1. 添加图标文件2. 添加保存图标变量3. 窗口启动初始化图标文件4. 构建后即可完成图标的更改 步骤 1. 添加图标文件 如下&#xff0c;添加一个名为 favicon.ico 的文件到.pro 工程文件所在的目录中。 2. 添加保存图标变量 RC_ICONS是一个变量&#xff0c;它被用于存储…

项目准备和启动

1.什么是项目建议书&#xff1f; 2.项目建议书的内容 3.可行性分析方法 4.项目组织结构&#xff08;职能型 项目型 矩阵型&#xff09; 5.项目管理层决策层执行层之间的关系 6.软件项目的可行性分析包括哪几个方面&#xff1f;影响决策的关键因素又是什么&#xff1f; 软件项目…

88. 合并两个有序数组(简单)

88. 合并两个有序数组 1. 题目描述2.详细题解3.代码实现3.1 Python3.2 Java 1. 题目描述 题目中转&#xff1a;88. 合并两个有序数组 2.详细题解 两个数组均有序&#xff08;非递减&#xff09;&#xff0c;要求合并两个数组&#xff0c;直观的思路&#xff0c;借助第三个数…

使用芯片为ZYNQ—7020,基于野火FPGA ZYNQ开发板

使用芯片为ZYNQ—7020&#xff0c;基于野火FPGA ZYNQ开发板 肤色模型简介 YCrCb也称为YUV&#xff0c;主要用于优化彩色视频信号的传输。与RGB视频信号传输相比&#xff0c;它最大的优点在于只需占用极少的频宽&#xff08;RGB要求三个独立的视频信号同时传输&#xff09;。其…

栈(Stack)

目录 一.栈&#xff08;Stack&#xff09; 1.概念 2.栈的使用 3.栈的模拟实现 二.栈相关习题 1.逆波兰表达式求值 &#xff08;1&#xff09;链接 &#xff08;2&#xff09;解析 &#xff08;3&#xff09;题解 2.括号匹配 &#xff08;1&#xff09;链接 &#xff…

【UIDynamic-动力学-UICollisionBehavior-碰撞行为-创建边界 Objective-C语言】

一、接下来,我们来说一个,碰撞的创建边界, 1.我们刚才呢,仅仅是让self.view,来变成边界, 实际上,这个边界呢,还可以自己去创建, 我们把之前的代码备份一份儿,改个名儿:05-碰撞行为-创建边界, 选中这一段儿,先删掉, command + R, 好,这一段儿,删掉啊, 接下来…

Altair 助力优化摩托车空气动力学性能,实现最佳的整流罩设计

案例简介 整流罩是绝大多数摩托车的重要组成部分&#xff0c;旨在提高车辆的空气动力学性能和稳定性。Altair 与 KTM 公司员工组成的项目团队&#xff0c;针对摩托车整流罩空气动力学方面的学生项目&#xff0c;展开了密切合作。 项目任务主要是对摩托车整流罩设计进行比较&…

基于QT和C++实现的中国象棋

一&#xff0c;源码 board.h #ifndef BOARD_H #define BOARD_H#include <QWidget> #include "Stone.h"class Board : public QWidget {Q_OBJECT public:explicit Board(QWidget *parent 0);bool _bRedTurn; // 红方先走int _currentPlayer; // 当前玩家&…

电子期刊制作秘籍:如何让你的出版物脱颖而出?

​如何让你的电子期刊在众多出版物中脱颖而出&#xff0c;吸引读者的目光呢&#xff1f;在微信公众号这个平台上&#xff0c;让你的电子期刊内容更具吸引力、专业性和创新性&#xff0c;是至关重要的。下面&#xff0c;我将教你制作电子期刊一些方法&#xff0c;助你打造出一本…

在得物的小程序生态实践

一、前言 提起微信小程序&#xff0c;相信所有人都不陌生&#xff0c;下面这个典型使用场景你一定经历过&#xff1a; 餐馆落座——微信扫桌角小程序码——使用微信小程序点餐&#x1f354; 微信小程序&#xff08;下文简称&#xff1a;小程序&#xff09;作为一种在微信平台…

怎么样判断真假单北斗

国产化替代正在中国各行各业逐步提升中&#xff0c;特别涉及重点产业——国家安全&#xff01; 只有仅支持B1I和B3信号的芯片才是真正的单北斗芯片。但凡你支持了B1C、B2a、B2b中的一个就是假的单北斗。 B1C/L1/E1、B2a/ L5/E5a、B2b/G3/E5b这些频点与其他GNSS系统是完全重合的…

举个栗子!Tableau 技巧(277):创建径向 WIFI 信号图

之前为大家分享了 &#x1f330;&#xff1a;学做径向柱状图&#xff08;Radial Column Chart&#xff09;。在此基础上&#xff0c;我们又做了进一步的延展&#xff1a;径向 WIFI 信号图。 它的用法和径向柱状图一致&#xff0c;区别在于它将柱体分切成多个弧线&#xff08;内…

豆包高质量声音有望复现-Seed-TTS

我们介绍了 Seed-TTS&#xff0c;这是一个大规模自回归文本转语音 &#xff08;TTS&#xff09; 模型系列&#xff0c;能够生成与人类语音几乎没有区别的语音。Seed-TTS 作为语音生成的基础模型&#xff0c;在语音上下文学习方面表现出色&#xff0c;在说话人的相似性和自然性方…

pg表空间和mysql表空间的区别

一、表空间的定义 1、在pg中表空间实际上是为表指定一个存储的目录。并且在创建数据库时可以为数据库指定默认的表空间。创建表和索引时可以指定表空间&#xff0c;这样表和索引就可以存储到表空间对应的目录下了。 在pg中一个库中可以有多个表空间&#xff0c;一个表空间可以…