Share a loading button animation

First look at the effect:
insert image description here
then look at the code:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>加载动画按钮</title>
  <script src="https://cdnjs.cloudflare.com/ajax/libs/animejs/3.2.1/anime.min.js"></script>
  <style>
    * {
      
      
      box-sizing: border-box;
    }

    body {
      
      
      width: 100%;
      height: 100vh;
      background-color: #000000;
      font-family: "Plus Jakarta Sans", sans-serif;
    }
    body main {
      
      
      display: flex;
      justify-content: center;
      align-items: center;
      width: inherit;
      height: inherit;
    }
    body main .btn-area {
      
      
      position: relative;
      width: calc(210px + 2px);
      height: calc(60px + 2px);
    }
    body main .btn-area:hover .main-btn {
      
      
      opacity: 0.25;
    }
    body main .btn-area .main-btn {
      
      
      width: inherit;
      height: inherit;
      opacity: 0.15;
      transition: 250ms;
    }
    body main .btn-area .main-btn rect {
      
      
      width: 210px;
      height: 60px;
      transition: 250ms;
    }
    body main .btn-area .progress-area,
    body main .btn-area .complete-area {
      
      
      position: absolute;
      display: flex;
      justify-content: space-evenly;
      align-items: center;
      top: 50%;
      left: 50%;
      transform: translate(-50%, -50%);
      width: 100%;
      color: #1578ff;
      user-select: none;
      opacity: 1;
    }
    body main .btn-area .progress-area .area-left,
    body main .btn-area .complete-area .area-left {
      
      
      display: flex;
    }
    body main .btn-area .progress-area .area-left svg,
    body main .btn-area .complete-area .area-left svg {
      
      
      margin-right: 1rem;
      opacity: 0;
    }
    body main .btn-area .progress-area .area-left span,
    body main .btn-area .complete-area .area-left span {
      
      
      width: 50px;
      text-align: right;
    }
    body main .btn-area .progress-area .area-right,
    body main .btn-area .complete-area .area-right {
      
      
      position: relative;
      display: flex;
      align-items: center;
      height: 20px;
      margin-right: -1rem;
      padding-left: 1rem;
      border-left: 1px solid #1578ff44;
      overflow: hidden;
      cursor: pointer;
    }
    body main .btn-area .progress-area .area-right:hover,
    body main .btn-area .complete-area .area-right:hover {
      
      
      height: 40px;
      transition: 250ms height;
    }
    body main .btn-area .progress-area .area-right:hover svg,
    body main .btn-area .complete-area .area-right:hover svg {
      
      
      opacity: 1;
    }
    body main .btn-area .progress-area .area-right svg,
    body main .btn-area .complete-area .area-right svg {
      
      
      opacity: 0.5;
      transition: 250ms;
    }
    body main .btn-area .progress-area .area-right .btn-play,
    body main .btn-area .complete-area .area-right .btn-play {
      
      
      position: absolute;
      transform: translateY(40px);
    }
    body main .btn-area .complete-area {
      
      
      justify-content: space-around;
      display: none;
    }
    body main .btn-area .complete-area svg {
      
      
      width: 40px;
      height: 20px;
      margin-right: 1rem;
      padding-right: 1rem;
      border-right: 1px solid #fff6;
    }
    body main .btn-area .complete-area svg path {
      
      
      stroke-linecap: round;
      stroke-linejoin: round;
      stroke-width: 6px;
      opacity: 0;
    }
    body main .btn-area .complete-area span {
      
      
      color: #fffd;
      padding-bottom: 1px;
      opacity: 0;
    }
    body main .btn-area .frame-btn {
      
      
      position: absolute;
      top: 0;
      left: 0;
      width: inherit;
      height: inherit;
    }
    body main .btn-area .frame-btn rect {
      
      
      width: 210px;
      height: 60px;
      stroke-linecap: round;
      stroke-linejoin: round;
    }
    body .instagram-link {
      
      
      position: absolute;
      right: 50px;
      bottom: 50px;
    }
    body .instagram-link i {
      
      
      color: #1578ff;
      transition: 150ms;
      font-size: 2rem;
      opacity: 0.5;
    }
    body .instagram-link i:hover {
      
      
      opacity: 1;
    }
  </style>
