transition不触发是因为缺少可动画属性的数值变化;需确保class间属性值不同、类型兼容,避免用all和不可动画属性,并通过重排或requestAnimationFrame保证浏览器识别状态差。
CSS transition 本身不关心你是用 :hover、:focus 还是 JS 切换 class —— 它只响应「某个可动画的 CSS 属性在两个状态间发生了数值变化」。如果你加了 transition: all 0.3s,但切换 class 后样式没变(比如目标 class 里没写新值),或者写了但属性不可动画(如 display、height 从 auto 变具体值),那过渡就静默失败。
opacit
y、transform、background-color)all,明确写出要过渡的属性: transition: opacity 0.3s, transform 0.3s
color: #000 → color: #fff 可动;color: red → color: transparent 也可动;但 visibility: hidden → visible 不会过渡,因为不可动画)这是最常踩的坑:浏览器在 JS 同步设置 class 后,**立刻计算样式并应用过渡起始态**,但若起始态和结束态“看起来一样”,或者 DOM 重排被跳过,就会丢掉过渡。典型场景是刚加 class 就改样式,浏览器来不及识别中间状态。
getComputedStyle 强制重排,再改样式:element.classList.add('active');
// 触发重排,让浏览器记住当前状态
window.getComputedStyle(element).transform;
element.style.transform = 'scale(1.2)';setTimeout 或 requestAnimationFrame 把样式变更延后一帧:element.classList.add('active');
requestAnimationFrame(() => {
element.style.transform = 'scale(1.2)';
});现代方案是监听 transitionend 事件 + 使用 transition-property 精确控制触发条件。它比靠时间猜更可靠,也避免重复绑定。
transitionend 在任意一个过渡完成时触发,不是全部;需检查 event.propertyName 来区分event.elapsedTime 会小于声明时长,可用于判断是否被中断once: true 选项或手动 removeEventListener
element.addEventListener('transitionend', (e) => {
if (e.propertyName === 'transform') {
console.log('缩放完成');
}
}, { once: true });
这些属性从 0 → auto 或 auto → 具体值,浏览器无法插值,transition 直接跳变。这不是 bug,是规范限制。
max-height 替代 height:设一个足够大的固定值(如 max-height: 500px),配合 overflow: hidden
transform: scaleY(0) → scaleY(1) 模拟高度展开,性能更好且可过渡margin/padding 做过渡,改用 transform 位移或 clip-path 裁剪