「Mac畅玩鸿蒙与硬件22」鸿蒙UI组件篇12 - Canvas 组件的动态进阶应用

在鸿蒙应用中,Canvas 组件可以实现丰富的动态效果,适合用于动画和实时更新的场景。本篇将介绍如何在 Canvas 中实现动画循环、动态进度条、旋转和缩放动画,以及性能优化策略。

在这里插入图片描述


关键词
  • Canvas 组件
  • 动态绘制
  • 动画效果
  • 动态进度条
  • 旋转和缩放
  • 性能优化

一、使用定时器实现动画循环

通过定时更新画布内容,可以让图形在 Canvas 中产生动画效果。setTimeout 方法可用于实现帧刷新,使图形流畅移动。

1.1 动画循环示例:水平移动

以下代码展示了如何在 Canvas 上创建一个自动移动的圆形。

@Entry
@Component
struct MovingCircleExample {
  private x: number = 0; // 圆心的 x 坐标
  private dx: number = 5; // 每帧的水平移动距离
  private context: CanvasRenderingContext2D = new CanvasRenderingContext2D(); // 创建 Canvas 渲染上下文

  build() {
    Column() {
      Canvas(this.context)
        .width('100%')
        .height(600)
        .onReady(() => {
          this.animateCircle(); // 开始动画
        });
    }
  }

  private animateCircle(): void {
    this.context.clearRect(0, 0, 600, 600); // 清除画布

    this.context.beginPath(); // 开始新路径
    this.context.arc(this.x, 300, 50, 0, 2 * Math.PI); // 绘制圆形
    this.context.fillStyle = '#FF4500'; // 设置填充颜色为橙色
    this.context.fill(); // 填充圆形

    this.x += this.dx; // 更新圆心的 x 坐标
    if (this.x > 600 || this.x < 0) {
      this.dx = -this.dx; // 碰到边界时反向
    }

    // 使用 setTimeout 模拟帧刷新
    setTimeout(() => this.animateCircle(), 16); // 约60帧每秒
  }
}

效果示例:一个橙色圆形在 Canvas 中水平往返移动,碰到边界会反弹。
在这里插入图片描述


二、动态进度条

动态进度条是 Canvas 的常见应用。以下示例展示了如何使用 CanvassetInterval 创建一个动态的进度条,进度条会在画布中自动增加。

@Entry
@Component
struct AnimatedProgressBarExample {
  private progress: number = 0; // 进度条当前进度
  private context: CanvasRenderingContext2D = new CanvasRenderingContext2D(); // 创建 Canvas 渲染上下文

  build() {
    Column() {
      Canvas(this.context)
        .width('100%')
        .height(300)
        .onReady(() => {
          this.startProgress(); // 开始进度条
        });
    }
  }

  // 动态绘制进度条的方法
  startProgress() {
    setInterval(() => {
      this.context.clearRect(0, 0, 600, 300); // 清除画布

      // 绘制进度条背景
      this.context.fillStyle = '#D3D3D3'; // 设置背景颜色为浅灰色
      this.context.fillRect(50, 100, 500, 50); // 绘制进度条背景

      // 绘制进度条
      this.context.fillStyle = '#32CD32'; // 设置进度条颜色为绿色
      this.context.fillRect(50, 100, (500 * this.progress) / 100, 50); // 绘制当前进度

      // 显示进度百分比文本
      this.context.font = '24px sans-serif'; // 设置字体样式
      this.context.fillStyle = '#000000'; // 设置文本颜色为黑色
      this.context.fillText(`${this.progress}%`, 280, 135); // 绘制进度文本

      this.progress = (this.progress + 1) % 101; // 更新进度
    }, 50); // 每50毫秒更新一次
  }
}

效果示例:一个绿色的进度条在 Canvas 上动态增加,并显示当前百分比。

在这里插入图片描述


三、复杂动画示例:旋转和缩放

结合 rotatescale 方法,可以在 Canvas 中实现图形的旋转和缩放动画。

@Entry
@Component
struct RotateAndScaleAnimation {
  private scale1: number = 1; // 当前缩放比例
  private rotation: number = 0; // 当前旋转角度
  private context: CanvasRenderingContext2D = new CanvasRenderingContext2D(); // 创建 Canvas 渲染上下文

  build() {
    Column() {
      Canvas(this.context)
        .width('100%')
        .height(600)
        .onReady(() => {
          this.animateShape(); // 开始动画
        });
    }
  }

