agent-ecosystem/src/renderer/components/extensions/plugins/PluginsPanel.tsx
iliya 126f8e2865 feat: add Extension Store with plugin catalog and MCP registry
Full Extension Store implementation (Phases 0-6):
- Plugin marketplace catalog with ETag caching and search/filter/sort
- MCP server registry with Official + Glama aggregation
- Install/uninstall flows for both plugins and MCP servers via CLI
- Per-tab UI state, skeleton loading, dashed empty states, card polish
- Input validation and security hardening (scope allowlists, env/header
  key regex, projectPath validation, HTTP body size limits)
- 8 test suites covering catalog, install, aggregation, normalizers
2026-03-08 01:00:18 +02:00

301 lines
9.6 KiB
TypeScript

/**
* PluginsPanel — search, filter, sort and browse the plugin catalog.
*/
import { useMemo } from 'react';
import { Button } from '@renderer/components/ui/button';
import { Checkbox } from '@renderer/components/ui/checkbox';
import { Label } from '@renderer/components/ui/label';
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from '@renderer/components/ui/select';
import { useStore } from '@renderer/store';
import { inferCapabilities, normalizeCategory } from '@shared/utils/extensionNormalizers';
import { Puzzle, Search } from 'lucide-react';
import { SearchInput } from '../common/SearchInput';
import { CapabilityChips } from './CapabilityChips';
import { CategoryChips } from './CategoryChips';
import { PluginCard } from './PluginCard';
import { PluginDetailDialog } from './PluginDetailDialog';
import type {
EnrichedPlugin,
PluginCapability,
PluginFilters,
PluginSortField,
} from '@shared/types/extensions';
interface PluginsPanelProps {
pluginFilters: PluginFilters;
pluginSort: { field: PluginSortField; order: 'asc' | 'desc' };
selectedPluginId: string | null;
updatePluginSearch: (search: string) => void;
toggleCategory: (category: string) => void;
toggleCapability: (capability: PluginCapability) => void;
toggleInstalledOnly: () => void;
setSelectedPluginId: (id: string | null) => void;
clearFilters: () => void;
hasActiveFilters: boolean;
setPluginSort: (sort: { field: PluginSortField; order: 'asc' | 'desc' }) => void;
}
const SORT_OPTIONS: { value: string; label: string }[] = [
{ value: 'popularity:desc', label: 'Popular' },
{ value: 'name:asc', label: 'Name A-Z' },
{ value: 'name:desc', label: 'Name Z-A' },
{ value: 'category:asc', label: 'Category' },
];
/** Pure function: filter + sort the catalog */
function selectFilteredPlugins(
catalog: EnrichedPlugin[],
filters: PluginFilters,
sort: { field: PluginSortField; order: 'asc' | 'desc' }
): EnrichedPlugin[] {
let result = catalog;
// Search
if (filters.search) {
const q = filters.search.toLowerCase();
result = result.filter(
(p) =>
p.name.toLowerCase().includes(q) ||
p.description.toLowerCase().includes(q) ||
p.pluginId.toLowerCase().includes(q)
);
}
// Categories
if (filters.categories.length > 0) {
result = result.filter((p) => filters.categories.includes(normalizeCategory(p.category)));
}
// Capabilities
if (filters.capabilities.length > 0) {
result = result.filter((p) => {
const caps = inferCapabilities(p);
return filters.capabilities.some((fc) => caps.includes(fc));
});
}
// Installed only
if (filters.installedOnly) {
result = result.filter((p) => p.isInstalled);
}
// Sort
const direction = sort.order === 'asc' ? 1 : -1;
result = [...result].sort((a, b) => {
switch (sort.field) {
case 'popularity':
return (b.installCount - a.installCount) * direction;
case 'name':
return a.name.localeCompare(b.name) * direction;
case 'category':
return a.category.localeCompare(b.category) * direction;
default:
return 0;
}
});
return result;
}
export const PluginsPanel = ({
pluginFilters,
pluginSort,
selectedPluginId,
updatePluginSearch,
toggleCategory,
toggleCapability,
toggleInstalledOnly,
setSelectedPluginId,
clearFilters,
hasActiveFilters,
setPluginSort,
}: PluginsPanelProps): React.JSX.Element => {
const catalog = useStore((s) => s.pluginCatalog);
const loading = useStore((s) => s.pluginCatalogLoading);
const error = useStore((s) => s.pluginCatalogError);
const filtered = useMemo(
() => selectFilteredPlugins(catalog, pluginFilters, pluginSort),
[catalog, pluginFilters, pluginSort]
);
const selectedPlugin = useMemo(
() =>
selectedPluginId ? (catalog.find((p) => p.pluginId === selectedPluginId) ?? null) : null,
[catalog, selectedPluginId]
);
const sortValue = `${pluginSort.field}:${pluginSort.order}`;
return (
<div className="flex flex-col gap-4">
{/* Search + Sort + Installed only row */}
<div className="flex items-center gap-3">
<div className="flex-1">
<SearchInput
value={pluginFilters.search}
onChange={updatePluginSearch}
placeholder="Search plugins..."
/>
</div>
<Select
value={sortValue}
onValueChange={(v) => {
const [field, order] = v.split(':') as [PluginSortField, 'asc' | 'desc'];
setPluginSort({ field, order });
}}
>
<SelectTrigger className="w-36">
<SelectValue />
</SelectTrigger>
<SelectContent>
{SORT_OPTIONS.map((opt) => (
<SelectItem key={opt.value} value={opt.value}>
{opt.label}
</SelectItem>
))}
</SelectContent>
</Select>
<div className="flex items-center gap-2">
<Checkbox
id="installed-only"
checked={pluginFilters.installedOnly}
onCheckedChange={toggleInstalledOnly}
/>
<Label htmlFor="installed-only" className="whitespace-nowrap text-xs text-text-secondary">
Installed only
</Label>
</div>
</div>
{/* Filters */}
<div className="bg-surface-raised/50 rounded-md border border-border p-3">
<div className="flex flex-col gap-2.5">
<div className="flex items-center justify-between">
<span className="text-[11px] font-semibold uppercase tracking-wider text-text-muted">
Categories
</span>
{hasActiveFilters && (
<Button
variant="link"
size="sm"
onClick={clearFilters}
className="h-auto p-0 text-xs text-[var(--color-accent)]"
>
Clear all
</Button>
)}
</div>
<CategoryChips
plugins={catalog}
selected={pluginFilters.categories}
onToggle={toggleCategory}
/>
<div className="border-b border-border" />
<span className="text-[11px] font-semibold uppercase tracking-wider text-text-muted">
Capabilities
</span>
<CapabilityChips
plugins={catalog}
selected={pluginFilters.capabilities}
onToggle={toggleCapability}
/>
</div>
</div>
{/* Result count */}
{!loading && !error && filtered.length > 0 && (
<p className="text-xs text-text-muted">
{filtered.length} plugin{filtered.length !== 1 ? 's' : ''}
</p>
)}
{/* Content */}
{loading && (
<div className="grid grid-cols-1 gap-3 sm:grid-cols-2 xl:grid-cols-3">
{Array.from({ length: 6 }, (_, i) => (
<div
key={i}
className="skeleton-card flex flex-col gap-2 rounded-lg border border-border p-4"
style={{ animationDelay: `${i * 80}ms` }}
>
<div className="flex items-start justify-between gap-2">
<div className="h-4 w-32 rounded bg-surface-raised" />
<div className="h-5 w-16 rounded-full bg-surface-raised" />
</div>
<div className="space-y-1.5">
<div className="h-3 w-full rounded bg-surface-raised" />
<div className="h-3 w-3/4 rounded bg-surface-raised" />
</div>
<div className="flex gap-1.5">
<div className="h-5 w-14 rounded-full bg-surface-raised" />
<div className="h-5 w-12 rounded-full bg-surface-raised" />
</div>
<div className="flex items-center justify-between">
<div className="h-3 w-24 rounded bg-surface-raised" />
<div className="h-7 w-16 rounded bg-surface-raised" />
</div>
</div>
))}
</div>
)}
{error && (
<div className="rounded-md border border-red-500/30 bg-red-500/5 p-4 text-sm text-red-400">
{error}
</div>
)}
{!loading && !error && filtered.length === 0 && (
<div className="flex flex-col items-center gap-3 rounded-sm border border-dashed border-border px-8 py-16">
<div className="flex size-10 items-center justify-center rounded-lg border border-border bg-surface-raised">
{hasActiveFilters ? (
<Search className="size-5 text-text-muted" />
) : (
<Puzzle className="size-5 text-text-muted" />
)}
</div>
<p className="text-sm text-text-secondary">
{hasActiveFilters ? 'No plugins match your filters' : 'No plugins available'}
</p>
<p className="text-xs text-text-muted">
{hasActiveFilters
? 'Try adjusting your search or filter criteria'
: 'Check back later for new plugins'}
</p>
{hasActiveFilters && (
<Button variant="outline" size="sm" onClick={clearFilters}>
Clear filters
</Button>
)}
</div>
)}
{!loading && !error && filtered.length > 0 && (
<div className="grid grid-cols-1 gap-3 sm:grid-cols-2 xl:grid-cols-3">
{filtered.map((plugin) => (
<PluginCard key={plugin.pluginId} plugin={plugin} onClick={setSelectedPluginId} />
))}
</div>
)}
{/* Detail dialog */}
<PluginDetailDialog
plugin={selectedPlugin}
open={selectedPluginId !== null}
onClose={() => setSelectedPluginId(null)}
/>
</div>
);
};