mirror of
https://github.com/aljazceru/goose.git
synced 2025-12-19 07:04:21 +01:00
Co-authored-by: Michael Neale <michael.neale@gmail.com> Co-authored-by: Wendy Tang <wendytang@squareup.com> Co-authored-by: Jarrod Sibbison <72240382+jsibbison-square@users.noreply.github.com> Co-authored-by: Alex Hancock <alex.hancock@example.com> Co-authored-by: Alex Hancock <alexhancock@block.xyz> Co-authored-by: Lifei Zhou <lifei@squareup.com> Co-authored-by: Wes <141185334+wesrblock@users.noreply.github.com> Co-authored-by: Max Novich <maksymstepanenko1990@gmail.com> Co-authored-by: Zaki Ali <zaki@squareup.com> Co-authored-by: Salman Mohammed <smohammed@squareup.com> Co-authored-by: Kalvin C <kalvinnchau@users.noreply.github.com> Co-authored-by: Alec Thomas <alec@swapoff.org> Co-authored-by: lily-de <119957291+lily-de@users.noreply.github.com> Co-authored-by: kalvinnchau <kalvin@block.xyz> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: Rizel Scarlett <rizel@squareup.com> Co-authored-by: bwrage <bwrage@squareup.com> Co-authored-by: Kalvin Chau <kalvin@squareup.com> Co-authored-by: Alice Hau <110418948+ahau-square@users.noreply.github.com> Co-authored-by: Alistair Gray <ajgray@stripe.com> Co-authored-by: Nahiyan Khan <nahiyan.khan@gmail.com> Co-authored-by: Alex Hancock <alexhancock@squareup.com> Co-authored-by: Nahiyan Khan <nahiyan@squareup.com> Co-authored-by: marcelle <1852848+laanak08@users.noreply.github.com> Co-authored-by: Yingjie He <yingjiehe@block.xyz> Co-authored-by: Yingjie He <yingjiehe@squareup.com> Co-authored-by: Lily Delalande <ldelalande@block.xyz> Co-authored-by: Adewale Abati <acekyd01@gmail.com> Co-authored-by: Ebony Louis <ebony774@gmail.com> Co-authored-by: Angie Jones <jones.angie@gmail.com> Co-authored-by: Ebony Louis <55366651+EbonyLouis@users.noreply.github.com>
112 lines
3.6 KiB
TypeScript
112 lines
3.6 KiB
TypeScript
import { Search } from "lucide-react";
|
|
import { Button } from "../components/ui/button";
|
|
import { Input } from "../components/ui/input";
|
|
import { ServerCard } from "../components/server-card";
|
|
import { useState, useEffect } from "react";
|
|
import type { MCPServer } from "../types/server";
|
|
import { fetchMCPServers, searchMCPServers } from "../mcp-servers";
|
|
import { motion, AnimatePresence } from "framer-motion";
|
|
|
|
export default function HomePage() {
|
|
const [servers, setServers] = useState<MCPServer[]>([]);
|
|
const [searchQuery, setSearchQuery] = useState("");
|
|
const [isLoading, setIsLoading] = useState(true);
|
|
const [error, setError] = useState<string | null>(null);
|
|
|
|
// Combined effect for initial load and search
|
|
useEffect(() => {
|
|
const loadServers = async () => {
|
|
try {
|
|
setIsLoading(true);
|
|
setError(null);
|
|
|
|
const trimmedQuery = searchQuery.trim();
|
|
const results = trimmedQuery
|
|
? await searchMCPServers(trimmedQuery)
|
|
: await fetchMCPServers();
|
|
|
|
setServers(results);
|
|
} catch (err) {
|
|
const errorMessage =
|
|
err instanceof Error ? err.message : "Unknown error";
|
|
setError(`Failed to load servers: ${errorMessage}`);
|
|
console.error("Error loading servers:", err);
|
|
} finally {
|
|
setIsLoading(false);
|
|
}
|
|
};
|
|
|
|
// Debounce all server loads
|
|
const timeoutId = setTimeout(loadServers, 300);
|
|
return () => clearTimeout(timeoutId);
|
|
}, [searchQuery]);
|
|
|
|
return (
|
|
<div className="pb-24">
|
|
<div className="pb-16">
|
|
<h1 className="text-[64px] font-medium text-textProminent">
|
|
Browse Extensions
|
|
</h1>
|
|
<p className="text-textProminent">
|
|
Your central directory for discovering and installing extensions.
|
|
</p>
|
|
</div>
|
|
|
|
<div className="relative">
|
|
{/* <Search className="absolute left-3 top-3 h-4 w-4 text-gray-500" /> */}
|
|
<Input
|
|
className="pl-0"
|
|
placeholder="Search for extensions"
|
|
value={searchQuery}
|
|
onChange={(e) => setSearchQuery(e.target.value)}
|
|
/>
|
|
</div>
|
|
|
|
{error && (
|
|
<div className="p-4 bg-red-50 text-red-600 rounded-md">{error}</div>
|
|
)}
|
|
|
|
<section className="pt-8">
|
|
<div className={`${searchQuery ? "pb-2" : "pb-8"}`}>
|
|
{/* <h2 className="text-xl">{searchQuery ? "Search Results" : ""}</h2> */}
|
|
<p className="text-textSubtle">
|
|
{searchQuery
|
|
? `${servers.length} result${
|
|
servers.length > 1 ? "s" : ""
|
|
} for "${searchQuery}"`
|
|
: ""}
|
|
</p>
|
|
</div>
|
|
|
|
{isLoading ? (
|
|
<div className="py-8 text-xl text-textSubtle">Loading servers...</div>
|
|
) : servers.length === 0 ? (
|
|
<div className="text-center py-8 text-gray-500">
|
|
{searchQuery
|
|
? "No servers found matching your search."
|
|
: "No servers available."}
|
|
</div>
|
|
) : (
|
|
<div className="grid gap-6 md:grid-cols-2 lg:grid-cols-3">
|
|
{/* <AnimatePresence> */}
|
|
{servers.map((server, index) => (
|
|
<motion.div
|
|
key={server.id}
|
|
initial={{
|
|
opacity: 0,
|
|
}}
|
|
animate={{ opacity: 1 }}
|
|
exit={{ opacity: 0 }}
|
|
transition={{ duration: 0.6 }}
|
|
>
|
|
<ServerCard key={server.id} server={server} />
|
|
</motion.div>
|
|
))}
|
|
{/* </AnimatePresence> */}
|
|
</div>
|
|
)}
|
|
</section>
|
|
</div>
|
|
);
|
|
}
|