Spaces:
Sleeping
Sleeping
| /* 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; | |