casmopedia / static /js /performance.js
aradhyapavan's picture
CospmoPedia
7e3b585 verified
/* 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;