Files
coco-app/src/components/AppAI/DropdownList.tsx

201 lines
6.3 KiB
TypeScript
Raw Normal View History

import { useEffect, useRef, useState } from "react";
2025-01-11 10:12:53 +08:00
import { CircleAlert, Bolt, X } from "lucide-react";
import { isTauri } from "@tauri-apps/api/core";
import { open } from "@tauri-apps/plugin-shell";
2025-01-11 10:12:53 +08:00
import { useAppStore } from "@/stores/appStore";
interface DropdownListProps {
selected: (item: any) => void;
suggests: any[];
2025-01-11 10:12:53 +08:00
IsError: boolean;
isSearchComplete: boolean;
}
2025-01-11 10:12:53 +08:00
function DropdownList({ selected, suggests, IsError }: DropdownListProps) {
const connector_data = useAppStore((state) => state.connector_data);
2025-01-17 16:17:08 +08:00
const datasourceData = useAppStore((state) => state.datasourceData);
const endpoint_http = useAppStore((state) => state.endpoint_http);
2025-01-11 10:12:53 +08:00
const [showError, setShowError] = useState<boolean>(IsError);
const [selectedItem, setSelectedItem] = useState<number | null>(null);
const [showIndex, setShowIndex] = useState<boolean>(false);
const containerRef = useRef<HTMLDivElement>(null);
const itemRefs = useRef<(HTMLDivElement | null)[]>([]);
const handleOpenURL = async (url: string) => {
if (!url) return;
try {
if (isTauri()) {
await open(url);
2025-01-17 16:17:08 +08:00
// console.log("URL opened in default browser");
}
} catch (error) {
console.error("Failed to open URL:", error);
}
};
const handleKeyDown = (e: KeyboardEvent) => {
2025-01-17 16:17:08 +08:00
// console.log(
// "handleKeyDown",
// e.key,
// showIndex,
// e.key >= "0" && e.key <= "9" && showIndex
// );
if (!suggests.length) return;
if (e.key === "ArrowUp") {
e.preventDefault();
setSelectedItem((prev) =>
prev === null || prev === 0 ? suggests.length - 1 : prev - 1
);
} else if (e.key === "ArrowDown") {
e.preventDefault();
setSelectedItem((prev) =>
prev === null || prev === suggests.length - 1 ? 0 : prev + 1
);
} else if (e.key === "Meta") {
e.preventDefault();
setShowIndex(true);
}
if (e.key === "Enter" && selectedItem !== null) {
2025-01-17 16:17:08 +08:00
// console.log("Enter key pressed", selectedItem);
const item = suggests[selectedItem];
if (item?._source?.url) {
handleOpenURL(item?._source?.url);
} else {
selected(item);
}
}
if (e.key >= "0" && e.key <= "9" && showIndex) {
2025-01-17 16:17:08 +08:00
// console.log(`number ${e.key}`);
const item = suggests[parseInt(e.key, 10)];
if (item?._source?.url) {
handleOpenURL(item?._source?.url);
} else {
selected(item);
}
}
};
const handleKeyUp = (e: KeyboardEvent) => {
2025-01-17 16:17:08 +08:00
// console.log("handleKeyUp", e.key);
if (!suggests.length) return;
if (!e.metaKey) {
setShowIndex(false);
}
};
useEffect(() => {
window.addEventListener("keydown", handleKeyDown);
window.addEventListener("keyup", handleKeyUp);
return () => {
window.removeEventListener("keydown", handleKeyDown);
window.removeEventListener("keyup", handleKeyUp);
};
}, [showIndex, selectedItem, suggests]);
useEffect(() => {
if (selectedItem !== null && itemRefs.current[selectedItem]) {
itemRefs.current[selectedItem]?.scrollIntoView({
behavior: "smooth",
block: "nearest",
});
}
}, [selectedItem]);
2025-01-11 10:12:53 +08:00
function getIcon(_source: any) {
2025-01-17 16:17:08 +08:00
const id = _source?.source?.id || "";
const result = datasourceData.find((item: any) => item._source.id === id);
const connector_id = result?._source?.connector?.id;
const result1 = connector_data.find(
(item: any) => item._source.id === connector_id
);
2025-01-17 16:17:08 +08:00
const icons = result1?._source?.assets?.icons || {};
if (icons[_source.icon]?.includes("http")) {
return icons[_source.icon];
} else {
return endpoint_http + icons[_source.icon];
}
2025-01-11 10:12:53 +08:00
}
return (
<div
ref={containerRef}
data-tauri-drag-region
className="h-[458px] w-full p-2 flex flex-col overflow-y-auto custom-scrollbar focus:outline-none"
tabIndex={0}
>
2025-01-11 10:12:53 +08:00
{showError ? (
<div className="flex items-center gap-2 text-sm text-[#333] p-2">
<CircleAlert className="text-[#FF0000] w-[14px] h-[14px]" />
Coco server is unavailable, only local results and available services
are displayed.
<Bolt className="text-[#000] w-[14px] h-[14px] cursor-pointer" />
<X
className="text-[#666] w-[16px] h-[16px] cursor-pointer"
onClick={() => setShowError(false)}
/>
</div>
) : null}
<div className="p-2 text-xs text-[#999] dark:text-[#666]">Results</div>
{suggests?.map((item, index) => {
const isSelected = selectedItem === index;
return (
<div
key={item._id}
ref={(el) => (itemRefs.current[index] = el)}
onMouseEnter={() => setSelectedItem(index)}
onClick={() => {
if (item?._source?.url) {
handleOpenURL(item?._source?.url);
} else {
selected(item);
}
}}
className={`w-full px-2 py-2.5 text-sm flex items-center justify-between rounded-lg transition-colors ${
isSelected
? "bg-[rgba(0,0,0,0.1)] dark:bg-[rgba(255,255,255,0.1)] hover:bg-[rgba(0,0,0,0.1)] dark:hover:bg-[rgba(255,255,255,0.1)]"
: ""
}`}
>
<div className="flex gap-2 items-center">
2025-01-11 10:12:53 +08:00
<img
className="w-5 h-5"
src={getIcon(item?._source)}
alt="icon"
/>
<span className="text-[#333] dark:text-[#d8d8d8] truncate w-80 text-left">
{item?._source?.title}
</span>
</div>
<div className="flex gap-2 items-center relative">
<span className="text-sm text-[#666] dark:text-[#666] truncate w-52 text-right">
{item?._source?.source?.name}
</span>
{showIndex && index < 10 ? (
<div
className={`absolute right-0 w-4 h-4 flex items-center justify-center font-normal text-xs text-[#333] leading-[14px] bg-[#ccc] dark:bg-[#6B6B6B] shadow-[-6px_0px_6px_2px_#e6e6e6] dark:shadow-[-6px_0px_6px_2px_#000] rounded-md`}
>
{index}
</div>
) : null}
</div>
</div>
);
})}
</div>
);
}
export default DropdownList;