452 lines
16 KiB
JavaScript
452 lines
16 KiB
JavaScript
// Mobile Navigation Toggle
|
|
document.addEventListener('DOMContentLoaded', function() {
|
|
const hamburger = document.querySelector('.hamburger');
|
|
const navMenu = document.querySelector('.nav-menu');
|
|
|
|
hamburger.addEventListener('click', function() {
|
|
hamburger.classList.toggle('active');
|
|
navMenu.classList.toggle('active');
|
|
});
|
|
|
|
// Close mobile menu when clicking on a link
|
|
document.querySelectorAll('.nav-link').forEach(n => n.addEventListener('click', () => {
|
|
hamburger.classList.remove('active');
|
|
navMenu.classList.remove('active');
|
|
}));
|
|
});
|
|
|
|
// Smooth Scrolling for Navigation Links
|
|
document.querySelectorAll('a[href^="#"]').forEach(anchor => {
|
|
anchor.addEventListener('click', function (e) {
|
|
e.preventDefault();
|
|
const target = document.querySelector(this.getAttribute('href'));
|
|
if (target) {
|
|
target.scrollIntoView({
|
|
behavior: 'smooth',
|
|
block: 'start'
|
|
});
|
|
}
|
|
});
|
|
});
|
|
|
|
// Portfolio Filter Functionality
|
|
document.addEventListener('DOMContentLoaded', function() {
|
|
const filterButtons = document.querySelectorAll('.filter-btn');
|
|
const portfolioItems = document.querySelectorAll('.portfolio-item');
|
|
|
|
filterButtons.forEach(button => {
|
|
button.addEventListener('click', function() {
|
|
// Remove active class from all buttons
|
|
filterButtons.forEach(btn => btn.classList.remove('active'));
|
|
// Add active class to clicked button
|
|
this.classList.add('active');
|
|
|
|
const filterValue = this.getAttribute('data-filter');
|
|
|
|
portfolioItems.forEach(item => {
|
|
if (filterValue === 'all' || item.getAttribute('data-category') === filterValue) {
|
|
item.style.display = 'block';
|
|
setTimeout(() => {
|
|
item.style.opacity = '1';
|
|
item.style.transform = 'scale(1)';
|
|
}, 100);
|
|
} else {
|
|
item.style.opacity = '0';
|
|
item.style.transform = 'scale(0.8)';
|
|
setTimeout(() => {
|
|
item.style.display = 'none';
|
|
}, 300);
|
|
}
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
// Form Handling with EmailJS
|
|
document.addEventListener('DOMContentLoaded', function() {
|
|
// Check if EmailJS is loaded
|
|
if (typeof emailjs === 'undefined') {
|
|
console.error('EmailJS not loaded. Please check the script tag.');
|
|
return;
|
|
}
|
|
|
|
// Initialize EmailJS
|
|
emailjs.init("QznwPi5vZti7xnjRj"); // Replace with your EmailJS public key
|
|
|
|
console.log('EmailJS initialized successfully');
|
|
|
|
const quoteForm = document.getElementById('quoteForm');
|
|
const contactForm = document.getElementById('contactForm');
|
|
|
|
// Function to send email via EmailJS
|
|
function sendEmail(templateId, templateParams) {
|
|
console.log('Attempting to send email with EmailJS:', templateId);
|
|
console.log('Template parameters:', templateParams);
|
|
|
|
return emailjs.send('service_c51669d', templateId, templateParams)
|
|
.then(function(response) {
|
|
console.log('EmailJS Success:', response.status, response.text);
|
|
return response;
|
|
})
|
|
.catch(function(error) {
|
|
console.error('EmailJS Error:', error);
|
|
throw error;
|
|
});
|
|
}
|
|
|
|
// Quote Form Submission
|
|
if (quoteForm) {
|
|
quoteForm.addEventListener('submit', function(e) {
|
|
e.preventDefault();
|
|
|
|
// Get form data
|
|
const formData = new FormData(this);
|
|
const data = {};
|
|
for (let [key, value] of formData.entries()) {
|
|
data[key] = value;
|
|
}
|
|
|
|
// Show loading state
|
|
const submitBtn = this.querySelector('button[type="submit"]');
|
|
const originalText = submitBtn.textContent;
|
|
submitBtn.textContent = 'Submitting...';
|
|
submitBtn.disabled = true;
|
|
|
|
// Prepare EmailJS template parameters
|
|
const templateParams = {
|
|
to_email: 'oli@ptslondon.co.uk',
|
|
from_name: data.name,
|
|
reply_to: data.email,
|
|
customer_name: data.name,
|
|
customer_email: data.email,
|
|
customer_phone: data.phone || 'Not provided',
|
|
customer_company: data.company || 'Not provided',
|
|
service_type: data.service,
|
|
project_location: data.location,
|
|
preferred_date: data.date || 'Not specified',
|
|
budget_range: data.budget || 'Not specified',
|
|
project_description: data.description,
|
|
form_type: 'Quote Request'
|
|
};
|
|
|
|
// Send email using EmailJS
|
|
sendEmail('quote_template', templateParams)
|
|
.then((result) => {
|
|
console.log('Quote email sent successfully:', result);
|
|
showNotification('Thank you for your quote request! We will contact you within 24 hours.', 'success');
|
|
this.reset();
|
|
})
|
|
.catch((error) => {
|
|
console.error('EmailJS error:', error);
|
|
let errorMessage = 'There was an error sending your quote request. ';
|
|
|
|
if (error.text && error.text.includes('template')) {
|
|
errorMessage += 'Email template not found. ';
|
|
} else if (error.text && error.text.includes('service')) {
|
|
errorMessage += 'Email service configuration error. ';
|
|
}
|
|
|
|
errorMessage += 'Please contact us directly at oli@ptslondon.co.uk.';
|
|
showNotification(errorMessage, 'error');
|
|
})
|
|
.finally(() => {
|
|
submitBtn.textContent = originalText;
|
|
submitBtn.disabled = false;
|
|
});
|
|
});
|
|
}
|
|
|
|
// Contact Form Submission
|
|
if (contactForm) {
|
|
contactForm.addEventListener('submit', function(e) {
|
|
e.preventDefault();
|
|
|
|
// Get form data
|
|
const formData = new FormData(this);
|
|
const data = {};
|
|
for (let [key, value] of formData.entries()) {
|
|
data[key] = value;
|
|
}
|
|
|
|
// Show loading state
|
|
const submitBtn = this.querySelector('button[type="submit"]');
|
|
const originalText = submitBtn.textContent;
|
|
submitBtn.textContent = 'Sending...';
|
|
submitBtn.disabled = true;
|
|
|
|
// Prepare EmailJS template parameters
|
|
const templateParams = {
|
|
to_email: 'oli@ptslondon.co.uk',
|
|
from_name: data.contact_name,
|
|
reply_to: data.contact_email,
|
|
customer_name: data.contact_name,
|
|
customer_email: data.contact_email,
|
|
message_subject: data.subject,
|
|
message_content: data.message,
|
|
form_type: 'Contact Message'
|
|
};
|
|
|
|
// Send email using EmailJS
|
|
sendEmail('contact_template', templateParams)
|
|
.then((result) => {
|
|
console.log('Contact email sent successfully:', result);
|
|
showNotification('Thank you for your message! We will get back to you soon.', 'success');
|
|
this.reset();
|
|
})
|
|
.catch((error) => {
|
|
console.error('Contact EmailJS error:', error);
|
|
let errorMessage = 'There was an error sending your message. ';
|
|
|
|
if (error.text && error.text.includes('template')) {
|
|
errorMessage += 'Email template not found. ';
|
|
} else if (error.text && error.text.includes('service')) {
|
|
errorMessage += 'Email service configuration error. ';
|
|
}
|
|
|
|
errorMessage += 'Please contact us directly at oli@ptslondon.co.uk.';
|
|
showNotification(errorMessage, 'error');
|
|
})
|
|
.finally(() => {
|
|
submitBtn.textContent = originalText;
|
|
submitBtn.disabled = false;
|
|
});
|
|
});
|
|
}
|
|
});
|
|
|
|
// Navbar Background on Scroll
|
|
window.addEventListener('scroll', function() {
|
|
const header = document.querySelector('.header');
|
|
if (window.scrollY > 100) {
|
|
header.style.background = 'rgba(255, 255, 255, 0.98)';
|
|
header.style.boxShadow = '0 2px 20px rgba(0, 0, 0, 0.1)';
|
|
} else {
|
|
header.style.background = 'rgba(255, 255, 255, 0.95)';
|
|
header.style.boxShadow = 'none';
|
|
}
|
|
});
|
|
|
|
// Intersection Observer for Animations
|
|
const observerOptions = {
|
|
threshold: 0.1,
|
|
rootMargin: '0px 0px -50px 0px'
|
|
};
|
|
|
|
const observer = new IntersectionObserver(function(entries) {
|
|
entries.forEach(entry => {
|
|
if (entry.isIntersecting) {
|
|
entry.target.classList.add('visible');
|
|
}
|
|
});
|
|
}, observerOptions);
|
|
|
|
// Observe elements for animation
|
|
document.addEventListener('DOMContentLoaded', function() {
|
|
const animateElements = document.querySelectorAll('.service-card, .portfolio-item, .about-content, .contact-item');
|
|
animateElements.forEach(el => {
|
|
el.classList.add('fade-in');
|
|
observer.observe(el);
|
|
});
|
|
});
|
|
|
|
// Form Validation
|
|
function validateEmail(email) {
|
|
const re = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
|
|
return re.test(email);
|
|
}
|
|
|
|
function validatePhone(phone) {
|
|
const re = /^[\+]?[1-9][\d]{0,15}$/;
|
|
return re.test(phone.replace(/\s/g, ''));
|
|
}
|
|
|
|
// Real-time form validation
|
|
document.addEventListener('DOMContentLoaded', function() {
|
|
const emailInputs = document.querySelectorAll('input[type="email"]');
|
|
const phoneInputs = document.querySelectorAll('input[type="tel"]');
|
|
|
|
emailInputs.forEach(input => {
|
|
input.addEventListener('blur', function() {
|
|
if (this.value && !validateEmail(this.value)) {
|
|
this.style.borderColor = '#ff6b6b';
|
|
this.style.boxShadow = '0 0 0 3px rgba(255, 107, 107, 0.1)';
|
|
} else {
|
|
this.style.borderColor = '#ddd';
|
|
this.style.boxShadow = 'none';
|
|
}
|
|
});
|
|
});
|
|
|
|
phoneInputs.forEach(input => {
|
|
input.addEventListener('blur', function() {
|
|
if (this.value && !validatePhone(this.value)) {
|
|
this.style.borderColor = '#ff6b6b';
|
|
this.style.boxShadow = '0 0 0 3px rgba(255, 107, 107, 0.1)';
|
|
} else {
|
|
this.style.borderColor = '#ddd';
|
|
this.style.boxShadow = 'none';
|
|
}
|
|
});
|
|
});
|
|
});
|
|
|
|
// Set minimum date for date inputs to today
|
|
document.addEventListener('DOMContentLoaded', function() {
|
|
const dateInput = document.getElementById('date');
|
|
if (dateInput) {
|
|
const today = new Date().toISOString().split('T')[0];
|
|
dateInput.setAttribute('min', today);
|
|
}
|
|
});
|
|
|
|
// Lazy loading for images
|
|
document.addEventListener('DOMContentLoaded', function() {
|
|
const images = document.querySelectorAll('img[data-src]');
|
|
|
|
const imageObserver = new IntersectionObserver((entries, observer) => {
|
|
entries.forEach(entry => {
|
|
if (entry.isIntersecting) {
|
|
const img = entry.target;
|
|
img.src = img.dataset.src;
|
|
img.removeAttribute('data-src');
|
|
imageObserver.unobserve(img);
|
|
}
|
|
});
|
|
});
|
|
|
|
images.forEach(img => imageObserver.observe(img));
|
|
});
|
|
|
|
// Add loading states for buttons
|
|
function addLoadingState(button, loadingText = 'Loading...') {
|
|
const originalText = button.textContent;
|
|
button.textContent = loadingText;
|
|
button.disabled = true;
|
|
button.classList.add('loading');
|
|
|
|
return function removeLoadingState() {
|
|
button.textContent = originalText;
|
|
button.disabled = false;
|
|
button.classList.remove('loading');
|
|
};
|
|
}
|
|
|
|
// Enhanced form submission with better error handling
|
|
function submitForm(form, endpoint) {
|
|
const submitButton = form.querySelector('button[type="submit"]');
|
|
const removeLoading = addLoadingState(submitButton);
|
|
|
|
const formData = new FormData(form);
|
|
|
|
// Convert FormData to regular object
|
|
const data = {};
|
|
for (let [key, value] of formData.entries()) {
|
|
data[key] = value;
|
|
}
|
|
|
|
// Simulate API call (replace with actual endpoint)
|
|
return new Promise((resolve, reject) => {
|
|
setTimeout(() => {
|
|
// Simulate success/failure
|
|
if (Math.random() > 0.1) { // 90% success rate
|
|
resolve(data);
|
|
} else {
|
|
reject(new Error('Network error'));
|
|
}
|
|
}, 2000);
|
|
})
|
|
.then(result => {
|
|
removeLoading();
|
|
showNotification('Form submitted successfully!', 'success');
|
|
form.reset();
|
|
return result;
|
|
})
|
|
.catch(error => {
|
|
removeLoading();
|
|
showNotification('There was an error submitting the form. Please try again.', 'error');
|
|
throw error;
|
|
});
|
|
}
|
|
|
|
// Notification system
|
|
function showNotification(message, type = 'info') {
|
|
// Remove existing notifications
|
|
const existingNotifications = document.querySelectorAll('.notification');
|
|
existingNotifications.forEach(notification => notification.remove());
|
|
|
|
const notification = document.createElement('div');
|
|
notification.className = `notification notification-${type}`;
|
|
notification.innerHTML = `
|
|
<span>${message}</span>
|
|
<button class="notification-close">×</button>
|
|
`;
|
|
|
|
// Add styles
|
|
Object.assign(notification.style, {
|
|
position: 'fixed',
|
|
top: '20px',
|
|
right: '20px',
|
|
padding: '1rem 1.5rem',
|
|
borderRadius: '5px',
|
|
color: 'white',
|
|
fontWeight: '500',
|
|
zIndex: '10000',
|
|
animation: 'slideInRight 0.3s ease',
|
|
display: 'flex',
|
|
alignItems: 'center',
|
|
gap: '1rem',
|
|
maxWidth: '300px'
|
|
});
|
|
|
|
// Set background color based on type
|
|
const colors = {
|
|
success: '#28a745',
|
|
error: '#dc3545',
|
|
warning: '#ffc107',
|
|
info: '#17a2b8'
|
|
};
|
|
notification.style.backgroundColor = colors[type] || colors.info;
|
|
|
|
// Add close functionality
|
|
const closeBtn = notification.querySelector('.notification-close');
|
|
closeBtn.style.background = 'none';
|
|
closeBtn.style.border = 'none';
|
|
closeBtn.style.color = 'white';
|
|
closeBtn.style.fontSize = '1.5rem';
|
|
closeBtn.style.cursor = 'pointer';
|
|
closeBtn.style.padding = '0';
|
|
closeBtn.style.marginLeft = 'auto';
|
|
|
|
closeBtn.addEventListener('click', () => notification.remove());
|
|
|
|
document.body.appendChild(notification);
|
|
|
|
// Auto remove after 5 seconds
|
|
setTimeout(() => {
|
|
if (document.body.contains(notification)) {
|
|
notification.remove();
|
|
}
|
|
}, 5000);
|
|
}
|
|
|
|
// Add CSS for notification animation
|
|
const style = document.createElement('style');
|
|
style.textContent = `
|
|
@keyframes slideInRight {
|
|
from {
|
|
transform: translateX(100%);
|
|
opacity: 0;
|
|
}
|
|
to {
|
|
transform: translateX(0);
|
|
opacity: 1;
|
|
}
|
|
}
|
|
|
|
.loading {
|
|
opacity: 0.7;
|
|
cursor: not-allowed;
|
|
}
|
|
`;
|
|
document.head.appendChild(style);
|