前言

在鸿蒙OS的生态中,开发者们不断探索和创新,以期为用户带来更丰富的交互体验。最近,我萌生了一个想法:制作一个能够随着声音动态变化的话筒组件。尽管网络上缺乏现成的参考案例,但我决定亲自动手,将这一创意变为现实。本文将深入解析这一开发过程,分享我的实战经验和技术细节。

一、前期准备

在开始之前,我们需要准备两张图片,它们将作为话筒动态效果的基础。将这些图片放置在项目的entry/ets目录下,并在代码中进行相应的配置。



最后效果:

代码示例:

private img: ImageBitmap = new ImageBitmap("image/img.png");
private img2: ImageBitmap = new ImageBitmap("image/img_1.png");

二、理解绘制所需类

为了实现动态效果,我们需要了解并使用一些关键的类和对象。这些类将帮助我们在Canvas上进行绘制和效果处理。

RenderingContextSettings:用于配置CanvasRenderingContext2D对象,包括是否开启抗锯齿功能。

private settings: RenderingContextSettings = new RenderingContextSettings(true);

CanvasRenderingContext2D:用于创建绘图上下文,通过它我们可以在Canvas中绘制图形。

private context: CanvasRenderingContext2D = new CanvasRenderingContext2D(this.settings);

OffscreenCanvas:一个离屏画布,允许我们在不影响主线程的情况下进行绘制操作。

private offCanvas: OffscreenCanvas = new OffscreenCanvas(600, 600);

三、制定话筒显示效果规则

话筒的动态效果将根据声音的强度变化。我们定义了五个级别,从0到5,分别代表声音的不同强度。为了模拟这一效果,我们编写了一个函数来随机生成声音等级。

模拟声音等级函数:

generateRandomNumberUpToFive(): number {
const randomInt = Math.floor(Math.random() * 5); // 0, 1, 2, 3, 4
// 有1/5的概率返回5
if (Math.random() < 0.2) {
return 5;
}
return randomInt;
}

为了实现动态效果,我们使用一个定时器每隔一段时间更新声音等级,并在组件初始化时开始这一过程。

组件初始化:

@State @Watch('draw') level: number = 3;

aboutToAppear(): void {
setInterval(() => {
this.level = this.generateRandomNumberUpToFive();
}, 300);
}

四、关键代码

在实现动态效果时,我们需要先清除之前的绘制效果,然后根据声音等级绘制话筒的动态层效果。最终,将这些效果叠加在一起,展示出最终的动态效果。

绘制逻辑:

draw() {
this.context.clearRect(0, 0, 192, 192);
let offContext = this.offCanvas.getContext("2d", this.settings);
offContext.drawImage(this.img2, 0, 0, 192, 192);
let h = this.level * 192 / 5;
let imagedata = offContext.getImageData(0, h, 192, 192);
offContext.drawImage(this.img, 0, 0, 192, 192);
offContext.putImageData(imagedata, 0, h);
let image = this.offCanvas.transferToImageBitmap();
this.context.transferFromImageBitmap(image);
}

五、整体代码效果

将上述代码整合到一个组件中,确保在组件初始化时绘制背景,并在需要时更新动态效果。

完整组件代码:

@Component
struct CanvasExample1 {
private settings: RenderingContextSettings = new RenderingContextSettings(true);
private context: CanvasRenderingContext2D = new CanvasRenderingContext2D(this.settings);
private offCanvas: OffscreenCanvas = new OffscreenCanvas(600, 600);
private img: ImageBitmap = new ImageBitmap("image/img.png");
private img2: ImageBitmap = new ImageBitmap("image/img_1.png"); @State @Watch('draw') level: number = 3; draw() {
this.context.clearRect(0, 0, 192, 192);
let offContext = this.offCanvas.getContext("2d", this.settings);
offContext.drawImage(this.img2, 0, 0, 192, 192);
let h = this.level * 192 / 5;
let imagedata = offContext.getImageData(0, h, 192, 192);
offContext.drawImage(this.img, 0, 0, 192, 192);
offContext.putImageData(imagedata, 0, h);
let image = this.offCanvas.transferToImageBitmap();
this.context.transferFromImageBitmap(image);
} generateRandomNumberUpToFive(): number {
const randomInt = Math.floor(Math.random() * 5); // 0, 1, 2, 3, 4
if (Math.random() < 0.2) {
return 5;
}
return randomInt;
} aboutToAppear(): void {
setInterval(() => {
this.level = this.generateRandomNumberUpToFive();
}, 300);
} build() {
Flex({ direction: FlexDirection.Column, alignItems: ItemAlign.Center, justifyContent: FlexAlign.Center }) {
Canvas(this.context)
.width('100%')
.height('100%')
.onReady(() => {
let offContext = this.offCanvas.getContext("2d", this.settings);
offContext.drawImage(this.img, 0, 0, 192, 192);
});
}
.width('100%')
.height('100%');
}
}

总结

通过本文的详细解析,我们不仅实现了一个动态声控话筒组件,还深入了解了鸿蒙OS中的绘图API和动态效果实现方法。希望这篇文章能够为其他开发者提供灵感和指导,共同推动鸿蒙生态的创新与发展。

