【时间盒子】-【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

相关文章

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

如何利用Java获取当天的开始和结束时间

《如何利用Java获取当天的开始和结束时间》:本文主要介绍如何使用Java8的LocalDate和LocalDateTime类获取指定日期的开始和结束时间,展示了如何通过这些类进行日期和时间的处... 目录前言1. Java日期时间API概述2. 获取当天的开始和结束时间代码解析运行结果3. 总结前言在J

Java数字转换工具类NumberUtil的使用

《Java数字转换工具类NumberUtil的使用》NumberUtil是一个功能强大的Java工具类,用于处理数字的各种操作,包括数值运算、格式化、随机数生成和数值判断,下面就来介绍一下Number... 目录一、NumberUtil类概述二、主要功能介绍1. 数值运算2. 格式化3. 数值判断4. 随机

修改若依框架Token的过期时间问题

《修改若依框架Token的过期时间问题》本文介绍了如何修改若依框架中Token的过期时间,通过修改`application.yml`文件中的配置来实现,默认单位为分钟,希望此经验对大家有所帮助,也欢迎... 目录修改若依框架Token的过期时间修改Token的过期时间关闭Token的过期时js间总结修改若依

Go Mongox轻松实现MongoDB的时间字段自动填充

《GoMongox轻松实现MongoDB的时间字段自动填充》这篇文章主要为大家详细介绍了Go语言如何使用mongox库,在插入和更新数据时自动填充时间字段,从而提升开发效率并减少重复代码,需要的可以... 目录前言时间字段填充规则Mongox 的安装使用 Mongox 进行插入操作使用 Mongox 进行更

Java使用POI-TL和JFreeChart动态生成Word报告

《Java使用POI-TL和JFreeChart动态生成Word报告》本文介绍了使用POI-TL和JFreeChart生成包含动态数据和图表的Word报告的方法,并分享了实际开发中的踩坑经验,通过代码... 目录前言一、需求背景二、方案分析三、 POI-TL + JFreeChart 实现3.1 Maven

Java导出Excel动态表头的示例详解

《Java导出Excel动态表头的示例详解》这篇文章主要为大家详细介绍了Java导出Excel动态表头的相关知识,文中的示例代码简洁易懂,具有一定的借鉴价值,有需要的小伙伴可以了解下... 目录前言一、效果展示二、代码实现1.固定头实体类2.动态头实现3.导出动态头前言本文只记录大致思路以及做法,代码不进

vue基于ElementUI动态设置表格高度的3种方法

《vue基于ElementUI动态设置表格高度的3种方法》ElementUI+vue动态设置表格高度的几种方法,抛砖引玉,还有其它方法动态设置表格高度,大家可以开动脑筋... 方法一、css + js的形式这个方法需要在表格外层设置一个div,原理是将表格的高度设置成外层div的高度,所以外层的div需要

对postgresql日期和时间的比较

《对postgresql日期和时间的比较》文章介绍了在数据库中处理日期和时间类型时的一些注意事项,包括如何将字符串转换为日期或时间类型,以及在比较时自动转换的情况,作者建议在使用数据库时,根据具体情况... 目录PostgreSQL日期和时间比较DB里保存到时分秒,需要和年月日比较db里存储date或者ti

SpringBoot实现动态插拔的AOP的完整案例

《SpringBoot实现动态插拔的AOP的完整案例》在现代软件开发中,面向切面编程(AOP)是一种非常重要的技术,能够有效实现日志记录、安全控制、性能监控等横切关注点的分离,在传统的AOP实现中,切... 目录引言一、AOP 概述1.1 什么是 AOP1.2 AOP 的典型应用场景1.3 为什么需要动态插