Реалізація Scroll-triggered Popup на сайті
Scroll-triggered popup з'являється, коли користувач прокрутив певний відсоток сторінки. Логіка: якщо людина прочитала до середини статті або сторінки товару — вона залучена, і пропозиція підписатися або отримати знижку потрапляє в потрібний момент.
Реалізація з Intersection Observer
IntersectionObserver ефективніше scroll-події: не блокує основний потік, не потребує requestAnimationFrame.
// scroll-popup.ts
interface ScrollPopupConfig {
triggerPercent?: number; // % прокручування сторінки (0-100)
triggerElement?: string; // CSS-селектор елемента-тригера
cooldownMs?: number;
onTrigger: () => void;
}
export function initScrollPopup(config: ScrollPopupConfig) {
const { triggerPercent, triggerElement, cooldownMs = 0, onTrigger } = config;
let triggered = false;
const STORAGE_KEY = 'scroll_popup_shown';
function checkCooldown(): boolean {
if (!cooldownMs) return true;
const last = localStorage.getItem(STORAGE_KEY);
if (last && Date.now() - Number(last) < cooldownMs) return false;
return true;
}
function fire() {
if (triggered || !checkCooldown()) return;
triggered = true;
if (cooldownMs) localStorage.setItem(STORAGE_KEY, String(Date.now()));
onTrigger();
}
// Варіант 1: відсоток прокручування
if (triggerPercent !== undefined) {
// Створюємо невидимий елемент-маркер на потрібній висоті
const marker = document.createElement('div');
marker.style.cssText = 'position:absolute;top:0;left:0;width:1px;height:1px;pointer-events:none;';
document.body.style.position = 'relative';
document.body.appendChild(marker);
// Позиціонуємо маркер на triggerPercent висоти документа
function updateMarker() {
const docHeight = document.documentElement.scrollHeight;
const viewportHeight = window.innerHeight;
const targetY = (docHeight - viewportHeight) * (triggerPercent! / 100);
marker.style.top = `${targetY}px`;
}
updateMarker();
window.addEventListener('resize', updateMarker, { passive: true });
const observer = new IntersectionObserver(
([entry]) => { if (entry.isIntersecting) fire(); },
{ threshold: 0 }
);
observer.observe(marker);
return () => {
observer.disconnect();
marker.remove();
window.removeEventListener('resize', updateMarker);
};
}
// Варіант 2: конкретний DOM-елемент
if (triggerElement) {
const el = document.querySelector(triggerElement);
if (!el) {
console.warn(`[scroll-popup] Елемент не знайдено: ${triggerElement}`);
return () => {};
}
const observer = new IntersectionObserver(
([entry]) => { if (entry.isIntersecting) fire(); },
{ threshold: 0.5 } // 50% елемента видно
);
observer.observe(el);
return () => observer.disconnect();
}
return () => {};
}
Використання в React
// BlogPost.tsx
import { useEffect } from 'react';
import { initScrollPopup } from './scroll-popup';
import { NewsletterPopup } from './NewsletterPopup';
import { useState } from 'react';
export function BlogPost({ content }: { content: string }) {
const [showPopup, setShowPopup] = useState(false);
useEffect(() => {
const cleanup = initScrollPopup({
triggerPercent: 60,
cooldownMs: 7 * 24 * 60 * 60 * 1000, // раз на тиждень
onTrigger: () => setShowPopup(true),
});
return cleanup;
}, []);
return (
<>
<article dangerouslySetInnerHTML={{ __html: content }} />
{showPopup && (
<NewsletterPopup onClose={() => setShowPopup(false)} />
)}
</>
);
}
Спливаюче вікно підписки
// NewsletterPopup.tsx
import { useRef, useEffect, useState } from 'react';
export function NewsletterPopup({ onClose }: { onClose: () => void }) {
const dialogRef = useRef<HTMLDialogElement>(null);
const [email, setEmail] = useState('');
const [status, setStatus] = useState<'idle' | 'loading' | 'done'>('idle');
useEffect(() => {
dialogRef.current?.showModal();
return () => dialogRef.current?.close();
}, []);
async function subscribe() {
if (!email || status !== 'idle') return;
setStatus('loading');
await fetch('/api/subscribe', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ email, source: 'scroll_popup' }),
});
setStatus('done');
setTimeout(onClose, 2000);
}
return (
<dialog
ref={dialogRef}
className="rounded-2xl p-8 max-w-sm w-full shadow-xl backdrop:bg-black/40"
>
{status === 'done' ? (
<p className="text-center text-green-700 font-medium">Ви підписалися!</p>
) : (
<>
<h2 className="text-lg font-bold mb-2">Сподобалася стаття?</h2>
<p className="text-sm text-gray-600 mb-4">
Отримуйте найкращі матеріали раз на тиждень. Без спаму.
</p>
<input
type="email"
value={email}
onChange={e => setEmail(e.target.value)}
onKeyDown={e => e.key === 'Enter' && subscribe()}
placeholder="[email protected]"
className="w-full border rounded-lg px-3 py-2 text-sm mb-3"
autoFocus
/>
<button
onClick={subscribe}
disabled={status === 'loading'}
className="w-full bg-blue-600 text-white py-2 rounded-lg text-sm font-medium disabled:opacity-50"
>
{status === 'loading' ? 'Підписуюся...' : 'Підписатися'}
</button>
<button
onClick={onClose}
className="mt-2 w-full text-xs text-gray-400 hover:text-gray-600"
>
Ні, дякую
</button>
</>
)}
</dialog>
);
}
Терміни
День-два з урахуванням адаптивного дизайну, тестування на реальних пристроях та налаштування cooldown-логіки. Якщо потрібна аналітика конверсій (GA4 + власне відстеження) — додайте півдня.







