Nyavokevin 1504c841e0
Some checks failed
linter / quality (push) Has been cancelled
tests / ci (push) Has been cancelled
card gasp
2025-09-19 15:32:25 +03:00

238 lines
8.6 KiB
Vue

<template>
<section
class="relative flex min-h-screen w-full items-center justify-center overflow-hidden bg-cover bg-center bg-no-repeat px-4 py-24 text-center"
>
<div class="pointer-events-none absolute inset-0 bg-gradient-to-b from-[var(--c-purple)]/60 via-[var(--c-deep-navy)]/60 to-black/80"></div>
<div class="relative z-10 mx-auto flex max-w-5xl flex-col items-center justify-between lg:flex-row">
<!-- Text Content -->
<div class="mb-10 lg:mb-0 lg:w-1/2">
<h1
class="mb-4 transform text-4xl font-black text-white transition-all duration-700 md:text-6xl"
:class="isMounted ? 'translate-y-0 opacity-100' : 'translate-y-10 opacity-0'"
>
Révélez Votre Voyage Intérieur
</h1>
<p
class="mb-8 transform text-lg text-white/80 transition-all delay-200 duration-1000 md:text-xl"
:class="isMounted ? 'translate-y-0 opacity-100' : 'translate-y-10 opacity-0'"
>
Embrassez la sagesse intemporelle de l'Oracle de Kris Saint Ange, un guide stratégique pour naviguer votre destin avec clarté et
confiance.
</p>
<!-- Buttons -->
<div
class="flex transform flex-wrap justify-center gap-6 transition-all delay-500 duration-1000"
:class="isMounted ? 'translate-y-0 opacity-100' : 'translate-y-10 opacity-0'"
>
<button
@click="goToShuffle"
class="group relative inline-flex h-14 min-w-[180px] items-center justify-center overflow-hidden rounded-full bg-gradient-to-r from-[var(--c-gold)] to-yellow-400 px-8 font-bold tracking-wide text-[var(--c-purple)] shadow-lg transition-all duration-300 hover:-translate-y-0.5 hover:shadow-[var(--c-gold)]/40"
>
<span class="relative z-10 truncate">Essayez 1 carte gratuite</span>
<span
class="pointer-events-none absolute inset-0 translate-x-[-120%] -skew-x-12 bg-gradient-to-r from-transparent via-white/30 to-transparent transition-transform duration-700 group-hover:translate-x-[120%]"
></span>
</button>
<button
@click="goToShuffle"
class="inline-flex h-14 min-w-[180px] items-center justify-center rounded-full border-2 border-[var(--c-purple)] bg-transparent px-8 font-bold text-white transition-all duration-300 hover:-translate-y-0.5 hover:bg-[var(--c-purple)]"
>
<span class="truncate">Découvrir les tirages</span>
</button>
</div>
</div>
<!-- Card Stack -->
<div class="relative flex h-96 items-center justify-center lg:w-1/2">
<div class="perspective-1000 relative h-96 w-64" @click.self="randomizeStack" title="Cliquez pour mélanger">
<!-- Card Stack -->
<div
v-for="(card, index) in cardStack"
:key="card.id"
class="absolute h-full w-full cursor-pointer transition-all duration-500"
:style="{
transform: `translateY(${index * -4}px) rotate(${index * -2}deg)`,
zIndex: cardStack.length - index,
}"
@click="flipCard(card)"
>
<div
class="preserve-3d h-full w-full rounded-xl shadow-xl transition-transform duration-700"
:class="{ 'rotate-y-180': card.isFlipped }"
>
<!-- Card Back -->
<div
class="border-gold/50 absolute h-full w-full overflow-hidden rounded-xl border-2 bg-gradient-to-br from-purple-900 via-blue-900 to-indigo-900 backface-hidden"
>
<img src="/cards/1.png" alt="Card Back" class="card-fill" />
</div>
<!-- Card Front -->
<div class="absolute h-full w-full rotate-y-180 overflow-hidden rounded-xl backface-hidden">
<img :src="card.image" class="card-fill" :alt="'Card ' + card.id" />
</div>
</div>
</div>
</div>
</div>
</div>
</section>
</template>
<script setup lang="ts">
import { router } from '@inertiajs/vue3';
import { onMounted, ref } from 'vue';
const isMounted = ref(false);
const container = ref<HTMLDivElement | null>(null);
// Sample card data
const cardStack = ref([
{ id: 2, title: "L'Intuition", image: '/cards/2.png', isFlipped: false },
{ id: 3, title: 'La Sagesse', image: '/cards/3.png', isFlipped: false },
{ id: 4, title: 'Le Destin', image: '/cards/4.png', isFlipped: false },
]);
// Background images
const backgroundImages = Array.from({ length: 12 }, (_, i) => `/cards/${(i % 4) + 1}.png`);
onMounted(() => {
setTimeout(() => {
isMounted.value = true;
}, 100);
// Initialize GSAP animations for background
if (container.value) {
const cards = container.value.querySelectorAll('.bg-card');
cards.forEach((el: Element, idx: number) => {
const startY = Math.random() * 300 - 150;
const rot = Math.random() * 30 - 15;
const drift = Math.random() * 160 + 120;
// Set initial state
(el as HTMLElement).style.transform = `translate(-50%, -50%) rotate(${rot}deg) translateY(${startY}px)`;
(el as HTMLElement).style.opacity = '0.5';
// Animate with requestAnimationFrame for simplicity
// In a real implementation, you would use GSAP here
let startTime: number | null = null;
const duration = 20000 + Math.random() * 10000;
function animate(timestamp: number) {
if (!startTime) startTime = timestamp;
const elapsed = timestamp - startTime;
const progress = (elapsed % duration) / duration;
const yPos = startY + drift * progress;
const rotation = rot + 10 * Math.sin(progress * Math.PI * 2);
(el as HTMLElement).style.transform = `translate(-50%, -50%) rotate(${rotation}deg) translateY(${yPos}px)`;
(el as HTMLElement).style.opacity = `${0.5 + 0.35 * Math.sin(progress * Math.PI)}`;
requestAnimationFrame(animate);
}
requestAnimationFrame(animate);
});
}
});
const goToShuffle = () => {
router.visit('/tirage');
};
function randomInt(min: number, max: number) {
return Math.floor(Math.random() * (max - min + 1)) + min;
}
function randomizeStack() {
const picked = new Set<number>();
while (picked.size < 3) picked.add(randomInt(1, 22));
const ids = Array.from(picked);
cardStack.value = ids.map((id, idx) => ({ id, title: `Carte ${id}`, image: `/cards/${id}.png`, isFlipped: false }));
}
const flipCard = (card: any) => {
card.isFlipped = !card.isFlipped;
};
</script>
<style scoped>
@keyframes float {
0%,
100% {
transform: translateY(0) rotate(0deg);
}
33% {
transform: translateY(-20px) rotate(5deg);
}
66% {
transform: translateY(10px) rotate(-5deg);
}
}
@keyframes textShine {
0% {
background-position: 0% 50%;
}
100% {
background-position: 100% 50%;
}
}
.animate-float {
animation: float 10s ease-in-out infinite;
}
.animate-text-shine {
animation: textShine 2s ease-in-out infinite alternate;
}
.card-fill {
width: 100%;
height: 100%;
object-fit: cover;
}
.perspective-1000 {
perspective: 1000px;
}
.preserve-3d {
transform-style: preserve-3d;
}
.rotate-y-180 {
transform: rotateY(180deg);
}
.backface-hidden {
backface-visibility: hidden;
}
.bg-card {
background-size: cover;
background-position: center;
filter: drop-shadow(0 12px 30px rgba(0, 0, 0, 0.35));
mix-blend-mode: screen;
}
/* Custom color variables */
:root {
--c-purple: #4c1d95;
--c-deep-navy: #1e1b4b;
--c-gold: rgba(245, 158, 11, 0.7);
}
@media (min-width: 1024px) {
.bg-card {
width: 10rem;
height: 15rem;
}
}
</style>