</head>
<body>
<main>
  <div class="btn-area">
    <svg class="main-btn" xmlns="http://www.w3.org/2000/svg" version="1.1">
      <rect fill="#fff" stroke="#1578ff" stroke-width="2" x="1" y="1" rx="10" ry="10" />
    </svg>
    <svg class="frame-btn" xmlns="http://www.w3.org/2000/svg" version="1.1">
      <rect fill="none" stroke="#1578ff" stroke-width="2" stroke-dasharray="525" stroke-dashoffset="525" x="1" y="1" rx="10" ry="10" />
    </svg>
    <div class="progress-area">
      <div class="area-left">
        <svg width="18" height="18" viewBox="0 0 36 36" fill="none" xmlns="http://www.w3.org/2000/svg">
          <path d="M18 3C17.1 3 16.5 3.6 16.5 4.5V8.25C16.5 9.15 17.1 9.75 18 9.75C18.9 9.75 19.5 9.15 19.5 8.25V4.5C19.5 3.6 18.9 3 18 3Z" fill="#1578ff" />
          <path d="M10.0504 12.1504C10.3504 12.4504 10.8004 12.6004 11.1004 12.6004C11.4004 12.6004 11.8504 12.4504 12.1504 12.1504C12.7504 11.5504 12.7504 10.6504 12.1504 10.0504L9.45039 7.35039C8.85039 6.75039 7.95039 6.75039 7.35039 7.35039C6.75039 7.95039 6.75039 8.85039 7.35039 9.45039L10.0504 12.1504Z" fill="#1578ff" />
          <path d="M9.75 18C9.75 17.1 9.15 16.5 8.25 16.5H4.5C3.6 16.5 3 17.1 3 18C3 18.9 3.6 19.5 4.5 19.5H8.25C9 19.5 9.75 18.9 9.75 18Z" fill="#1578ff" />
          <path d="M10.0508 23.8504L7.50078 26.4004C6.90078 27.0004 6.90078 27.9004 7.50078 28.5004C7.80078 28.8004 8.25078 28.9504 8.55078 28.9504C8.85078 28.9504 9.30078 28.8004 9.60078 28.5004L12.1508 25.9504C12.7508 25.3504 12.7508 24.4504 12.1508 23.8504C11.5508 23.2504 10.6508 23.2504 10.0508 23.8504Z" fill="#1578ff" />
          <path d="M18 26.25C17.1 26.25 16.5 26.85 16.5 27.75V31.5C16.5 32.4 17.1 33 18 33C18.9 33 19.5 32.4 19.5 31.5V27.75C19.5 27 18.9 26.25 18 26.25Z" fill="#1578ff" />
          <path d="M25.9504 23.8504C25.3504 23.2504 24.4504 23.2504 23.8504 23.8504C23.2504 24.4504 23.2504 25.3504 23.8504 25.9504L26.4004 28.5004C26.7004 28.8004 27.1504 28.9504 27.4504 28.9504C27.7504 28.9504 28.2004 28.8004 28.5004 28.5004C29.1004 27.9004 29.1004 27.0004 28.5004 26.4004L25.9504 23.8504Z" fill="#1578ff" />
          <path d="M31.5 16.5H27.75C26.85 16.5 26.25 17.1 26.25 18C26.25 18.9 26.85 19.5 27.75 19.5H31.5C32.4 19.5 33 18.9 33 18C33 17.1 32.4 16.5 31.5 16.5Z" fill="#1578ff" />
          <path d="M24.9004 12.5998C25.3504 12.5998 25.6504 12.4498 25.9504 12.1498L28.5004 9.5998C29.1004 8.9998 29.1004 8.0998 28.5004 7.4998C27.9004 6.8998 27.0004 6.8998 26.4004 7.4998L23.8504 10.0498C23.2504 10.6498 23.2504 11.5498 23.8504 12.1498C24.1504 12.4498 24.6004 12.5998 24.9004 12.5998Z" fill="#1578ff" />
        </svg>
        <span>0 %</span>
      </div>
      <div class="area-right">
        <svg width="18" height="18" viewBox="0 0 36 36" fill="none" xmlns="http://www.w3.org/2000/svg" class="btn-pause">
          <path fill-rule="evenodd" clip-rule="evenodd" d="M8.25 6C6.97843 6 6 6.97843 6 8.25V27.75C6 29.0216 6.97843 30 8.25 30H10.65C11.7358 30 12.7805 29.0469 12.9 27.8353V8.25C12.9 6.97843 11.9216 6 10.65 6H8.25ZM3 8.25C3 5.32157 5.32157 3 8.25 3H10.65C13.5784 3 15.9 5.32157 15.9 8.25V27.9C15.9 27.9384 15.8985 27.9768 15.8956 28.015C15.6906 30.6796 13.4492 33 10.65 33H8.25C5.32157 33 3 30.6784 3 27.75V8.25Z" fill="#ff542b" />
          <path fill-rule="evenodd" clip-rule="evenodd" d="M25.3496 6C24.078 6 23.0996 6.97843 23.0996 8.25V27.75C23.0996 29.0216 24.078 30 25.3496 30H27.7496C29.0613 30 29.9996 28.982 29.9996 27.9V8.25C29.9996 6.97843 29.0212 6 27.7496 6H25.3496ZM20.0996 8.25C20.0996 5.32157 22.4212 3 25.3496 3H27.7496C30.678 3 32.9996 5.32157 32.9996 8.25V27.9C32.9996 30.718 30.6379 33 27.7496 33H25.3496C22.4212 33 20.0996 30.6784 20.0996 27.75V8.25Z" fill="#ff542b" />
        </svg>
        <svg width="18" height="18" viewBox="0 0 36 36" fill="none" xmlns="http://www.w3.org/2000/svg" class="btn-play">
          <path fill-rule="evenodd" clip-rule="evenodd" d="M3 7.33761C3 2.58639 8.08368 0.177285 12.0217 2.27008C12.0446 2.28222 12.0671 2.29495 12.0893 2.30826L30.0453 13.0771C33.9849 15.2998 33.9849 20.6354 30.0453 22.8581L12.0893 33.627L12.0701 33.6383C8.06754 35.9588 3 33.3067 3 28.7474V7.33761ZM10.5822 4.90258C8.33198 3.73209 6 5.21176 6 7.33761V28.7474C6 30.7729 8.36369 32.3115 10.5566 31.048L28.5205 20.2745C28.5334 20.2667 28.5465 20.2591 28.5597 20.2518C30.4801 19.1775 30.4801 16.7578 28.5597 15.6835C28.5521 15.6792 28.5445 15.6749 28.537 15.6705C28.5315 15.6673 28.526 15.6641 28.5205 15.6608L10.5822 4.90258Z" fill="#1578ff" />
        </svg>
      </div>
    </div>
    <div class="complete-area">
      <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 52.71 28.57">
        <path class="tick-1" fill="none" stroke="#ffff" stroke-dasharray="52" stroke-dashoffset="52" d="M2,12.42,16.14,26.57,34.71,2" />
        <path class="tick-2" fill="none" stroke="#fff8" stroke-dasharray="52" stroke-dashoffset="52" d="M18,12.42,32.14,26.57,50.71,2" />
      </svg>
      <span>完毕</span>
    </div>
  </div>
