summaryrefslogtreecommitdiff
path: root/frontend/src/components
diff options
context:
space:
mode:
Diffstat (limited to 'frontend/src/components')
-rw-r--r--frontend/src/components/LandingPage.tsx15
-rw-r--r--frontend/src/components/TypewriterRotator.tsx84
2 files changed, 98 insertions, 1 deletions
diff --git a/frontend/src/components/LandingPage.tsx b/frontend/src/components/LandingPage.tsx
index 931fe68..708cbb1 100644
--- a/frontend/src/components/LandingPage.tsx
+++ b/frontend/src/components/LandingPage.tsx
@@ -1,6 +1,15 @@
import React, { useState, useEffect, useRef } from 'react'
import { LoadingScreen } from './LoadingScreen'
import { HeartLogo } from './HeartLogo'
+import { TypewriterRotator } from './TypewriterRotator'
+
+const HERO_PHRASES = [
+ '低遅延ストリーミング · LOW-LATENCY OBSERVABILITY',
+ 'リアルタイム監視 · REAL-TIME SURVEILLANCE',
+ 'ミッションクリティカル · MISSION-CRITICAL INFRASTRUCTURE',
+ 'エンドツーエンド可視化 · END-TO-END VISIBILITY',
+ '安全な意思決定 · SECURE DECISIONS AT THE EDGE',
+]
interface LandingPageProps {
onLogin: () => void
@@ -189,7 +198,11 @@ export function LandingPage({ onLogin }: LandingPageProps) {
</p>
</div>
) : (
- <div className="hero" />
+ <div className="hero">
+ <div className="hero-tagline" aria-label="Soryu capabilities">
+ <TypewriterRotator phrases={HERO_PHRASES} />
+ </div>
+ </div>
)}
{/* CTA row spanning full width: left Mission/MAKIMA, right Login */}
diff --git a/frontend/src/components/TypewriterRotator.tsx b/frontend/src/components/TypewriterRotator.tsx
new file mode 100644
index 0000000..4cd89fa
--- /dev/null
+++ b/frontend/src/components/TypewriterRotator.tsx
@@ -0,0 +1,84 @@
+import React, { useEffect, useRef, useState } from 'react'
+
+/**
+ * TypewriterRotator
+ *
+ * Cycles through a list of phrases with a typing-and-deleting effect.
+ * Designed for the soryu landing hero — sits inside a centred container,
+ * so the caller controls position and the component just renders text.
+ *
+ * Honours `prefers-reduced-motion` by rendering the first phrase only,
+ * statically.
+ */
+
+interface TypewriterRotatorProps {
+ phrases: string[]
+ /** ms per character while typing */
+ typeMs?: number
+ /** ms per character while deleting */
+ deleteMs?: number
+ /** ms to hold a fully-typed phrase before deleting */
+ holdMs?: number
+ /** ms to wait after deletion before the next phrase types in */
+ gapMs?: number
+ className?: string
+}
+
+export function TypewriterRotator({
+ phrases,
+ typeMs = 55,
+ deleteMs = 28,
+ holdMs = 2200,
+ gapMs = 320,
+ className,
+}: TypewriterRotatorProps) {
+ const [text, setText] = useState('')
+ const [index, setIndex] = useState(0)
+ const [phase, setPhase] = useState<'typing' | 'holding' | 'deleting' | 'gap'>('typing')
+ const reduceMotion = useRef(
+ typeof window !== 'undefined' &&
+ window.matchMedia &&
+ window.matchMedia('(prefers-reduced-motion: reduce)').matches,
+ )
+
+ useEffect(() => {
+ if (reduceMotion.current) {
+ setText(phrases[0] ?? '')
+ return
+ }
+ const current = phrases[index % phrases.length] ?? ''
+ let timer: number
+
+ if (phase === 'typing') {
+ if (text.length < current.length) {
+ timer = window.setTimeout(() => setText(current.slice(0, text.length + 1)), typeMs)
+ } else {
+ timer = window.setTimeout(() => setPhase('holding'), 0)
+ }
+ } else if (phase === 'holding') {
+ // Skip the hold-then-delete cycle entirely if there's only one phrase
+ if (phrases.length <= 1) return
+ timer = window.setTimeout(() => setPhase('deleting'), holdMs)
+ } else if (phase === 'deleting') {
+ if (text.length > 0) {
+ timer = window.setTimeout(() => setText(current.slice(0, text.length - 1)), deleteMs)
+ } else {
+ timer = window.setTimeout(() => setPhase('gap'), 0)
+ }
+ } else if (phase === 'gap') {
+ timer = window.setTimeout(() => {
+ setIndex((i) => (i + 1) % phrases.length)
+ setPhase('typing')
+ }, gapMs)
+ }
+
+ return () => window.clearTimeout(timer)
+ }, [text, phase, index, phrases, typeMs, deleteMs, holdMs, gapMs])
+
+ return (
+ <span className={className} role="status" aria-live="polite">
+ <span className="tw-text">{text}</span>
+ <span className="tw-caret" aria-hidden="true" />
+ </span>
+ )
+}