AnswerBun.com

Как анимировать Skeleton Screen с помощью маски CSS?

Я пытаюсь добавить тонкую анимацию мерцания к skeleton, похожему на этот. В настоящее время у меня есть код, который выглядит следующим образом (см. На CodePen)

введите сюда описание изображения

Я пытаюсь написать скелетный компонент, который может принимать SVG следующим образом:

<div class="skeleton" aria-busy="true">
  <svg width="233" height="68" viewBox="0 0 233 68" xmlns="http://www.w3.org/2000/svg">
    <g opacity="0.8">
      <rect x="79" y="32" width="154" height="11" rx="2" fill="black" fill-opacity="0.07"/>
      <rect width="179" height="20" rx="2" fill="black" fill-opacity="0.07"/> 
      <rect x="79" y="52" width="84" height="11" rx="2" fill="black" fill-opacity="0.07"/>
      <rect y="26" width="67" height="42" rx="2" fill="black" fill-opacity="0.07"/>
    </g>
  </svg>
</div>

Вот CSS, который я использую для анимации мерцания над SVG:

.skeleton {
  overflow: hidden;
  position: relative;
}

.skeleton::before {
  content: '';
  position: absolute;
  top: 0;
  left: 0;
  bottom: 0;
  right: 0;
  background: linear-gradient(to right, rgb(243, 242, 241) 0%, rgb(237, 235, 233) 50%, rgb(243, 242, 241) 100%) 0px 0px / 90% 100% no-repeat rgb(243, 242, 241);
  transform: translateX(-100%);
  
  animation-name: skeleton-animation;
  animation-duration: 2s;
  animation-timing-function: ease-in-out;
  animation-direction: normal;
  animation-iteration-count: infinite;
}

@keyframes skeleton-animation {
  0% {
    transform: translateX(-100%);
  }
  100% {
    transform: translateX(100%);
  }
}

Я пытаюсь понять, как использовать маску, как описано здесь, чтобы мерцание анимации происходило только над SVG.

Свободный перевод вопроса How to Animate a Skeleton Screen with a CSS Mask от участника @Muhammad Rehan.

2 Answers

Один из способов использовать SVG в качестве изображения-маски и позволить обновлять положение фона линейного градиента.

Пример (при необходимости, обновите свои собственные цвета градиента):

.skeleton {
  overflow: hidden;
  position: relative;
  width: 233px;
  height: 68px;
  background: linear-gradient(to right, rgb(143, 142, 141) 0%, rgb(237, 235, 233) 50%, rgb(143, 142, 141) 100%) 0px 0px / 100% 100% rgb(243, 242, 241);
  -webkit-mask-image: url("");
  margin: 1em;
  animation: linearAnim 2s infinite linear
}

@keyframes linearAnim {
  0% {
    background-position: 0px 0px;
  }
  100% {
    background-position: 230px 0px;
  }
}

/* demo purpose only */
.skeleton + .skeleton {

  background: linear-gradient(to right, rgba(143, 142, 141,0.75) 0%, rgba(237, 235, 233, 0.75) 50%, rgba(143, 142, 141, 0.75) 100%) 0px 0px / 100% 100% rgba(243, 242, 241, 0.5);
  }
  .skeleton.blue {
  background-color:blue;
  }
  .skeleton.red {
  background-color:red;
  }
  .skeleton.yellow {
  background-color:yellow;
  }
  .skeleton.green {
  background-color:green;
  }
  .flex {display:flex;align-items:center;justify-content:center;box-sizing:border-box;padding-top:0.6em;font-size:10px;color:#fff8;}
body {background:#bee; display:grid;grid-template-columns:repeat(auto-fill, 300px);}
<div class="skeleton" aria-busy="true"></div>
<div class="skeleton blue" aria-busy="true"></div>
<div class="skeleton red" aria-busy="true"></div>
<div class="skeleton yellow" aria-busy="true"></div>
<div class="skeleton green flex" aria-busy="true">On its way ...</div>

вот аналогичный вопрос с другим методом

Свободный перевод ответа от участника @G-Cyrillus.

Answered by Alexandr_TT on January 1, 2021

Если не нужно использовать SVG, то это будет немного проще в HTML.

Но вот как это сделать с SVG.

.shimmer-rect {
  animation-name: skeleton-animation;
  animation-duration: 2s;
  animation-timing-function: linear;
  animation-direction: normal;
  animation-iteration-count: infinite;
}

@keyframes skeleton-animation {
  0% {
    transform: translateX(0px);
  }
  100% {
    transform: translateX(2000px);
  }
}
<div class="skeleton" aria-busy="true">
  <svg width="233" height="68" viewBox="0 0 233 68" overflow="visible">
    <defs>
      <linearGradient id="shimmer" gradientUnits="userSpaceOnUse" x2="2000" spreadMethod="repeat">
        <stop offset="0" stop-color="rgb(237,235,233)"/>
        <stop offset="25%" stop-color="rgb(243,242,241)"/>
        <stop offset="50%" stop-color="rgb(237,235,233)"/>
      </linearGradient>
      <mask id="shimmer-mask">
        <rect x="79" y="32" width="154" height="11" rx="2" fill="white"/>
        <rect width="179" height="20" rx="2" fill="white"/> 
        <rect x="79" y="52" width="84" height="11" rx="2" fill="white"/>
        <rect y="26" width="67" height="42" rx="2" fill="white"/>
      </mask>
    </defs>
    <g mask="url(#shimmer-mask)">
      <rect class="shimmer-rect" x="-2000" width="3000" height="100%" fill="url(#shimmer)"/>
    </g>
  </svg>
</div>

Свободный перевод ответа от участника @Paul LeBeau.

Answered by Alexandr_TT on January 1, 2021

Add your own answers!

Related Questions

Как отправить с FFmpeg на ffserver

0  Asked on November 5, 2021 by uiopio

 

ошибка при использовании команды pip

1  Asked on November 5, 2021 by pavel-framer

       

Ask a Question

Get help from others!

© 2022 AnswerBun.com. All rights reserved.