  private animateShape(): void {
    this.context.clearRect(0, 0, 600, 600); // 清除画布
    this.context.save(); // 保存当前绘图状态

    // 平移到中心点
    this.context.translate(300, 300); // 将原点移动到 (300, 300)
    this.context.rotate(this.rotation); // 应用旋转
    this.context.scale(this.scale1, this.scale1); // 应用缩放

    // 绘制图形
    this.context.fillStyle = '#4682B4'; // 设置填充颜色为蓝色
    this.context.fillRect(-50, -50, 100, 100); // 绘制中心点为 (300, 300) 的方块

    this.context.restore(); // 恢复到之前的状态

    // 更新旋转角度和缩放比例
    this.rotation += 0.1; // 每帧增加旋转角度
    this.scale1 = this.scale1 >= 1.5 ? 1 : this.scale1 + 0.01; // 循环缩放

    // 使用 setTimeout 模拟帧刷新
    setTimeout(() => this.animateShape(), 16); // 约60帧每秒
  }
}

效果示例:一个蓝色的方块在画布上旋转并缩放,循环往复。

在这里插入图片描述


四、性能优化策略

当应用复杂图形或大量动画时,优化 Canvas 性能变得至关重要。以下是几种常用的性能优化策略。

4.1 减少不必要的重绘

在动态绘制中,避免重复重绘不需要更新的区域。使用 clearRect 精确清除指定区域,减少全局重绘的开销。

@Entry
@Component
struct OptimizedDrawingExample {
  private x: number = 0; // 圆心的 x 坐标
  private context: CanvasRenderingContext2D = new CanvasRenderingContext2D(); // 创建 Canvas 渲染上下文

  build() {
    Canvas(this.context)
      .width('100%')
      .height(500)
      .onReady(() => {
        this.animate(); // 开始动画
      });
  }

  private animate = () => {
    // 清除上一帧的圆形
    this.context.clearRect(this.x - 2, 150, 52, 52); // 精确清除指定区域

    this.x += 2; // 更新 x 坐标
    this.context.beginPath(); // 开始新路径
    this.context.arc(this.x, 200, 25, 0, 2 * Math.PI); // 绘制圆形
    this.context.fillStyle = '#32CD32'; // 设置填充颜色为绿色
    this.context.fill(); // 填充圆形

    if (this.x > 500) this.x = 0; // 碰到边界时重置位置

    setTimeout(this.animate, 16); // 约60帧每秒
  }
}

效果示例:圆形在水平移动时仅清除上一位置的小范围区域,从而优化重绘性能。

在这里插入图片描述


4.2 使用离屏 Canvas

使用离屏 Canvas 可以在不影响主线程的情况下进行复杂的图形运算,绘制完成后再一次性渲染到主 Canvas 上。

@Entry
@Component
struct OffscreenCanvasExample {
  private offscreenCanvas: OffscreenCanvas = new OffscreenCanvas(500, 500); // 定义离屏 Canvas
  private offscreenContext!: OffscreenCanvasRenderingContext2D; // 声明离屏 Canvas 渲染上下文
  private context: CanvasRenderingContext2D = new CanvasRenderingContext2D(); // 创建主 Canvas渲染上下文

  build() {
    Canvas(this.context)
      .width('100%')
      .height(500)
      .onReady(() => {
        this.offscreenContext = this.offscreenCanvas.getContext('2d')!; // 获取离屏 Canvas 渲染上下文
        this.drawOffscreen(); // 绘制离屏 Canvas
        const imageBitmap = this.offscreenCanvas.transferToImageBitmap(); // 转换为 ImageBitmap
        this.context.drawImage(imageBitmap, 0, 0); // 将离屏 Canvas 绘制到主 Canvas
      });
  }

  private drawOffscreen() {
    this.offscreenContext.fillStyle = '#8A2BE2'; // 设置填充颜色为紫色
    this.offscreenContext.fillRect(50, 50, 400, 400); // 在离屏 Canvas 上绘制矩形

    this.offscreenContext.beginPath(); // 开始新路径
    this.offscreenContext.arc(250, 250, 100, 0, 2 * Math.PI); // 绘制圆形
    this.offscreenContext.fillStyle = '#FFD700'; // 设置填充颜色为金色
    this.offscreenContext.fill(); // 填充圆形
  }
}

