必须将自定义动画定义在theme.extend.keyframes下,值为纯对象,键为'0%'/'100%',值为'transform: translateX(...)'格式的完整CSS声明;animation需在theme.extend.animation中配置为'marquee 8s linear infinite'格式。
直接在 tailwind.config.js 里定义自定义 @keyframes 是可行的,但必须严格遵循 Tailwind 的扩展语法 —— 错一个键名或值格式,animate- 类就完全不会生效。
theme.extend.keyframes 下Tailwind 不识别 theme.keyframes 或 theme.animation.keyframes 这类路径。只认 theme.extend.keyframes 这个固定位置,且值必须是纯对象:
'from'、'to' 或 '0%'、'100%'(不支持中间帧如 '50%')transform: 'translateX(0)',不能简写成 translateX(0)
错误示例:{ '0%': 'translateX(100vw)' }(缺 transform:);正确写法:{ '0%': 'transform: translateX(100vw)' }
theme.extend.animation 里的值不是 CSS 动画名,而是完整的动画声明字符串,格式为:'keyframes-name duration timing-function':
立即学习“前端免费学习笔记(深入)”;
duration 必须带单位,如 '2s'、'3000ms',不能只写 2
timing-function 推荐用 'linear' 实现匀速滚动;若用 'ease-in-out' 会导致启停卡顿,破坏循环感iteration-count 或 direction,这些必须靠 HTML 元素 class 或内联样式控制示例配置:
module.exports = { theme: { extend: { keyframes: { 'marquee': { '0%': 'transform: translateX(100vw)', '100%': 'transform: translateX(-100%)', } }, animation: { 'marquee': 'marquee 8s linear infinite', } } }}
仅加 animate-marquee 不够 —— 跑马灯失效最常见的原因是元素本身没“跑起来”,根源在布局:
overflow-hidden,否则动画移出区域的内容仍可见whitespace-nowrap(如果是文字)或明确 inline-block(如果是图片/子容器),避免换行打断连续位移translateX(100vw) 是从视口右边缘开始,translateX(-100%) 是向左移自身宽度;两者配合才能无缝衔接,若用 -100vw 会因容器宽度变化导致跳帧典型结构:
<div class="overflow-hidden"> <div class="animate-marquee inline-block whitespace-nowrap"> <span>内容1</span> <span>内容2</span> <span>内容3</span> </div></div>
Tailwind 不报错也不提示 keyframes 缺失,它只是静默忽略非法配置。最有效的验证方式是:
@keyframes marquee,确认是否出现在生成的 CSS 中tailwind.config.js 是否漏了 extend 层级,或 keyframes 对象被意外写到了 plugins 里getComputedStyle 检查元素是否真应用了 animation 属性,常见干扰是父级 display: flex 导致子元素被拉伸变形,影响 translateX 计算真正容易被忽略的是:Tailwind 的 keyframes 扩展不支持动态参数(比如根据内容宽度自动算 duration),所有时间值都得硬编码,需要滚动速度变化时,只能手动改配置再重启构建。