鸿蒙OS创新实践:动态声控话筒开发指南的更多相关文章

  1. 终于有人把鸿蒙OS讲明白了,大佬讲解!快收藏!

    来自 | GitHub科技 本文面向的是开发人员,主要想通过科普让大家了解一下鸿蒙开发.接下来,我想给大家科普一下这个这么火的鸿蒙系统. 到底什么是鸿蒙 OS 在官网上看到鸿蒙 OS 的简介是,分布式 ...

  2. 鸿蒙OS前端开发入门指南:网络图片_Image渲染网络图片

    目录: 1.开启明文传输 2.权限申请 3.引入http插件 4.案例展示 5.<鸿蒙OS前端开发入门指南>文章合集 开启明文传输 在config.json配置文件添加如下配置(如果不开启 ...

  3. 华为鸿蒙OS发布!方舟支持混合编译,终将可替换安卓?

    前言 有关于鸿蒙的消息之前也有说过,就在昨天下午,华为举行了2019开发大会,正式推出了鸿蒙os系统(Harmony).其相关负责人表示,也是基于微软内核的全场景分布式OS   鸿蒙凭借微内核的优势, ...

  4. 鸿蒙OS

    8月9日,华为消费者业务在其全球开发者大会上正式发布其全新的基于微内核的面向全场景的分布式操作系统——鸿蒙OS(HarmonyOS)! 鸿蒙的定义是基于微内核的全场景分布式操作系统.其中,微内核是技术 ...

  5. 【转帖】知乎关于鸿蒙OS的思考

    作者:bacon xu链接:https://www.zhihu.com/question/339567108/answer/782431141来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非 ...

  6. 阿里聚安全受邀参加SFDC安全大会,分享互联网业务面临问题和安全创新实践

    现今,技术引领的商业变革已无缝渗透入我们的日常生活,「技术改变生活」的开发者们被推向了创新浪潮的顶端.国内知名的开发者技术社区 SegmentFault 至今已有四年多了,自技术问答开始,他们已经发展 ...

  7. 鸿蒙OS与微内核

    目前主流面向个人端操作系统都不是像鸿蒙一样采用微内核的,或许在这些操作系统上诞生时面对的需求场景用微内核是行不通的.但在当前场景下是不是能走通要实际验证才知道.面对当前操作系统已成熟的市场生态.专利技 ...

  8. 鸿蒙OS与谷歌Fuchsia

    鸿蒙,意在“开天辟地”,它的征程是物联网.跨终端,是一款战略性产品.它真正对标的不是安卓,而是谷歌最新研发的操作系统Fuchsia. 根据Fuchsia中文社区的介绍,在安卓和 Chrome OS 两 ...

  9. 5G能带来什么改变-从鸿蒙OS说起

    背景 从5G投票事件开始,开始关注5G.许多文章都说到5G的特点有速度快.时延低,其中,时延低是最重要的特点.然而,时延低能给社会带来什么改变呢? 2G是短信的时代,3G促成了语音视频,4G促成了短视 ...

  10. [github] 关于华为鸿蒙OS

    English Docs | 中文文档 | Türkçe Dökümanlar HarmonyOS Ⅰ. 鸿蒙系统简介 鸿蒙系统(HarmonyOS),是第一款基于微内核的全场景分布式OS,是华为自主 ...

随机推荐

  1. 使用 KubeSphere 和极狐GitLab 打造云原生持续交付系统

    KubeSphere 简介 Kubernetes 是一个非常复杂的容器编排平台,学习成本非常高,KubeSphere 所做的事情就是高度产品化和抽象了底层 Kubernetes,是一个面向云原生的操作 ...

  2. appium-解决uiautomatorviewer不能定位android7以上版本元素的方法

    上图是android10模拟器,发现我们是无法连接模拟器识别的 解决方法: 在模拟器内打开指定页面然后截图,最后保存为uix文件,然后用adb拉到本地,一共四个命令,可以保存为.bat文件快捷执行(这 ...

  3. Python实现回数

    题目:回数是指从左向右读和从右向左读都是一样的数,例如 12321,909.请利用 filter()滤掉非回数: 思路:要实现回数判断,主要是将输入的数找到其各个位置的值,然后判断前后相对应的位置是否 ...

  4. 为什么会有unknown模块

    当一个模块在free后,他的线程仍然在运行时,会导致GetModuleName失败,返回unknown

  5. 详解 Hough 变换(基本原理与直线检测)

    Hough 变换原理与应用 前言: 详细介绍了 Hough 变换的基本思想.基本原理和应用等.其中大多都是自己的理解,难免有偏差,仅供参考. 文章目录 Hough 变换原理与应用 1. 基本概述 1. ...

  6. OpenCV开发笔记(八十一):通过棋盘格使用鱼眼方式标定相机内参矩阵矫正摄像头图像

    前言   对于广角摄像头通过相机图片可以识别出棋盘角点计算相机内参矩阵,通过畸变校准可以得到较好的效果,但是鱼眼摄像头通过这种方式获得周围四周的图像效果并不是很好.所以,鱼眼摄像头在校准上与普通摄像头 ...

  7. 噢!JavaScript (2):对数组要小心使用delete

    最近在重写我自己的静态博客生成器,虽然遇到的小问题挺多,但今早这个问题令我印象深刻. 发现问题 博客的文章基础数据储存在main.json中,其中专门有数组dateindex来储存经过排列后的文章顺序 ...

  8. 《大话设计模式》java实现:第二章-策略模式

    <大话设计模式>java实现:第二章-策略模式 第二章是使用策略模式实现商场收银系统,使收银系统的促销策略可以灵活更改. 1. 原始代码实现: package gof; /* * < ...

  9. 题解:CF1551D1 Domino (easy version)

    题解:CF1551D1 Domino (easy version) 分析 题目中保证 \(n\times m\) 为偶数,下面进行分类讨论. 情况一 如果 \(n\) 和 \(m\) 都是偶数,那么可 ...

  10. 卢卡斯定理/Lucas 定理

    卢卡斯定理/Lucas 定理 引入 求 \(C_{n+m}^n \mod p\). \(n,m,p \leq 10^5\). 如果直接用阶乘求,可能在阶乘过程中出现了 \(p\),而最后的结果没有出现 ...