【raect.js + hooks】useRef 搭配 Houdini 创造 useRipple

水波纹点击特效 really cool,实现水波纹的方案也有很多,笔者经常使用 material 组件,非常喜欢 mui 中的 ripple,他家的 ripple 特效就是通过 css Houdini 实现的。
今天,我们将复刻一个 ripple,并封装成 hooks 来使用!

CSS Houdini

首先,我们需要了解下 CSS Houdini 的相关知识:

Houdini 是一组底层 API,它们公开了 CSS 引擎的各个部分,从而使开发人员能够通过加入浏览器渲染引擎的样式和布局过程来扩展 CSS。Houdini 是一组 API,它们使开发人员可以直接访问CSS 对象模型 (CSSOM),使开发人员可以编写浏览器可以解析为 CSS 的代码,从而创建新的 CSS 功能,而无需等待它们在浏览器中本地实现。
Houdini 的 CSS Typed OM 是一个包含类型和方法的 CSS 对象、并且暴露出了作为 JavaScript 对象的值。比起先前基于字符串的,对 HTMLElement.style 进行操作的方案,对 JavaScript 对象进行操作更符合直觉。每个元素和样式表规则都拥有一个样式对应表,该对应表可以通过 StylePropertyMap 来获得。

<script>CSS.paintWorklet.addModule('csscomponent.js');</script>

csscomponents.js 里面定义一个 具名 类,然后应用到元素即可

li {background-image: paint(myComponent, stroke, 10px);--highlights: blue;--lowlights: green;
}

一个 CSS Houdini 的特性就是 Worklet (en-US)。在它的帮助下,你可以通过引入一行 JavaScript 代码来引入配置化的组件,从而创建模块式的 CSS。不依赖任何前置处理器、后置处理器或者 JavaScript 框架。

没有明白?没事,直接实操就明白了。

实现思路

点击元素时获取点击坐标(js 点击事件),将坐标,颜色,时常等参数传递给 css 变量,并从坐标处展开一个涟漪动画(houdini worklet),worklet 获取参数并渲染 canvas 动画即可。
涟漪变化的相关参数是时间,--ripple-time 将会在后面的js点击事件中实时更新。

创建 ripple 绘制 worklet

注册一个名为 “ripple” 的 paint 类,获取涟漪动画的 css 变量然后渲染涟漪。

// ripple-worklet.js
try {registerPaint("ripple",class {static get inputProperties() {return ["--ripple-x", "--ripple-y", "--ripple-color", "--ripple-time"];}paint(ctx, geom, properties) {const x = parseFloat(properties.get("--ripple-x").toString());const y = parseFloat(properties.get("--ripple-y").toString());const color = properties.get("--ripple-color").toString();const time = parseFloat(properties.get("--ripple-time").toString());ctx.fillStyle = color;ctx.globalAlpha = Math.max(1 - time, 0);ctx.arc(x, y, geom.width * time, 0, 2 * Math.PI);ctx.fill();}});
} catch (error) {if (error.name !== "DOMException") {throw error;}
}

封装 useRipple hook

为简化使用,将点击事件,涟漪样式都绑定到 ref 传递给需要使用涟漪的元素,并将应用 ripple worklet 的过程也添加到 useRipple 内;useRipple 再设置一下传参,传递 color(涟漪层颜色), duration(涟漪时常)和 trigger(触发时机),用于提高涟漪的可定制能力。
其中,为了让动画持续更新,通过 requestAnimationFrame 递归调用 animate 函数,实时更新 --ripple-time 参数

在外部定义 isWorkletRegistered 标志,避免重复注册 ripple worklet.