效果示例:复杂图形先在离屏 Canvas 中完成,再一次性绘制到主 Canvas 上,提升了渲染性能。

在这里插入图片描述


小结

本篇介绍了鸿蒙 Canvas 组件的动态进阶应用,包括动画循环、动态进度条、旋转和缩放动画,以及性能优化策略。通过这些技术,开发者可以实现流畅的动态效果,为应用带来更生动的视觉体验。


下一篇预告

在下一篇中,我们将探索鸿蒙的自定义组件功能,学习如何创建可复用的自定义组件来提升代码的灵活性和复用性。


上一篇:「Mac畅玩鸿蒙与硬件21」鸿蒙UI组件篇11 - Canvas 组件的静态进阶应用
下一篇:「Mac畅玩鸿蒙与硬件23」鸿蒙UI组件篇13 - 自定义组件的创建与使用

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

大家知道输电线路微风振动在线监测有哪些先进技术?

特力康TLKS-PMG-WD输电线路微风振动在线监测装置&#xff08;输电线路北斗导线舞动在线监测装置&#xff09;集成了多项先进技术&#xff0c;堪称输电线路监测领域的佼佼者&#xff01;它利用高精度的舞动传感器实时监测导线数据&#xff0c;并通过无线网络发送到监控中心&…

WPF+MVVM案例实战(十七)- 自定义字体图标按钮的封装与实现(ABC类)

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 1、案例效果1、按钮分类2、ABC类按钮实现1、文件创建2、字体图标资源3、自定义依赖属性4、按钮特效样式实现 3、按钮案例演示1、页面实现与文件创建2、依赖注入3 运…

【算法】(Python)贪心算法

贪心算法&#xff1a; 又称贪婪算法&#xff0c;greedy algorithm。贪心地追求局部最优解&#xff0c;即每一步当前状态下最优选择。试图通过各局部最优解达到最终全局最优解。但不从整体最优上考虑&#xff0c;不一定全局最优解。步骤&#xff1a;从初始状态拆分成一步一步的…

01简介——基于全志V3S的Linux开发板教程笔记

声明&#xff1a;本笔记内容为个人在使用自制的基于全志V3S的Linux开发板的学习笔记文章&#xff0c;仅用于记录学习与开发过程中的问题处理过程、方法操作记录、参考的网络资源等内容。 一、前言 一次偶然的机会&#xff0c;发现了全志V3S这款芯片&#xff0c;基于Cortex-A7内…

【数据库】elasticsearch

1、架构 es会为每个索引创建一定数量的主分片和副本分片。 分片&#xff08;Shard&#xff09;&#xff1a; 将索引数据分割成多个部分&#xff0c;每个部分都是一个独立的索引。 主要目的是实现数据的分布式存储和并行处理&#xff0c;从而提高系统的扩展性和性能。 在创建索…

C6.【C++ Cont】cout的格式输出

