2025-01-29 19:07:31 +01:00
|
|
|
import { Controller } from "@hotwired/stimulus";
|
2025-01-30 19:20:29 +01:00
|
|
|
import { loadStripe } from "@stripe/stripe-js";
|
2025-01-29 16:44:20 +01:00
|
|
|
|
|
|
|
export default class extends Controller {
|
2025-01-30 20:51:13 +01:00
|
|
|
static targets = ["key", "submit", "firstname", "lastname", "email", "phone", "ticketType", "foodType", "note"];
|
2025-01-29 20:38:10 +01:00
|
|
|
|
|
|
|
stripe;
|
|
|
|
|
|
|
|
async connect() {
|
|
|
|
this.stripe = await loadStripe(this.keyTarget.textContent);
|
|
|
|
}
|
|
|
|
|
2025-01-29 19:07:31 +01:00
|
|
|
addEntry(event) {
|
|
|
|
event.preventDefault();
|
2025-01-29 19:06:48 +01:00
|
|
|
|
2025-01-30 20:51:13 +01:00
|
|
|
const template = this.element.querySelector(".forms").firstElementChild.cloneNode(true);
|
|
|
|
|
|
|
|
template.querySelectorAll('select, input').forEach(element => {
|
|
|
|
element.value = '';
|
|
|
|
if (element.id) {
|
|
|
|
element.id = `${element.id}_${Date.now()}`;
|
|
|
|
}
|
|
|
|
});
|
2025-01-29 19:06:48 +01:00
|
|
|
|
2025-01-30 20:51:13 +01:00
|
|
|
template.querySelectorAll('label').forEach(label => {
|
|
|
|
const forAttribute = label.getAttribute('for');
|
|
|
|
if (forAttribute) {
|
|
|
|
label.setAttribute('for', `${forAttribute}_${Date.now()}`);
|
|
|
|
}
|
2025-01-29 19:07:31 +01:00
|
|
|
});
|
2025-01-29 16:44:20 +01:00
|
|
|
|
2025-01-30 20:51:13 +01:00
|
|
|
template.classList.add('opacity-0');
|
|
|
|
this.element.querySelector(".forms").appendChild(template);
|
|
|
|
|
|
|
|
requestAnimationFrame(() => {
|
|
|
|
template.classList.remove('opacity-0');
|
|
|
|
template.classList.add('transition-all', 'duration-300');
|
|
|
|
});
|
2025-01-29 19:07:31 +01:00
|
|
|
}
|
2025-01-29 16:44:20 +01:00
|
|
|
|
2025-01-29 19:07:31 +01:00
|
|
|
removeEntry(event) {
|
|
|
|
event.preventDefault();
|
2025-01-29 19:06:48 +01:00
|
|
|
|
2025-01-30 20:51:13 +01:00
|
|
|
const forms = this.element.querySelector(".forms");
|
|
|
|
if (forms.childElementCount <= 1) {
|
2025-01-29 19:07:31 +01:00
|
|
|
return;
|
2025-01-29 16:44:20 +01:00
|
|
|
}
|
|
|
|
|
2025-01-30 20:51:13 +01:00
|
|
|
const ticketElement = event.currentTarget.closest(".bg-white\\/80");
|
|
|
|
if (!ticketElement) return;
|
|
|
|
|
|
|
|
ticketElement.classList.add('opacity-0', 'transition-all', 'duration-300');
|
|
|
|
setTimeout(() => {
|
|
|
|
ticketElement.remove();
|
|
|
|
}, 300);
|
2025-01-29 19:07:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
submit(event) {
|
|
|
|
event.preventDefault();
|
2025-01-30 20:51:13 +01:00
|
|
|
if (!this.validateForm()) {
|
|
|
|
return;
|
|
|
|
}
|
2025-01-29 19:07:31 +01:00
|
|
|
|
2025-01-30 19:20:29 +01:00
|
|
|
this.submitTarget.querySelector("span").classList.add("hidden");
|
2025-01-31 09:10:35 +01:00
|
|
|
this.submitTarget.querySelector("svg.arrow").classList.add("hidden");
|
|
|
|
this.submitTarget.querySelector("svg.loader").classList.remove("hidden");
|
2025-01-30 12:26:26 +01:00
|
|
|
|
2025-01-30 20:51:13 +01:00
|
|
|
const tickets = this.element.querySelectorAll(".forms > div");
|
2025-01-30 19:12:23 +01:00
|
|
|
const personalData = this.getPersonalData();
|
2025-01-30 20:51:13 +01:00
|
|
|
const ticketData = this.getTicketData(tickets);
|
2025-01-30 19:12:23 +01:00
|
|
|
|
|
|
|
const formData = {
|
2025-01-30 19:20:29 +01:00
|
|
|
personal: personalData,
|
2025-01-30 20:51:13 +01:00
|
|
|
tickets: ticketData
|
2025-01-30 19:20:29 +01:00
|
|
|
};
|
2025-01-29 19:07:31 +01:00
|
|
|
|
|
|
|
fetch("/ticket/submit", {
|
|
|
|
method: "POST",
|
|
|
|
body: JSON.stringify(formData),
|
|
|
|
headers: {
|
|
|
|
"Content-Type": "application/json",
|
|
|
|
},
|
|
|
|
}).then((response) => {
|
|
|
|
if (!response.ok) {
|
2025-01-31 09:10:35 +01:00
|
|
|
this.submitTarget.querySelector("svg.error").classList.add("hidden");
|
|
|
|
this.submitTarget.querySelector("svg.loader").classList.remove("hidden");
|
2025-01-30 19:20:29 +01:00
|
|
|
this.submitTarget.querySelector("span").classList.remove("hidden");
|
2025-01-30 20:51:13 +01:00
|
|
|
this.showError("Ein Fehler ist aufgetreten. Bitte versuchen Sie es erneut.");
|
2025-01-29 20:38:10 +01:00
|
|
|
} else {
|
|
|
|
response.json().then((data) => {
|
|
|
|
this.stripe.redirectToCheckout({
|
|
|
|
sessionId: data.id,
|
|
|
|
});
|
|
|
|
});
|
2025-01-29 19:07:31 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2025-01-30 20:51:13 +01:00
|
|
|
validateForm() {
|
|
|
|
let isValid = true;
|
|
|
|
|
|
|
|
['firstname', 'lastname', 'email'].forEach(field => {
|
|
|
|
const target = this[`${field}Target`];
|
|
|
|
if (!target.value.trim()) {
|
|
|
|
this.showFieldError(target);
|
|
|
|
isValid = false;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
this.element.querySelectorAll(".forms > div").forEach(ticket => {
|
|
|
|
const selects = ticket.querySelectorAll('select[required]');
|
|
|
|
selects.forEach(select => {
|
|
|
|
if (!select.value) {
|
|
|
|
this.showFieldError(select);
|
|
|
|
isValid = false;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
return isValid;
|
|
|
|
}
|
|
|
|
|
|
|
|
showFieldError(element) {
|
|
|
|
element.classList.add('border-red-500', 'focus:border-red-500', 'focus:ring-red-200');
|
|
|
|
element.addEventListener('input', () => {
|
|
|
|
element.classList.remove('border-red-500', 'focus:border-red-500', 'focus:ring-red-200');
|
|
|
|
}, { once: true });
|
|
|
|
}
|
|
|
|
|
|
|
|
showError(message) {
|
|
|
|
const errorDiv = document.createElement('div');
|
|
|
|
errorDiv.className = 'fixed top-4 right-4 bg-red-50 border-l-4 border-red-500 p-4 rounded shadow-lg transform transition-all duration-500 opacity-0 translate-x-4';
|
|
|
|
errorDiv.innerHTML = `
|
|
|
|
<div class="flex items-center">
|
|
|
|
<div class="flex-shrink-0">
|
|
|
|
<twig:ux:icon name="mdi:alert-circle" class="w-5 h-5 text-red-400" />
|
|
|
|
</div>
|
|
|
|
<div class="ml-3">
|
|
|
|
<p class="text-sm text-red-700">${message}</p>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
`;
|
|
|
|
|
|
|
|
document.body.appendChild(errorDiv);
|
|
|
|
requestAnimationFrame(() => {
|
|
|
|
errorDiv.classList.remove('opacity-0', 'translate-x-4');
|
2025-01-29 19:07:31 +01:00
|
|
|
});
|
|
|
|
|
2025-01-30 20:51:13 +01:00
|
|
|
setTimeout(() => {
|
|
|
|
errorDiv.classList.add('opacity-0', 'translate-x-4');
|
|
|
|
setTimeout(() => errorDiv.remove(), 500);
|
|
|
|
}, 5000);
|
|
|
|
}
|
|
|
|
|
|
|
|
getTicketData(tickets) {
|
|
|
|
return Array.from(tickets).map(ticket => this.extractTicketData(ticket));
|
2025-01-29 19:07:31 +01:00
|
|
|
}
|
2025-01-30 19:12:23 +01:00
|
|
|
|
|
|
|
getPersonalData() {
|
|
|
|
return {
|
2025-01-30 20:51:13 +01:00
|
|
|
firstname: this.firstnameTarget.value.trim(),
|
|
|
|
lastname: this.lastnameTarget.value.trim(),
|
|
|
|
email: this.emailTarget.value.trim(),
|
|
|
|
phone: this.phoneTarget.value.trim(),
|
2025-01-30 19:12:23 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2025-01-30 20:51:13 +01:00
|
|
|
extractTicketData(ticket) {
|
2025-01-30 19:12:23 +01:00
|
|
|
return {
|
2025-01-30 20:51:13 +01:00
|
|
|
ticket: parseInt(ticket.querySelector('select[name="ticket"]').value),
|
|
|
|
food: parseInt(ticket.querySelector('select[name="food"]').value),
|
|
|
|
note: ticket.querySelector('input[name="note"]').value.trim(),
|
2025-01-30 19:20:29 +01:00
|
|
|
};
|
2025-01-30 19:12:23 +01:00
|
|
|
}
|
2025-01-29 19:07:31 +01:00
|
|
|
}
|