import { useRef, useEffect } from "react";export type RippleConfig = {color?: React.CSSProperties["color"];duration?: number;trigger?: "click" | "mousedown" | "pointerdown";
};let isWorkletRegistered = false;const useRipple = <T extends HTMLElement = HTMLButtonElement>(config: RippleConfig = {color: "rgba(31, 143, 255, 0.5)",duration: 500,}
): React.RefObject<T> => {const ref = useRef<T>(null);const mounted = useRef<boolean>(false);useEffect(() => {if (mounted.current) return;try {if ("paintWorklet" in CSS && !isWorkletRegistered) {if (!isWorkletRegistered) {// @ts-ignoreCSS.paintWorklet.addModule("houdini/ripple.js");isWorkletRegistered = true;console.log("Ripple worklet is registered");} else {console.warn("Ripple worklet is already registered");}} else {console.warn("Your browser doesn't support CSS Paint API");}} catch (error) {console.error(error);}mounted.current = true;}, []);useEffect(() => {const button = ref.current;if (!button) return;let animationFrameId: number | null = null;const handleClick = (event: MouseEvent) => {const rect = button.getBoundingClientRect();const x = event.clientX - rect.left;const y = event.clientY - rect.top;const startTime = performance.now();button.style.setProperty("--ripple-color", config.color ?? "rgba(31, 143, 255, 0.5)");button.style.setProperty("--ripple-x", `${x}px`);button.style.setProperty("--ripple-y", `${y}px`);button.style.setProperty("--ripple-time", "0");button.style.setProperty("background-image", "paint(ripple)");const animate = (time: number) => {const progress = (time - startTime) / (config.duration ?? 500); // Convert time to secondsbutton.style.setProperty("--ripple-time", `${progress}`);if (progress < 1) {animationFrameId = requestAnimationFrame(animate);} else {if (animationFrameId) {cancelAnimationFrame(animationFrameId);}}};animationFrameId = requestAnimationFrame(animate);};button.addEventListener(config.trigger ?? "mousedown", handleClick);return () => {if (animationFrameId) {cancelAnimationFrame(animationFrameId);}button.removeEventListener(config.trigger ?? "mousedown", handleClick);};}, []);return ref;
};export default useRipple;

ripple-worklet 转 Blob

上面的 ripple.js 我们只能放在 public 下或者公网地址,通过路径传给 CSS.paintWorklet.addModule,放在 useRipple 目录下通过"./ripple.js" 传是无效的。有没有解决办法呢?注意,这个路径其实是 URL,我们可以通过 URL.createObjectURL 封装 ripple.js,再传给 addModule:

// rippleWorklet.ts
const rippleWorklet = URL.createObjectURL(new Blob([`try {registerPaint("ripple",class {static get inputProperties() {return ["--ripple-x", "--ripple-y", "--ripple-color", "--ripple-time"];}paint(ctx, geom, properties) {const x = parseFloat(properties.get("--ripple-x").toString());const y = parseFloat(properties.get("--ripple-y").toString());const color = properties.get("--ripple-color").toString();const time = parseFloat(properties.get("--ripple-time").toString());ctx.fillStyle = color;ctx.globalAlpha = Math.max(1 - time, 0);ctx.arc(x, y, geom.width * time, 0, 2 * Math.PI);ctx.fill();}});} catch (error) {if (err.name !== "DOMException") {throw err;}}`,],{type: "application/javascript",})
);export default rippleWorklet;

然后调整 useRipple:

CSS.paintWorklet.addModule(rippleWorklet); // "Houdini/ripple.js"

此时效果是一样的,不再需要额外配置 ripple.js.

使用示例

以下代码用 useRipple 创建了一个附带 ripple 特效的 div 组件,你可以用相同的方式为任意元素添加 ripple,也可以直接用这个 Ripple 组件包裹其他元素。

import { useRipple } from "@/hooks";export default Ripple() {const rippleRef = useRipple<HTMLDivElement>();return(<div ref={rippleRef}>水波纹特效</div>)
}

结合 useRipple 高仿 @mui/Button 的效果:
涟漪按钮效果

.confirm-modal__actions__button--cancel {color: dodgerblue;
}.confirm-modal__actions__button--confirm {color: #fff;background-color: dodgerblue;
}.confirm-modal__actions__button {border-radius: 4px;margin-left: 0.5rem;text-transform: uppercase;font-size: 12px;
}

Bingo! 一个便捷的 useRipple 就这样实现了!

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring中@DependsOn 使用详解

一、注解源码 Target({ElementType.TYPE, ElementType.METHOD}) Retention(RetentionPolicy.RUNTIME) Documented public interface DependsOn {String[] value() default {}; } 二、基础概念 DependsOn是Spring框架用来指定bean之间依赖关系的注解之一&#xff0c;即可用户类…

淘宝商品详情数据接口(店铺搬家、数据分析、代购商城、ERP选品、无货源铺货、品牌监控)

使用淘宝API接口需要以下步骤&#xff1a; 注册开发者账号&#xff1a;在淘宝开放平台&#xff08;https://o0b.cn/anzexi&#xff09;上注册一个开发者账号&#xff0c;并创建一个应用。 获取API密钥&#xff1a;在应用页面上获取API密钥&#xff0c;这是后续调用API接口的凭…

Mendix组件推荐:灵活的在线表格

- 视频 mendix在线表格.mp4 20.95MB - 客户需求 如果你是一个中小型企业的负责人&#xff0c;你可能面临着&#xff1a; 多人协作录入数据展示数据库中的数据对数据安全有要求、希望本地离线部署并且IT人员配置有限等挑战 为了更好地管理你的业务数据&#xff0c;你需要一个…

如何防止网络被入侵?

随着互联网的普及&#xff0c;网络安全问题越来越受到人们的关注。其中&#xff0c;如何防止网络被入侵是一个重要的问题。本文将介绍一些防止网络被入侵的方法&#xff0c;帮助大家保护自己的网络安全。 一、使用强密码 强密码是防止网络被入侵的第一道防线。一个好的密码应该…

【brpc学习实践十二】bthread

概览 bthread(代码)是baidu-rpc使用的M:N线程库,是其稳定和高效的关键组件。能更好地利用多核cpu,能在pthread中运行,需要注意的是,bthread的work stealing机制会da让任务pthread发生切换,从而让thread_local变量不可信,通常在bthread_usleep或这join的时候就有可能发生…

Agent举例与应用

什么是Agent OpenAI 应用研究主管 Lilian Weng 在一篇长文中提出了 Agent LLM&#xff08;大型语言模型&#xff09;记忆规划技能工具使用这一概念&#xff0c;并详细解释了Agent的每个模块的功能。她对Agent未来的应用前景充满信心&#xff0c;但也表明到挑战无处不在。 现…

Linux RN6752 驱动编写

一、概述 关于 RN6752V1 这个芯片这里就不做介绍了&#xff0c;看到这篇笔记的小伙伴应该都明白&#xff0c;虽然说 RN6752V1 芯片是 AHD 信号的解码芯片&#xff0c;但是也可以把芯片当做是一个 YUV 信号的 MIPI 摄像头&#xff0c;所以驱动的编写和 MIPI 摄像头无太大的区别。…

虚拟机虚拟化原理

目录 什么是虚拟化广义虚拟化狭义虚拟化 虚拟化指令集敏感指令集虚拟化指令集的工作模式监视器对敏感指令的处理过程&#xff1a; 虚拟化类型全虚拟化类虚拟化硬件辅助虚拟化 虚拟化架构裸金属架构宿主机模式架构 什么是虚拟化 虚拟化就是通过模仿下层原有的功能模块创造接口来…

软件著作权申请都应该提供哪些材料?

作为一个苦逼的程序员&#xff0c;一直致力于搬砖式的增删改查&#xff0c;偶尔突然有了一个小小的创意&#xff0c;然而却无法用法律的武器保护自己。 想申请著作权又不知道该如何实现&#xff0c;那么&#xff0c;我就把当年自己申请著作权需要的材料列一下&#xff0c;希望广…

中国版的 GPTs:InsCode AI 生成应用

前言 在上一篇文章 《InsCode&#xff1a;这可能是下一代应用开发平台&#xff1f;》中&#xff0c;我们介绍了一个新的应用开发平台 InsCode&#xff0c;它是基于云原生开发环境 云 IDE AI 辅助编程的一站式在线开发平台。 最近&#xff0c;InsCode 又推出了另一种全新的开…

OpenCV项目开发实战-通过轮廓检测最小化图像眩光

该过程涉及使用两个输入;输入 1 是参考图像,而输入 2 是要扭曲/对齐的图像。匹配图像后,图像 2 通过透视与图像 1 对齐。下一步是通过阈值处理获得二值图像,然后根据轮廓的面积对其进行排序。最后对二值图像进行清洗后得到绝对差图像。 目录 轮廓检测:减少图像 opencv上的…

Python三十个常见的脚本汇总

1、冒泡排序 2、计算x的n次方的方法 3、计算a*a b*b c*c …… 4、计算阶乘 n! 5、列出当前目录下的所有文件和目录名 6、把一个list中所有的字符串变成小写&#xff1a; 7、输出某个路径下的所有文件和文件夹的路径 8、输出某个路径及其子目录下的所有文件路径 9、输出某个路…

NMap扫描进阶

NMap扫描进阶 一, 基础扫描 扫描IP地址: nmap -sn 192.168.112.0/24 192.168.112.1-255扫描端口号: 基于SYN包扫描: nmap -sS 192.168.112.200 基于三次握手扫描: nmap -sT 192.168.112.200指定端口扫描: nmap -p10-200 192.168.112.200 nmap -p21,22,25,80,445,3306,1521…

用bat写一个定时备份数据库某几个表的脚本

1.首先是bat脚本的固定框架 echo off setlocalendlocal echo off 其中抑制当前命令的回显&#xff0c;echo off关闭命令回显setlocal 和 endlocal 是 Windows 命令行脚本中用于限定本地环境变量作用范围的命令 2.然后设置下要备份的数据库的信息 rem 目标MySQL数据库信息 s…

08 C++中的运算符

系列文章目录 08 C中常见的运算符 目录 系列文章目录 文章目录 前言 一、 C中常见的运算符有哪些&#xff1f; 二、各个运算符的具体介绍 1.算术运算符 2.关系运算符 3.逻辑运算符 4.位运算符 5.赋值运算符 6. 其他运算符 三、C 中的运算符优先级 总结 前言 运算…

深度学习毕设项目 医学大数据分析 - 心血管疾病分析

# 1 前言 &#x1f6a9; 基于大数据的心血管疾病分析 &#x1f947;学长这里给一个题目综合评分(每项满分5分) 难度系数&#xff1a;3分工作量&#xff1a;3分创新点&#xff1a;4分 1 课题背景 本项目的任务是利用患者的检查结果预测心血管疾病(CVD)的存在与否。 2 数据…

新型Python环境与依赖管理工具——pipenv

文章目录 pipenv介绍pipenv安装pipenv使用创建虚拟环境删除虚拟环境安装依赖查看包之间的依赖图卸载依赖在虚拟环境中执行命令shell环境下通过requirements.txt安装依赖导出requirements.txt文件查看虚拟环境的路径 pipenv介绍 pipenv可以看做是pip和virtualenv的组合体&#…

【开源视频联动物联网平台】流媒体传输协议HLS,FLV的功能和特点

HLS&#xff08;HTTP Live Streaming&#xff09;和FLV&#xff08;Flash Video&#xff09;都是用于视频流传输的协议或容器格式&#xff0c;但它们在某些方面有着显著的区别和特点。 HLS是一种由苹果公司开发的用于流媒体传输的协议&#xff0c;而FLV则是Adobe公司开发的用于…

【ArcGIS Pro二次开发】:CC工具箱1.1.4更新_免费_50+工具

CC工具箱1.1.4更新【2023.11.30】 使用环境要求&#xff1a;ArcGIS Pro 3.0 一、下载链接 工具安装文件及使用文档&#xff1a; https://pan.baidu.com/s/1OJmO6IPtMfX_vob3bMtvEg?pwduh5rhttps://pan.baidu.com/s/1OJmO6IPtMfX_vob3bMtvEg?pwduh5r 二、使用方法 1、在下…

从物理机到K8S:应用系统部署方式的演进及其影响

公众号「架构成长指南」&#xff0c;专注于生产实践、云原生、分布式系统、大数据技术分享。 概述 随着科技的进步&#xff0c;软件系统的部署架构也在不断演进&#xff0c;从以前传统的物理机到虚拟机、Docker和Kubernetes&#xff0c;我们经历了一系列变化。 这些技术的引入…