| import React, { useState, useMemo } from "react"; |
| import { Input } from "./ui/input"; |
| import { |
| Dialog, |
| DialogContent, |
| DialogHeader, |
| DialogTitle, |
| DialogTrigger, |
| } from "./ui/dialog"; |
| import { Button } from "./ui/button"; |
| import { fetchAllAuthorsData } from "../utils/authors"; |
| import { generateCalendarData } from "../utils/calendar"; |
| import Heatmap from "./Heatmap"; |
| import { ModelData } from "../types/heatmap"; |
| import { fetchAuthorData, fetchUserData } from "@/utils/authors"; |
|
|
| const UserSearchDialog = () => { |
| const [isOpen, setIsOpen] = useState(false); |
| const [isLoading, setIsLoading] = useState(false); |
| const [searchInput, setSearchInput] = useState(""); |
| const [searchedData, setSearchedData] = useState<ModelData[] | null>(null); |
| const [isCopied, setIsCopied] = useState(false); |
| const [currentSearchTerm, setCurrentSearchTerm] = useState(""); |
| const [userInfo, setUserInfo] = useState<{ fullName: string; avatarUrl: string | null } | null>(null); |
|
|
| const handleSearch = async () => { |
| if (searchInput.trim()) { |
| setIsLoading(true); |
| try { |
| const authorData = await fetchAllAuthorsData([searchInput.trim()]); |
| const authorInfo = await fetchUserData([searchInput.trim()]); |
| setSearchedData(authorData); |
| setUserInfo(authorInfo); |
| setCurrentSearchTerm(searchInput.trim()); |
| } catch (error) { |
| console.error("Error fetching data for searched user:", error); |
| setSearchedData(null); |
| setUserInfo(null); |
| setCurrentSearchTerm(""); |
| } |
| setIsLoading(false); |
| } else { |
| setSearchedData(null); |
| setUserInfo(null); |
| setCurrentSearchTerm(""); |
| } |
| }; |
|
|
| const handleKeyPress = (e: React.KeyboardEvent<HTMLInputElement>) => { |
| if (e.key === 'Enter') { |
| handleSearch(); |
| } |
| }; |
|
|
| const handleInputChange = (e: React.ChangeEvent<HTMLInputElement>) => { |
| setSearchInput(e.target.value); |
| }; |
|
|
| const getIframeCode = (username: string) => { |
| return `<iframe |
| src="https://cfahlgren1-model-release-heatmap.hf.space/${username}" |
| class="w-full h-[300px]" |
| frameborder="0" |
| ></iframe>`; |
| }; |
|
|
| const handleCopyCode = () => { |
| const iframeCode = getIframeCode(searchInput); |
| navigator.clipboard.writeText(iframeCode).then(() => { |
| setIsCopied(true); |
| setTimeout(() => setIsCopied(false), 2000); |
| }); |
| }; |
|
|
| const searchedHeatmapData = useMemo(() => { |
| if (searchedData && searchedData.length > 0 && userInfo) { |
| return generateCalendarData(searchedData, [{ |
| authors: [currentSearchTerm], |
| color: "#0088cc", |
| fullName: userInfo.fullName, |
| avatarUrl: userInfo.avatarUrl |
| }])[currentSearchTerm]; |
| } |
| return null; |
| }, [searchedData, currentSearchTerm, userInfo]); |
|
|
| const handleDialogOpenChange = (open: boolean) => { |
| setIsOpen(open); |
| if (!open) { |
| setSearchInput(""); |
| setSearchedData(null); |
| setCurrentSearchTerm(""); |
| setIsLoading(false); |
| setIsCopied(false); |
| setUserInfo(null); |
| } |
| }; |
|
|
| return ( |
| <Dialog open={isOpen} onOpenChange={handleDialogOpenChange}> |
| <DialogTrigger asChild> |
| <Button variant="outline">Search</Button> |
| </DialogTrigger> |
| <DialogContent className="w-full max-w-[95vw] sm:max-w-4xl p-4 sm:p-6 max-h-[90vh] flex flex-col"> |
| <DialogHeader> |
| <DialogTitle className="text-lg sm:text-xl mb-4">Get your Hugging Face Heatmap</DialogTitle> |
| </DialogHeader> |
| <div className="flex-grow overflow-y-auto"> |
| <div className="grid gap-4 py-4"> |
| <div className="flex items-center space-x-2 p-2 bg-background rounded-md"> |
| <Input |
| type="text" |
| placeholder="Enter username" |
| value={searchInput} |
| onChange={handleInputChange} |
| onKeyDown={handleKeyPress} |
| className="flex-grow" |
| /> |
| </div> |
| {isLoading ? ( |
| <p className="text-center">Loading...</p> |
| ) : searchedHeatmapData && userInfo ? ( |
| <div className="mt-4 space-y-4"> |
| <div className="overflow-x-auto pb-2"> |
| <Heatmap |
| data={searchedHeatmapData} |
| color="#FF9D00" |
| providerName={currentSearchTerm} |
| fullName={userInfo.fullName} |
| avatarUrl={userInfo.avatarUrl || ''} |
| /> |
| </div> |
| <div> |
| <div className="flex justify-between items-center mb-2"> |
| <h3 className="font-semibold text-sm sm:text-base">Embed in iFrame</h3> |
| <Button onClick={handleCopyCode} variant="link" size="sm"> |
| {isCopied ? "Copied!" : "Copy"} |
| </Button> |
| </div> |
| <div className="overflow-x-auto"> |
| <pre className="bg-secondary p-2 rounded text-xs whitespace-pre-wrap break-all"> |
| <code>{getIframeCode(searchInput)}</code> |
| </pre> |
| </div> |
| </div> |
| </div> |
| ) : searchedData !== null && searchedData.length === 0 ? ( |
| <p className="text-center text-slate-500 text-sm italic">User or Organization not found</p> |
| ) : null} |
| </div> |
| </div> |
| </DialogContent> |
| </Dialog> |
| ); |
| }; |
|
|
| export default UserSearchDialog; |