创建无限动画文字效果,以遵循用户的动作偏好。
完整文章 · YouTube 上的视频 · GitHub 上的来源
HTML
<p split-by="word" word-animation="trampoline">
  split a paragraph of content 🤘💀
</p>CSS
        @keyframes trampoline {
  0% {
    transform: translateY(100%);
    animation-timing-function: ease-out
  }
  50% {
    transform: translateY(0);
    animation-timing-function: ease-in
  }
}
@media (prefers-reduced-motion:no-preference) {
  [word-animation] {
    display: inline-flex;
    flex-wrap: wrap;
    gap: 1ch
  }
  [word-animation=trampoline]>span {
    display: inline-block;
    transform: translateY(100%);
    animation: trampoline 3s ease calc(var(--index) * 150 * 1ms) infinite alternate
  }
}
        JS
        const span = (text, index) => {
  const node = document.createElement('span')
  node.textContent = text
  node.style.setProperty('--index', index)
  
  return node
}
export const byWord = text =>
  text.split(' ').map(span)
const {matches:motionOK} = window.matchMedia(
  '(prefers-reduced-motion: no-preference)'
)
if (motionOK) {
  const splitTargets = document.querySelectorAll('[split-by]')
  splitTargets.forEach(node => {
    let nodes = byWord(node.innerText)
    if (nodes)
      node.firstChild.replaceWith(...nodes)
  })
}