abiball/assets/controllers/form_controller.js

205 lines
5.3 KiB
JavaScript
Raw Normal View History

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-31 11:01:45 +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-31 11:01:45 +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-31 11:01:45 +01:00
template.querySelectorAll("label").forEach((label) => {
const forAttribute = label.getAttribute("for");
if (forAttribute) {
2025-01-31 11:01:45 +01:00
label.setAttribute("for", `${forAttribute}_${Date.now()}`);
}
2025-01-29 19:07:31 +01:00
});
2025-01-29 16:44:20 +01:00
2025-01-31 11:01:45 +01:00
template.classList.add("opacity-0");
this.element.querySelector(".forms").appendChild(template);
2025-01-31 11:01:45 +01:00
requestAnimationFrame(() => {
2025-01-31 11:01:45 +01:00
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
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
}
const ticketElement = event.currentTarget.closest(".bg-white\\/80");
if (!ticketElement) return;
2025-01-31 11:01:45 +01:00
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();
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
const tickets = this.element.querySelectorAll(".forms > div");
2025-01-30 19:12:23 +01:00
const personalData = this.getPersonalData();
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-31 11:01:45 +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");
2025-01-31 11:01:45 +01:00
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-31 11:01:45 +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
}
});
}
validateForm() {
let isValid = true;
2025-01-31 11:01:45 +01:00
["firstname", "lastname", "email"].forEach((field) => {
const target = this[`${field}Target`];
if (!target.value.trim()) {
this.showFieldError(target);
isValid = false;
}
});
2025-01-31 11:01:45 +01:00
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) {
2025-01-31 11:01:45 +01:00
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) {
2025-01-31 11:01:45 +01:00
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>
`;
2025-01-31 11:01:45 +01:00
document.body.appendChild(errorDiv);
requestAnimationFrame(() => {
2025-01-31 11:01:45 +01:00
errorDiv.classList.remove("opacity-0", "translate-x-4");
2025-01-29 19:07:31 +01:00
});
setTimeout(() => {
2025-01-31 11:01:45 +01:00
errorDiv.classList.add("opacity-0", "translate-x-4");
setTimeout(() => errorDiv.remove(), 500);
}, 5000);
}
getTicketData(tickets) {
2025-01-31 11:01:45 +01:00
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 {
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
};
}
extractTicketData(ticket) {
2025-01-30 19:12:23 +01:00
return {
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
}