</main>

</body>
<script>
  const buttonArea = document.querySelector('.btn-area')
  const progressArea = document.querySelector('.progress-area')
  const completeArea = document.querySelector('.complete-area')
  const mainBtn = document.querySelector('.main-btn rect')
  const frameBtn = document.querySelector('.frame-btn rect')
  const loadingIcon = document.querySelector('.progress-area .area-left svg')
  const loadingProgress = document.querySelector('.progress-area .area-left span')
  const loadingBtn = document.querySelector('.progress-area .area-right')
  const pauseBtn = document.querySelector('.progress-area .area-right .btn-pause')
  const playBtn = document.querySelector('.progress-area .area-right .btn-play')
  const tick1 = document.querySelector('.complete-area .tick-1')
  const tick2 = document.querySelector('.complete-area .tick-2')
  const doneText = document.querySelector('.complete-area span')

  let loadingTime = 5500
  let progress = {
      
      
    value: '0 %'
  }
  let loadingStatus = true

  document.body.onload = function() {
      
      
    anime({
      
      
      targets: loadingIcon,
      opacity: [0, 1],
      duration: 750,
      easing: 'easeOutQuad'
    })

    let aniLoadingIcon = anime({
      
      
      targets: loadingIcon,
      rotateZ: 360,
      duration: 2000,
      loop: true,
      easing: 'linear'
    })

    anime({
      
      
      targets: loadingProgress,
      translateY: ['15px', '0'],
      opacity: [0, 1],
      delay: 250,
      duration: 750,
      easing: 'easeOutQuart'
    })

    anime({
      
      
      targets: loadingBtn,
      translateY: ['15px', '0'],
      opacity: [0, 1],
      delay: 350,
      duration: 1000,
      easing: 'easeOutQuart'
    })

    let aniProgress = anime({
      
      
      targets: progress,
      value: '100 %',
      duration: loadingTime,
      easing: 'cubicBezier(.5, .05, .3, .9)',
      delay: 1000,
      round: 1,
      update: function() {
      
      
        loadingProgress.innerHTML = JSON.stringify(progress.value).replace(/^"(.*)"$/, '$1')
      }
    })

    let aniFrameBtn = anime({
      
      
      targets: frameBtn,
      strokeDashoffset: [525, 0],
      duration: loadingTime,
      easing: 'cubicBezier(.5, .05, .3, .9)',
      delay: 1000,
      complete: function() {
      
      
        completeLoading()
      }
    })

    loadingBtn.addEventListener('click', () => {
      
      
      if (loadingStatus) {
      
      
        aniLoadingIcon.pause()
        aniProgress.pause()
        aniFrameBtn.pause()
        pauseBtn.style.transform = 'translateY(-40px)'
        playBtn.style.transform = 'translateY(0px)'
        loadingStatus = false
      } else {
      
      
        aniLoadingIcon.play()
        aniProgress.play()
        aniFrameBtn.play()
        pauseBtn.style.transform = 'translateY(0px)'
        playBtn.style.transform = 'translateY(40px)'
        loadingStatus = true
      }

    })
  }

  function completeLoading() {
      
      
    anime({
      
      
      targets: loadingIcon,
      translateX: [0, -20],
      opacity: [1, 0],
      duration: 500,
      delay: 0,
      easing: 'easeInQuad'
    })

    anime({
      
      
      targets: loadingProgress,
      translateY: [0, -20],
      opacity: [1, 0],
      duration: 500,
      delay: 250,
      easing: 'easeInQuad'
    })

    anime({
      
      
      targets: loadingBtn,
      translateY: [0, -20],
      opacity: [1, 0],
      duration: 500,
      delay: 500,
      easing: 'easeInQuad',
      complete: function() {
      
      
        progressArea.style.display = 'none'
        completeArea.style.display = 'flex'
      }
    })

    anime({
      
      
      targets: frameBtn,
      fill: ['#f5f9fe', '#1578ff'],
      duration: 500,
      delay: 750,
      easing: 'easeInQuad'
    })

    anime({
      
      
      targets: tick1,
      strokeDashoffset: [52, 0],
      opacity: [0, 1],
      duration: 500,
      easing: 'cubicBezier(.5, .05, .3, .9)',
      delay: 1000
    })

    anime({
      
      
      targets: tick2,
      strokeDashoffset: [52, 0],
      opacity: [0, 1],
      duration: 500,
      easing: 'cubicBezier(.5, .05, .3, .9)',
      delay: 1250
    })

    anime({
      
      
      targets: doneText,
      opacity: [0, 1],
      translateY: ['25', '0'],
      duration: 1000,
      easing: 'easeOutQuad',
      delay: 1250
    })
  }
</script>
</html>

Guess you like

Origin blog.csdn.net/qq_35241329/article/details/131790228