Реализация Noise/Grain эффектов на сайте

Наша компания занимается разработкой, поддержкой и обслуживанием сайтов любой сложности. От простых одностраничных сайтов до масштабных кластерных систем построенных на микро сервисах. Опыт разработчиков подтвержден сертификатами от вендоров.

Разработка и обслуживание любых видов сайтов:

Информационные сайты или веб-приложения
Сайты визитки, landing page, корпоративные сайты, онлайн каталоги, квиз, промо-сайты, блоги, новостные ресурсы, информационные порталы, форумы, агрегаторы
Сайты или веб-приложения электронной коммерции
Интернет-магазины, B2B-порталы, маркетплейсы, онлайн-обменники, кэшбэк-сайты, биржи, дропшиппинг-платформы, парсеры товаров
Веб-приложения для управления бизнес-процессами
CRM-системы, ERP-системы, корпоративные порталы, системы управления производством, парсеры информации
Сайты или веб-приложения электронных услуг
Доски объявлений, онлайн-школы, онлайн-кинотеатры, конструкторы сайтов, порталы предоставления электронных услуг, видеохостинги, тематические порталы

Это лишь некоторые из технических типов сайтов, с которыми мы работаем, и каждый из них может иметь свои специфические особенности и функциональность, а также быть адаптированным под конкретные потребности и цели клиента

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Реализация Noise/Grain эффектов на сайте
Средняя
от 1 рабочего дня до 3 рабочих дней
Часто задаваемые вопросы

Наши компетенции:

Этапы разработки

Последние работы

  • image_website-b2b-advance_0.png
    Разработка сайта компании B2B ADVANCE
    1262
  • image_web-applications_feedme_466_0.webp
    Разработка веб-приложения для компании FEEDME
    1171
  • image_websites_belfingroup_462_0.webp
    Разработка веб-сайта для компании БЕЛФИНГРУПП
    874
  • image_ecommerce_furnoro_435_0.webp
    Разработка интернет магазина для компании FURNORO
    1094
  • image_crm_enviok_479_0.webp
    Разработка веб-приложения для компании Enviok
    831
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Разработка веб-сайта для компании ФИКСПЕР
    851

Реализация Noise/Grain эффектов на сайте

Grain (зернистость, шум плёнки) — один из самых часто запрашиваемых визуальных эффектов в современном веб-дизайне. Он убирает стерильность экрана, добавляет текстуру, делает градиенты более органичными. Реализуется несколькими способами — от статичного SVG-фильтра до анимированного Canvas-шума.

Метод 1: SVG filter + CSS (самый лёгкий)

Браузер применяет процедурный шум через SVG feTurbulence. Почти нулевая нагрузка на CPU/GPU.

<!-- Скрытый SVG с фильтром -->
<svg xmlns="http://www.w3.org/2000/svg" style="position:absolute;width:0;height:0">
  <defs>
    <filter id="grain-filter" x="0%" y="0%" width="100%" height="100%"
            color-interpolation-filters="sRGB">
      <feTurbulence
        type="fractalNoise"
        baseFrequency="0.65"
        numOctaves="3"
        stitchTiles="stitch"
        result="noise"
      />
      <feColorMatrix type="saturate" values="0" in="noise" result="grayNoise"/>
      <feBlend in="SourceGraphic" in2="grayNoise" mode="overlay" result="blended"/>
      <feComponentTransfer in="blended">
        <feFuncA type="linear" slope="1"/>
      </feComponentTransfer>
    </filter>
  </defs>
