【时间盒子】-【5.绘制闹钟】动态绘制钟表和数字时间

2024-09-05 19:44

本文主要是介绍【时间盒子】-【5.绘制闹钟】动态绘制钟表和数字时间,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Tips:

  • @Preview装饰器,支持组件可预览;

  • @Component装饰器,自定义组件;

  • Canvas组件的使用;

  • 使用RenderingContext在Canvas组件上绘制图形,请参考官方文档:https://developer.huawei.com/consumer/cn/doc/harmonyos-references-V5/ts-canvasrenderingcontext2d-V5

一、自定义闹钟组件

  1. 新建component目录用来存放自定义组件,在其下再新建ArkTS文件,命名为ClockArea.ets。

2.页面主要包括画布组件Canvas,点击画布切换显示时钟表盘或数字时间样式,对其属性设置及布局如下。

@State showClock: boolean = true; // 是否显示时钟
private renderContextSettings: RenderingContextSettings = new RenderingContextSettings(true);
private renderContext: CanvasRenderingContext2D = new CanvasRenderingContext2D(this.renderContextSettings);
// 画布尺寸
private canvasSize: number = 252;
// 绘制间隔时间
private drawInterval: number = -1;
private clockRadius: number = this.canvasSize / 2 - 2;build() {Column() {Canvas(this.renderContext).width(this.canvasSize).aspectRatio(1).onClick(() => {this.showClock = !this.showClock;}).onReady(() => {if (this.drawInterval === -1) {// 开始绘制this.startDrawTask();}})}
}

 3.启动定时任务,每秒绘制一次闹钟,与时间同步秒针动态在走的效果。

/*** 启动定时绘制任务*/
private startDrawTask(): void {// console.log("开始绘制");this.renderContext.translate(this.canvasSize / 2, this.canvasSize / 2);this.drawClockArea()this.drawInterval = setInterval(() => {this.drawClockArea()}, 1000);
}

4.定义绘制闹钟区域的方法drawClockArea

/*** 绘制闹钟区域*/
private drawClockArea(): void {// console.log("绘制时区");let date = new Date();let hour = date.getHours();let minute = date.getMinutes();let second = date.getSeconds();this.renderContext.clearRect(-this.canvasSize,-this.canvasSize / 2,this.canvasSize * 2,this.canvasSize);if (this.showClock) {this.drawClockPan();this.drawClockHands(30 * (hour > 12 ? hour - 12 : hour)+ minute / 12 * 6,BaseConstant.HOUR_HAND_IMAGE_URL);this.drawClockHands(minute * 6, BaseConstant.MINUTE_HAND_IMAGE_URL);this.drawClockHands(second * 6, BaseConstant.SECOND_HAND_IMAGE_URL);} else {// 回显数字时间this.drawTimeHHMMSS(hour, minute, second);}
}

5.定义绘制闹钟的表盘的方法drawClockPan,其实就是一张表盘时刻的背景图。

/*** 绘制表盘*/
private drawClockPan(): void {let imgWidth = this.clockRadius * 2;let secondImg = new ImageBitmap(BaseConstant.CLOCK_PAN_IMAGE_URL);this.renderContext.beginPath();this.renderContext.drawImage(secondImg,-this.clockRadius,-this.clockRadius,imgWidth, imgWidth);this.renderContext.restore();
}

6.定义绘制表针的方法drawClockHands,包括时针、分针和秒针,每个针对应一个图片。

/*** 绘制表针:时针、分针、秒针*/
private drawClockHands(degree: number, handImgRes: string): void {let imgWidth = 10;let handImg = new ImageBitmap(handImgRes);let theta = (degree + 180) * Math.PI / 180;this.renderContext.save();this.renderContext.rotate(theta);this.renderContext.beginPath();this.renderContext.drawImage(handImg,-imgWidth / 2,-this.clockRadius,imgWidth,this.clockRadius * 2);this.renderContext.restore();
}

7.定义绘制数字时间的方法,时间的显示格式为24小时制HH:MM:SS。

/*** 绘制数字时间HH:MM:SS*/
private drawTimeHHMMSS(hour: number, minute: number, second: number): void {let hh = hour > 9 ? hour.toString() : "0" + hour;let mm = minute > 9 ? minute.toString() : "0" + minute;let ss = second > 9 ? second.toString() : "0" + second;let time = `${hh}:${mm}:${ss}`;this.renderContext.save();this.renderContext.font = SizeUtil.getPx($r("app.float.clock_time_font_size")) + "px";this.renderContext.beginPath();this.renderContext.textAlign = "center";this.renderContext.fillText(time, 0, 0);this.renderContext.restore();
}

到此,会发现以上代码缺少2个重要的类文件,分别是BaseConstant.ets和SizeUtil.ets,以及资源文件float.json的参数定义。

 

二、定义常量类文件BaseConstant.ets

新建目录constants,在其下新建ArkTS文件BaseConstant.ets。程序中的常量可以定义在这个类文件中,比如:图片路径等。

export class BaseConstant {static readonly CLOCK_PAN_IMAGE_URL: string = "images/icon_clock_pan.png";static readonly HOUR_HAND_IMAGE_URL: string = "images/icon_hour_hand.png";static readonly MINUTE_HAND_IMAGE_URL: string = "images/icon_minute_hand.png";static readonly SECOND_HAND_IMAGE_URL: string = "images/icon_second_hand.png";
}

三、定义单位转换类文件SizeUtil.ets

新建目录utils,在其下新建ArkTS文件SizeUtil.ets。为什么要封装这个单位转换公共类,可参考我的帖子:https://developer.huawei.com/consumer/cn/forum/topic/0208151714177357329?fid=0101587866109860105

import display from '@ohos.display';
import { GlobalContext } from './GlobalContext';let context = getContext(this);
const DESIGN_WIDTH = 360; // 设计稿宽度
const DESIGN_HEIGHT = 780; // 设计稿高度/*** 尺寸适配工具类*/
export default class SizeUtil {/*** 尺寸适配* @param value 设计稿尺寸*/static adaptSize(value: number): number {let deviceDisplay = GlobalContext.getContext().getObject("globalDisplay") as display.Display;let widthScale = deviceDisplay.width / DESIGN_WIDTH;let virtualHeight = DESIGN_HEIGHT * widthScale;let designDim = Math.sqrt(DESIGN_WIDTH * DESIGN_WIDTH + DESIGN_HEIGHT * DESIGN_HEIGHT);let virtualDim = Math.sqrt(deviceDisplay.width * deviceDisplay.width + virtualHeight * virtualHeight);return virtualDim * value / designDim; // 放缩后长度}/*** 获取px* @param value 设计稿尺寸*/static getPx(value: Resource): number {console.log("context:", context);let beforeVp = context.resourceManager.getNumber(value.id);return SizeUtil.adaptSize(beforeVp);}/*** 获取vp* @param value 设计稿尺寸*/static getVp(value: Resource): number {let beforeVp = context.resourceManager.getNumber(value.id);return px2vp(SizeUtil.adaptSize(beforeVp));}/*** 获取fp* @param value 设计稿尺寸*/static getFp(value: Resource): number {let beforeFp = context.resourceManager.getNumber(value.id);return px2fp(SizeUtil.adaptSize(beforeFp));}
}

四、定义全局上下文类文件GlobalContext.ets

在utils目录下新建ArkTS文件GlobalContext.ets。

/*** 全局上下文*/
export class GlobalContext {private constructor() {}private static instance: GlobalContext;private objects = new Map<string, Object>();/*** 获取全局上下文*/public static getContext(): GlobalContext {if (!GlobalContext.instance) {GlobalContext.instance = new GlobalContext();}return GlobalContext.instance;}/*** 获取对象*/getObject(name: string): Object | undefined {return this.objects.get(name);}/*** 设置对象*/setObject(key: string, objectClass: Object): void {this.objects.set(key, objectClass);}
}

五、资源文件float.json

在文件中定义常用的数值变量,比如:显示数字时间的字体大小。

{"name": "clock_time_font_size","value": "50"
},

请查阅官方文档:https://developer.huawei.com/consumer/cn/doc/harmonyos-guides/resource-usage-0000001820880417

六、图片文件

新建images目录,在其下添加已设计好的表盘、时针、分针和秒针的图片。

 (图片素材见文章顶部的附件)

七、运行效果

注意:需要在真机上才可见秒针走动的效果,及点击切换显示数字时钟。

 ​​​​​​

这篇关于【时间盒子】-【5.绘制闹钟】动态绘制钟表和数字时间的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

C# LiteDB处理时间序列数据的高性能解决方案

《C#LiteDB处理时间序列数据的高性能解决方案》LiteDB作为.NET生态下的轻量级嵌入式NoSQL数据库,一直是时间序列处理的优选方案,本文将为大家大家简单介绍一下LiteDB处理时间序列数... 目录为什么选择LiteDB处理时间序列数据第一章:LiteDB时间序列数据模型设计1.1 核心设计原则

MySQL按时间维度对亿级数据表进行平滑分表

《MySQL按时间维度对亿级数据表进行平滑分表》本文将以一个真实的4亿数据表分表案例为基础,详细介绍如何在不影响线上业务的情况下,完成按时间维度分表的完整过程,感兴趣的小伙伴可以了解一下... 目录引言一、为什么我们需要分表1.1 单表数据量过大的问题1.2 分表方案选型二、分表前的准备工作2.1 数据评估

基于Python实现数字限制在指定范围内的五种方式

《基于Python实现数字限制在指定范围内的五种方式》在编程中,数字范围限制是常见需求,无论是游戏开发中的角色属性值、金融计算中的利率调整,还是传感器数据处理中的异常值过滤,都需要将数字控制在合理范围... 目录引言一、基础条件判断法二、数学运算巧解法三、装饰器模式法四、自定义类封装法五、NumPy数组处理

MySQL中DATE_FORMAT时间函数的使用小结

《MySQL中DATE_FORMAT时间函数的使用小结》本文主要介绍了MySQL中DATE_FORMAT时间函数的使用小结,用于格式化日期/时间字段,可提取年月、统计月份数据、精确到天,对大家的学习或... 目录前言DATE_FORMAT时间函数总结前言mysql可以使用DATE_FORMAT获取日期字段

Python标准库datetime模块日期和时间数据类型解读

《Python标准库datetime模块日期和时间数据类型解读》文章介绍Python中datetime模块的date、time、datetime类,用于处理日期、时间及日期时间结合体,通过属性获取时间... 目录Datetime常用类日期date类型使用时间 time 类型使用日期和时间的结合体–日期时间(

Java获取当前时间String类型和Date类型方式

《Java获取当前时间String类型和Date类型方式》:本文主要介绍Java获取当前时间String类型和Date类型方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录Java获取当前时间String和Date类型String类型和Date类型输出结果总结Java获取

Python实现批量提取BLF文件时间戳

《Python实现批量提取BLF文件时间戳》BLF(BinaryLoggingFormat)作为Vector公司推出的CAN总线数据记录格式,被广泛用于存储车辆通信数据,本文将使用Python轻松提取... 目录一、为什么需要批量处理 BLF 文件二、核心代码解析:从文件遍历到数据导出1. 环境准备与依赖库

go动态限制并发数量的实现示例

《go动态限制并发数量的实现示例》本文主要介绍了Go并发控制方法,通过带缓冲通道和第三方库实现并发数量限制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录带有缓冲大小的通道使用第三方库其他控制并发的方法因为go从语言层面支持并发,所以面试百分百会问到