[Khitan] Starter
This commit is contained in:
		
							parent
							
								
									b343fd7b01
								
							
						
					
					
						commit
						bd176aad5c
					
				| @ -0,0 +1,29 @@ | ||||
| <template> | ||||
|   <section class="min-h-screen flex flex-col items-center justify-center bg-blue-50 text-center px-6"> | ||||
|     <div class="max-w-2xl bg-white rounded-2xl shadow-lg p-8 space-y-4"> | ||||
|       <h2 class="text-2xl font-bold text-blue-700 mb-4">📅 Acara Tasyakuran</h2> | ||||
| 
 | ||||
|       <p><strong>Hari & Tanggal:</strong> {{ hari_tanggal_acara }}</p> | ||||
|       <p><strong>Waktu:</strong> {{ waktu }}</p> | ||||
|       <p><strong>Alamat:</strong> {{ alamat }}</p> | ||||
| 
 | ||||
|       <a | ||||
|         v-if="link_gmaps" | ||||
|         :href="link_gmaps" | ||||
|         target="_blank" | ||||
|         class="inline-block mt-4 text-blue-600 font-semibold underline hover:text-blue-800" | ||||
|       > | ||||
|         Lihat di Google Maps | ||||
|       </a> | ||||
|     </div> | ||||
|   </section> | ||||
| </template> | ||||
| 
 | ||||
| <script setup> | ||||
| defineProps({ | ||||
|   hari_tanggal_acara: String, | ||||
|   waktu: String, | ||||
|   alamat: String, | ||||
|   link_gmaps: String | ||||
| }) | ||||
| </script> | ||||
| @ -0,0 +1,28 @@ | ||||
| <template> | ||||
|   <section class="min-h-screen bg-white py-12 px-6"> | ||||
|     <div class="text-center mb-10"> | ||||
|       <h2 class="text-3xl font-bold text-blue-700">Galeri</h2> | ||||
|       <p class="text-gray-600">Momen istimewa kami</p> | ||||
|     </div> | ||||
| 
 | ||||
|     <div class="grid grid-cols-1 sm:grid-cols-2 gap-6 max-w-3xl mx-auto"> | ||||
|       <div | ||||
|         v-for="(img, index) in images" | ||||
|         :key="index" | ||||
|         class="overflow-hidden rounded-2xl shadow-md" | ||||
|       > | ||||
|         <img | ||||
|           :src="img" | ||||
|           alt="Foto Galeri" | ||||
|           class="w-full h-64 object-cover hover:scale-105 transition-transform duration-500" | ||||
|         /> | ||||
|       </div> | ||||
|     </div> | ||||
|   </section> | ||||
| </template> | ||||
| 
 | ||||
| <script setup> | ||||
| defineProps({ | ||||
|   images: { type: Array, default: () => [] } | ||||
| }) | ||||
| </script> | ||||
| @ -0,0 +1,25 @@ | ||||
| <template> | ||||
|   <section class="min-h-screen flex flex-col justify-center items-center bg-white px-6 text-center"> | ||||
|     <div class="max-w-2xl space-y-6"> | ||||
|       <h2 class="text-3xl font-bold text-blue-700 mb-4">Assalamu’alaikum Warahmatullahi Wabarakatuh</h2> | ||||
|       <p class="text-gray-700 leading-relaxed"> | ||||
|         Dengan penuh rasa syukur ke hadirat Allah SWT, kami bermaksud mengadakan | ||||
|         acara tasyakuran khitan putra kami: | ||||
|       </p> | ||||
| 
 | ||||
|       <h3 class="text-4xl text-blue-800 font-bold mt-6">{{ nama_lengkap }}</h3> | ||||
| 
 | ||||
|       <p class="text-gray-700 mt-6"> | ||||
|         Putra dari pasangan <strong>{{ nama_bapak }}</strong> dan <strong>{{ nama_ibu }}</strong>. | ||||
|       </p> | ||||
|     </div> | ||||
|   </section> | ||||
| </template> | ||||
| 
 | ||||