</svg>
.grain-overlay {
  position: fixed;
  inset: 0;
  z-index: 1000;
  pointer-events: none;
  opacity: 0.15;
  filter: url(#grain-filter);
  background: transparent;
}

/* Или на конкретный элемент */
.hero-with-grain {
  position: relative;
}

.hero-with-grain::after {
  content: '';
  position: absolute;
  inset: 0;
  opacity: 0.12;
  filter: url(#grain-filter);
  pointer-events: none;
}

Статичный — шум не двигается. Подходит для текстурирования градиентов.

Метод 2: CSS pseudo-element с base64 PNG

Pre-rendered PNG с шумом, тайлится через background-size. Меньше артефактов на некоторых браузерах.

.grain-texture::after {
  content: '';
  position: fixed;
  inset: -200%;  /* выходим за пределы для анимации */
  width: 400%;
  height: 400%;
  background-image: url('/textures/grain.png');
  background-size: 200px 200px;
  opacity: 0.08;
  pointer-events: none;
  z-index: 9999;
  animation: grain-shift 0.2s steps(1) infinite;
}

@keyframes grain-shift {
  0%  { transform: translate(0, 0); }
  10% { transform: translate(-5%, -10%); }
  20% { transform: translate(-15%, 5%); }
  30% { transform: translate(7%, -25%); }
  40% { transform: translate(-5%, 25%); }
  50% { transform: translate(-15%, 10%); }
  60% { transform: translate(15%, 0%); }
  70% { transform: translate(0%, 15%); }
  80% { transform: translate(3%, 35%); }
  90% { transform: translate(-10%, 10%); }
  100%{ transform: translate(0%, 5%); }
}

Генерация grain.png через Node.js:

// scripts/generate-grain.js
const { createCanvas } = require('canvas')
const fs = require('fs')

const SIZE = 256
const canvas = createCanvas(SIZE, SIZE)
const ctx = canvas.getContext('2d')

const imageData = ctx.createImageData(SIZE, SIZE)
const data = imageData.data

for (let i = 0; i < data.length; i += 4) {
  const value = Math.floor(Math.random() * 255)
  data[i] = value      // R
  data[i + 1] = value  // G
  data[i + 2] = value  // B
  data[i + 3] = 255    // A
}

ctx.putImageData(imageData, 0, 0)
fs.writeFileSync('./public/textures/grain.png', canvas.toBuffer('image/png'))

Метод 3: Canvas с анимированным шумом

Полный контроль: скорость обновления, размер зерна, opacity, цвет.

class GrainCanvas {
  private canvas: HTMLCanvasElement
  private ctx: CanvasRenderingContext2D
  private rafId: number | null = null
  private frameCount = 0
  private readonly FRAME_SKIP = 2  // обновлять каждые N кадров

  constructor(container: HTMLElement = document.body, opacity = 0.1) {
    this.canvas = document.createElement('canvas')
    this.canvas.style.cssText = `
      position: fixed;
      inset: 0;
      width: 100%;
      height: 100%;
      pointer-events: none;
      z-index: 9999;
      opacity: ${opacity};
      mix-blend-mode: overlay;
    `
    container.appendChild(this.canvas)

    this.ctx = this.canvas.getContext('2d')!
    this.resize()
    window.addEventListener('resize', this.resize)
    this.start()
  }

  private resize = () => {
    // Используем меньшее разрешение для производительности
    const scale = 0.5
    this.canvas.width = window.innerWidth * scale
    this.canvas.height = window.innerHeight * scale
  }

  private generateNoise() {
    const { width, height } = this.canvas
    const imageData = this.ctx.createImageData(width, height)
    const buffer = new Uint32Array(imageData.data.buffer)

    for (let i = 0; i < buffer.length; i++) {
      const v = (Math.random() * 256) | 0
      // Packed RGBA (little-endian): AABBGGRR
      buffer[i] = (255 << 24) | (v << 16) | (v << 8) | v
    }

    this.ctx.putImageData(imageData, 0, 0)
  }

  private start() {
    const tick = () => {
      this.frameCount++
      if (this.frameCount % this.FRAME_SKIP === 0) {
        this.generateNoise()
      }
      this.rafId = requestAnimationFrame(tick)
    }
    this.rafId = requestAnimationFrame(tick)
  }

  destroy() {
    if (this.rafId) cancelAnimationFrame(this.rafId)
    window.removeEventListener('resize', this.resize)
    this.canvas.remove()
  }
}

new GrainCanvas(document.body, 0.08)

Метод 4: WebGL shader noise (GLSL)

Для grain поверх WebGL-сцены или для процедурного шума с управлением частотой:

// Fragment shader — animated film grain
uniform float uTime;
uniform float uIntensity;
uniform vec2 uResolution;

varying vec2 vUv;

// Псевдослучайная функция
float rand(vec2 co) {
  return fract(sin(dot(co.xy, vec2(12.9898, 78.233))) * 43758.5453);
}

void main() {
  // Меняем seed каждый кадр — зерно "живёт"
  vec2 seed = vUv + fract(uTime * 37.0);
  float grain = rand(seed * uResolution) * 2.0 - 1.0;

  // Добавляем к существующему цвету
  vec4 base = texture2D(uTexture, vUv);
  base.rgb += grain * uIntensity;

  gl_FragColor = base;
}

Grain поверх градиента: устранение banding

Градиенты на экранах с ограниченной глубиной цвета показывают полосы. Grain эффективно маскирует banding:

.gradient-section {
  background: linear-gradient(135deg, #1a0050 0%, #0a1628 50%, #001a2e 100%);
  position: relative;
}

.gradient-section::after {
  content: '';
  position: absolute;
  inset: 0;
  background-image: url('/textures/grain.png');
  background-size: 150px;
  opacity: 0.05;
  animation: grain-shift 0.3s steps(1) infinite;
  pointer-events: none;
}

Производительность

Метод CPU GPU Анимация
SVG feTurbulence статик ~0 низкий нет
CSS pseudo + PNG ~0 низкий да
Canvas средний ~0 да
WebGL shader ~0 минимал да

Canvas с полным разрешением при 60fps создаёт нагрузку. Решения:

  • Уменьшить размер canvas (scale = 0.5) и растянуть через CSS
  • Обновлять каждые 2–3 кадра (FRAME_SKIP)
  • OffscreenCanvas + Worker для отдельного потока
// OffscreenCanvas в Web Worker
// main.js
const canvas = document.getElementById('grain')
const offscreen = canvas.transferControlToOffscreen()
const worker = new Worker('/workers/grain-worker.js')
worker.postMessage({ canvas: offscreen }, [offscreen])

Сроки

SVG или PNG метод с CSS анимацией — 2–3 часа. Canvas с настройкой интенсивности, переключением для prefers-reduced-motion и React-компонентом — 1 день.