Реализация анимации загрузки (Preloader) сайта

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

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

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

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

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Реализация анимации загрузки (Preloader) сайта
Средняя
от 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

Реализация анимации загрузки (Preloader) сайта

Preloader — экран, который показывается пока страница загружается. Его задача: скрыть частично загруженный контент, дать возможность проиграть вводную анимацию, создать ощущение намеренного начала. Плохой preloader раздражает. Хороший — органично переходит в первый экран.

Когда preloader оправдан

Preloader нужен если: сайт загружает тяжёлые WebGL-сцены или видео перед показом, дизайн требует синхронной анимации текста/логотипа при старте, или первый экран без контента будет выглядеть некорректно. Для обычных информационных сайтов preloader — лишняя задержка.

Базовая структура

Preloader вставляется первым в <body>, работает на чистом CSS без JS-зависимостей для начального рендера:

<body>
  <div id="preloader" class="preloader">
    <div class="preloader__content">
      <div class="preloader__logo">
        <svg>...</svg>
      </div>
      <div class="preloader__counter">
        <span id="preloader-count">0</span>
        <span>%</span>
      </div>
    </div>
    <div class="preloader__overlay"></div>
  </div>

  <main id="app">
    <!-- Контент скрыт до окончания preloader -->
  </main>
</body>
.preloader {
  position: fixed;
  inset: 0;
  z-index: 9999;
  background: #0a0a0a;
  display: flex;
  align-items: center;
  justify-content: center;
  overflow: hidden;
}

/* Контент скрыт во время загрузки */
body.is-loading #app {
  visibility: hidden;
}

Отслеживание прогресса загрузки

class Preloader {
  private counter: HTMLElement
  private preloader: HTMLElement
  private currentCount = 0
  private targetCount = 0
  private resources: string[]
  private loaded = 0
  private rafId: number | null = null

  constructor(resources: string[] = []) {
    this.counter = document.getElementById('preloader-count')!
    this.preloader = document.getElementById('preloader')!
    this.resources = resources

    document.body.classList.add('is-loading')
  }

  async load(): Promise<void> {
    if (this.resources.length === 0) {
      // Если ресурсов нет — имитируем прогресс
      await this.fakeProgress()
    } else {
      await this.loadResources()
    }

    await this.hide()
    document.body.classList.remove('is-loading')
  }

  private async loadResources(): Promise<void> {
    const total = this.resources.length

    const promises = this.resources.map((url) =>
      this.loadAsset(url).then(() => {
        this.loaded++
        this.targetCount = Math.round((this.loaded / total) * 100)
      })
    )

    // Анимируем счётчик параллельно с загрузкой
    this.animateCounter()

    await Promise.all(promises)
    this.targetCount = 100
  }

  private loadAsset(url: string): Promise<void> {
    return new Promise((resolve) => {
      if (/\.(jpg|png|webp|svg|gif)$/i.test(url)) {
        const img = new Image()
        img.onload = () => resolve()
        img.onerror = () => resolve()  // не блокировать на ошибке
        img.src = url
      } else if (/\.(mp4|webm)$/i.test(url)) {
        const video = document.createElement('video')
        video.oncanplaythrough = () => resolve()
        video.onerror = () => resolve()
        video.src = url
        video.load()
      } else {
        fetch(url)
          .then(() => resolve())
          .catch(() => resolve())
      }
    })
  }

  private async fakeProgress(): Promise<void> {
    return new Promise((resolve) => {
      let progress = 0
      const intervals = [
        { target: 30, duration: 400 },
        { target: 70, duration: 600 },
        { target: 90, duration: 300 },
        { target: 100, duration: 200 },
      ]

      let step = 0

      const tick = () => {
        const { target, duration } = intervals[step]
        const speed = (target - progress) / (duration / 16)

        progress = Math.min(progress + speed, target)
        this.targetCount = Math.round(progress)

        if (progress >= target) {
          step++
          if (step >= intervals.length) {
            resolve()
            return
          }
        }

        requestAnimationFrame(tick)
      }

      this.animateCounter()
      tick()
    })
  }

  private animateCounter() {
    const tick = () => {
      if (this.currentCount < this.targetCount) {
        this.currentCount = Math.min(
          this.currentCount + Math.ceil((this.targetCount - this.currentCount) * 0.1),
          this.targetCount
        )
        this.counter.textContent = String(this.currentCount)
      }

      if (this.currentCount < 100) {
        this.rafId = requestAnimationFrame(tick)
      }
    }

    this.rafId = requestAnimationFrame(tick)
  }

  private async hide(): Promise<void> {
    // Ждём пока счётчик дойдёт до 100
    await new Promise<void>((resolve) => {
      const check = setInterval(() => {
        if (this.currentCount >= 100) {
          clearInterval(check)
          resolve()
        }
      }, 50)
    })

    return new Promise((resolve) => {
      // Анимация исчезновения через GSAP
      import('gsap').then(({ default: gsap }) => {
        const tl = gsap.timeline({ onComplete: () => {
          this.preloader.style.display = 'none'
          resolve()
        }})

        tl.to('.preloader__counter', { opacity: 0, duration: 0.3 })
          .to('.preloader', {
            clipPath: 'inset(0 0 100% 0)',
            duration: 0.8,
            ease: 'power3.inOut',
          })
          .from('#app', { opacity: 0, duration: 0.4 }, '-=0.2')
      })
    })
  }
}

// Инициализация
const preloader = new Preloader([
  '/images/hero-bg.webp',
  '/images/about-photo.webp',
])

preloader.load().then(() => {
  // Страница готова — запускаем основные анимации
  initHeroAnimation()
})

Анимация логотипа

Типовой вариант: SVG логотип с path анимацией через stroke-dashoffset.

.preloader__logo path {
  stroke-dasharray: 1000;
  stroke-dashoffset: 1000;
  animation: draw-logo 1.5s ease forwards;
}

@keyframes draw-logo {
  to { stroke-dashoffset: 0; }
}
// После окончания draw-анимации — fill
gsap.to('.preloader__logo path', {
  fill: '#ffffff',
  stroke: 'transparent',
  duration: 0.4,
  delay: 1.6,
})

Сохранение состояния — показывать один раз

Если preloader должен показываться только при первом визите в сессии:

const PRELOADER_KEY = 'preloader_shown'

function shouldShowPreloader(): boolean {
  if (sessionStorage.getItem(PRELOADER_KEY)) return false
  sessionStorage.setItem(PRELOADER_KEY, '1')
  return true
}

if (shouldShowPreloader()) {
  const preloader = new Preloader()
  preloader.load()
} else {
  document.getElementById('preloader')?.remove()
  document.body.classList.remove('is-loading')
}

Сроки

Простой preloader с прогресс-баром и fade-out — 4–6 часов. Анимация логотипа, отслеживание реальных ресурсов, плавный переход к первому экрану — 1–2 дня.