| <script setup> | ||||
| defineProps({ | ||||
|   nama_lengkap: String, | ||||
|   nama_bapak: String, | ||||
|   nama_ibu: String | ||||
| }) | ||||
| </script> | ||||
| @ -0,0 +1,46 @@ | ||||
| <template> | ||||
|   <div class="min-h-screen flex flex-col items-center justify-center bg-gradient-to-b from-blue-50 to-blue-200 text-center relative overflow-hidden"> | ||||
|     <!-- Background Ornaments --> | ||||
|     <div class="absolute inset-0 opacity-20 bg-[radial-gradient(circle_at_top_left,_white,_transparent_70%)]"></div> | ||||
| 
 | ||||
|     <!-- Title --> | ||||
|     <div class="z-10 space-y-4 animate-fade-in-down"> | ||||
|       <h1 class="text-blue-700 text-3xl md:text-5xl font-bold">Undangan Tasyakuran Khitan</h1> | ||||
|       <h2 class="text-gray-700 text-2xl md:text-4xl font-semibold">{{ childName }}</h2> | ||||
|     </div> | ||||
| 
 | ||||
|     <!-- Guest --> | ||||
|     <div class="z-10 mt-10 bg-white/60 backdrop-blur-sm rounded-xl p-5 shadow-md animate-fade-in-up"> | ||||
|       <p class="text-gray-600 text-sm mb-1">Kepada Yth.</p> | ||||
|       <p class="text-blue-800 text-xl font-bold">{{ guestName || 'Tamu Undangan' }}</p> | ||||
|     </div> | ||||
| 
 | ||||
|     <!-- Button --> | ||||
|     <button | ||||
|       @click="$emit('next-page')" | ||||
|       class="z-10 mt-12 px-8 py-3 bg-blue-600 text-white font-semibold rounded-full shadow-md hover:bg-blue-700 transition duration-300" | ||||
|     > | ||||
|       Buka Undangan | ||||
|     </button> | ||||
|   </div> | ||||
| </template> | ||||
| 
 | ||||
| <script setup> | ||||
| defineProps({ | ||||
|   childName: String, | ||||
|   guestName: String | ||||
| }) | ||||
| </script> | ||||
| 
 | ||||
| <style scoped> | ||||
| @keyframes fadeInDown { | ||||
|   from { opacity: 0; transform: translateY(-20px); } | ||||
|   to { opacity: 1; transform: translateY(0); } | ||||
| } | ||||
| @keyframes fadeInUp { | ||||
|   from { opacity: 0; transform: translateY(20px); } | ||||
|   to { opacity: 1; transform: translateY(0); } | ||||
| } | ||||
| .animate-fade-in-down { animation: fadeInDown 0.8s ease-out; } | ||||
| .animate-fade-in-up { animation: fadeInUp 0.8s ease-out; } | ||||
| </style> | ||||
| @ -0,0 +1,24 @@ | ||||
| <template> | ||||
|   <section class="min-h-screen flex flex-col items-center justify-center bg-gradient-to-b from-blue-100 to-blue-300 text-center px-6"> | ||||
|     <div class="max-w-xl space-y-6"> | ||||
|       <h2 class="text-3xl font-bold text-blue-700">Terima Kasih</h2> | ||||
|       <p class="text-gray-700 leading-relaxed"> | ||||
|         Terima kasih atas doa dan kehadiran Bapak/Ibu/Saudara/i dalam acara kami. | ||||
|         Semoga mendapatkan keberkahan dan kebahagiaan selalu. | ||||
|       </p> | ||||
| 
 | ||||
|       <p class="text-xl font-semibold text-blue-800 mt-4"> | ||||
|         {{ childName }} | ||||
|       </p> | ||||
| 
 | ||||
|       <p class="text-gray-600">Beserta keluarga</p> | ||||
|     </div> | ||||
|   </section> | ||||
| </template> | ||||
| 
 | ||||
