/* Performance Optimization & Lazy Loading */ class PerformanceOptimizer { constructor() { this.init(); } init() { this.setupLazyLoading(); this.optimizeAnimations(); this.setupImageLoadingStates(); this.optimizeScrollEvents(); this.preloadCriticalAssets(); } // Lazy Loading with Intersection Observer setupLazyLoading() { const imageObserver = new IntersectionObserver((entries, observer) => { entries.forEach(entry => { if (entry.isIntersecting) { const img = entry.target; // Add loading state img.classList.add('loading'); // Load the image if (img.dataset.src) { img.src = img.dataset.src; img.removeAttribute('data-src'); } // Handle load completion img.onload = () => { img.classList.remove('loading'); img.classList.add('loaded'); }; // Handle errors img.onerror = () => { img.classList.remove('loading'); img.classList.add('error'); // Set fallback image img.src = ''; }; observer.unobserve(img); } }); }, { rootMargin: '50px 0px', threshold: 0.01 }); // Observe all images with data-src attribute document.querySelectorAll('img[data-src]').forEach(img => { imageObserver.observe(img); }); // Also setup for dynamically loaded images this.setupDynamicImageObserver(imageObserver); } setupDynamicImageObserver(observer) { // Watch for new images added to the DOM const mutationObserver = new MutationObserver(mutations => { mutations.forEach(mutation => { mutation.addedNodes.forEach(node => { if (node.nodeType === 1) { // Element node const images = node.querySelectorAll ? node.querySelectorAll('img[data-src]') : []; images.forEach(img => observer.observe(img)); // If the node itself is an image if (node.tagName === 'IMG' && node.hasAttribute('data-src')) { observer.observe(node); } } }); }); }); mutationObserver.observe(document.body, { childList: true, subtree: true }); } // Optimize animations using requestAnimationFrame optimizeAnimations() { let ticking = false; function updateAnimations() { // Batch DOM updates here ticking = false; } function requestTick() { if (!ticking) { requestAnimationFrame(updateAnimations); ticking = true; } } // Throttle scroll events window.addEventListener('scroll', requestTick, { passive: true }); } // Add loading states for images setupImageLoadingStates() { // Add CSS for loading states if not already present if (!document.getElementById('lazy-loading-styles')) { const style = document.createElement('style'); style.id = 'lazy-loading-styles'; style.textContent = ` img.loading { background: linear-gradient(90deg, #1a1a2e 25%, #2d2d4a 50%, #1a1a2e 75%); background-size: 200% 100%; animation: loading-shimmer 1.5s infinite; min-height: 200px; } img.loaded { animation: fadeIn 0.3s ease-in-out; } img.error { background: #333; color: #999; display: flex; align-items: center; justify-content: center; min-height: 200px; } @keyframes loading-shimmer { 0% { background-position: -200% 0; } 100% { background-position: 200% 0; } } @keyframes fadeIn { from { opacity: 0; } to { opacity: 1; } } .card-img-top { transition: transform 0.3s ease; } .card:hover .card-img-top { transform: scale(1.05); } `; document.head.appendChild(style); } } // Optimize scroll events optimizeScrollEvents() { let scrollTimeout; window.addEventListener('scroll', () => { // Clear existing timeout clearTimeout(scrollTimeout); // Set a new timeout scrollTimeout = setTimeout(() => { // Update scroll progress this.updateScrollProgress(); }, 10); }, { passive: true }); } updateScrollProgress() { const scrollProgress = document.getElementById('scrollProgress'); if (scrollProgress) { const scrollTop = window.pageYOffset || document.documentElement.scrollTop; const docHeight = document.body.scrollHeight - window.innerHeight; const scrollPercent = (scrollTop / docHeight) * 100; scrollProgress.style.width = Math.min(scrollPercent, 100) + '%'; } } // Preload critical assets preloadCriticalAssets() { const criticalAssets = [ 'static/css/space-theme.css', 'static/js/space-theme.js' ]; criticalAssets.forEach(asset => { const link = document.createElement('link'); link.rel = 'preload'; link.href = asset; link.as = asset.endsWith('.css') ? 'style' : 'script'; document.head.appendChild(link); }); } // Convert existing images to lazy loading static convertToLazyLoading() { document.querySelectorAll('img:not([data-src])').forEach(img => { if (img.src && !img.src.startsWith('data:')) { img.dataset.src = img.src; img.src = ''; img.classList.add('lazy-load'); } }); } // Debounce function for performance static debounce(func, wait) { let timeout; return function executedFunction(...args) { const later = () => { clearTimeout(timeout); func(...args); }; clearTimeout(timeout); timeout = setTimeout(later, wait); }; } // Throttle function for performance static throttle(func, limit) { let inThrottle; return function() { const args = arguments; const context = this; if (!inThrottle) { func.apply(context, args); inThrottle = true; setTimeout(() => inThrottle = false, limit); } }; } } // Initialize performance optimizations when DOM is ready if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', () => { new PerformanceOptimizer(); }); } else { new PerformanceOptimizer(); } // Export for use in other scripts window.PerformanceOptimizer = PerformanceOptimizer;