import React from "react"; import { useNavigation } from "react-router"; import { useDispatch, useSelector } from "react-redux"; import { RootState } from "#/store"; import { addFile, removeFile } from "#/state/initial-query-slice"; import { SuggestionBubble } from "#/components/features/suggestions/suggestion-bubble"; import { SUGGESTIONS } from "#/utils/suggestions"; import { convertImageToBase64 } from "#/utils/convert-image-to-base-64"; import { ChatInput } from "#/components/features/chat/chat-input"; import { getRandomKey } from "#/utils/get-random-key"; import { cn } from "#/utils/utils"; import { AttachImageLabel } from "../features/images/attach-image-label"; import { ImageCarousel } from "../features/images/image-carousel"; import { UploadImageInput } from "../features/images/upload-image-input"; import { useCreateConversation } from "#/hooks/mutation/use-create-conversation"; import { LoadingSpinner } from "./loading-spinner"; interface TaskFormProps { ref: React.RefObject; } export function TaskForm({ ref }: TaskFormProps) { const dispatch = useDispatch(); const navigation = useNavigation(); const { files } = useSelector((state: RootState) => state.initialQuery); const [text, setText] = React.useState(""); const [suggestion, setSuggestion] = React.useState(() => { const key = getRandomKey(SUGGESTIONS["non-repo"]); return { key, value: SUGGESTIONS["non-repo"][key] }; }); const [inputIsFocused, setInputIsFocused] = React.useState(false); const { mutate: createConversation, isPending } = useCreateConversation(); const onRefreshSuggestion = () => { const suggestions = SUGGESTIONS["non-repo"]; // remove current suggestion to avoid refreshing to the same suggestion const suggestionCopy = { ...suggestions }; delete suggestionCopy[suggestion.key]; const key = getRandomKey(suggestionCopy); setSuggestion({ key, value: suggestions[key] }); }; const onClickSuggestion = () => { setText(suggestion.value); }; const handleSubmit = (event: React.FormEvent) => { event.preventDefault(); const formData = new FormData(event.currentTarget); const q = formData.get("q")?.toString(); createConversation({ q }); }; return (
{isPending ? (
) : ( { if (typeof ref !== "function") ref?.current?.requestSubmit(); }} onChange={(message) => setText(message)} onFocus={() => setInputIsFocused(true)} onBlur={() => setInputIsFocused(false)} onImagePaste={async (imageFiles) => { const promises = imageFiles.map(convertImageToBase64); const base64Images = await Promise.all(promises); base64Images.forEach((base64) => { dispatch(addFile(base64)); }); }} value={text} maxRows={15} showButton={!!text} className="text-[17px] leading-5 py-[17px]" buttonClassName="pb-[17px]" disabled={navigation.state === "submitting"} /> )}
{ const promises = uploadedFiles.map(convertImageToBase64); const base64Images = await Promise.all(promises); base64Images.forEach((base64) => { dispatch(addFile(base64)); }); }} label={} /> {files.length > 0 && ( dispatch(removeFile(index))} /> )}
); }