Вот моя не идеальная идея, где я буду полагаться только на CSS и немного элементов.Я подробно опишу каждую часть, а затем объединю это в одну анимацию.
Для части с пиктограммой (звездочкой) я бы сделал то же самое, но я, вероятно, рассмотрю фильтр grayscale
, чтобы иметь общий эффект, которыйработа с любым элементом и любым цветом.
.magic i{
color:red;
filter:grayscale(100%);
}
.magic:hover i{
animation:change 1s forwards;
}
@keyframes change{
50% {
transform:scale(0);
filter:grayscale(100%);
}
51% {
filter:grayscale(0%);
}
100% {
transform:scale(1);
filter:grayscale(0%);
}
}
<link rel="stylesheet" href="https://use.fontawesome.com/releases/v5.7.1/css/all.css">
<span class="magic">
<i class="fas fa-star fa-5x"></i>
</span>
Для круга я рассмотрю только один элемент, и хитрость здесь заключается в том, чтобы полагаться на окраску границ по сравнению с окраской фона.Изначально мы делаем высоту / ширину равной 0
, и у нас есть только граница, поэтому это будет полный круг.Затем мы просто уменьшаем толщину границы, сохраняя общую ширину одинаковой.Итак, мы сделаем:
- Начальная ширина / высота формы
0
и border-width 0
- Увеличиваем border-width для создания эффекта масштаба
- Мы уменьшаем ширину границы, увеличивая ширину / высоту, чтобы сохранить общую ширину / высоту одинаковой.
.circle {
display:inline-block;
width:0px;
height:0px;
border-radius:50%;
border-color:orange;
border-style:solid;
border-width:0px;
box-sizing:border-box;
}
body:hover .circle {
animation:change 1s forwards;
}
@keyframes change {
50% {
border-width:25px;
}
100% {
border-width:0;
width:50px;
height:50px;
}
}
body {
min-height:100px;
}
<span class="circle"></span>
У этого решения есть небольшой недостаток, так как элемент будет увеличиваться сверху слева, а не по центру.Мы можем исправить это, используя масштаб вместо изменения ширины / высоты:
.circle {
display:inline-block;
width:50px;
height:50px;
border-radius:50%;
border-color:orange;
border-style:solid;
border-width:25px;
transform:scale(0);
box-sizing:border-box;
}
body:hover .circle {
animation:change 1s linear forwards;
}
@keyframes change {
50% {
transform:scale(1);
border-width:25px;
}
100% {
transform:scale(1);
border-width:0;
}
}
body {
min-height:100px;
}
<span class="circle"></span>
Мы все еще можем упростить рассмотрение простого перехода:
.circle {
display:inline-block;
width:50px;
height:50px;
border-radius:50%;
border-color:orange;
border-style:solid;
border-width:25px;
transform:scale(0);
box-sizing:border-box;
transition:
transform 0.5s,
border-width 0.5s 0.5s;
}
body:hover .circle {
border-width:0;
transform:scale(1);
}
body {
min-height:100px;
}
<span class="circle"></span>
Теперь сложная часть и маленькие кружочки.Для этого я буду полагаться на radial-gradient
и масштаб.идея состоит в том, чтобы создать маленькие круги с градиентом внутри одного элемента, и используя масштаб, мы создадим эффект расширения.
.small {
display:inline-block;
width:100px;
height:100px;
background:
/*4 reds*/
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
/*4 oranges*/
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%);
background-size:16px 16px;
background-position:
calc(50% - 30px) calc(50% - 30px),
calc(50% + 30px) calc(50% - 30px),
calc(50% - 30px) calc(50% + 30px),
calc(50% + 30px) calc(50% + 30px),
calc(50% + 0px) calc(50% + 40px),
calc(50% + 40px) calc(50% + 0px),
calc(50% - 40px) calc(50% + 0px),
calc(50% + 0px) calc(50% - 40px);
background-repeat:no-repeat;
border-radius:50%;
}
<span class="small"></span>
Я создал 8 кругов и разместил их, смещая их от центра (проверьте этот ответ, чтобы получить более подробную информацию о том, как работает background-position
: https://stackoverflow.com/a/51734530/8620333). Вам просто нужно настроить размер, положение и цвет круга, как вы хотите.
А вот и с анимацией:
.small {
display:inline-block;
width:100px;
height:100px;
background:
/*4 reds*/
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
/*4 oranges*/
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%);
background-size:16px 16px;
background-position:
calc(50% - 30px) calc(50% - 30px),
calc(50% + 30px) calc(50% - 30px),
calc(50% - 30px) calc(50% + 30px),
calc(50% + 30px) calc(50% + 30px),
calc(50% + 0px) calc(50% + 40px),
calc(50% + 40px) calc(50% + 0px),
calc(50% - 40px) calc(50% + 0px),
calc(50% + 0px) calc(50% - 40px);
background-repeat:no-repeat;
border-radius:50%;
transform:scale(0);
transition:transform 0.5s,opacity 0.4s 0.4s;
}
body {
min-height:200px;
}
body:hover .small {
transform:scale(1);
opacity:0;
}
<span class="small"></span>
Если вы хотите более точную анимацию, вы также можете рассмотреть вопрос об уменьшении кругов путем уменьшения background-size
.
.small {
display:inline-block;
width:100px;
height:100px;
background:
/*4 reds*/
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
/*4 oranges*/
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%);
background-size:16px 16px; /*at least 2x7px */
background-position:
calc(50% - 30px) calc(50% - 30px),
calc(50% + 30px) calc(50% - 30px),
calc(50% - 30px) calc(50% + 30px),
calc(50% + 30px) calc(50% + 30px),
calc(50% + 0px) calc(50% + 40px),
calc(50% + 40px) calc(50% + 0px),
calc(50% - 40px) calc(50% + 0px),
calc(50% + 0px) calc(50% - 40px);
background-repeat:no-repeat;
border-radius:50%;
transform:scale(0);
transition:transform 0.5s,opacity 0.4s 0.4s,background-size 0.5s 0.4s;
}
body {
min-height:200px;
}
body:hover .small {
transform:scale(1);
opacity:0;
background-size:0 0;
}
<span class="small"></span>
Теперь вы просто делаете то же самое с другими маленькими кружками, изменяя некоторые значения.
Давайте соединим все это!
.magic {
display:inline-block;
margin:50px;
position:relative;
}
.magic i{
color:orange;
filter:grayscale(100%);
position:relative;
}
.magic:hover i{
animation:change 1s forwards;
}
@keyframes change{
50% {
transform:scale(0);
filter:grayscale(100%);
}
51% {
filter:grayscale(0%);
}
100% {
transform:scale(1);
filter:grayscale(0%);
}
}
/**/
.magic:before {
content:"";
position:absolute;
top:calc(50% - 45px);
left:calc(50% - 45px);
width:90px;
height:90px;
border-radius:50%;
border-color:orange;
border-style:solid;
border-width:45px;
transform:scale(0);
box-sizing:border-box;
}
.magic:hover::before {
transition:
transform 0.5s,
border-width 0.5s 0.5s;
border-width:0;
transform:scale(1);
}
/**/
.magic::after {
content:"";
position:absolute;
width:160px;
height:160px;
left:calc(50% - 80px);
top:calc(50% - 80px);
background:
/*4 reds*/
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
/*4 oranges*/
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%);
background-size:16px 16px;
background-position:
calc(50% - 50px) calc(50% - 50px),
calc(50% + 50px) calc(50% - 50px),
calc(50% - 50px) calc(50% + 50px),
calc(50% + 50px) calc(50% + 50px),
calc(50% + 0px) calc(50% + 70px),
calc(50% + 70px) calc(50% + 0px),
calc(50% - 70px) calc(50% + 0px),
calc(50% + 0px) calc(50% - 70px);
background-repeat:no-repeat;
border-radius:50%;
transform:scale(0);
}
.magic:hover:after {
transform:scale(1);
opacity:0;
background-size:0 0;
transition:
transform 0.5s 0.5s,
opacity 0.4s 0.9s,
background-size 0.5s 0.9s;
}
<link rel="stylesheet" href="https://use.fontawesome.com/releases/v5.7.1/css/all.css">
<span class="magic">
<i class="fas fa-star fa-5x"></i>
</span>
<span class="magic">
<i class="fas fa-user fa-5x"></i>
</span>
Как я уже сказал, он не идеален, но очень близок к тому, что вы хотите, с меньшим количеством элементов и с необходимыми деталями, так что вы можете легко настроить различные значения.Его также легко использовать с любым значком, поскольку вам нужно только добавить обертку к значку.
Я не добавил крошечные кружочки для простоты, но мы можем рассмотреть другой псевдоэлемент и легко добавить их:
.magic {
display:inline-block;
margin:50px;
position:relative;
}
.magic i{
color:orange;
filter:grayscale(100%);
}
.magic:hover i{
animation:change 1s forwards;
}
@keyframes change{
50% {
transform:scale(0);
filter:grayscale(100%);
}
51% {
filter:grayscale(0%);
}
100% {
transform:scale(1);
filter:grayscale(0%);
}
}
/**/
.magic:before {
content:"";
position:absolute;
top:calc(50% - 45px);
left:calc(50% - 45px);
width:90px;
height:90px;
border-radius:50%;
border-color:orange;
border-style:solid;
border-width:45px;
transform:scale(0);
box-sizing:border-box;
}
.magic:hover::before {
border-width:0;
transform:scale(1);
transition:
transform 0.5s,
border-width 0.5s 0.5s;
}
/**/
.magic::after,
.magic i::after{
content:"";
position:absolute;
width:160px;
height:160px;
left:calc(50% - 80px);
top:calc(50% - 80px);
background:
/*4 reds*/
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
radial-gradient(circle,red 50%,transparent 60%),
/*4 oranges*/
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%),
radial-gradient(circle,orange 50%,transparent 60%);
background-size:16px 16px;
background-position:
calc(50% - 50px) calc(50% - 50px),
calc(50% + 50px) calc(50% - 50px),
calc(50% - 50px) calc(50% + 50px),
calc(50% + 50px) calc(50% + 50px),
calc(50% + 0px) calc(50% + 70px),
calc(50% + 70px) calc(50% + 0px),
calc(50% - 70px) calc(50% + 0px),
calc(50% + 0px) calc(50% - 70px);
background-repeat:no-repeat;
border-radius:50%;
transform:scale(0);
}
.magic i::after {
background-size:10px 10px;
transform:rotate(10deg) scale(0);
}
.magic:hover:after {
transform:scale(1);
opacity:0;
background-size:0 0;
transition:transform 0.5s 0.5s,opacity 0.4s 0.9s,background-size 0.5s 0.9s;
}
.magic:hover i:after {
transform:rotate(10deg) scale(1);
opacity:0;
background-size:0 0;
transition:transform 0.5s 0.5s,opacity 0.4s 0.9s,background-size 0.5s 0.9s;
}
/**/
<link rel="stylesheet" href="https://use.fontawesome.com/releases/v5.7.1/css/all.css">
<span class="magic">
<i class="fas fa-star fa-5x"></i>
</span>
<span class="magic">
<i class="fas fa-user fa-5x"></i>
</span>
ОБНОВЛЕНИЕ
Это улучшение кода, учитывающее некоторую переменную CSS и использование более гибкого модуля для простого управленияразмерность:
.magic {
display:inline-block;
margin:50px;
position:relative;
--r:45px;
}
.magic i{
color:orange;
filter:grayscale(100%);
}
.magic:hover i{
animation:change 1s forwards;
}
@keyframes change{
50% {
transform:scale(0);
filter:grayscale(100%);
}
51% {
filter:grayscale(0%);
}
100% {
transform:scale(1);
filter:grayscale(0%);
}
}
/**/
.magic:before {
content:"";
position:absolute;
top:calc(50% - var(--r));
left:calc(50% - var(--r));
width:calc(2*var(--r));
height:calc(2*var(--r));
border-radius:50%;
border:solid orange var(--r);
transform:scale(0);
box-sizing:border-box;
}
.magic:hover::before {
border-width:0;
transform:scale(1);
transition:
transform 0.5s,
border-width 0.5s 0.5s;
}
/**/
.magic::after,
.magic i::after{
content:"";
position:absolute;
width: calc(4*var(--r));
height:calc(4*var(--r));
left:calc(50% - 2*var(--r));
top: calc(50% - 2*var(--r));
--c1:radial-gradient(circle,red 50% ,transparent 60%);
--c2:radial-gradient(circle,orange 50%,transparent 60%);
background:
/*4 reds*/
var(--c1),var(--c1),var(--c1),var(--c1),
/*4 oranges*/
var(--c2),var(--c2),var(--c2),var(--c2);
background-size:calc(var(--r)/3) calc(var(--r)/3);
background-position:
calc(50% - var(--r)) calc(50% - var(--r)),
calc(50% + var(--r)) calc(50% - var(--r)),
calc(50% - var(--r)) calc(50% + var(--r)),
calc(50% + var(--r)) calc(50% + var(--r)),
calc(50% + 0px) calc(50% + var(--r)*1.414),
calc(50% + var(--r)*1.414) calc(50% + 0px),
calc(50% - var(--r)*1.414) calc(50% + 0px),
calc(50% + 0px) calc(50% - var(--r)*1.414);
background-repeat:no-repeat;
transform:scale(0);
}
.magic i::after {
background-size:calc(var(--r)/5) calc(var(--r)/5);
transform:rotate(55deg) scale(0);
}
.magic:hover:after {
transform:scale(1);
opacity:0;
background-size:0 0;
transition:
transform 0.5s 0.5s,
opacity 0.4s 0.9s,
background-size 0.5s 0.9s;
}
.magic:hover i:after {
transform:rotate(55deg) scale(1);
opacity:0;
background-size:0 0;
transition:
transform 0.5s 0.5s,
opacity 0.4s 0.9s,
background-size 0.5s 0.9s;
}
/**/
<link rel="stylesheet" href="https://use.fontawesome.com/releases/v5.7.1/css/all.css">
<span class="magic" style="--r:80px;">
<i class="fas fa-star fa-10x"></i>
</span>
<span class="magic">
<i class="fas fa-user fa-5x"></i>
</span>
<span class="magic" style="--r:20px;">
<i class="far fa-bell fa-3x"></i>
</span>
По существу, переменная r
будет определять радиус всей фигуры, и вы можете легко изменять ее в зависимости от размера вашей иконки.