目录 1.头文件 2.使用 1.控制宽度和填充 setw函数(全称set field width设置字段宽度) setfill函数(全称Set fill character设置填充字符) 2.控制数值格式 3.控制整数格式 4.控制对齐方式 1.头文件 用cout进行格式化输出前,先引用头文件iomanip(全称input&output m…

【Unity】Unity拖拽在Android设备有延迟和卡顿问题的解决

一、介绍 在制作Block类游戏时&#xff0c;其核心的逻辑就是拖拽方块放入到地图中&#xff0c;这里最先想到的就是Unity的拖拽接口IDragHandler,然后通过 IPointerDownHandler, IPointerUpHandler 这两个接口判断按下和松手&#xff0c;具体的实现逻辑就是下面 public void On…

零基础快速入门MATLAB

文章目录 前言1.向量1.1 创建方式1.1.1 直接输入各个元素1.1.2 冒号创建1.1.3 使用linspace函数 1.2 向量的运算1.2.1 加法1.2.2 相乘 2.输入与输出2.1 输入函数--input()2.2 输出函数 3.分支结构3.1 if语句3.2 switch语句 4.循环结构4.1 for循环4.2 while循环4.3 特殊语句 5.函…

gitmakegdb

git git reset 命令 | 菜鸟教程 (runoob.com) 像嫁接一样 make Makefile | 爱编程的大丙 (subingwen.cn) # 举例: 有源文件 a.c b.c c.c head.h, 需要生成可执行程序 app ################# 例1 ################# app:a.c b.c c.cgcc a.c b.c c.c -o app################# 例…

记一次微信云托管搭建Redis服务

背景 最近在做一个微信小程序&#xff0c;规划服务全部部署在云托管上面&#xff0c;本次使用了对象存储、mysql、java服务、Redis服务&#xff08;pc端用的&#xff09;。 由于对部署Redis不理解&#xff0c;查看了官方文档&#xff0c;首先看到的是这个架构图&#xff0c;看…

gerrit 搭建遇到的问题

1、启动Apache&#xff0c;端口被占用 : AH00072: make sock: could not bind to address (0S 10048)通常每个套接字地址(协议/网络地址/端口)只允许使用一次。: AH00072: make sock: could not bind to address 0.0.0.:443 a AH00451: no listening sockets available, shutti…

STM32之看门狗

STM32有独立看门狗&#xff08;IWDG&#xff09;和窗口看门狗(WWDG)。 采用窗口看门狗&#xff08;WWDG&#xff09;&#xff0c;有一个死前中断&#xff0c;可以用来作一个报警的功能。 独立看门狗超时时间计算公式 假设LSI是32KHz,超时时间等于 预分频系数&#xff08;4&…

Python爬虫基础-正则表达式!

前言 正则表达式是对字符串的一种逻辑公式&#xff0c;用事先定义好的一些特定字符、及这些特定字符的组合&#xff0c;组成一个“规则的字符串”&#xff0c;此字符串用来表示对字符串的一种“过滤”逻辑。正在在很多开发语言中都存在&#xff0c;而非python独有。对其知识点…

lvgl白屏问题(LCD长时间白屏)和优化lvgl

开机白屏时间过长 -- 这里我们不考虑是lvgl占的内存太大的问题&#xff0c;这里考虑的是为什么lcd屏幕启动后会有长时间的白屏。 首先我们要了解lvgl的相关操作&#xff0c;主要集中在一个函数中。只有程序执行到了这个函数&#xff0c;lvgl的屏幕才会显现出来 总结来说就是l…

雷池社区版 7.1.0 LTS 发布了

LTS&#xff08;Long Term Support&#xff0c;长期支持版本&#xff09;是软件开发中的一个概念&#xff0c;表示该版本将获得较长时间的支持和更新&#xff0c;通常包含稳定性、性能改进和安全修复&#xff0c;但不包含频繁的新特性更新。 作为最受欢迎的社区waf&#xff0c…

【系统分析师】-案例综合知识大全

1、表示处理流程的工具 图形工具、表格工具和语言工具。 其中常见的图形工具包括程序流程图、IPO 图、盒图、问题分析图、判定树&#xff0c; 表格工具包括判定表&#xff0c; 语言工具包括过程设计语言 2、用例建模过程 识别参与者、合并需求获得用例、细化用例描述和调…

python爬取旅游攻略(1)

参考网址&#xff1a; https://blog.csdn.net/m0_61981943/article/details/131262987 导入相关库&#xff0c;用get请求方式请求网页方式&#xff1a; import requests import parsel import csv import time import random url fhttps://travel.qunar.com/travelbook/list.…

G. Welcome to Join the Online Meeting!【CCPC2024哈尔滨站】

G. Welcome to Join the Online Meeting 思路: 挺简单的BFS思路 图论题写的比较少&#xff0c;算是补题吧 代码: #include <bits/stdc.h> #define endl \n #define int long long #define pb push_back #define pii pair<int,int> const int MOD 1e97; const …

《图像滤波算法综述》

一、引言 在数字图像处理的世界里&#xff0c;滤波是一项关键技术。通过对图像应用滤波算法&#xff0c;可以有效去除噪声、增强图像的细节并显著提升图像质量。本篇内容将为您深入介绍几种常见的图像滤波算法及其原理和应用场景。 二、图像滤波算法的分类 图像滤波算法可以…

RK3568开发板静态IP地址配置

1. 连接SSH MYD-LR3568 开发板设置了静态 eth0:1 192.168.0.10 和 eth1:1 192.168.1.10&#xff0c;在没有串口时调试开发板&#xff0c;可以用工具 SSH 登陆到开发板。 首先需要用一根网线直连电脑和开发板&#xff0c;或者通过路由器连接到开发板&#xff0c;将电脑 IP 手动设…