В этой статье мы подробно рассмотрим, как реализовать анимацию с эффектом Hover для карточек, как показано ниже.

Основные трудности, которые нам предстоит решить:

  1. При движении мыши отображается граница и световой эффект текущего края карты.

  2. Эффект только рядом с курсором: Существует несколько подходов для реализации этого эффекта. Можно рассчитать зону вокруг курсора и применять эффект только в этой области, но это может быть слишком ресурсоемко.

Альтернативный подход — использовать маску. Можно заранее создать общий эффект с градиентной границей и внутренним свечением, а затем сделать так, чтобы маска следовала за курсором.

Чтобы создать такой эффект, потребуется адаптировать его к движению мыши и переключать эффекты в зависимости от положения курсора. На данный момент чистый CSS не справляется с этим, поэтому понадобится немного JavaScript.

В следующем разделе мы разберем, как пошагово реализовать этот эффект, учитывая упомянутые трудности.

Создание статического эффекта

Сначала мы создаем статический эффект, который виден, когда мышь не находится над карточкой. Вот пример:

Каждая карточка имеет размытое фоновое изображение, которое должно быть индивидуальным для каждой картинки, поэтому одну универсальную фоновую картинку использовать нельзя.

Кроме того, размытое фоновое изображение должно примерно соответствовать цвету настоящего изображения.

Для решения этой задачи можно использовать CSS-фильтр filter: blur(). Код для этого достаточно простой.

<div></div>

:root {
    --pic: url("https://oss.aiyuzhou8.com/2023/05/08-.jpg");
}
div {
    position: relative;
    margin: auto;
    width: 350px;
    height: 500px;
    border-radius: 30px;
    overflow: hidden;
    
    &::before,
    &::after {
        content: "";
        position: absolute;
        background: var(--pic);
        background-size: cover;
        background-position: center;
        border-radius: 30px;
    }
    
    &::before {
        inset: 0;
        filter: blur(20px);
    }
    
    &::after {
        inset: 50px;
    }
}

Мы используем псевдоэлементы: один для исходного изображения, другой для размытого фона.

Это позволяет адаптировать размытое изображение к любому фону.

Реализация градиентной границы

Следующий шаг – создание градиентной границы.

Для этого используется conic-gradient. Потребуется дополнительный элемент div, чтобы создать нужный эффект. Мы накладываем этот элемент на предыдущий эффект, немного увеличив его размер.