| <script setup> | ||||
| defineProps({ | ||||
|   childName: String, | ||||
|   jsonData: Object | ||||
| }) | ||||
| </script> | ||||
| @ -1,51 +0,0 @@ | ||||
| <template> | ||||
|   <div class="min-h-screen bg-gradient-to-br from-yellow-200 to-orange-200 py-10 px-6 text-center"> | ||||
|     <h2 class="text-3xl font-bold text-orange-700 mb-6">📖 Buku Tamu</h2> | ||||
| 
 | ||||
|     <p class="mb-4">Tinggalkan ucapan terbaikmu untuk {{ guestName || 'Teman Kami' }} 💌</p> | ||||
| 
 | ||||
|     <form @submit.prevent="submitMessage" class="max-w-md mx-auto bg-white p-6 rounded-2xl shadow-md"> | ||||
|       <textarea | ||||
|         v-model="message" | ||||
|         rows="3" | ||||
|         placeholder="Tulis ucapan di sini..." | ||||
|         class="w-full border rounded-md p-3 focus:outline-none focus:ring-2 focus:ring-orange-400" | ||||
|       ></textarea> | ||||
| 
 | ||||
|       <button | ||||
|         type="submit" | ||||
|         class="mt-4 bg-orange-600 text-white py-2 px-6 rounded-full hover:bg-orange-700 transition" | ||||
|       > | ||||
|         Kirim | ||||
|       </button> | ||||
|     </form> | ||||
| 
 | ||||
|     <div v-if="messages.length" class="mt-8 max-w-md mx-auto text-left"> | ||||
|       <h3 class="text-lg font-semibold text-gray-700 mb-3">Ucapan:</h3> | ||||
|       <ul class="space-y-3"> | ||||
|         <li v-for="(msg, i) in messages" :key="i" class="bg-white p-4 rounded-lg shadow"> | ||||
|           {{ msg }} | ||||
|         </li> | ||||
|       </ul> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| 
 | ||||
| <script setup> | ||||
| import { ref } from 'vue' | ||||
| 
 | ||||
| defineProps({ | ||||
|   guestName: String, | ||||
|   messages: Array | ||||
| }) | ||||
| 
 | ||||
| const emit = defineEmits(['addMessage']) | ||||
| const message = ref('') | ||||
| 
 | ||||
