File size: 3,149 Bytes
e9a2815 65676ec e9a2815 8ec33d8 e9a2815 65676ec e9a2815 8ec33d8 6864389 e9a2815 6864389 e9a2815 8ec33d8 e9a2815 8ec33d8 e9a2815 6864389 8ec33d8 6864389 e9a2815 aeb9637 e9a2815 8ec33d8 e9a2815 6864389 e9a2815 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 |
import { KeyboardEvent, useRef, useEffect } from "react";
import { Input } from "@/components/ui/input";
import { Button } from "@/components/ui/button";
import { useTranslation } from "@/hooks/useTranslation";
interface SentenceWord {
word: string;
provider: 'player' | 'ai';
}
interface InputFormProps {
playerInput: string;
onInputChange: (value: string) => void;
onSubmitWord: (e: React.FormEvent) => void;
onMakeGuess: () => void;
isAiThinking: boolean;
hasMultipleWords: boolean;
containsTargetWord: boolean;
isTooLong: boolean;
isValidInput: boolean;
sentence: SentenceWord[];
}
export const InputForm = ({
playerInput,
onInputChange,
onSubmitWord,
onMakeGuess,
isAiThinking,
hasMultipleWords,
containsTargetWord,
isTooLong,
isValidInput,
sentence
}: InputFormProps) => {
const inputRef = useRef<HTMLInputElement>(null);
const t = useTranslation();
useEffect(() => {
if (!isAiThinking) {
setTimeout(() => {
inputRef.current?.focus();
}, 100);
}
}, [isAiThinking]);
const handleKeyDown = (e: KeyboardEvent<HTMLInputElement>) => {
if (e.shiftKey && e.key === 'Enter') {
e.preventDefault();
if (!hasMultipleWords && !containsTargetWord && !isTooLong && !isAiThinking && isValidInput) {
onMakeGuess();
}
}
};
const getInputError = () => {
if (hasMultipleWords) return t.game.singleWordOnly;
if (containsTargetWord) return t.game.cantUseTargetWord;
if (isTooLong) return t.game.shorterWord;
if (!isValidInput && playerInput) return t.game.lettersOnly;
return null;
};
const error = getInputError();
const canMakeGuess = (sentence.length > 0 || playerInput.trim().length > 0) &&
!hasMultipleWords && !containsTargetWord && !isTooLong && isValidInput && !isAiThinking;
return (
<form onSubmit={onSubmitWord} className="mb-4">
<div className="relative mb-4">
<Input
ref={inputRef}
type="text"
value={playerInput}
onChange={(e) => onInputChange(e.target.value)}
onKeyDown={handleKeyDown}
placeholder={t.game.inputPlaceholder}
className={`w-full ${error ? 'border-red-500 focus-visible:ring-red-500' : ''}`}
disabled={isAiThinking}
autoComplete="words"
/>
{error && (
<p className="text-sm text-red-500 mt-1">
{error}
</p>
)}
</div>
<div className="flex gap-4">
<Button
type="submit"
className="flex-1 bg-primary text-lg hover:bg-primary/90"
disabled={!playerInput.trim() || isAiThinking || hasMultipleWords || containsTargetWord || isTooLong || !isValidInput}
>
{isAiThinking ? t.game.aiThinking : `${t.game.addWord} ⏎`}
</Button>
<Button
type="button"
onClick={onMakeGuess}
className="flex-1 bg-secondary text-lg hover:bg-secondary/90"
disabled={!canMakeGuess}
>
{isAiThinking ? t.game.aiThinking : `${t.game.makeGuess} ⇧⏎`}
</Button>
</div>
</form>
);
}; |