<div></div>
div {
    width: 350px;
    height: 500px;
    border-radius: 30px;
    background: conic-gradient(#03a9f4, #e91e63, #9c27b0, #ff5722, #03a9f4);
}

Получаем такой градиент.

Мы накладываем этот элемент на предыдущий эффект, немного увеличив его размер.

При внимательном рассмотрении можно заметить, что помимо градиентной границы, текущий эффект также имеет внутреннее свечение. Он очень кривой, это не совсем то, чего мы хотим добиться:

Исследование прозрачности с filter: blur()

Эффект прозрачности возникает из-за того, что элемент с фильтром filter: blur() создаёт плавное затухание прозрачности от краёв к центру.

Проведем простой эксперимент

<div></div>
<div></div>
div {
    position: relative;
    width: 200px;
    height: 300px;
    border-radius: 10px;
    border: 1px solid #000;
    background: conic-gradient(#03a9f4, #e91e63, #9c27b0, #ff5722, #03a9f4);
    
    &::before {
        content: "";
        position: absolute;
        inset: 10px;
        border-radius: 10px;
        background: #fff;
        border: 1px solid #000;
    }
}

Мы создаем два одинаковых div, где сам элемент имеет угловой градиентный фон.

Затем, используя его псевдоэлемент, мы устанавливаем белый фон в середине элемента, на расстоянии 10px от границы. Эффект выглядит следующим образом:

На данный момент два элемента не отличаются друг от друга. Но далее мы добавляем фильтр: blur() эффект "Размытие по Гауссу" к псевдоэлементу второго элемента:

div:nth-child(2) {
    &::before {
        filter: blur(20px);
    }
}

В этот момент снова посмотрите на эффекты:

На краях белого элемента действительно наблюдается эффект уменьшения прозрачности в направлении внутрь.

Конечно, поскольку размытие по Гауссу распространяется и наружу, приведенный выше DEMO выглядит не очень четко, поэтому мы можем предотвратить распространение гауссова размытия наружу, применив дополнительный слой контейнеров с overflow: hidden.

Давайте еще немного подправим макет:

<div class="g-father">
    <div class="g-child"></div>
</div>
<div class="g-father">
    <div class="g-child"></div>
</div>
.g-father {
    position: relative;
    width: 200px;
    height: 300px;
    border-radius: 10px;
    border: 1px solid #000;
    background: conic-gradient(#03a9f4, #e91e63, #9c27b0, #ff5722, #03a9f4);
    
    .g-child {
        position: absolute;
        inset: 10px;
        border-radius: 10px;
        border: 1px solid #000;
        overflow: hidden;
        
        &::before {
            content: "";
            position: absolute;
            inset: 0;
            background: #fff;
            border-radius: 10px;
            
        }
    }
}

.g-father:nth-child(2) {
    .g-child::before {
        filter: blur(20px);
    }
}

Сейчас если мы посмотрим на весь эффект, элемент с установленным фильтром: blur() будет иметь прозрачный распад от краев к центру, и эффект будет очень очевиден:

Добавление событий мыши и масок для достижения эффекта

Итак, к этому моменту нам удалось добиться такого эффекта:

Исходя из вышеописанного эффекта, в итоге мы добиваемся такого эффекта:

Мы будем использовать обработчик события mousemove для отслеживания перемещения курсора и комбинировать его с маской для создания желаемого эффекта.

В реализации эффекта используются несколько слоев. Конкретно, граница и внутреннее свечение — это отдельные слои, которые формируют фоновый эффект.

Что нам нужно сделать:

  1. Создать маску с радиальным градиентом с помощью radial-gradient().

  2. Отслеживать событие перемещения мыши и перемещать центр маски в зависимости от положения курсора.

  3. Добавить дополнительный слой, чтобы градиентный элемент фона появлялся только при наведении мыши (Hover), и исчезал, когда курсор покидает область элемента.

Примерный код выглядит следующим образом:

<div id="g-container">
    <div id="g-img"></div>
</div>
:root {
    --x: 0;
    --y: 0;
}
#g-container {
    position: relative;
    width: 350px;
    height: 500px;
    border-radius: 30px;
}
#g-img {
    position: absolute;
    inset: 0px;
    border-radius: 30px;
    background: conic-gradient(#03a9f4, #e91e63, #9c27b0, #ff5722, #03a9f4);
    mask: radial-gradient(
        circle at var(--x) var(--y),
        #000,
        #000,
        transparent,
        transparent,
        transparent
    );
}
const container = document.getElementById("g-container");
const img = document.getElementById("g-img");

container.addEventListener("mousemove", (event) => {
    img.style.visibility = 'visible';

    const target = event.target;
    const rect = target.getBoundingClientRect();

    var offsetX = event.clientX - rect.left;
    var offsetY = event.clientY - rect.top;

    var percentX = (Math.min(Math.max(offsetX / rect.width, 0), 1) * 100).toFixed(2);
    var percentY = (Math.min(Math.max(offsetY / rect.height, 0), 1) * 100).toFixed(2);;

    console.log('X: ' + percentX + '%');
    console.log('Y: ' + percentY + '%');

    container.setAttribute('style', `--x: ${percentX}%;--y: ${percentY}%;`);

});

container.addEventListener("mouseout", (event) => {
    img.style.visibility = 'hidden';
});

Перемещение мыши по графику дает такой эффект:

Объедините два первых слоя, чтобы в итоге получился идеальный эффект:

Комментарии (4)


  1. Finesse
    15.08.2024 13:32
    +1

    В вашей градиентной рамке середина не прозрачная, а белая. Каким образом вы делаете её прозрачной, чтобы наложить на фоновый слой?

    Мы накладываем этот элемент на предыдущий эффект, немного увеличив его размер

    (почти готовое изображение)

    Тут либо картинка не та, либо пропущено много шагов.

    Ну и ссылочку бы на CodePen, чтобы посмотреть код целиком.


    1. AlexeyRybakov Автор
      15.08.2024 13:32
      +2

      https://codepen.io/Chokcoco/pen/NWmQLvZ


      1. Finesse
        15.08.2024 13:32
        +1

        Понятно, фоновый слой находится над слоем конусного градиента, а не под ним. Получается, размытие фонового слоя убивает двух зайцев: собственно размытие и мягкое свечение рамки. Теперь картинка после «Мы накладываем этот элемент на предыдущий эффект» выглядит логично. Было бы намного понятнее, если бы вы показали схему слоёв как в начале статьи.

        (оригинальная ссылка от автора)


  1. monochromer
    15.08.2024 13:32

    Нужно отметить, что статья является переводом https://www.cnblogs.com/coco1s/p/18358267.