332 lines
12 KiB
TypeScript
332 lines
12 KiB
TypeScript
'use client'
|
|
|
|
import { useState, useEffect, useMemo, useCallback } from 'react'
|
|
import { createPortal } from 'react-dom'
|
|
|
|
interface WordAnalysis {
|
|
word: string
|
|
translation: string
|
|
definition: string
|
|
partOfSpeech: string
|
|
pronunciation: string
|
|
synonyms: string[]
|
|
antonyms?: string[]
|
|
isPhrase: boolean
|
|
isHighValue?: boolean
|
|
learningPriority?: 'high' | 'medium' | 'low'
|
|
phraseInfo?: {
|
|
phrase: string
|
|
meaning: string
|
|
warning: string
|
|
colorCode: string
|
|
}
|
|
difficultyLevel: string
|
|
costIncurred?: number
|
|
example?: string
|
|
exampleTranslation?: string
|
|
}
|
|
|
|
interface ClickableTextProps {
|
|
text: string
|
|
analysis?: Record<string, WordAnalysis>
|
|
onWordClick?: (word: string, analysis: WordAnalysis) => void
|
|
onSaveWord?: (word: string, analysis: WordAnalysis) => Promise<{ success: boolean; error?: string }>
|
|
remainingUsage?: number
|
|
showPhrasesInline?: boolean
|
|
}
|
|
|
|
const POPUP_CONFIG = {
|
|
WIDTH: 320,
|
|
HEIGHT: 400,
|
|
PADDING: 16,
|
|
MOBILE_BREAKPOINT: 640
|
|
} as const
|
|
|
|
export function ClickableTextV2({
|
|
text,
|
|
analysis,
|
|
onWordClick,
|
|
onSaveWord,
|
|
remainingUsage = 5,
|
|
showPhrasesInline = true
|
|
}: ClickableTextProps) {
|
|
const [selectedWord, setSelectedWord] = useState<string | null>(null)
|
|
const [popupPosition, setPopupPosition] = useState({ x: 0, y: 0, showBelow: false })
|
|
const [isSavingWord, setIsSavingWord] = useState(false)
|
|
const [mounted, setMounted] = useState(false)
|
|
|
|
useEffect(() => {
|
|
setMounted(true)
|
|
}, [])
|
|
|
|
const getCEFRColor = useCallback((level: string) => {
|
|
switch (level) {
|
|
case 'A1': return 'bg-green-100 text-green-700 border-green-200'
|
|
case 'A2': return 'bg-blue-100 text-blue-700 border-blue-200'
|
|
case 'B1': return 'bg-yellow-100 text-yellow-700 border-yellow-200'
|
|
case 'B2': return 'bg-orange-100 text-orange-700 border-orange-200'
|
|
case 'C1': return 'bg-red-100 text-red-700 border-red-200'
|
|
case 'C2': return 'bg-purple-100 text-purple-700 border-purple-200'
|
|
default: return 'bg-gray-100 text-gray-700 border-gray-200'
|
|
}
|
|
}, [])
|
|
|
|
const getWordProperty = useCallback((wordData: any, propName: string) => {
|
|
if (!wordData) return undefined;
|
|
|
|
const variations = [
|
|
propName,
|
|
propName.toLowerCase(),
|
|
propName.charAt(0).toUpperCase() + propName.slice(1),
|
|
propName.charAt(0).toLowerCase() + propName.slice(1)
|
|
];
|
|
|
|
for (const variation of variations) {
|
|
if (wordData[variation] !== undefined) {
|
|
return wordData[variation];
|
|
}
|
|
}
|
|
|
|
if (propName === 'synonyms') {
|
|
return [];
|
|
}
|
|
|
|
return undefined;
|
|
}, [])
|
|
|
|
const findWordAnalysis = useCallback((word: string) => {
|
|
const cleanWord = word.toLowerCase().replace(/[.,!?;:]/g, '')
|
|
return analysis?.[cleanWord] || analysis?.[word] || analysis?.[word.toLowerCase()] || null
|
|
}, [analysis])
|
|
|
|
const getLevelIndex = useCallback((level: string): number => {
|
|
const levels = ['A1', 'A2', 'B1', 'B2', 'C1', 'C2']
|
|
return levels.indexOf(level)
|
|
}, [])
|
|
|
|
const getWordClass = useCallback((word: string) => {
|
|
const wordAnalysis = findWordAnalysis(word)
|
|
const baseClass = "cursor-pointer transition-all duration-200 rounded relative mx-0.5 px-1 py-0.5"
|
|
|
|
if (!wordAnalysis) return ""
|
|
|
|
const isPhrase = getWordProperty(wordAnalysis, 'isPhrase')
|
|
if (isPhrase) return ""
|
|
|
|
const difficultyLevel = getWordProperty(wordAnalysis, 'difficultyLevel') || 'A1'
|
|
const userLevel = typeof window !== 'undefined' ? localStorage.getItem('userEnglishLevel') || 'A2' : 'A2'
|
|
|
|
const userIndex = getLevelIndex(userLevel)
|
|
const wordIndex = getLevelIndex(difficultyLevel)
|
|
|
|
if (userIndex > wordIndex) {
|
|
return `${baseClass} bg-gray-50 border border-dashed border-gray-300 hover:bg-gray-100 hover:border-gray-400 text-gray-600 opacity-80`
|
|
} else if (userIndex === wordIndex) {
|
|
return `${baseClass} bg-green-50 border border-green-200 hover:bg-green-100 hover:shadow-lg transform hover:-translate-y-0.5 text-green-700 font-medium`
|
|
} else {
|
|
return `${baseClass} bg-orange-50 border border-orange-200 hover:bg-orange-100 hover:shadow-lg transform hover:-translate-y-0.5 text-orange-700 font-medium`
|
|
}
|
|
}, [findWordAnalysis, getWordProperty, getLevelIndex])
|
|
|
|
const getWordIcon = (word: string) => {
|
|
// 移除所有圖標,保持簡潔設計
|
|
return null
|
|
}
|
|
|
|
const words = useMemo(() => text.split(/(\s+|[.,!?;:])/g), [text])
|
|
|
|
const calculatePopupPosition = useCallback((rect: DOMRect) => {
|
|
const popupWidth = 320 // w-80 = 320px
|
|
const popupHeight = 400 // estimated popup height
|
|
const margin = 16
|
|
|
|
const viewportWidth = window.innerWidth
|
|
const viewportHeight = window.innerHeight
|
|
|
|
let x = rect.left + rect.width / 2
|
|
let y = rect.bottom + 10
|
|
let showBelow = true
|
|
|
|
// Check if popup would go off right edge
|
|
if (x + popupWidth / 2 > viewportWidth - margin) {
|
|
x = viewportWidth - popupWidth / 2 - margin
|
|
}
|
|
|
|
// Check if popup would go off left edge
|
|
if (x - popupWidth / 2 < margin) {
|
|
x = popupWidth / 2 + margin
|
|
}
|
|
|
|
// Check if popup would go off bottom edge
|
|
if (y + popupHeight > viewportHeight - margin) {
|
|
y = rect.top - 10
|
|
showBelow = false
|
|
}
|
|
|
|
// Check if popup would go off top edge (when showing above)
|
|
if (!showBelow && y - popupHeight < margin) {
|
|
y = rect.bottom + 10
|
|
showBelow = true
|
|
}
|
|
|
|
return { x, y, showBelow }
|
|
}, [])
|
|
|
|
const handleWordClick = useCallback(async (word: string, event: React.MouseEvent) => {
|
|
const cleanWord = word.toLowerCase().replace(/[.,!?;:]/g, '')
|
|
const wordAnalysis = findWordAnalysis(word)
|
|
|
|
if (!wordAnalysis) return
|
|
|
|
const rect = event.currentTarget.getBoundingClientRect()
|
|
const position = calculatePopupPosition(rect)
|
|
|
|
setPopupPosition(position)
|
|
setSelectedWord(cleanWord)
|
|
onWordClick?.(cleanWord, wordAnalysis)
|
|
}, [findWordAnalysis, onWordClick, calculatePopupPosition])
|
|
|
|
const closePopup = useCallback(() => {
|
|
setSelectedWord(null)
|
|
}, [])
|
|
|
|
const handleSaveWord = useCallback(async () => {
|
|
if (!selectedWord || !analysis?.[selectedWord] || !onSaveWord) return
|
|
|
|
setIsSavingWord(true)
|
|
try {
|
|
const result = await onSaveWord(selectedWord, analysis[selectedWord])
|
|
if (result?.success) {
|
|
setSelectedWord(null)
|
|
} else {
|
|
console.error('Save word error:', result?.error || '保存失敗')
|
|
}
|
|
} catch (error) {
|
|
console.error('Save word error:', error)
|
|
} finally {
|
|
setIsSavingWord(false)
|
|
}
|
|
}, [selectedWord, analysis, onSaveWord])
|
|
|
|
const VocabPopup = () => {
|
|
if (!selectedWord || !analysis?.[selectedWord] || !mounted) return null
|
|
|
|
return createPortal(
|
|
<>
|
|
<div
|
|
className="fixed inset-0 bg-black bg-opacity-50 z-40"
|
|
onClick={closePopup}
|
|
/>
|
|
|
|
<div
|
|
className="fixed z-50 bg-white rounded-xl shadow-lg w-80 sm:w-96 max-w-[90vw] sm:max-w-md overflow-hidden"
|
|
style={{
|
|
left: `${popupPosition.x}px`,
|
|
top: `${popupPosition.y}px`,
|
|
transform: popupPosition.showBelow ? 'translate(-50%, 8px)' : 'translate(-50%, -100%)',
|
|
maxHeight: '85vh',
|
|
overflowY: 'auto'
|
|
}}
|
|
>
|
|
<div className="bg-gradient-to-br from-blue-50 to-indigo-50 p-4 sm:p-5 border-b border-blue-200">
|
|
<div className="flex justify-end mb-3">
|
|
<button
|
|
onClick={closePopup}
|
|
className="text-gray-400 hover:text-gray-600 w-6 h-6 rounded-full bg-white bg-opacity-80 hover:bg-opacity-100 transition-all flex items-center justify-center"
|
|
>
|
|
✕
|
|
</button>
|
|
</div>
|
|
|
|
<div className="mb-3">
|
|
<h3 className="text-xl sm:text-2xl font-bold text-gray-900 break-words">{getWordProperty(analysis[selectedWord], 'word')}</h3>
|
|
</div>
|
|
|
|
<div className="flex items-center justify-between">
|
|
<div className="flex flex-col sm:flex-row sm:items-center gap-2 sm:gap-3">
|
|
<span className="text-xs sm:text-sm bg-gray-100 text-gray-700 px-2 sm:px-3 py-1 rounded-full w-fit">
|
|
{getWordProperty(analysis[selectedWord], 'partOfSpeech')}
|
|
</span>
|
|
<span className="text-sm sm:text-base text-gray-600 break-all">{getWordProperty(analysis[selectedWord], 'pronunciation')}</span>
|
|
</div>
|
|
|
|
<span className={`px-3 py-1 rounded-full text-sm font-medium border ${getCEFRColor(getWordProperty(analysis[selectedWord], 'difficultyLevel'))}`}>
|
|
{getWordProperty(analysis[selectedWord], 'difficultyLevel')}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="p-3 sm:p-4 space-y-3 sm:space-y-4">
|
|
<div className="bg-green-50 rounded-lg p-3 border border-green-200">
|
|
<h4 className="font-semibold text-green-900 mb-2 text-left text-sm">中文翻譯</h4>
|
|
<p className="text-green-800 font-medium text-left">{getWordProperty(analysis[selectedWord], 'translation')}</p>
|
|
</div>
|
|
|
|
<div className="bg-gray-50 rounded-lg p-3 border border-gray-200">
|
|
<h4 className="font-semibold text-gray-900 mb-2 text-left text-sm">英文定義</h4>
|
|
<p className="text-gray-700 text-left text-sm leading-relaxed">{getWordProperty(analysis[selectedWord], 'definition')}</p>
|
|
</div>
|
|
|
|
{(() => {
|
|
const example = getWordProperty(analysis[selectedWord], 'example');
|
|
return example && example !== 'null' && example !== 'undefined';
|
|
})() && (
|
|
<div className="bg-blue-50 rounded-lg p-3 border border-blue-200">
|
|
<h4 className="font-semibold text-blue-900 mb-2 text-left text-sm">例句</h4>
|
|
<div className="space-y-2">
|
|
<p className="text-blue-800 text-left text-sm italic">
|
|
"{getWordProperty(analysis[selectedWord], 'example')}"
|
|
</p>
|
|
<p className="text-blue-700 text-left text-sm">
|
|
{getWordProperty(analysis[selectedWord], 'exampleTranslation')}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
{onSaveWord && (
|
|
<div className="p-3 sm:p-4 pt-2">
|
|
<button
|
|
onClick={handleSaveWord}
|
|
disabled={isSavingWord}
|
|
className="w-full bg-primary text-white py-3 rounded-lg font-medium hover:bg-primary-hover transition-colors flex items-center justify-center gap-2"
|
|
>
|
|
<span className="font-medium">{isSavingWord ? '保存中...' : '保存到詞卡'}</span>
|
|
</button>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</>,
|
|
document.body
|
|
)
|
|
}
|
|
|
|
return (
|
|
<div className="relative">
|
|
<div className="text-lg" style={{lineHeight: '2.5'}}>
|
|
{words.map((word, index) => {
|
|
if (word.trim() === '' || /^[.,!?;:\s]+$/.test(word)) {
|
|
return <span key={index}>{word}</span>
|
|
}
|
|
|
|
const className = getWordClass(word)
|
|
const icon = getWordIcon(word)
|
|
|
|
return (
|
|
<span
|
|
key={index}
|
|
className={className}
|
|
onClick={(e) => handleWordClick(word, e)}
|
|
>
|
|
{word}
|
|
{icon}
|
|
</span>
|
|
)
|
|
})}
|
|
</div>
|
|
|
|
<VocabPopup />
|
|
</div>
|
|
)
|
|
} |