如何使用 CSS 自定义属性平滑动画径向渐变

2次阅读

如何使用 CSS 自定义属性平滑动画径向渐变

本文详解如何通过 CSS @property 声明自定义动画属性,结合 radial-gradient 实现圆心半径的连续、流畅缩放动画,并提供 Firefox 兼容方案。

本文详解如何通过 css `@property` 声明自定义动画属性,结合 `radial-gradient` 实现圆心半径的连续、流畅缩放动画,并提供 firefox 兼容方案。

在 CSS 中直接对 background-image 中的数值(如 radial-gradient(#363636 10px, transparent 1px) 中的 10px)做关键帧动画是 无效的 ——浏览器无法自动插值两个不同的渐变函数字符串,导致动画“跳变”而非平滑过渡。要实现从大圆到小圆的细腻缩放效果,核心思路是: 将可变尺寸抽离为受控的 CSS 自定义属性,并确保该属性支持插值动画

✅ 正确方案:@property + CSS 变量驱动

CSS @property 规则(属 Houdini API 的一部分)允许开发者显式声明自定义属性的类型、初始值和继承性,从而启用其在 @keyframes 中的 平滑插值能力。以下是完整、可运行的解决方案:

/* 声明可动画的长度型自定义属性 */ @property --d {syntax: "<length>";   initial-value: 10px;   inherits: false;}  .loader {position: absolute;   z-index: 1;   width: 200px;   height: 200px;   background-color: transparent;   /* 将半径绑定到 --d 变量 */   background-image: radial-gradient(#363636 var(--d), transparent 1px);   background-size: 30px 30px;   animation: loaderout 2s ease-in-out forwards; }  @keyframes loaderout {0%   { --d: 10px;}   100% {--d: 0.7px;} /* 注意:此处可设为任意合法长度值,如 0.7px、1px、0px */ }
<div class="loader"></div>

? 关键点说明

  • syntax: “<length>” 显式声明 –d 为长度类型,使浏览器能正确解析并插值 10px → 0.7px;
  • initial-value: 10px 避免未设置时的计算异常;
  • inherits: false 提升性能(本例中无需继承);
  • 动画使用 ease-in-out 而非默认线性,视觉更自然。

⚠️ 兼容性处理:Firefox 回退方案

目前 Firefox 尚未支持 @property(截至 Firefox 128)。为保障跨浏览器一致性,可利用 @supports 进行特性检测,并采用 font-size 作为代理变量(因其天然支持插值且不影响布局):

立即学习 前端免费学习笔记(深入)”;

@supports not (--d: 0px) {.loader {     /* 利用 font-size 间接控制 --d */     font-size: 10px;}   .loader::before {content: "";     position: absolute;     inset: 0;     background-image: radial-gradient(#363636 calc(var(--d, 1em) * 1px), transparent 1px);     background-size: 30px 30px;   }    @keyframes loaderout {0%   { font-size: 10px;}     100% {font-size: 0.7px;}   }    /* 通过 font-size 计算 --d(需配合 JS 或 calc 处理)*/   /* 更简洁的实践:直接在 background-image 中用 calc 引用 font-size */   .loader {background-image: radial-gradient(#363636 calc(1em * 1), transparent 1px);   }   @keyframes loaderout {0%   { font-size: 10px;}     100% {font-size: 0.7px;}   } }

? 推荐简化回退写法(无 JS)
若仅需基础兼容,可改用 transform: scale() 模拟缩放感(虽非真渐变变化,但视觉效果接近且 100% 兼容):

.loader {background-image: radial-gradient(#363636 10px, transparent 1px);   background-size: 30px 30px;   transform-origin: center; } @keyframes loaderout {0%   { transform: scale(1); }   100% {transform: scale(0.5); } }

✅ 总结与最佳实践

  • 优先使用 @property:现代浏览器(Chrome ≥ 102、Edge ≥ 102、Safari ≥ 16.4)已全面支持,语义清晰、性能优异;
  • 避免字符串插值陷阱:切勿在 background-image 中硬编码数值并期望其动画化;
  • 单位一致性很重要:@property 的 syntax 必须匹配实际使用的单位(如 <length> 对应 px/em/rem);
  • 动画缓动建议:ease-in-out 或 cubic-bezier(.3,1.2,.7,1) 可增强“呼吸感”,比 linear 更符合设计直觉。

通过以上方法,你就能稳定实现径向渐变中核心圆半径的丝滑缩放动画,兼顾表现力与工程健壮性。

星耀云
版权声明:本站原创文章,由 星耀云 2026-03-27发表,共计2245字。
转载说明:转载本网站任何内容,请按照转载方式正确书写本站原文地址。本站提供的一切软件、教程和内容信息仅限用于学习和研究目的;不得将上述内容用于商业或者非法用途,否则,一切后果请用户自负。本站信息来自网络,版权争议与本站无关。
text=ZqhQzanResources