| const submitMessage = () => { | ||||
|   if (message.value.trim()) { | ||||
|     emit('addMessage', message.value) | ||||
|     message.value = '' | ||||
|   } | ||||
| } | ||||
| </script> | ||||
| @ -0,0 +1,134 @@ | ||||
| <template> | ||||
|     <div | ||||
|       class="min-h-screen bg-gradient-to-b from-blue-100 via-blue-200 to-blue-300 relative overflow-hidden" | ||||
|     > | ||||
|       <!-- ================= NAVIGATION ================= --> | ||||
|       <nav | ||||
|         v-if="currentSection !== 'landing'" | ||||
|         class="absolute top-4 left-1/2 transform -translate-x-1/2 z-20" | ||||
|       > | ||||
|         <ul | ||||
|           class="flex space-x-6 bg-white/40 backdrop-blur-md px-6 py-3 rounded-full shadow-md text-sm font-semibold text-gray-800" | ||||
|         > | ||||
|           <li><button @click="switchSection('introduction')" :class="navClass('introduction')">Intro</button></li> | ||||
|           <li><button @click="switchSection('event')" :class="navClass('event')">Event</button></li> | ||||
|           <li><button @click="switchSection('gallery')" :class="navClass('gallery')">Gallery</button></li> | ||||
|           <li><button @click="switchSection('thanks')" :class="navClass('thanks')">Thanks</button></li> | ||||
|         </ul> | ||||
|       </nav> | ||||
|    | ||||
|       <!-- ================= MUSIK CONTROL ================= --> | ||||
|       <div class="fixed bottom-4 left-4 z-30" v-if="currentSection !== 'landing'"> | ||||
|         <button @click="toggleMusic" class="bg-blue-600 p-3 rounded-full text-white shadow-lg"> | ||||
|           {{ isPlaying ? '⏸️' : '▶️' }} | ||||
|         </button> | ||||
|         <audio ref="audioPlayer" :src="musicUrl" loop></audio> | ||||
|       </div> | ||||
|    | ||||
|       <!-- ================= MAIN CONTENT ================= --> | ||||
|       <main | ||||
|         class="relative z-10 min-h-screen flex items-center justify-center p-4 transition-all duration-700 ease-in-out" | ||||
|       > | ||||
|         <!-- Landing Page --> | ||||
|         <KhitanAStarter | ||||
|           v-if="currentSection === 'landing'" | ||||
|           :childName="formData.nama_lengkap" | ||||
|           :guestName="data.nama_tamu" | ||||
|           @next-page="switchSection('introduction')" | ||||
|         /> | ||||
|    | ||||
|         <!-- Introduction --> | ||||
|         <KhitanIntroductionStarter | ||||
|           v-if="currentSection === 'introduction'" | ||||
|           :nama_lengkap="formData.nama_lengkap" | ||||
|           :nama_bapak="formData.nama_bapak" | ||||
|           :nama_ibu="formData.nama_ibu" | ||||
|         /> | ||||
|    | ||||
|         <!-- Event --> | ||||
|         <KhitanEventStarter | ||||
|           v-if="currentSection === 'event'" | ||||
|           :hari_tanggal_acara="formData.hari_tanggal_acara" | ||||
|           :waktu="formData.waktu" | ||||
|           :alamat="formData.alamat" | ||||
|         /> | ||||
|    | ||||
|         <!-- Gallery --> | ||||
|         <KhitanGalleryStarter | ||||
|           v-if="currentSection === 'gallery'" | ||||
|           :images="galleryImages" | ||||
|         /> | ||||
|    | ||||
|         <!-- Thank You --> | ||||
|         <KhitanThankYouStarter | ||||
|           v-if="currentSection === 'thanks'" | ||||
|           :childName="formData.nama_lengkap" | ||||
|           :jsonData="formData" | ||||
|         /> | ||||
|       </main> | ||||
|     </div> | ||||
|   </template> | ||||
|    | ||||
|   <script setup> | ||||
|   import { ref, computed } from 'vue' | ||||
|   import { useRuntimeConfig } from '#app' | ||||
|    | ||||
|   // ================== IMPORT KOMPONEN ================== | ||||
|   import KhitanAStarter from '~/components/templates/KhitanStarter/KhitanA.vue' | ||||
|   import KhitanIntroductionStarter from '~/components/templates/KhitanStarter/Introduction.vue' | ||||
|   import KhitanEventStarter from '~/components/templates/KhitanStarter/Event.vue' | ||||
|   import KhitanGalleryStarter from '~/components/templates/KhitanStarter/Gallery.vue' | ||||
|   import KhitanThankYouStarter from '~/components/templates/KhitanStarter/ThankYou.vue' | ||||
|    | ||||
|   // ================== PROPS ================== | ||||
|   const props = defineProps({ | ||||
|     data: { type: Object, required: true } | ||||
|   }) | ||||
|    | ||||
|   // ================== BACKEND CONFIG ================== | ||||
|   const config = useRuntimeConfig() | ||||
|   const backendUrl = config.public.apiBaseUrl | ||||
|    | ||||
|   // ================== FORM DATA ================== | ||||
|   const formData = computed(() => props.data.form || {}) | ||||
|    | ||||
|   // ================== GALERI ================== | ||||
|   const galleryImages = computed(() => { | ||||
|     const f = formData.value | ||||
|     return [f.foto_1, f.foto_2].filter(Boolean).map(img => `${backendUrl}/${img}`) | ||||
|   }) | ||||
|    | ||||
|   // ================== NAVIGASI SECTION ================== | ||||
|   const currentSection = ref('landing') | ||||
|   const switchSection = (s) => (currentSection.value = s) | ||||
|    | ||||
|   // ================== MUSIK ================== | ||||
|   const audioPlayer = ref(null) | ||||
|   const isPlaying = ref(false) | ||||
|   const musicUrl = computed(() => | ||||
|     formData.value.link_music ? `${backendUrl}/${formData.value.link_music}` : '' | ||||
|   ) | ||||
|    | ||||
|   const toggleMusic = () => { | ||||
|     if (!audioPlayer.value) return | ||||
|     if (isPlaying.value) { | ||||
|       audioPlayer.value.pause() | ||||
|     } else { | ||||
|       audioPlayer.value.play() | ||||
|     } | ||||
|     isPlaying.value = !isPlaying.value | ||||
|   } | ||||
|    | ||||
|   // ================== STYLE NAV ================== | ||||
|   const navClass = (s) => | ||||
|     currentSection.value === s | ||||
|       ? 'text-blue-800 underline' | ||||
|       : 'hover:text-blue-700' | ||||
|   </script> | ||||
|    | ||||
|   <style scoped> | ||||
|   main { | ||||
|     transition: all 0.7s ease-in-out; | ||||
|   } | ||||
|   </style> | ||||
|    | ||||
| @ -103,7 +103,6 @@ | ||||
| import Introduction from '~/components/templates/UltahStarter/Introduction.vue' | ||||
| import Event from '~/components/templates/UltahStarter/Event.vue' | ||||
| import Gallery from '~/components/templates/UltahStarter/Gallery.vue' | ||||
| import GuestBook from '~/components/templates/UltahStarter/GuestBook.vue' | ||||
| import ThankYou from '~/components/templates/UltahStarter/ThankYou.vue' | ||||
| 
 | ||||
