2026-01-24 22:59:20 +00:00
|
|
|
<template>
|
|
|
|
|
<Transition name="splash-fade">
|
|
|
|
|
<div v-if="showSplash" class="fixed inset-0 z-[2000] flex items-center justify-center bg-black" style="will-change: opacity, transform;">
|
|
|
|
|
<!-- Video background - shown during Welcome Noderunner and Logo (seamless, no zoom) -->
|
|
|
|
|
<video
|
|
|
|
|
v-if="showWelcome || showLogo"
|
|
|
|
|
ref="videoElement"
|
|
|
|
|
class="absolute inset-0 w-full h-full object-cover"
|
|
|
|
|
:style="{ opacity: backgroundOpacity, transform: 'scale(1)', transition: 'opacity 1.2s ease-out' }"
|
|
|
|
|
autoplay
|
|
|
|
|
loop
|
|
|
|
|
muted
|
|
|
|
|
playsinline
|
|
|
|
|
preload="auto"
|
2026-02-18 08:18:14 +00:00
|
|
|
poster="/assets/img/bg-intro.jpg"
|
2026-01-24 22:59:20 +00:00
|
|
|
>
|
|
|
|
|
<source src="/assets/video/video-intro.mp4?v=7" type="video/mp4">
|
|
|
|
|
<!-- Fallback to image if video fails -->
|
|
|
|
|
<div
|
|
|
|
|
class="absolute inset-0"
|
|
|
|
|
:style="{
|
2026-02-18 08:18:14 +00:00
|
|
|
backgroundImage: 'url(/assets/img/bg-intro.jpg)',
|
2026-01-24 22:59:20 +00:00
|
|
|
backgroundSize: 'auto 100vh',
|
|
|
|
|
backgroundPosition: 'center top',
|
|
|
|
|
backgroundRepeat: 'no-repeat',
|
|
|
|
|
}"
|
|
|
|
|
/>
|
|
|
|
|
</video>
|
|
|
|
|
|
|
|
|
|
<!-- Static image background - shown during alien intro -->
|
|
|
|
|
<div
|
|
|
|
|
v-else
|
|
|
|
|
class="absolute inset-0"
|
|
|
|
|
:style="{
|
2026-02-18 08:18:14 +00:00
|
|
|
backgroundImage: 'url(/assets/img/bg-intro.jpg)',
|
2026-01-24 22:59:20 +00:00
|
|
|
backgroundSize: 'auto 100vh',
|
|
|
|
|
backgroundPosition: 'center top',
|
|
|
|
|
backgroundRepeat: 'no-repeat',
|
|
|
|
|
opacity: backgroundOpacity,
|
|
|
|
|
transform: 'scale(1)',
|
|
|
|
|
transition: 'opacity 1.2s ease-out',
|
|
|
|
|
}"
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
<!-- Alien Intro -->
|
|
|
|
|
<Transition name="fade">
|
|
|
|
|
<div
|
|
|
|
|
v-if="!alienIntroComplete"
|
|
|
|
|
class="absolute inset-0 z-10 flex items-center justify-center transition-opacity duration-800"
|
|
|
|
|
:class="{ 'opacity-0': fadeAlienIntro }"
|
|
|
|
|
>
|
|
|
|
|
<div class="font-mono text-white px-4 sm:px-5 max-w-[95vw] sm:max-w-[90vw] md:max-w-[1200px] text-base sm:text-lg md:text-[24px] leading-relaxed break-words">
|
|
|
|
|
<div v-if="showLine1" class="flex items-start mb-4 sm:mb-6 opacity-0" :class="{ 'opacity-100': showLine1 }">
|
2026-02-17 20:40:26 +00:00
|
|
|
<span class="text-[#fbbf24] mr-3 sm:mr-6 flex-shrink-0">></span>
|
2026-02-17 19:19:54 +00:00
|
|
|
<span class="text-white break-words">{{ displayLine1 }}</span><span v-if="isTypingLine1" class="intro-typing-caret" aria-hidden="true"></span>
|
2026-01-24 22:59:20 +00:00
|
|
|
</div>
|
|
|
|
|
<div v-if="showLine2" class="flex items-start mb-4 sm:mb-6 opacity-0" :class="{ 'opacity-100': showLine2 }">
|
2026-02-17 20:40:26 +00:00
|
|
|
<span class="text-[#fbbf24] mr-3 sm:mr-6 flex-shrink-0">></span>
|
2026-02-17 19:19:54 +00:00
|
|
|
<span class="text-white break-words">{{ displayLine2 }}</span><span v-if="isTypingLine2" class="intro-typing-caret" aria-hidden="true"></span>
|
2026-01-24 22:59:20 +00:00
|
|
|
</div>
|
|
|
|
|
<div v-if="showLine3" class="flex items-start mb-4 sm:mb-6 opacity-0" :class="{ 'opacity-100': showLine3 }">
|
2026-02-17 20:40:26 +00:00
|
|
|
<span class="text-[#fbbf24] mr-3 sm:mr-6 flex-shrink-0">></span>
|
2026-02-17 19:19:54 +00:00
|
|
|
<span class="text-white break-words">{{ displayLine3 }}</span><span v-if="isTypingLine3" class="intro-typing-caret" aria-hidden="true"></span>
|
2026-01-24 22:59:20 +00:00
|
|
|
</div>
|
|
|
|
|
<div v-if="showLine4" class="flex items-start mb-8 sm:mb-12 opacity-0" :class="{ 'opacity-100': showLine4 }">
|
2026-02-17 20:40:26 +00:00
|
|
|
<span class="text-[#fbbf24] mr-3 sm:mr-6 flex-shrink-0">></span>
|
2026-02-17 19:19:54 +00:00
|
|
|
<span class="text-white break-words">{{ displayLine4 }}</span><span v-if="isTypingLine4" class="intro-typing-caret" aria-hidden="true"></span>
|
2026-01-24 22:59:20 +00:00
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</Transition>
|
|
|
|
|
|
|
|
|
|
<!-- Welcome Message -->
|
|
|
|
|
<Transition name="welcome-fade">
|
|
|
|
|
<div
|
|
|
|
|
v-if="showWelcome"
|
|
|
|
|
class="absolute inset-0 z-[15] flex items-center justify-center font-mono text-3xl sm:text-4xl md:text-5xl px-4"
|
|
|
|
|
:class="{ 'welcome-fade-out': fadeWelcome }"
|
|
|
|
|
>
|
|
|
|
|
<div class="typing-container">
|
|
|
|
|
<span class="text-white" :class="{ 'typing-text': typingWelcome }">
|
|
|
|
|
Welcome Noderunner
|
|
|
|
|
</span>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</Transition>
|
|
|
|
|
|
|
|
|
|
<!-- Logo - Archipelago logo for splash -->
|
|
|
|
|
<Transition name="logo-zoom">
|
|
|
|
|
<div v-if="showLogo" class="relative z-20 logo-container">
|
|
|
|
|
<img
|
|
|
|
|
src="/assets/img/logo-archipelago.svg"
|
|
|
|
|
alt="Archipelago"
|
|
|
|
|
class="w-[min(80vw,900px)] max-w-[90vw] h-auto filter drop-shadow-[0_6px_24px_rgba(0,0,0,0.35)] m-5 object-contain logo-zoom-bounce"
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
</Transition>
|
|
|
|
|
|
2026-02-18 08:18:14 +00:00
|
|
|
<!-- Tap to start - logo + "Enter the Exit" behind (like screensaver) -->
|
2026-02-17 19:42:59 +00:00
|
|
|
<div
|
|
|
|
|
v-if="showTapToStart"
|
2026-02-18 10:10:12 +00:00
|
|
|
class="absolute inset-0 z-[100] flex items-center justify-center cursor-pointer overflow-hidden"
|
|
|
|
|
:class="tapStartTransitioning ? 'tap-overlay-zoom-out' : 'bg-black/40'"
|
2026-02-17 19:42:59 +00:00
|
|
|
@click="handleTapToStart"
|
|
|
|
|
>
|
2026-02-18 10:10:12 +00:00
|
|
|
<div class="tap-to-start-content relative flex items-center justify-center perspective-1000">
|
|
|
|
|
<span
|
|
|
|
|
class="tap-to-start-text font-archipelago font-extrabold text-[rgba(0,0,0,0.35)] text-6xl sm:text-7xl md:text-8xl lg:text-9xl tracking-widest uppercase whitespace-nowrap select-none transition-opacity duration-300"
|
|
|
|
|
:class="{ 'opacity-0': tapStartTransitioning }"
|
|
|
|
|
>
|
2026-02-18 08:18:14 +00:00
|
|
|
Enter the Exit
|
|
|
|
|
</span>
|
2026-02-18 10:10:12 +00:00
|
|
|
<div
|
|
|
|
|
class="tap-to-start-logo absolute"
|
|
|
|
|
:class="{ 'tap-logo-launch': tapStartTransitioning }"
|
|
|
|
|
>
|
2026-02-18 08:18:14 +00:00
|
|
|
<ScreensaverLogo />
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
2026-02-17 19:42:59 +00:00
|
|
|
</div>
|
|
|
|
|
|
2026-01-24 22:59:20 +00:00
|
|
|
<!-- Skip Button -->
|
|
|
|
|
<button
|
2026-02-17 19:42:59 +00:00
|
|
|
v-if="!alienIntroComplete && !showTapToStart"
|
2026-02-17 19:19:54 +00:00
|
|
|
@click="handleSkipClick"
|
2026-01-24 22:59:20 +00:00
|
|
|
class="absolute bottom-8 right-8 z-20 bg-black/60 border border-white/30 text-white/70 font-mono text-xs px-4 py-2 rounded backdrop-blur-[10px] hover:bg-black/80 hover:text-white/90 hover:border-white/50 hover:-translate-y-0.5 active:translate-y-0 transition-all duration-300"
|
|
|
|
|
>
|
|
|
|
|
Skip Intro
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
</Transition>
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<script setup lang="ts">
|
2026-02-17 19:19:54 +00:00
|
|
|
import { ref, onMounted, onBeforeUnmount, watch } from 'vue'
|
2026-02-18 08:18:14 +00:00
|
|
|
import ScreensaverLogo from '@/components/ScreensaverLogo.vue'
|
|
|
|
|
import { playIntroTyping, playLoopStart, playWelcomeNoderunnerSpeech, resumeAudioContext, startSynthwave, stopIntroTyping } from '@/composables/useLoginSounds'
|
2026-01-24 22:59:20 +00:00
|
|
|
|
|
|
|
|
const emit = defineEmits<{
|
|
|
|
|
complete: []
|
|
|
|
|
}>()
|
|
|
|
|
|
2026-02-17 19:19:54 +00:00
|
|
|
const INTRO_LINES = [
|
|
|
|
|
'In the future there will be 3 types of humans',
|
|
|
|
|
'Government Employees',
|
|
|
|
|
'Corporate Employees',
|
|
|
|
|
'And Noderunners...',
|
|
|
|
|
] as const
|
|
|
|
|
const MS_PER_CHAR = 55
|
|
|
|
|
const BLINK_AFTER_TYPING = 1500
|
|
|
|
|
|
2026-01-24 22:59:20 +00:00
|
|
|
const showSplash = ref(true)
|
2026-02-17 19:42:59 +00:00
|
|
|
const showTapToStart = ref(true)
|
2026-02-18 10:10:12 +00:00
|
|
|
const tapStartTransitioning = ref(false)
|
2026-01-24 22:59:20 +00:00
|
|
|
const backgroundOpacity = ref(0)
|
|
|
|
|
const alienIntroComplete = ref(false)
|
|
|
|
|
const fadeAlienIntro = ref(false)
|
|
|
|
|
const showWelcome = ref(false)
|
|
|
|
|
const fadeWelcome = ref(false)
|
|
|
|
|
const typingWelcome = ref(false)
|
|
|
|
|
const showLogo = ref(false)
|
|
|
|
|
const showLine1 = ref(false)
|
|
|
|
|
const showLine2 = ref(false)
|
|
|
|
|
const showLine3 = ref(false)
|
|
|
|
|
const showLine4 = ref(false)
|
2026-02-17 19:19:54 +00:00
|
|
|
const displayLine1 = ref('')
|
|
|
|
|
const displayLine2 = ref('')
|
|
|
|
|
const displayLine3 = ref('')
|
|
|
|
|
const displayLine4 = ref('')
|
|
|
|
|
const isTypingLine1 = ref(false)
|
|
|
|
|
const isTypingLine2 = ref(false)
|
|
|
|
|
const isTypingLine3 = ref(false)
|
|
|
|
|
const isTypingLine4 = ref(false)
|
2026-01-24 22:59:20 +00:00
|
|
|
const videoElement = ref<HTMLVideoElement | null>(null)
|
2026-02-17 19:19:54 +00:00
|
|
|
let introTypingTimeout: ReturnType<typeof setTimeout> | null = null
|
2026-01-24 22:59:20 +00:00
|
|
|
|
|
|
|
|
// Ensure video plays continuously from Welcome Noderunner through logo
|
|
|
|
|
watch([showWelcome, showLogo], ([welcome, logo]) => {
|
|
|
|
|
if ((welcome || logo) && videoElement.value) {
|
|
|
|
|
// Ensure video is playing and doesn't pause
|
|
|
|
|
if (videoElement.value.paused) {
|
|
|
|
|
videoElement.value.play().catch(err => {
|
|
|
|
|
console.warn('Video autoplay failed:', err)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
// Keep video playing - prevent any pauses
|
|
|
|
|
videoElement.value.addEventListener('pause', (e) => {
|
|
|
|
|
if (welcome || logo) {
|
|
|
|
|
e.preventDefault()
|
|
|
|
|
videoElement.value?.play()
|
|
|
|
|
}
|
|
|
|
|
}, { once: false })
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// Start video as soon as welcome appears
|
|
|
|
|
watch(showWelcome, (isShowing) => {
|
|
|
|
|
if (isShowing && videoElement.value) {
|
|
|
|
|
// Start video immediately when welcome appears
|
|
|
|
|
videoElement.value.play().catch(err => {
|
|
|
|
|
console.warn('Video autoplay failed on welcome:', err)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// Store video currentTime continuously and before unmounting for seamless transition
|
|
|
|
|
watch(showSplash, (isShowing) => {
|
|
|
|
|
if (!isShowing && videoElement.value) {
|
|
|
|
|
// Store current video time for seamless transition
|
|
|
|
|
const currentTime = videoElement.value.currentTime
|
|
|
|
|
const wasPlaying = !videoElement.value.paused
|
|
|
|
|
sessionStorage.setItem('video_intro_currentTime', currentTime.toString())
|
|
|
|
|
sessionStorage.setItem('video_intro_wasPlaying', wasPlaying.toString())
|
|
|
|
|
// Store video playback rate to maintain smooth playback
|
|
|
|
|
sessionStorage.setItem('video_intro_playbackRate', videoElement.value.playbackRate.toString())
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// Continuously update video time while playing (for more accurate restoration)
|
|
|
|
|
let videoTimeUpdateInterval: number | null = null
|
|
|
|
|
watch([showWelcome, showLogo], ([welcome, logo]) => {
|
|
|
|
|
if ((welcome || logo) && videoElement.value) {
|
|
|
|
|
// Update stored time every 50ms for better accuracy and smoother transition
|
|
|
|
|
videoTimeUpdateInterval = window.setInterval(() => {
|
|
|
|
|
if (videoElement.value && !videoElement.value.paused) {
|
|
|
|
|
sessionStorage.setItem('video_intro_currentTime', videoElement.value.currentTime.toString())
|
|
|
|
|
sessionStorage.setItem('video_intro_wasPlaying', 'true')
|
|
|
|
|
sessionStorage.setItem('video_intro_playbackRate', videoElement.value.playbackRate.toString())
|
|
|
|
|
}
|
|
|
|
|
}, 50) // More frequent updates for smoother transition
|
|
|
|
|
} else {
|
|
|
|
|
if (videoTimeUpdateInterval) {
|
|
|
|
|
clearInterval(videoTimeUpdateInterval)
|
|
|
|
|
videoTimeUpdateInterval = null
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// Check if user has seen intro
|
|
|
|
|
const seenIntro = localStorage.getItem('neode_intro_seen') === '1'
|
|
|
|
|
|
2026-02-17 19:42:59 +00:00
|
|
|
function handleTapToStart() {
|
2026-02-18 10:10:12 +00:00
|
|
|
if (!showTapToStart.value || tapStartTransitioning.value) return
|
2026-02-17 19:42:59 +00:00
|
|
|
resumeAudioContext()
|
2026-02-18 10:10:12 +00:00
|
|
|
tapStartTransitioning.value = true
|
|
|
|
|
// Logo: grow (150ms) then zoom out to background (850ms). Total 1s.
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
showTapToStart.value = false
|
|
|
|
|
tapStartTransitioning.value = false
|
|
|
|
|
startAlienIntro()
|
|
|
|
|
}, 1000)
|
2026-02-17 19:42:59 +00:00
|
|
|
}
|
|
|
|
|
|
2026-02-17 19:19:54 +00:00
|
|
|
function handleSkipClick() {
|
|
|
|
|
resumeAudioContext()
|
|
|
|
|
skipIntro()
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-24 22:59:20 +00:00
|
|
|
function skipIntro() {
|
|
|
|
|
// Jump to "Welcome Noderunner" part
|
2026-02-17 19:19:54 +00:00
|
|
|
if (introTypingTimeout) {
|
|
|
|
|
clearTimeout(introTypingTimeout)
|
|
|
|
|
introTypingTimeout = null
|
|
|
|
|
}
|
2026-01-24 22:59:20 +00:00
|
|
|
alienIntroComplete.value = true
|
|
|
|
|
fadeAlienIntro.value = true
|
|
|
|
|
showWelcome.value = true
|
|
|
|
|
typingWelcome.value = true
|
2026-02-17 19:19:54 +00:00
|
|
|
stopIntroTyping()
|
|
|
|
|
playLoopStart()
|
|
|
|
|
startSynthwave()
|
2026-02-18 08:18:14 +00:00
|
|
|
playWelcomeNoderunnerSpeech()
|
|
|
|
|
|
2026-02-17 19:19:54 +00:00
|
|
|
// Stop alien intro typing and any playing typing sound
|
|
|
|
|
stopIntroTyping()
|
|
|
|
|
isTypingLine1.value = false
|
|
|
|
|
isTypingLine2.value = false
|
|
|
|
|
isTypingLine3.value = false
|
|
|
|
|
isTypingLine4.value = false
|
2026-01-24 22:59:20 +00:00
|
|
|
|
|
|
|
|
// Start background fade in at 0.3 opacity when welcome appears
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
backgroundOpacity.value = 0.3
|
|
|
|
|
}, 0)
|
|
|
|
|
|
|
|
|
|
// Continue with welcome fade out after typing (2s) + cursor continues (1.5s) + 3 blinks (1.35s)
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
fadeWelcome.value = true
|
|
|
|
|
typingWelcome.value = false
|
|
|
|
|
}, 4850)
|
|
|
|
|
|
|
|
|
|
// Show logo - no zoom, just fade
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
showLogo.value = true
|
|
|
|
|
// Keep background at 0.3 opacity during logo display
|
|
|
|
|
}, 5500)
|
|
|
|
|
|
|
|
|
|
// Hide welcome after logo starts appearing
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
showWelcome.value = false
|
|
|
|
|
}, 6000)
|
|
|
|
|
|
|
|
|
|
// Fade background to full opacity just before completing (for smooth transition to modal)
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
backgroundOpacity.value = 1
|
|
|
|
|
}, 9000)
|
|
|
|
|
|
|
|
|
|
// Complete splash with smooth transition - wait for zoom to complete
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
// Add a small delay to ensure smooth transition
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
showSplash.value = false
|
|
|
|
|
document.body.classList.add('splash-complete')
|
|
|
|
|
localStorage.setItem('neode_intro_seen', '1')
|
|
|
|
|
emit('complete')
|
|
|
|
|
}, 500)
|
|
|
|
|
}, 9500)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function startAlienIntro() {
|
2026-02-17 19:19:54 +00:00
|
|
|
function typeLine(
|
|
|
|
|
lineIndex: number,
|
|
|
|
|
displayRef: { value: string },
|
|
|
|
|
isTypingRef: { value: boolean },
|
|
|
|
|
onDone: () => void
|
|
|
|
|
) {
|
|
|
|
|
const text = INTRO_LINES[lineIndex]!
|
|
|
|
|
let i = 0
|
|
|
|
|
displayRef.value = ''
|
|
|
|
|
isTypingRef.value = true
|
|
|
|
|
|
|
|
|
|
function tick() {
|
|
|
|
|
if (i === 0) {
|
|
|
|
|
playIntroTyping()
|
|
|
|
|
}
|
|
|
|
|
if (i < text.length) {
|
|
|
|
|
displayRef.value = text.slice(0, i + 1)
|
|
|
|
|
i++
|
|
|
|
|
introTypingTimeout = setTimeout(tick, MS_PER_CHAR)
|
|
|
|
|
} else {
|
|
|
|
|
stopIntroTyping()
|
|
|
|
|
isTypingRef.value = false
|
|
|
|
|
introTypingTimeout = setTimeout(onDone, BLINK_AFTER_TYPING)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
tick()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function scheduleLine1() {
|
2026-01-24 22:59:20 +00:00
|
|
|
showLine1.value = true
|
2026-02-17 19:19:54 +00:00
|
|
|
typeLine(0, displayLine1, isTypingLine1, scheduleLine2)
|
|
|
|
|
}
|
2026-01-24 22:59:20 +00:00
|
|
|
|
2026-02-17 19:19:54 +00:00
|
|
|
function scheduleLine2() {
|
2026-01-24 22:59:20 +00:00
|
|
|
showLine2.value = true
|
2026-02-17 19:19:54 +00:00
|
|
|
typeLine(1, displayLine2, isTypingLine2, scheduleLine3)
|
|
|
|
|
}
|
2026-01-24 22:59:20 +00:00
|
|
|
|
2026-02-17 19:19:54 +00:00
|
|
|
function scheduleLine3() {
|
2026-01-24 22:59:20 +00:00
|
|
|
showLine3.value = true
|
2026-02-17 19:19:54 +00:00
|
|
|
typeLine(2, displayLine3, isTypingLine3, scheduleLine4)
|
|
|
|
|
}
|
2026-01-24 22:59:20 +00:00
|
|
|
|
2026-02-17 19:19:54 +00:00
|
|
|
function scheduleLine4() {
|
2026-01-24 22:59:20 +00:00
|
|
|
showLine4.value = true
|
2026-02-17 19:19:54 +00:00
|
|
|
typeLine(3, displayLine4, isTypingLine4, () => {
|
|
|
|
|
isTypingLine4.value = false
|
|
|
|
|
fadeAlienIntro.value = true
|
|
|
|
|
introTypingTimeout = setTimeout(showWelcomePhase, 800)
|
|
|
|
|
})
|
|
|
|
|
}
|
2026-01-24 22:59:20 +00:00
|
|
|
|
2026-02-17 19:19:54 +00:00
|
|
|
function showWelcomePhase() {
|
2026-01-24 22:59:20 +00:00
|
|
|
alienIntroComplete.value = true
|
|
|
|
|
showWelcome.value = true
|
|
|
|
|
typingWelcome.value = true
|
2026-02-17 19:19:54 +00:00
|
|
|
stopIntroTyping()
|
|
|
|
|
playLoopStart()
|
|
|
|
|
startSynthwave()
|
2026-02-18 08:18:14 +00:00
|
|
|
playWelcomeNoderunnerSpeech()
|
2026-01-24 22:59:20 +00:00
|
|
|
if (videoElement.value) {
|
|
|
|
|
videoElement.value.play().catch(err => {
|
|
|
|
|
console.warn('Video autoplay failed on welcome:', err)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
backgroundOpacity.value = 0.3
|
|
|
|
|
|
2026-02-17 19:19:54 +00:00
|
|
|
introTypingTimeout = setTimeout(() => {
|
|
|
|
|
fadeWelcome.value = true
|
|
|
|
|
typingWelcome.value = false
|
|
|
|
|
}, 4850)
|
2026-01-24 22:59:20 +00:00
|
|
|
|
2026-02-17 19:19:54 +00:00
|
|
|
introTypingTimeout = setTimeout(() => {
|
|
|
|
|
showLogo.value = true
|
|
|
|
|
}, 5500)
|
2026-01-24 22:59:20 +00:00
|
|
|
|
2026-02-17 19:19:54 +00:00
|
|
|
introTypingTimeout = setTimeout(() => {
|
|
|
|
|
showWelcome.value = false
|
|
|
|
|
}, 6000)
|
|
|
|
|
|
|
|
|
|
introTypingTimeout = setTimeout(() => {
|
|
|
|
|
backgroundOpacity.value = 1
|
|
|
|
|
}, 9000)
|
|
|
|
|
|
|
|
|
|
introTypingTimeout = setTimeout(() => {
|
|
|
|
|
if (videoElement.value && !videoElement.value.paused) {
|
|
|
|
|
sessionStorage.setItem('video_intro_currentTime', videoElement.value.currentTime.toString())
|
|
|
|
|
sessionStorage.setItem('video_intro_wasPlaying', 'true')
|
|
|
|
|
}
|
|
|
|
|
showSplash.value = false
|
|
|
|
|
document.body.classList.add('splash-complete')
|
|
|
|
|
localStorage.setItem('neode_intro_seen', '1')
|
|
|
|
|
emit('complete')
|
|
|
|
|
}, 9500)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
introTypingTimeout = setTimeout(scheduleLine1, 500)
|
2026-01-24 22:59:20 +00:00
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onMounted(() => {
|
|
|
|
|
if (seenIntro) {
|
|
|
|
|
showSplash.value = false
|
|
|
|
|
document.body.classList.add('splash-complete')
|
|
|
|
|
emit('complete')
|
2026-02-17 19:19:54 +00:00
|
|
|
}
|
2026-02-17 19:42:59 +00:00
|
|
|
// Typing starts only after user taps "Tap to start" (required for loop-start + music)
|
2026-02-17 19:19:54 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
|
|
onBeforeUnmount(() => {
|
|
|
|
|
if (introTypingTimeout) {
|
|
|
|
|
clearTimeout(introTypingTimeout)
|
|
|
|
|
introTypingTimeout = null
|
2026-01-24 22:59:20 +00:00
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
.splash-fade-enter-active {
|
|
|
|
|
transition: opacity 0.5s ease;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.splash-fade-leave-active {
|
|
|
|
|
transition: opacity 1s ease-out, transform 1s ease-out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.splash-fade-enter-from,
|
|
|
|
|
.splash-fade-leave-to {
|
|
|
|
|
opacity: 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.splash-fade-leave-to {
|
|
|
|
|
transform: scale(1.1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.fade-enter-active,
|
|
|
|
|
.fade-leave-active {
|
|
|
|
|
transition: opacity 0.5s ease;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.fade-enter-from,
|
|
|
|
|
.fade-leave-to {
|
|
|
|
|
opacity: 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Welcome message fade out */
|
|
|
|
|
.welcome-fade-enter-active {
|
|
|
|
|
transition: opacity 0.8s ease-out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.welcome-fade-leave-active {
|
|
|
|
|
transition: opacity 0.6s ease-in;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.welcome-fade-enter-from,
|
|
|
|
|
.welcome-fade-leave-to {
|
|
|
|
|
opacity: 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.welcome-fade-out {
|
|
|
|
|
opacity: 0;
|
|
|
|
|
transition: opacity 0.6s ease-in;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Logo zoom bounce animation - smooth and buttery */
|
|
|
|
|
.logo-zoom-enter-active {
|
|
|
|
|
transition: all 1s cubic-bezier(0.25, 0.46, 0.45, 0.94);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.logo-zoom-leave-active {
|
|
|
|
|
transition: opacity 0.5s ease-out, transform 0.5s ease-out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.logo-zoom-enter-from {
|
|
|
|
|
opacity: 0;
|
|
|
|
|
transform: scale(0.7);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.logo-zoom-enter-to {
|
|
|
|
|
opacity: 1;
|
|
|
|
|
transform: scale(1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.logo-zoom-leave-from {
|
|
|
|
|
opacity: 1;
|
|
|
|
|
transform: scale(1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.logo-zoom-leave-to {
|
|
|
|
|
opacity: 0;
|
|
|
|
|
transform: scale(1.05);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.logo-zoom-bounce {
|
|
|
|
|
animation: logoZoomBounce 1.2s cubic-bezier(0.25, 0.46, 0.45, 0.94) forwards;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@keyframes logoZoomBounce {
|
|
|
|
|
0% {
|
|
|
|
|
transform: scale(0.85);
|
|
|
|
|
opacity: 0;
|
|
|
|
|
}
|
|
|
|
|
50% {
|
|
|
|
|
transform: scale(1.02);
|
|
|
|
|
opacity: 0.9;
|
|
|
|
|
}
|
|
|
|
|
75% {
|
|
|
|
|
transform: scale(0.98);
|
|
|
|
|
opacity: 0.95;
|
|
|
|
|
}
|
|
|
|
|
100% {
|
|
|
|
|
transform: scale(1);
|
|
|
|
|
opacity: 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Container to keep the typing text centered */
|
|
|
|
|
.typing-container {
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
align-items: center;
|
|
|
|
|
min-width: 0;
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-17 20:40:26 +00:00
|
|
|
/* Intro typing cursor - block style, yellow blink (Archipelago style) */
|
2026-02-17 19:19:54 +00:00
|
|
|
.intro-typing-caret {
|
|
|
|
|
display: inline-block;
|
|
|
|
|
width: 4px;
|
|
|
|
|
min-width: 4px;
|
|
|
|
|
height: 1.2em;
|
2026-02-17 20:40:26 +00:00
|
|
|
background: #fbbf24;
|
2026-02-17 19:19:54 +00:00
|
|
|
margin-left: 2px;
|
|
|
|
|
vertical-align: text-bottom;
|
|
|
|
|
animation: intro-caret-blink 0.5s step-end infinite;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@keyframes intro-caret-blink {
|
|
|
|
|
0%, 100% { opacity: 1; }
|
|
|
|
|
50% { opacity: 0; }
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-24 22:59:20 +00:00
|
|
|
/* Ensure text wraps smoothly on mobile */
|
|
|
|
|
.font-mono {
|
|
|
|
|
word-wrap: break-word;
|
|
|
|
|
overflow-wrap: break-word;
|
|
|
|
|
hyphens: auto;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Smooth line transitions for mobile */
|
|
|
|
|
@media (max-width: 640px) {
|
|
|
|
|
.font-mono span {
|
|
|
|
|
display: inline-block;
|
|
|
|
|
max-width: 100%;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Background zoom transition - matches OnboardingWrapper style */
|
|
|
|
|
.bg-zoom-transition {
|
|
|
|
|
transition: transform 1.5s cubic-bezier(0.4, 0, 0.2, 1), opacity 1.2s ease-out;
|
|
|
|
|
transform: scale(1);
|
|
|
|
|
transform-origin: center center;
|
|
|
|
|
will-change: transform, opacity;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.bg-zoom-transition.bg-zoom-in {
|
|
|
|
|
transform: scale(1.15);
|
|
|
|
|
}
|
2026-02-18 08:18:14 +00:00
|
|
|
|
2026-02-18 10:10:12 +00:00
|
|
|
/* Tap to start - logo grow then zoom out to background */
|
|
|
|
|
.tap-overlay-zoom-out {
|
|
|
|
|
background-color: rgba(0, 0, 0, 0.4);
|
|
|
|
|
transition: background-color 0.6s cubic-bezier(0.4, 0, 0.2, 1);
|
|
|
|
|
animation: tap-overlay-fade 1s ease-out forwards;
|
|
|
|
|
}
|
|
|
|
|
@keyframes tap-overlay-fade {
|
|
|
|
|
0% { background-color: rgba(0, 0, 0, 0.4); }
|
|
|
|
|
30% { background-color: rgba(0, 0, 0, 0.35); }
|
|
|
|
|
100% { background-color: rgba(0, 0, 0, 0); }
|
|
|
|
|
}
|
|
|
|
|
.perspective-1000 {
|
|
|
|
|
perspective: 1000px;
|
|
|
|
|
}
|
|
|
|
|
.tap-logo-launch {
|
|
|
|
|
animation: tap-logo-launch 1s cubic-bezier(0.22, 1, 0.36, 1) forwards;
|
|
|
|
|
transform-origin: center center;
|
|
|
|
|
will-change: transform, opacity;
|
|
|
|
|
}
|
|
|
|
|
@keyframes tap-logo-launch {
|
|
|
|
|
0% { transform: scale(1); opacity: 1; }
|
|
|
|
|
15% { transform: scale(1.2); opacity: 1; }
|
|
|
|
|
25% { transform: scale(1.15); opacity: 1; }
|
|
|
|
|
100% { transform: scale(0); opacity: 0; }
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-18 08:18:14 +00:00
|
|
|
/* Tap to start - "Enter the Exit" big behind logo */
|
|
|
|
|
.tap-to-start-content {
|
|
|
|
|
min-height: 12rem;
|
|
|
|
|
}
|
|
|
|
|
.tap-to-start-text {
|
|
|
|
|
position: absolute;
|
|
|
|
|
z-index: 0;
|
|
|
|
|
pointer-events: none;
|
|
|
|
|
}
|
|
|
|
|
.tap-to-start-logo {
|
|
|
|
|
position: relative;
|
|
|
|
|
z-index: 1;
|
|
|
|
|
}
|
|
|
|
|
.tap-to-start-logo {
|
|
|
|
|
filter: drop-shadow(0 0 40px rgba(255, 255, 255, 0.15));
|
|
|
|
|
}
|
|
|
|
|
.tap-to-start-logo :deep(.logo-gradient-border) {
|
|
|
|
|
width: 12rem;
|
|
|
|
|
height: 12rem;
|
|
|
|
|
}
|
|
|
|
|
@media (min-width: 640px) {
|
|
|
|
|
.tap-to-start-content {
|
|
|
|
|
min-height: 14rem;
|
|
|
|
|
}
|
|
|
|
|
.tap-to-start-logo :deep(.logo-gradient-border) {
|
|
|
|
|
width: 14rem;
|
|
|
|
|
height: 14rem;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
@media (min-width: 768px) {
|
|
|
|
|
.tap-to-start-content {
|
|
|
|
|
min-height: 16rem;
|
|
|
|
|
}
|
|
|
|
|
.tap-to-start-logo :deep(.logo-gradient-border) {
|
|
|
|
|
width: 16rem;
|
|
|
|
|
height: 16rem;
|
|
|
|
|
}
|
|
|
|
|
}
|
2026-01-24 22:59:20 +00:00
|
|
|
</style>
|
|
|
|
|
|