- lib/solar-terms.ts: solarlunar → lunar-javascript로 전면 교체 - getSolarTermDate(): LunarYear.fromYear().getJieQiJulianDays() 사용 (시분 단위 정밀도) - 소한(22)/대한(23)은 year-1로 조회해 해당 연도 1월 날짜 정확히 반환 - getCurrentSolarTerm(): 입춘 기준 두 구간 분리, Date.UTC() 비교 - lib/daeun-calculator.ts: getSolarTermDate 정확도 향상으로 termYear 수동 보정 제거 - lib/saju-calculator.ts: 일주 기준일 甲戌, Date.UTC(), 오호둔월법 공식 적용 - lib/ai-interpretation.ts: 신약 용신 후보 내림차순 정렬 수정 - app/saju/result/page.tsx: Python 엔진(fetchFromPythonEngine) 완전 제거, TS 전용 - app/api/saju/calculate/route.ts: Python 프록시 라우트 삭제 - app/saju/page.tsx: fromHistory 파라미터 제거 - types/lunar-javascript.d.ts: 타입 선언 파일 추가 검증 케이스(1992-12-23 16:30 남성): 壬申/壬子/癸酉/庚申 Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
626 lines
32 KiB
TypeScript
626 lines
32 KiB
TypeScript
import { calculateSaju } from '@/lib/saju-calculator';
|
|
import Link from 'next/link';
|
|
import { calculateDaeun, getCurrentDaeun, getDaeunDescription } from '@/lib/daeun-calculator';
|
|
import { getCurrentSolarTerm, getSolarTermName, getSolarTermMonthBranch } from '@/lib/solar-terms';
|
|
import { EARTHLY_BRANCHES_KR, FIVE_ELEMENTS_KR, FIVE_ELEMENTS } from '@/lib/saju-calculator';
|
|
import { calculateElementScore, performFullAnalysis } from '@/lib/ai-interpretation';
|
|
import { createClient } from '@/lib/supabase/server';
|
|
import SajuAISection from './SajuAISection';
|
|
import SajuLottoSection from './SajuLottoSection';
|
|
|
|
interface PageProps {
|
|
searchParams: Promise<{
|
|
year: string;
|
|
month: string;
|
|
day: string;
|
|
hour?: string;
|
|
gender: 'male' | 'female';
|
|
calendarType: 'solar' | 'lunar';
|
|
originalYear?: string;
|
|
originalMonth?: string;
|
|
originalDay?: string;
|
|
isLeapMonth?: string;
|
|
}>;
|
|
}
|
|
|
|
export default async function SajuResultPage({ searchParams }: PageProps) {
|
|
const params = await searchParams;
|
|
const { year, month, day, hour, gender, calendarType, originalYear, originalMonth, originalDay, isLeapMonth } = params;
|
|
|
|
const yearNum = parseInt(year, 10);
|
|
const monthNum = parseInt(month, 10);
|
|
const dayNum = parseInt(day, 10);
|
|
const hourNum = hour ? parseInt(hour, 10) : null;
|
|
|
|
if (isNaN(yearNum) || isNaN(monthNum) || isNaN(dayNum)) {
|
|
return (
|
|
<div className="min-h-full bg-[#f0f5ff] flex items-center justify-center">
|
|
<div className="text-center py-20">
|
|
<p className="text-slate-500 text-sm mb-4">잘못된 접근입니다. 생년월일을 다시 입력해주세요.</p>
|
|
<a href="/saju/input" className="text-blue-600 underline text-sm">사주 입력하기</a>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
const inputYear = originalYear ? parseInt(originalYear) : yearNum;
|
|
const inputMonth = originalMonth ? parseInt(originalMonth) : monthNum;
|
|
const inputDay = originalDay ? parseInt(originalDay) : dayNum;
|
|
const isLunar = calendarType === 'lunar';
|
|
const isLeap = isLeapMonth === 'true';
|
|
|
|
// ── 사주팔자 계산 (TypeScript — lunar-javascript 기반 정밀 절기 계산) ──
|
|
const sajuData = calculateSaju(yearNum, monthNum, dayNum, hourNum, gender);
|
|
|
|
// 추가 분석 (신강신약, 용신, 오행균형, 세운)
|
|
const analysis = performFullAnalysis(sajuData);
|
|
const elementScores = analysis.elementScores;
|
|
|
|
// 대운
|
|
const currentYear = new Date().getFullYear();
|
|
const daeunList = calculateDaeun(
|
|
yearNum, monthNum, dayNum, gender,
|
|
sajuData.month.stem, sajuData.month.branch
|
|
);
|
|
const currentDaeun = getCurrentDaeun(daeunList, currentYear);
|
|
|
|
// 지지 상호작용 / 신살 / 공망 / 지장간
|
|
const branchInteractions = analysis.branchInteractions;
|
|
const shinsal = analysis.shinsal;
|
|
const gongmang = analysis.gongmang;
|
|
const hiddenStems = analysis.hiddenStems;
|
|
|
|
// ── 절기 정보 (표시용) ────────────────────────────────────────────────
|
|
const solarTermIndex = getCurrentSolarTerm(yearNum, monthNum, dayNum);
|
|
const solarTermName = getSolarTermName(solarTermIndex);
|
|
|
|
// ── 결제 여부 + 저장된 AI 해석 + 로또 구독 확인 ─────────────────────
|
|
let hasPaid = false;
|
|
let savedInterpretation: string | null = null;
|
|
let hasLottoSubscription = false;
|
|
try {
|
|
const supabase = await createClient();
|
|
const { data: { user } } = await supabase.auth.getUser();
|
|
if (user) {
|
|
// 사주 결제 확인 (anon client — 본인 orders는 RLS 허용 가정)
|
|
const { data: order } = await supabase
|
|
.from('orders').select('id')
|
|
.eq('user_id', user.id).eq('product_id', 'saju_detail').eq('status', 'paid')
|
|
.maybeSingle();
|
|
hasPaid = !!order;
|
|
|
|
if (hasPaid) {
|
|
const birthKey: Record<string, unknown> = { birth_year: yearNum, birth_month: monthNum, birth_day: dayNum, gender };
|
|
if (hourNum !== null) birthKey.birth_hour = hourNum;
|
|
const { data: record } = await supabase
|
|
.from('saju_records').select('interpretation')
|
|
.eq('user_id', user.id).eq('is_paid', true)
|
|
.contains('saju_data', birthKey).maybeSingle();
|
|
savedInterpretation = record?.interpretation ?? null;
|
|
}
|
|
|
|
// 로또 구독 확인 — subscriptions 테이블 (세션 클라이언트로 RLS select_own 통과)
|
|
const { data: lottoSub } = await supabase
|
|
.from('subscriptions')
|
|
.select('id')
|
|
.eq('user_id', user.id)
|
|
.eq('status', 'active')
|
|
.in('product_id', ['lotto_gold', 'lotto_platinum', 'lotto_diamond', 'lotto_annual'])
|
|
.maybeSingle();
|
|
hasLottoSubscription = !!lottoSub;
|
|
|
|
// subscriptions에서 못 찾으면 orders 테이블로 폴백 (구독 마이그레이션 전 데이터)
|
|
if (!hasLottoSubscription) {
|
|
const now = new Date().toISOString();
|
|
const thirtyOneDaysAgo = new Date(Date.now() - 31 * 24 * 60 * 60 * 1000).toISOString();
|
|
const { data: lottoOrder } = await supabase
|
|
.from('orders')
|
|
.select('id, created_at')
|
|
.eq('user_id', user.id)
|
|
.eq('status', 'paid')
|
|
.in('product_id', ['lotto_gold', 'lotto_platinum', 'lotto_diamond', 'lotto_annual'])
|
|
.gte('created_at', thirtyOneDaysAgo)
|
|
.maybeSingle();
|
|
hasLottoSubscription = !!lottoOrder;
|
|
}
|
|
}
|
|
} catch {
|
|
// 미로그인 시 무시
|
|
}
|
|
|
|
// ── 오행 색상 ──────────────────────────────────────────────────────────
|
|
const elementColors: { [k: string]: string } = {
|
|
'木': 'text-green-700', '火': 'text-red-600', '土': 'text-yellow-700',
|
|
'金': 'text-amber-600', '水': 'text-blue-700',
|
|
};
|
|
const elementBgColors: { [k: string]: string } = {
|
|
'木': 'bg-green-50 border-green-400', '火': 'bg-red-50 border-red-400',
|
|
'土': 'bg-yellow-50 border-yellow-400', '金': 'bg-amber-50 border-amber-400',
|
|
'水': 'bg-blue-50 border-blue-400',
|
|
};
|
|
|
|
// ── 띠 계산 ────────────────────────────────────────────────────────────
|
|
const zodiacAnimals = ['쥐', '소', '호랑이', '토끼', '용', '뱀', '말', '양', '원숭이', '닭', '개', '돼지'];
|
|
const zodiacIdx = (yearNum - 4) % 12;
|
|
const zodiacAnimal = zodiacAnimals[zodiacIdx >= 0 ? zodiacIdx : zodiacIdx + 12];
|
|
|
|
const engineBadge = <span className="text-[10px] bg-blue-50 border border-blue-200 text-blue-600 px-2 py-0.5 rounded-full font-semibold">TS 계산</span>;
|
|
|
|
return (
|
|
<div className="min-h-full bg-[#f0f5ff]">
|
|
{/* 헤더 */}
|
|
<div className="bg-gradient-to-br from-[#04102b] via-[#0a1f5c] to-[#04102b] px-6 py-10">
|
|
<div className="max-w-4xl mx-auto text-center">
|
|
<div className="inline-flex items-center gap-2 bg-violet-400/10 border border-violet-400/25 text-violet-300 text-xs font-semibold px-4 py-1.5 rounded-full mb-4">
|
|
<span className="w-1.5 h-1.5 rounded-full bg-amber-400" />
|
|
사주팔자 감정서
|
|
</div>
|
|
<h1 className="text-3xl font-extrabold text-white mb-2">사주팔자 분석 결과</h1>
|
|
<p className="text-blue-200/60 text-sm">전통 명리학과 AI 기술의 만남</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="px-6 py-8 max-w-4xl mx-auto">
|
|
<div className="grid lg:grid-cols-[280px_1fr] gap-6">
|
|
|
|
{/* 사이드바 */}
|
|
<aside className="lg:sticky lg:top-6 h-fit">
|
|
<div className="bg-[#04102b] rounded-2xl p-6 text-white">
|
|
<h2 className="text-base font-bold mb-5 text-center pb-4 border-b border-white/10">기본 정보</h2>
|
|
<div className="space-y-4 text-sm">
|
|
<div>
|
|
<div className="text-blue-300/60 mb-1">생년월일</div>
|
|
<div className="font-bold">
|
|
{isLunar ? (
|
|
<div>
|
|
<div>음력 {inputYear}.{inputMonth}.{inputDay}{isLeap ? ' (윤달)' : ''}</div>
|
|
<div className="text-xs text-blue-300/50 mt-0.5">양력 {yearNum}.{monthNum}.{dayNum}</div>
|
|
</div>
|
|
) : (
|
|
<div>{yearNum}.{monthNum}.{dayNum}</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
{hourNum !== null && (
|
|
<div>
|
|
<div className="text-blue-300/60 mb-1">태어난 시간</div>
|
|
<div className="font-bold">{hourNum}시</div>
|
|
</div>
|
|
)}
|
|
<div>
|
|
<div className="text-blue-300/60 mb-1">성별</div>
|
|
<div className="font-bold">{gender === 'male' ? '남성' : '여성'}</div>
|
|
</div>
|
|
<div>
|
|
<div className="text-blue-300/60 mb-1">띠</div>
|
|
<div className="font-bold">{zodiacAnimal}띠</div>
|
|
</div>
|
|
<div>
|
|
<div className="text-blue-300/60 mb-1">태어난 절기</div>
|
|
<div className="font-bold text-amber-300">{solarTermName} 이후</div>
|
|
</div>
|
|
<div>
|
|
<div className="text-blue-300/60 mb-1">일간</div>
|
|
<div className="font-bold text-2xl text-amber-400">
|
|
{sajuData.day.stem} ({sajuData.day.stemKr})
|
|
</div>
|
|
<div className="text-xs text-blue-300/60 mt-1">
|
|
{FIVE_ELEMENTS_KR[sajuData.day.element as keyof typeof FIVE_ELEMENTS_KR]}({sajuData.day.element})
|
|
</div>
|
|
</div>
|
|
<div className="flex items-center gap-2">
|
|
<div className="text-blue-300/60 text-xs">계산 엔진</div>
|
|
{engineBadge}
|
|
</div>
|
|
</div>
|
|
|
|
<div className="mt-5 pt-5 border-t border-white/10 space-y-2">
|
|
<Link href="/saju/input"
|
|
className="block w-full text-center bg-white/10 hover:bg-white/20 text-white px-4 py-2 rounded-lg transition text-sm font-medium">
|
|
다시 입력하기
|
|
</Link>
|
|
<Link href="/saju"
|
|
className="block w-full text-center bg-violet-500/20 hover:bg-violet-500/30 text-violet-300 px-4 py-2 rounded-lg transition text-sm font-medium">
|
|
서비스 소개
|
|
</Link>
|
|
</div>
|
|
</div>
|
|
</aside>
|
|
|
|
{/* 메인 */}
|
|
<main className="space-y-6">
|
|
|
|
{/* 사주팔자 표 */}
|
|
<div className="bg-white rounded-2xl border border-[#dbe8ff] p-6">
|
|
<h2 className="text-xl font-extrabold text-[#04102b] mb-5 text-center">사주팔자 (四柱八字)</h2>
|
|
|
|
<div className="overflow-x-auto">
|
|
<table className="w-full border-collapse text-sm">
|
|
<thead>
|
|
<tr className="bg-[#04102b] text-white">
|
|
<th className="py-2.5 px-3 text-center font-bold text-xs">구분</th>
|
|
{sajuData.hour && <th className="py-2.5 px-3 text-center font-bold text-xs">시주</th>}
|
|
<th className="py-2.5 px-3 text-center font-bold text-xs">일주</th>
|
|
<th className="py-2.5 px-3 text-center font-bold text-xs">월주</th>
|
|
<th className="py-2.5 px-3 text-center font-bold text-xs">년주</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{/* 천간 */}
|
|
<tr className="border-b border-slate-100">
|
|
<td className="py-2.5 px-3 text-center font-semibold text-[#04102b] bg-[#f0f5ff] text-xs">천간</td>
|
|
{sajuData.hour && (
|
|
<td className="py-2.5 px-3 text-center">
|
|
<div className="text-xl font-bold text-[#04102b]">{sajuData.hour.stem}</div>
|
|
<div className="text-xs text-slate-500 mt-0.5">{sajuData.hour.stemKr}</div>
|
|
</td>
|
|
)}
|
|
<td className="py-2.5 px-3 text-center bg-amber-50">
|
|
<div className="text-xl font-bold text-[#04102b]">{sajuData.day.stem}</div>
|
|
<div className="text-xs text-slate-500 mt-0.5">{sajuData.day.stemKr}</div>
|
|
<div className="text-xs text-amber-600 font-bold mt-0.5">일간</div>
|
|
</td>
|
|
<td className="py-2.5 px-3 text-center">
|
|
<div className="text-xl font-bold text-[#04102b]">{sajuData.month.stem}</div>
|
|
<div className="text-xs text-slate-500 mt-0.5">{sajuData.month.stemKr}</div>
|
|
</td>
|
|
<td className="py-2.5 px-3 text-center">
|
|
<div className="text-xl font-bold text-[#04102b]">{sajuData.year.stem}</div>
|
|
<div className="text-xs text-slate-500 mt-0.5">{sajuData.year.stemKr}</div>
|
|
</td>
|
|
</tr>
|
|
|
|
{/* 지지 */}
|
|
<tr className="border-b border-slate-100">
|
|
<td className="py-2.5 px-3 text-center font-semibold text-[#04102b] bg-[#f0f5ff] text-xs">지지</td>
|
|
{sajuData.hour && (
|
|
<td className="py-2.5 px-3 text-center">
|
|
<div className="text-xl font-bold text-[#04102b]">{sajuData.hour.branch}</div>
|
|
<div className="text-xs text-slate-500 mt-0.5">{sajuData.hour.branchKr}</div>
|
|
</td>
|
|
)}
|
|
<td className="py-2.5 px-3 text-center bg-amber-50">
|
|
<div className="text-xl font-bold text-[#04102b]">{sajuData.day.branch}</div>
|
|
<div className="text-xs text-slate-500 mt-0.5">{sajuData.day.branchKr}</div>
|
|
</td>
|
|
<td className="py-2.5 px-3 text-center">
|
|
<div className="text-xl font-bold text-[#04102b]">{sajuData.month.branch}</div>
|
|
<div className="text-xs text-slate-500 mt-0.5">{sajuData.month.branchKr}</div>
|
|
</td>
|
|
<td className="py-2.5 px-3 text-center">
|
|
<div className="text-xl font-bold text-[#04102b]">{sajuData.year.branch}</div>
|
|
<div className="text-xs text-slate-500 mt-0.5">{sajuData.year.branchKr}</div>
|
|
</td>
|
|
</tr>
|
|
|
|
{/* 지장간 */}
|
|
<tr className="border-b border-slate-100">
|
|
<td className="py-2.5 px-3 text-center font-semibold text-[#04102b] bg-[#f0f5ff] text-xs">
|
|
<div>지장간</div>
|
|
<div className="text-[10px] text-slate-400 font-normal">숨은 천간</div>
|
|
</td>
|
|
{(() => {
|
|
const order = sajuData.hour
|
|
? ['시주', '일주', '월주', '년주']
|
|
: ['일주', '월주', '년주'];
|
|
return order.map((pillarName, idx) => {
|
|
const h = hiddenStems.find((hs: any) => hs.pillar === pillarName);
|
|
return (
|
|
<td key={idx} className={`py-2 px-2 text-center ${pillarName === '일주' ? 'bg-amber-50' : ''}`}>
|
|
{h && (
|
|
<div className="flex flex-wrap justify-center gap-1">
|
|
{h.stems.map((s: any, si: number) => (
|
|
<span key={si}
|
|
className={`inline-block px-1.5 py-0.5 rounded text-xs font-semibold border ${elementBgColors[s.element] || 'bg-gray-100'}`}
|
|
title={s.role}>
|
|
{s.stemKr}
|
|
</span>
|
|
))}
|
|
</div>
|
|
)}
|
|
</td>
|
|
);
|
|
});
|
|
})()}
|
|
</tr>
|
|
|
|
{/* 십성 */}
|
|
<tr className="border-b border-slate-100">
|
|
<td className="py-2.5 px-3 text-center font-semibold text-[#04102b] bg-[#f0f5ff] text-xs">십성</td>
|
|
{sajuData.hour && (
|
|
<td className="py-2.5 px-3 text-center">
|
|
<div className="text-xs font-bold text-[#04102b]">{sajuData.hour.tenGod}</div>
|
|
</td>
|
|
)}
|
|
<td className="py-2.5 px-3 text-center bg-amber-50">
|
|
<div className="text-xs font-bold text-[#04102b]">{sajuData.day.tenGod}</div>
|
|
</td>
|
|
<td className="py-2.5 px-3 text-center">
|
|
<div className="text-xs font-bold text-[#04102b]">{sajuData.month.tenGod}</div>
|
|
</td>
|
|
<td className="py-2.5 px-3 text-center">
|
|
<div className="text-xs font-bold text-[#04102b]">{sajuData.year.tenGod}</div>
|
|
</td>
|
|
</tr>
|
|
|
|
{/* 십이운성 */}
|
|
<tr>
|
|
<td className="py-2.5 px-3 text-center font-semibold text-[#04102b] bg-[#f0f5ff] text-xs">십이운성</td>
|
|
{sajuData.hour && (
|
|
<td className="py-2.5 px-3 text-center">
|
|
<div className="text-xs font-bold text-[#04102b]">{sajuData.hour.fortune}</div>
|
|
</td>
|
|
)}
|
|
<td className="py-2.5 px-3 text-center bg-amber-50">
|
|
<div className="text-xs font-bold text-[#04102b]">{sajuData.day.fortune}</div>
|
|
</td>
|
|
<td className="py-2.5 px-3 text-center">
|
|
<div className="text-xs font-bold text-[#04102b]">{sajuData.month.fortune}</div>
|
|
</td>
|
|
<td className="py-2.5 px-3 text-center">
|
|
<div className="text-xs font-bold text-[#04102b]">{sajuData.year.fortune}</div>
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
|
|
{/* 지지 상호작용 */}
|
|
{branchInteractions.length > 0 && (
|
|
<div className="mt-5 pt-5 border-t border-slate-100">
|
|
<h3 className="text-sm font-bold text-[#04102b] mb-3 text-center">지지 상호작용</h3>
|
|
<div className="flex flex-wrap justify-center gap-2">
|
|
{branchInteractions.map((inter: any, idx: number) => {
|
|
const isPositive = inter.type.includes('합');
|
|
const isNegative = inter.type.includes('충') || inter.type.includes('형');
|
|
const colorClass = isPositive
|
|
? 'bg-emerald-50 border-emerald-400 text-emerald-800'
|
|
: isNegative
|
|
? 'bg-red-50 border-red-400 text-red-800'
|
|
: 'bg-amber-50 border-amber-400 text-amber-800';
|
|
return (
|
|
<span key={idx} className={`inline-flex items-center px-3 py-1 rounded-full text-xs font-bold border ${colorClass}`}
|
|
title={inter.description}>
|
|
{inter.type} {inter.branchesKr.join('')}
|
|
{inter.resultElement && ` → ${FIVE_ELEMENTS_KR[inter.resultElement as keyof typeof FIVE_ELEMENTS_KR]}`}
|
|
</span>
|
|
);
|
|
})}
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
{/* 오행 균형 */}
|
|
<div className="mt-5 pt-5 border-t border-slate-100">
|
|
<h3 className="text-sm font-bold text-[#04102b] mb-4 text-center">오행 균형</h3>
|
|
<div className="grid grid-cols-5 gap-2">
|
|
{Object.entries(elementScores).map(([element, score]) => (
|
|
<div key={element} className="text-center">
|
|
<div className={`text-lg font-bold mb-1 ${elementColors[element] || ''}`}>{element}</div>
|
|
<div className="text-xs text-slate-500 mb-2">
|
|
{FIVE_ELEMENTS_KR[element as keyof typeof FIVE_ELEMENTS_KR]}
|
|
</div>
|
|
<div className="w-full bg-slate-200 rounded-full h-1.5 mb-1">
|
|
<div
|
|
className={`h-1.5 rounded-full transition-all ${element === sajuData.day.element
|
|
? 'bg-gradient-to-r from-[#1a56db] to-[#7c3aed]'
|
|
: 'bg-slate-400'
|
|
}`}
|
|
style={{ width: `${Math.max(score, 5)}%` }}
|
|
/>
|
|
</div>
|
|
<div className="text-xs font-bold text-[#04102b]">{score}%</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{/* 분석 카드 그리드 */}
|
|
<div className="grid md:grid-cols-2 gap-6">
|
|
|
|
{/* 신강/신약 + 용신 */}
|
|
<div className="bg-white rounded-2xl border border-[#dbe8ff] p-6">
|
|
<h3 className="text-base font-extrabold text-[#04102b] mb-4">일간 세력 분석</h3>
|
|
<div className="flex items-center gap-3 mb-4">
|
|
<span className={`inline-block px-4 py-1.5 rounded-xl text-sm font-bold ${
|
|
analysis.dayMasterStrength.result === '신강'
|
|
? 'bg-red-100 text-red-700 border-2 border-red-400'
|
|
: analysis.dayMasterStrength.result === '신약'
|
|
? 'bg-blue-100 text-blue-700 border-2 border-blue-400'
|
|
: 'bg-green-100 text-green-700 border-2 border-green-400'
|
|
}`}>
|
|
{analysis.dayMasterStrength.result}
|
|
</span>
|
|
<span className="text-slate-500 text-xs">점수: {analysis.dayMasterStrength.score}</span>
|
|
</div>
|
|
<ul className="space-y-1 text-xs text-slate-500 mb-5">
|
|
{analysis.dayMasterStrength.reasons.map((r: string, i: number) => (
|
|
<li key={i} className="flex items-start">
|
|
<span className="text-amber-500 mr-1.5">-</span>
|
|
<span>{r}</span>
|
|
</li>
|
|
))}
|
|
</ul>
|
|
|
|
<div className="border-t border-slate-100 pt-4">
|
|
<h4 className="font-bold text-[#04102b] mb-2.5 text-sm">용신 / 희신 / 기신</h4>
|
|
<div className="flex flex-wrap gap-2 mb-3">
|
|
<span className={`px-2.5 py-1 rounded-lg text-xs font-bold border ${elementBgColors[analysis.yongShin.yongShin] || 'bg-gray-100'}`}>
|
|
용신: {analysis.yongShin.yongShinKr}
|
|
</span>
|
|
<span className={`px-2.5 py-1 rounded-lg text-xs font-bold border ${elementBgColors[analysis.yongShin.heeShin] || 'bg-gray-100'}`}>
|
|
희신: {analysis.yongShin.heeShinKr}
|
|
</span>
|
|
<span className="px-2.5 py-1 rounded-lg text-xs font-bold bg-slate-100 border border-slate-300 text-slate-700">
|
|
기신: {analysis.yongShin.giShinKr}
|
|
</span>
|
|
</div>
|
|
<p className="text-xs text-slate-500 leading-relaxed">{analysis.yongShin.explanation}</p>
|
|
</div>
|
|
</div>
|
|
|
|
{/* 신살 + 공망 */}
|
|
<div className="bg-white rounded-2xl border border-[#dbe8ff] p-6">
|
|
<h3 className="text-base font-extrabold text-[#04102b] mb-4">신살 (神煞)</h3>
|
|
{shinsal.length > 0 ? (
|
|
<div className="space-y-2 mb-5">
|
|
{shinsal.map((s: any, i: number) => (
|
|
<div key={i} className="flex items-start gap-2 p-3 rounded-xl bg-[#f0f5ff]">
|
|
<span className="inline-block px-2 py-0.5 bg-[#04102b] text-white rounded-lg text-xs font-bold whitespace-nowrap">
|
|
{s.name}
|
|
</span>
|
|
<div>
|
|
<div className="text-xs font-semibold text-[#04102b]">
|
|
{s.pillar} {s.branchKr}
|
|
</div>
|
|
<div className="text-xs text-slate-500 mt-0.5">{s.description}</div>
|
|
</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
) : (
|
|
<p className="text-slate-500 text-xs mb-5">특별한 신살이 발견되지 않았습니다.</p>
|
|
)}
|
|
|
|
<div className="border-t border-slate-100 pt-4">
|
|
<h4 className="font-bold text-[#04102b] mb-2 text-sm">공망 (空亡)</h4>
|
|
<div className="flex gap-2 mb-2">
|
|
{gongmang.branchesKr.map((bk: string, i: number) => (
|
|
<span key={i} className="px-2.5 py-1 bg-[#04102b] text-white rounded-lg text-xs font-bold">
|
|
{bk}
|
|
</span>
|
|
))}
|
|
</div>
|
|
<p className="text-xs text-slate-500 leading-relaxed">{gongmang.description}</p>
|
|
</div>
|
|
|
|
{/* 세운 정보 */}
|
|
<div className="border-t border-slate-100 pt-4 mt-4">
|
|
<h4 className="font-bold text-[#04102b] mb-2 text-sm">
|
|
{analysis.seun.year}년 세운
|
|
</h4>
|
|
<div className="flex items-center gap-2 mb-2">
|
|
<span className={`px-2.5 py-1 rounded-lg text-xs font-bold border ${elementBgColors[analysis.seun.element] || 'bg-gray-100'}`}>
|
|
{analysis.seun.stemKr}{analysis.seun.branchKr}
|
|
</span>
|
|
<span className="text-xs text-slate-500">{analysis.seun.elementKr} 기운</span>
|
|
</div>
|
|
{analysis.seun.interactions.length > 0 && (
|
|
<div className="flex flex-wrap gap-1.5 mt-2">
|
|
{analysis.seun.interactions.map((si: any, i: number) => (
|
|
<span key={i} className={`text-xs px-2 py-0.5 rounded-full font-semibold ${
|
|
si.type.includes('합') ? 'bg-emerald-50 text-emerald-700' : 'bg-red-50 text-red-700'
|
|
}`}>
|
|
{si.type} {si.branchesKr.join('')}
|
|
</span>
|
|
))}
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{/* AI 상세 해석 섹션 */}
|
|
{(() => {
|
|
const birthKey = {
|
|
birth_year: yearNum, birth_month: monthNum, birth_day: dayNum, gender,
|
|
...(hourNum !== null ? { birth_hour: hourNum } : {}),
|
|
};
|
|
const currentUrl = `/saju/result?year=${yearNum}&month=${monthNum}&day=${dayNum}${hourNum !== null ? `&hour=${hourNum}` : ''}&gender=${gender}&calendarType=${calendarType}${originalYear ? `&originalYear=${originalYear}&originalMonth=${originalMonth}&originalDay=${originalDay}` : ''}${isLeap ? '&isLeapMonth=true' : ''}`;
|
|
return (
|
|
<SajuAISection
|
|
hasPaid={hasPaid}
|
|
savedInterpretation={savedInterpretation}
|
|
sajuData={sajuData}
|
|
daeun={currentDaeun}
|
|
daeunList={daeunList}
|
|
gender={gender}
|
|
birthKey={birthKey}
|
|
currentUrl={currentUrl}
|
|
engineData={undefined}
|
|
/>
|
|
);
|
|
})()}
|
|
|
|
{/* 사주 연동 로또 번호 추천 (사주 결제 시 표시) */}
|
|
{hasPaid && (
|
|
<SajuLottoSection
|
|
yongShin={analysis.yongShin.yongShin}
|
|
yongShinKr={analysis.yongShin.yongShinKr}
|
|
heeShin={analysis.yongShin.heeShin}
|
|
heeShinKr={analysis.yongShin.heeShinKr}
|
|
dayBranch={sajuData.day.branch}
|
|
dayStemKr={sajuData.day.stemKr}
|
|
currentDaeun={currentDaeun ? {
|
|
stemKr: currentDaeun.stemKr,
|
|
branchKr: currentDaeun.branchKr,
|
|
startYear: currentDaeun.startYear,
|
|
endYear: currentDaeun.endYear,
|
|
age: currentDaeun.age,
|
|
} : null}
|
|
yearNum={yearNum}
|
|
monthNum={monthNum}
|
|
dayNum={dayNum}
|
|
hasLottoSubscription={hasLottoSubscription}
|
|
/>
|
|
)}
|
|
|
|
{/* 대운 */}
|
|
<div className="bg-white rounded-2xl border border-[#dbe8ff] p-6">
|
|
<h2 className="text-lg font-extrabold text-[#04102b] mb-5 text-center">
|
|
대운 (大運) — 10년 주기 운세
|
|
</h2>
|
|
|
|
{currentDaeun && (
|
|
<div className="bg-gradient-to-r from-[#04102b] to-[#0a2060] rounded-2xl p-5 mb-5 text-white">
|
|
<h3 className="text-sm font-bold mb-3 text-center text-blue-300">현재 대운</h3>
|
|
<div className="text-center mb-3">
|
|
<div className="text-3xl font-bold mb-1">
|
|
{currentDaeun.stem}{currentDaeun.branch}
|
|
</div>
|
|
<div className="text-base text-blue-200">
|
|
{currentDaeun.stemKr}{currentDaeun.branchKr}
|
|
</div>
|
|
<div className="text-xs text-blue-300/70 mt-1">
|
|
{currentDaeun.age}세 ~ {currentDaeun.age + 9}세 ({currentDaeun.startYear} ~ {currentDaeun.endYear}년)
|
|
</div>
|
|
</div>
|
|
<p className="text-center leading-relaxed text-xs text-blue-200/80">
|
|
{getDaeunDescription(currentDaeun, sajuData.day.stem)}
|
|
</p>
|
|
</div>
|
|
)}
|
|
|
|
<div className="grid grid-cols-2 md:grid-cols-4 gap-3">
|
|
{daeunList.map((daeun: any, index: number) => {
|
|
const isCurrent = currentDaeun &&
|
|
daeun.startYear === currentDaeun.startYear &&
|
|
daeun.endYear === currentDaeun.endYear;
|
|
return (
|
|
<div key={index}
|
|
className={`rounded-xl p-3 border-2 transition ${isCurrent ? 'bg-amber-50 border-amber-400' : 'bg-white border-[#dbe8ff]'}`}>
|
|
<div className="text-center">
|
|
<div className="text-xl font-bold text-[#04102b] mb-0.5">{daeun.stem}{daeun.branch}</div>
|
|
<div className="text-xs text-slate-500 mb-1.5">{daeun.stemKr}{daeun.branchKr}</div>
|
|
<div className="text-xs text-slate-400">{daeun.age}세 ~ {daeun.age + 9}세</div>
|
|
<div className="text-xs text-slate-400">{daeun.startYear} ~ {daeun.endYear}</div>
|
|
{isCurrent && (
|
|
<div className="mt-1.5">
|
|
<span className="inline-block bg-[#04102b] text-white text-xs px-2.5 py-0.5 rounded-full font-semibold">현재</span>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
</div>
|
|
|
|
</main>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|