|    | ||||
|  | ||||
| @ -108,7 +108,8 @@ const componentMap = { | ||||
|   'undangan-khitan-premium': defineAsyncComponent(() => import('~/components/undangan/undangan-khitan-premium.vue')), | ||||
|   'undangan-ulang-tahun-basic': defineAsyncComponent(() => import('~/components/undangan/undangan-ulang-tahun-basic.vue')), | ||||
|   'undangan-ulang-tahun-starter': defineAsyncComponent(() => import('~/components/undangan/undangan-ulang-tahun-starter.vue')), | ||||
|   'undangan-khitan-basic': defineAsyncComponent(() => import('~/components/undangan/undangan-khitan-basic.vue')) | ||||
|   'undangan-khitan-basic': defineAsyncComponent(() => import('~/components/undangan/undangan-khitan-basic.vue')), | ||||
|   'undangan-khitan-starter': defineAsyncComponent(() => import('~/components/undangan/undangan-khitan-starter.vue')) | ||||
|   // Add more mappings as templates are developed | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -281,7 +281,8 @@ const componentMap = { | ||||
|   'undangan-pernikahan-premium': defineAsyncComponent(() => import('~/components/undangan/undangan-pernikahan-premium.vue')), | ||||
|   'undangan-ulang-tahun-basic': defineAsyncComponent(() => import('~/components/undangan/undangan-ulang-tahun-basic.vue')), | ||||
|   'undangan-ulang-tahun-starter': defineAsyncComponent(() => import('~/components/undangan/undangan-ulang-tahun-starter.vue')), | ||||
|   'undangan-khitan-basic': defineAsyncComponent(() => import('~/components/undangan/undangan-khitan-basic.vue')) | ||||
|   'undangan-khitan-basic': defineAsyncComponent(() => import('~/components/undangan/undangan-khitan-basic.vue')), | ||||
|   'undangan-khitan-starter': defineAsyncComponent(() => import('~/components/undangan/undangan-khitan-starter.vue')) | ||||
|   // Add more mappings as templates are developed | ||||
| } | ||||
| 
 | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user