【时间盒子】-【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 controller接口出入参时间序列化转换操作方法(两种)

《Javacontroller接口出入参时间序列化转换操作方法(两种)》:本文主要介绍Javacontroller接口出入参时间序列化转换操作方法,本文给大家列举两种简单方法,感兴趣的朋友一起看... 目录方式一、使用注解方式二、统一配置场景:在controller编写的接口,在前后端交互过程中一般都会涉及

python处理带有时区的日期和时间数据

《python处理带有时区的日期和时间数据》这篇文章主要为大家详细介绍了如何在Python中使用pytz库处理时区信息,包括获取当前UTC时间,转换为特定时区等,有需要的小伙伴可以参考一下... 目录时区基本信息python datetime使用timezonepandas处理时区数据知识延展时区基本信息

springboot使用Scheduling实现动态增删启停定时任务教程

《springboot使用Scheduling实现动态增删启停定时任务教程》:本文主要介绍springboot使用Scheduling实现动态增删启停定时任务教程,具有很好的参考价值,希望对大家有... 目录1、配置定时任务需要的线程池2、创建ScheduledFuture的包装类3、注册定时任务,增加、删

Python的time模块一些常用功能(各种与时间相关的函数)

《Python的time模块一些常用功能(各种与时间相关的函数)》Python的time模块提供了各种与时间相关的函数,包括获取当前时间、处理时间间隔、执行时间测量等,:本文主要介绍Python的... 目录1. 获取当前时间2. 时间格式化3. 延时执行4. 时间戳运算5. 计算代码执行时间6. 转换为指

SpringBoot基于配置实现短信服务策略的动态切换

《SpringBoot基于配置实现短信服务策略的动态切换》这篇文章主要为大家详细介绍了SpringBoot在接入多个短信服务商(如阿里云、腾讯云、华为云)后,如何根据配置或环境切换使用不同的服务商,需... 目录目标功能示例配置(application.yml)配置类绑定短信发送策略接口示例:阿里云 & 腾

Java中字符串转时间与时间转字符串的操作详解

《Java中字符串转时间与时间转字符串的操作详解》Java的java.time包提供了强大的日期和时间处理功能,通过DateTimeFormatter可以轻松地在日期时间对象和字符串之间进行转换,下面... 目录一、字符串转时间(一)使用预定义格式(二)自定义格式二、时间转字符串(一)使用预定义格式(二)自

Python实现特殊字符判断并去掉非字母和数字的特殊字符

《Python实现特殊字符判断并去掉非字母和数字的特殊字符》在Python中,可以通过多种方法来判断字符串中是否包含非字母、数字的特殊字符,并将这些特殊字符去掉,本文为大家整理了一些常用的,希望对大家... 目录1. 使用正则表达式判断字符串中是否包含特殊字符去掉字符串中的特殊字符2. 使用 str.isa

Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码

《Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码》:本文主要介绍Java中日期时间转换的多种方法,包括将Date转换为LocalD... 目录一、Date转LocalDateTime二、Date转LocalDate三、LocalDateTim

MySQL中动态生成SQL语句去掉所有字段的空格的操作方法

《MySQL中动态生成SQL语句去掉所有字段的空格的操作方法》在数据库管理过程中,我们常常会遇到需要对表中字段进行清洗和整理的情况,本文将详细介绍如何在MySQL中动态生成SQL语句来去掉所有字段的空... 目录在mysql中动态生成SQL语句去掉所有字段的空格准备工作原理分析动态生成SQL语句在MySQL

golang获取当前时间、时间戳和时间字符串及它们之间的相互转换方法

《golang获取当前时间、时间戳和时间字符串及它们之间的相互转换方法》:本文主要介绍golang获取当前时间、时间戳和时间字符串及它们之间的相互转换,本文通过实例代码给大家介绍的非常详细,感兴趣... 目录1、获取当前时间2、获取当前时间戳3、获取当前时间的字符串格式4、它们之间的相互转化上篇文章给大家介