DiscoverPlugins.tsx
commands/plugin/DiscoverPlugins.tsx
781
Lines
106666
Bytes
1
Exports
31
Imports
10
Keywords
What this is
This page documents one file from the repository and includes its full source so you can read it without leaving the docs site.
Beginner explanation
This file lives in the command layer. It likely turns a user action into concrete program behavior.
How it is used
Start from the exports list and related files. Those are the easiest clues for where this file fits into the system.
Expert explanation
Architecturally, this file intersects with commands. It contains 781 lines, 31 detected imports, and 1 detected exports.
Important relationships
Detected exports
DiscoverPlugins
Keywords
textentryelsepluginspluginpluginiddimcolorselectedpluginnamevoid
Detected imports
react/compiler-runtimefiguresreactreact../../components/ConfigurableShortcutHint.js../../components/design-system/Byline.js../../components/SearchBox.js../../hooks/useSearchInput.js../../hooks/useTerminalSize.js../../ink.js../../keybindings/useKeybinding.js../../types/plugin.js../../utils/array.js../../utils/browser.js../../utils/debug.js../../utils/errors.js../../utils/plugins/cacheUtils.js../../utils/plugins/installCounts.js../../utils/plugins/installedPluginsManager.js../../utils/plugins/marketplaceHelpers.js../../utils/plugins/marketplaceManager.js../../utils/plugins/officialMarketplace.js../../utils/plugins/pluginInstallationHelpers.js../../utils/plugins/pluginPolicy.js../../utils/stringUtils.js../../utils/truncate.js./PluginOptionsFlow.js./PluginTrustWarning.js./pluginDetailsHelpers.js./types.js./usePagination.js
Source notes
This page embeds the full file contents. Small or leaf files are still indexed honestly instead of being over-explained.
Full source
import { c as _c } from "react/compiler-runtime";
import figures from 'figures';
import * as React from 'react';
import { useCallback, useEffect, useMemo, useState } from 'react';
import { ConfigurableShortcutHint } from '../../components/ConfigurableShortcutHint.js';
import { Byline } from '../../components/design-system/Byline.js';
import { SearchBox } from '../../components/SearchBox.js';
import { useSearchInput } from '../../hooks/useSearchInput.js';
import { useTerminalSize } from '../../hooks/useTerminalSize.js';
// eslint-disable-next-line custom-rules/prefer-use-keybindings -- useInput needed for raw search mode text input
import { Box, Text, useInput, useTerminalFocus } from '../../ink.js';
import { useKeybinding, useKeybindings } from '../../keybindings/useKeybinding.js';
import type { LoadedPlugin } from '../../types/plugin.js';
import { count } from '../../utils/array.js';
import { openBrowser } from '../../utils/browser.js';
import { logForDebugging } from '../../utils/debug.js';
import { errorMessage } from '../../utils/errors.js';
import { clearAllCaches } from '../../utils/plugins/cacheUtils.js';
import { formatInstallCount, getInstallCounts } from '../../utils/plugins/installCounts.js';
import { isPluginGloballyInstalled } from '../../utils/plugins/installedPluginsManager.js';
import { createPluginId, detectEmptyMarketplaceReason, type EmptyMarketplaceReason, formatFailureDetails, formatMarketplaceLoadingErrors, loadMarketplacesWithGracefulDegradation } from '../../utils/plugins/marketplaceHelpers.js';
import { loadKnownMarketplacesConfig } from '../../utils/plugins/marketplaceManager.js';
import { OFFICIAL_MARKETPLACE_NAME } from '../../utils/plugins/officialMarketplace.js';
import { installPluginFromMarketplace } from '../../utils/plugins/pluginInstallationHelpers.js';
import { isPluginBlockedByPolicy } from '../../utils/plugins/pluginPolicy.js';
import { plural } from '../../utils/stringUtils.js';
import { truncateToWidth } from '../../utils/truncate.js';
import { findPluginOptionsTarget, PluginOptionsFlow } from './PluginOptionsFlow.js';
import { PluginTrustWarning } from './PluginTrustWarning.js';
import { buildPluginDetailsMenuOptions, extractGitHubRepo, type InstallablePlugin } from './pluginDetailsHelpers.js';
import type { ViewState as ParentViewState } from './types.js';
import { usePagination } from './usePagination.js';
type Props = {
error: string | null;
setError: (error: string | null) => void;
result: string | null;
setResult: (result: string | null) => void;
setViewState: (state: ParentViewState) => void;
onInstallComplete?: () => void | Promise<void>;
onSearchModeChange?: (isActive: boolean) => void;
targetPlugin?: string;
};
type ViewState = 'plugin-list' | 'plugin-details' | {
type: 'plugin-options';
plugin: LoadedPlugin;
pluginId: string;
};
export function DiscoverPlugins({
error,
setError,
result: _result,
setResult,
setViewState: setParentViewState,
onInstallComplete,
onSearchModeChange,
targetPlugin
}: Props): React.ReactNode {
// View state
const [viewState, setViewState] = useState<ViewState>('plugin-list');
const [selectedPlugin, setSelectedPlugin] = useState<InstallablePlugin | null>(null);
// Data state
const [availablePlugins, setAvailablePlugins] = useState<InstallablePlugin[]>([]);
const [loading, setLoading] = useState(true);
const [installCounts, setInstallCounts] = useState<Map<string, number> | null>(null);
// Search state
const [isSearchMode, setIsSearchModeRaw] = useState(false);
const setIsSearchMode = useCallback((active: boolean) => {
setIsSearchModeRaw(active);
onSearchModeChange?.(active);
}, [onSearchModeChange]);
const {
query: searchQuery,
setQuery: setSearchQuery,
cursorOffset: searchCursorOffset
} = useSearchInput({
isActive: viewState === 'plugin-list' && isSearchMode && !loading,
onExit: () => {
setIsSearchMode(false);
}
});
const isTerminalFocused = useTerminalFocus();
const {
columns: terminalWidth
} = useTerminalSize();
// Filter plugins based on search query
const filteredPlugins = useMemo(() => {
if (!searchQuery) return availablePlugins;
const lowerQuery = searchQuery.toLowerCase();
return availablePlugins.filter(plugin => plugin.entry.name.toLowerCase().includes(lowerQuery) || plugin.entry.description?.toLowerCase().includes(lowerQuery) || plugin.marketplaceName.toLowerCase().includes(lowerQuery));
}, [availablePlugins, searchQuery]);
// Selection state
const [selectedIndex, setSelectedIndex] = useState(0);
const [selectedForInstall, setSelectedForInstall] = useState<Set<string>>(new Set());
const [installingPlugins, setInstallingPlugins] = useState<Set<string>>(new Set());
// Pagination for plugin list (continuous scrolling)
const pagination = usePagination<InstallablePlugin>({
totalItems: filteredPlugins.length,
selectedIndex
});
// Reset selection when search query changes
useEffect(() => {
setSelectedIndex(0);
}, [searchQuery]);
// Details view state
const [detailsMenuIndex, setDetailsMenuIndex] = useState(0);
const [isInstalling, setIsInstalling] = useState(false);
const [installError, setInstallError] = useState<string | null>(null);
// Warning state for non-critical errors
const [warning, setWarning] = useState<string | null>(null);
// Empty state reason
const [emptyReason, setEmptyReason] = useState<EmptyMarketplaceReason | null>(null);
// Load all plugins from all marketplaces
useEffect(() => {
async function loadAllPlugins() {
try {
const config = await loadKnownMarketplacesConfig();
// Load marketplaces with graceful degradation
const {
marketplaces,
failures
} = await loadMarketplacesWithGracefulDegradation(config);
// Collect all plugins from all marketplaces
const allPlugins: InstallablePlugin[] = [];
for (const {
name,
data: marketplace
} of marketplaces) {
if (marketplace) {
for (const entry of marketplace.plugins) {
const pluginId = createPluginId(entry.name, name);
allPlugins.push({
entry,
marketplaceName: name,
pluginId,
// Only block when globally installed (user/managed scope).
// Project/local-scope installs don't block — user may want to
// promote to user scope so it's available everywhere (gh-29997).
isInstalled: isPluginGloballyInstalled(pluginId)
});
}
}
}
// Filter out installed and policy-blocked plugins
const uninstalledPlugins = allPlugins.filter(p => !p.isInstalled && !isPluginBlockedByPolicy(p.pluginId));
// Fetch install counts and sort by popularity
try {
const counts = await getInstallCounts();
setInstallCounts(counts);
if (counts) {
// Sort by install count (descending), then alphabetically
uninstalledPlugins.sort((a_0, b_0) => {
const countA = counts.get(a_0.pluginId) ?? 0;
const countB = counts.get(b_0.pluginId) ?? 0;
if (countA !== countB) return countB - countA;
return a_0.entry.name.localeCompare(b_0.entry.name);
});
} else {
// No counts available - sort alphabetically
uninstalledPlugins.sort((a_1, b_1) => a_1.entry.name.localeCompare(b_1.entry.name));
}
} catch (error_0) {
// Log the error, then gracefully degrade to alphabetical sort
logForDebugging(`Failed to fetch install counts: ${errorMessage(error_0)}`);
uninstalledPlugins.sort((a, b) => a.entry.name.localeCompare(b.entry.name));
}
setAvailablePlugins(uninstalledPlugins);
// Detect empty reason if no plugins available
const configuredCount = Object.keys(config).length;
if (uninstalledPlugins.length === 0) {
const reason = await detectEmptyMarketplaceReason({
configuredMarketplaceCount: configuredCount,
failedMarketplaceCount: failures.length
});
setEmptyReason(reason);
}
// Handle marketplace loading errors/warnings
const successCount = count(marketplaces, m => m.data !== null);
const errorResult = formatMarketplaceLoadingErrors(failures, successCount);
if (errorResult) {
if (errorResult.type === 'warning') {
setWarning(errorResult.message + '. Showing available plugins.');
} else {
throw new Error(errorResult.message);
}
}
// Handle targetPlugin - navigate directly to plugin details
// Search in allPlugins (before filtering) to handle installed plugins gracefully
if (targetPlugin) {
const foundPlugin = allPlugins.find(p_0 => p_0.entry.name === targetPlugin);
if (foundPlugin) {
if (foundPlugin.isInstalled) {
setError(`Plugin '${foundPlugin.pluginId}' is already installed. Use '/plugin' to manage existing plugins.`);
} else {
setSelectedPlugin(foundPlugin);
setViewState('plugin-details');
}
} else {
setError(`Plugin "${targetPlugin}" not found in any marketplace`);
}
}
} catch (err) {
setError(err instanceof Error ? err.message : 'Failed to load plugins');
} finally {
setLoading(false);
}
}
void loadAllPlugins();
}, [setError, targetPlugin]);
// Install selected plugins
const installSelectedPlugins = async () => {
if (selectedForInstall.size === 0) return;
const pluginsToInstall = availablePlugins.filter(p_1 => selectedForInstall.has(p_1.pluginId));
setInstallingPlugins(new Set(pluginsToInstall.map(p_2 => p_2.pluginId)));
let successCount_0 = 0;
let failureCount = 0;
const newFailedPlugins: Array<{
name: string;
reason: string;
}> = [];
for (const plugin_0 of pluginsToInstall) {
const result = await installPluginFromMarketplace({
pluginId: plugin_0.pluginId,
entry: plugin_0.entry,
marketplaceName: plugin_0.marketplaceName,
scope: 'user'
});
if (result.success) {
successCount_0++;
} else {
failureCount++;
newFailedPlugins.push({
name: plugin_0.entry.name,
reason: result.error
});
}
}
setInstallingPlugins(new Set());
setSelectedForInstall(new Set());
clearAllCaches();
// Handle installation results
if (failureCount === 0) {
const message = `✓ Installed ${successCount_0} ${plural(successCount_0, 'plugin')}. ` + `Run /reload-plugins to activate.`;
setResult(message);
} else if (successCount_0 === 0) {
setError(`Failed to install: ${formatFailureDetails(newFailedPlugins, true)}`);
} else {
const message_0 = `✓ Installed ${successCount_0} of ${successCount_0 + failureCount} plugins. ` + `Failed: ${formatFailureDetails(newFailedPlugins, false)}. ` + `Run /reload-plugins to activate successfully installed plugins.`;
setResult(message_0);
}
if (successCount_0 > 0) {
if (onInstallComplete) {
await onInstallComplete();
}
}
setParentViewState({
type: 'menu'
});
};
// Install single plugin from details view
const handleSinglePluginInstall = async (plugin_1: InstallablePlugin, scope: 'user' | 'project' | 'local' = 'user') => {
setIsInstalling(true);
setInstallError(null);
const result_0 = await installPluginFromMarketplace({
pluginId: plugin_1.pluginId,
entry: plugin_1.entry,
marketplaceName: plugin_1.marketplaceName,
scope
});
if (result_0.success) {
const loaded = await findPluginOptionsTarget(plugin_1.pluginId);
if (loaded) {
setIsInstalling(false);
setViewState({
type: 'plugin-options',
plugin: loaded,
pluginId: plugin_1.pluginId
});
return;
}
setResult(result_0.message);
if (onInstallComplete) {
await onInstallComplete();
}
setParentViewState({
type: 'menu'
});
} else {
setIsInstalling(false);
setInstallError(result_0.error);
}
};
// Handle error state
useEffect(() => {
if (error) {
setResult(error);
}
}, [error, setResult]);
// Escape in plugin-details view - go back to plugin-list
useKeybinding('confirm:no', () => {
setViewState('plugin-list');
setSelectedPlugin(null);
}, {
context: 'Confirmation',
isActive: viewState === 'plugin-details'
});
// Escape in plugin-list view (not search mode) - exit to parent menu
useKeybinding('confirm:no', () => {
setParentViewState({
type: 'menu'
});
}, {
context: 'Confirmation',
isActive: viewState === 'plugin-list' && !isSearchMode
});
// Handle entering search mode (non-escape keys)
useInput((input, _key) => {
const keyIsNotCtrlOrMeta = !_key.ctrl && !_key.meta;
if (!isSearchMode) {
// Enter search mode with '/' or any printable character
if (input === '/' && keyIsNotCtrlOrMeta) {
setIsSearchMode(true);
setSearchQuery('');
} else if (keyIsNotCtrlOrMeta && input.length > 0 && !/^\s+$/.test(input) &&
// Don't enter search mode for navigation keys
input !== 'j' && input !== 'k' && input !== 'i') {
setIsSearchMode(true);
setSearchQuery(input);
}
}
}, {
isActive: viewState === 'plugin-list' && !loading
});
// Plugin-list navigation (non-search mode)
useKeybindings({
'select:previous': () => {
if (selectedIndex === 0) {
setIsSearchMode(true);
} else {
pagination.handleSelectionChange(selectedIndex - 1, setSelectedIndex);
}
},
'select:next': () => {
if (selectedIndex < filteredPlugins.length - 1) {
pagination.handleSelectionChange(selectedIndex + 1, setSelectedIndex);
}
},
'select:accept': () => {
if (selectedIndex === filteredPlugins.length && selectedForInstall.size > 0) {
void installSelectedPlugins();
} else if (selectedIndex < filteredPlugins.length) {
const plugin_2 = filteredPlugins[selectedIndex];
if (plugin_2) {
if (plugin_2.isInstalled) {
setParentViewState({
type: 'manage-plugins',
targetPlugin: plugin_2.entry.name,
targetMarketplace: plugin_2.marketplaceName
});
} else {
setSelectedPlugin(plugin_2);
setViewState('plugin-details');
setDetailsMenuIndex(0);
setInstallError(null);
}
}
}
}
}, {
context: 'Select',
isActive: viewState === 'plugin-list' && !isSearchMode
});
useKeybindings({
'plugin:toggle': () => {
if (selectedIndex < filteredPlugins.length) {
const plugin_3 = filteredPlugins[selectedIndex];
if (plugin_3 && !plugin_3.isInstalled) {
const newSelection = new Set(selectedForInstall);
if (newSelection.has(plugin_3.pluginId)) {
newSelection.delete(plugin_3.pluginId);
} else {
newSelection.add(plugin_3.pluginId);
}
setSelectedForInstall(newSelection);
}
}
},
'plugin:install': () => {
if (selectedForInstall.size > 0) {
void installSelectedPlugins();
}
}
}, {
context: 'Plugin',
isActive: viewState === 'plugin-list' && !isSearchMode
});
// Plugin-details navigation
const detailsMenuOptions = React.useMemo(() => {
if (!selectedPlugin) return [];
const hasHomepage = selectedPlugin.entry.homepage;
const githubRepo = extractGitHubRepo(selectedPlugin);
return buildPluginDetailsMenuOptions(hasHomepage, githubRepo);
}, [selectedPlugin]);
useKeybindings({
'select:previous': () => {
if (detailsMenuIndex > 0) {
setDetailsMenuIndex(detailsMenuIndex - 1);
}
},
'select:next': () => {
if (detailsMenuIndex < detailsMenuOptions.length - 1) {
setDetailsMenuIndex(detailsMenuIndex + 1);
}
},
'select:accept': () => {
if (!selectedPlugin) return;
const action = detailsMenuOptions[detailsMenuIndex]?.action;
const hasHomepage_0 = selectedPlugin.entry.homepage;
const githubRepo_0 = extractGitHubRepo(selectedPlugin);
if (action === 'install-user') {
void handleSinglePluginInstall(selectedPlugin, 'user');
} else if (action === 'install-project') {
void handleSinglePluginInstall(selectedPlugin, 'project');
} else if (action === 'install-local') {
void handleSinglePluginInstall(selectedPlugin, 'local');
} else if (action === 'homepage' && hasHomepage_0) {
void openBrowser(hasHomepage_0);
} else if (action === 'github' && githubRepo_0) {
void openBrowser(`https://github.com/${githubRepo_0}`);
} else if (action === 'back') {
setViewState('plugin-list');
setSelectedPlugin(null);
}
}
}, {
context: 'Select',
isActive: viewState === 'plugin-details' && !!selectedPlugin
});
if (typeof viewState === 'object' && viewState.type === 'plugin-options') {
const {
plugin: plugin_4,
pluginId: pluginId_0
} = viewState;
function finish(msg: string): void {
setResult(msg);
if (onInstallComplete) {
void onInstallComplete();
}
setParentViewState({
type: 'menu'
});
}
return <PluginOptionsFlow plugin={plugin_4} pluginId={pluginId_0} onDone={(outcome, detail) => {
switch (outcome) {
case 'configured':
finish(`✓ Installed and configured ${plugin_4.name}. Run /reload-plugins to apply.`);
break;
case 'skipped':
finish(`✓ Installed ${plugin_4.name}. Run /reload-plugins to apply.`);
break;
case 'error':
finish(`Installed but failed to save config: ${detail}`);
break;
}
}} />;
}
// Loading state
if (loading) {
return <Text>Loading…</Text>;
}
// Error state
if (error) {
return <Text color="error">{error}</Text>;
}
// Plugin details view
if (viewState === 'plugin-details' && selectedPlugin) {
const hasHomepage_1 = selectedPlugin.entry.homepage;
const githubRepo_1 = extractGitHubRepo(selectedPlugin);
const menuOptions = buildPluginDetailsMenuOptions(hasHomepage_1, githubRepo_1);
return <Box flexDirection="column">
<Box marginBottom={1}>
<Text bold>Plugin details</Text>
</Box>
<Box flexDirection="column" marginBottom={1}>
<Text bold>{selectedPlugin.entry.name}</Text>
<Text dimColor>from {selectedPlugin.marketplaceName}</Text>
{selectedPlugin.entry.version && <Text dimColor>Version: {selectedPlugin.entry.version}</Text>}
{selectedPlugin.entry.description && <Box marginTop={1}>
<Text>{selectedPlugin.entry.description}</Text>
</Box>}
{selectedPlugin.entry.author && <Box marginTop={1}>
<Text dimColor>
By:{' '}
{typeof selectedPlugin.entry.author === 'string' ? selectedPlugin.entry.author : selectedPlugin.entry.author.name}
</Text>
</Box>}
</Box>
<PluginTrustWarning />
{installError && <Box marginBottom={1}>
<Text color="error">Error: {installError}</Text>
</Box>}
<Box flexDirection="column">
{menuOptions.map((option, index) => <Box key={option.action}>
{detailsMenuIndex === index && <Text>{'> '}</Text>}
{detailsMenuIndex !== index && <Text>{' '}</Text>}
<Text bold={detailsMenuIndex === index}>
{isInstalling && option.action.startsWith('install-') ? 'Installing…' : option.label}
</Text>
</Box>)}
</Box>
<Box marginTop={1}>
<Text dimColor>
<Byline>
<ConfigurableShortcutHint action="select:accept" context="Select" fallback="Enter" description="select" />
<ConfigurableShortcutHint action="confirm:no" context="Confirmation" fallback="Esc" description="back" />
</Byline>
</Text>
</Box>
</Box>;
}
// Empty state
if (availablePlugins.length === 0) {
return <Box flexDirection="column">
<Box marginBottom={1}>
<Text bold>Discover plugins</Text>
</Box>
<EmptyStateMessage reason={emptyReason} />
<Box marginTop={1}>
<Text dimColor italic>
Esc to go back
</Text>
</Box>
</Box>;
}
// Get visible plugins from pagination
const visiblePlugins = pagination.getVisibleItems(filteredPlugins);
return <Box flexDirection="column">
<Box>
<Text bold>Discover plugins</Text>
{pagination.needsPagination && <Text dimColor>
{' '}
({pagination.scrollPosition.current}/
{pagination.scrollPosition.total})
</Text>}
</Box>
{/* Search box */}
<Box marginBottom={1}>
<SearchBox query={searchQuery} isFocused={isSearchMode} isTerminalFocused={isTerminalFocused} width={terminalWidth - 4} cursorOffset={searchCursorOffset} />
</Box>
{/* Warning banner */}
{warning && <Box marginBottom={1}>
<Text color="warning">
{figures.warning} {warning}
</Text>
</Box>}
{/* No search results */}
{filteredPlugins.length === 0 && searchQuery && <Box marginBottom={1}>
<Text dimColor>No plugins match "{searchQuery}"</Text>
</Box>}
{/* Scroll up indicator */}
{pagination.scrollPosition.canScrollUp && <Box>
<Text dimColor> {figures.arrowUp} more above</Text>
</Box>}
{/* Plugin list - use startIndex in key to force re-render on scroll */}
{visiblePlugins.map((plugin_5, visibleIndex) => {
const actualIndex = pagination.toActualIndex(visibleIndex);
const isSelected = selectedIndex === actualIndex;
const isSelectedForInstall = selectedForInstall.has(plugin_5.pluginId);
const isInstallingThis = installingPlugins.has(plugin_5.pluginId);
const isLast = visibleIndex === visiblePlugins.length - 1;
return <Box key={`${pagination.startIndex}-${plugin_5.pluginId}`} flexDirection="column" marginBottom={isLast && !error ? 0 : 1}>
<Box>
<Text color={isSelected && !isSearchMode ? 'suggestion' : undefined}>
{isSelected && !isSearchMode ? figures.pointer : ' '}{' '}
</Text>
<Text>
{isInstallingThis ? figures.ellipsis : isSelectedForInstall ? figures.radioOn : figures.radioOff}{' '}
{plugin_5.entry.name}
<Text dimColor> · {plugin_5.marketplaceName}</Text>
{plugin_5.entry.tags?.includes('community-managed') && <Text dimColor> [Community Managed]</Text>}
{installCounts && plugin_5.marketplaceName === OFFICIAL_MARKETPLACE_NAME && <Text dimColor>
{' · '}
{formatInstallCount(installCounts.get(plugin_5.pluginId) ?? 0)}{' '}
installs
</Text>}
</Text>
</Box>
{plugin_5.entry.description && <Box marginLeft={4}>
<Text dimColor>
{truncateToWidth(plugin_5.entry.description, 60)}
</Text>
</Box>}
</Box>;
})}
{/* Scroll down indicator */}
{pagination.scrollPosition.canScrollDown && <Box>
<Text dimColor> {figures.arrowDown} more below</Text>
</Box>}
{/* Error messages */}
{error && <Box marginTop={1}>
<Text color="error">
{figures.cross} {error}
</Text>
</Box>}
<DiscoverPluginsKeyHint hasSelection={selectedForInstall.size > 0} canToggle={selectedIndex < filteredPlugins.length && !filteredPlugins[selectedIndex]?.isInstalled} />
</Box>;
}
function DiscoverPluginsKeyHint(t0) {
const $ = _c(10);
const {
hasSelection,
canToggle
} = t0;
let t1;
if ($[0] !== hasSelection) {
t1 = hasSelection && <ConfigurableShortcutHint action="plugin:install" context="Plugin" fallback="i" description="install" bold={true} />;
$[0] = hasSelection;
$[1] = t1;
} else {
t1 = $[1];
}
let t2;
if ($[2] === Symbol.for("react.memo_cache_sentinel")) {
t2 = <Text>type to search</Text>;
$[2] = t2;
} else {
t2 = $[2];
}
let t3;
if ($[3] !== canToggle) {
t3 = canToggle && <ConfigurableShortcutHint action="plugin:toggle" context="Plugin" fallback="Space" description="toggle" />;
$[3] = canToggle;
$[4] = t3;
} else {
t3 = $[4];
}
let t4;
let t5;
if ($[5] === Symbol.for("react.memo_cache_sentinel")) {
t4 = <ConfigurableShortcutHint action="select:accept" context="Select" fallback="Enter" description="details" />;
t5 = <ConfigurableShortcutHint action="confirm:no" context="Confirmation" fallback="Esc" description="back" />;
$[5] = t4;
$[6] = t5;
} else {
t4 = $[5];
t5 = $[6];
}
let t6;
if ($[7] !== t1 || $[8] !== t3) {
t6 = <Box marginTop={1}><Text dimColor={true} italic={true}><Byline>{t1}{t2}{t3}{t4}{t5}</Byline></Text></Box>;
$[7] = t1;
$[8] = t3;
$[9] = t6;
} else {
t6 = $[9];
}
return t6;
}
/**
* Context-aware empty state message for the Discover screen
*/
function EmptyStateMessage(t0) {
const $ = _c(6);
const {
reason
} = t0;
switch (reason) {
case "git-not-installed":
{
let t1;
if ($[0] === Symbol.for("react.memo_cache_sentinel")) {
t1 = <><Text dimColor={true}>Git is required to install marketplaces.</Text><Text dimColor={true}>Please install git and restart Claude Code.</Text></>;
$[0] = t1;
} else {
t1 = $[0];
}
return t1;
}
case "all-blocked-by-policy":
{
let t1;
if ($[1] === Symbol.for("react.memo_cache_sentinel")) {
t1 = <><Text dimColor={true}>Your organization policy does not allow any external marketplaces.</Text><Text dimColor={true}>Contact your administrator.</Text></>;
$[1] = t1;
} else {
t1 = $[1];
}
return t1;
}
case "policy-restricts-sources":
{
let t1;
if ($[2] === Symbol.for("react.memo_cache_sentinel")) {
t1 = <><Text dimColor={true}>Your organization restricts which marketplaces can be added.</Text><Text dimColor={true}>Switch to the Marketplaces tab to view allowed sources.</Text></>;
$[2] = t1;
} else {
t1 = $[2];
}
return t1;
}
case "all-marketplaces-failed":
{
let t1;
if ($[3] === Symbol.for("react.memo_cache_sentinel")) {
t1 = <><Text dimColor={true}>Failed to load marketplace data.</Text><Text dimColor={true}>Check your network connection.</Text></>;
$[3] = t1;
} else {
t1 = $[3];
}
return t1;
}
case "all-plugins-installed":
{
let t1;
if ($[4] === Symbol.for("react.memo_cache_sentinel")) {
t1 = <><Text dimColor={true}>All available plugins are already installed.</Text><Text dimColor={true}>Check for new plugins later or add more marketplaces.</Text></>;
$[4] = t1;
} else {
t1 = $[4];
}
return t1;
}
case "no-marketplaces-configured":
default:
{
let t1;
if ($[5] === Symbol.for("react.memo_cache_sentinel")) {
t1 = <><Text dimColor={true}>No plugins available.</Text><Text dimColor={true}>Add a marketplace first using the Marketplaces tab.</Text></>;
$[5] = t1;
} else {
t1 = $[5];
}
return t1;
}
}
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["figures","React","useCallback","useEffect","useMemo","useState","ConfigurableShortcutHint","Byline","SearchBox","useSearchInput","useTerminalSize","Box","Text","useInput","useTerminalFocus","useKeybinding","useKeybindings","LoadedPlugin","count","openBrowser","logForDebugging","errorMessage","clearAllCaches","formatInstallCount","getInstallCounts","isPluginGloballyInstalled","createPluginId","detectEmptyMarketplaceReason","EmptyMarketplaceReason","formatFailureDetails","formatMarketplaceLoadingErrors","loadMarketplacesWithGracefulDegradation","loadKnownMarketplacesConfig","OFFICIAL_MARKETPLACE_NAME","installPluginFromMarketplace","isPluginBlockedByPolicy","plural","truncateToWidth","findPluginOptionsTarget","PluginOptionsFlow","PluginTrustWarning","buildPluginDetailsMenuOptions","extractGitHubRepo","InstallablePlugin","ViewState","ParentViewState","usePagination","Props","error","setError","result","setResult","setViewState","state","onInstallComplete","Promise","onSearchModeChange","isActive","targetPlugin","type","plugin","pluginId","DiscoverPlugins","_result","setParentViewState","ReactNode","viewState","selectedPlugin","setSelectedPlugin","availablePlugins","setAvailablePlugins","loading","setLoading","installCounts","setInstallCounts","Map","isSearchMode","setIsSearchModeRaw","setIsSearchMode","active","query","searchQuery","setQuery","setSearchQuery","cursorOffset","searchCursorOffset","onExit","isTerminalFocused","columns","terminalWidth","filteredPlugins","lowerQuery","toLowerCase","filter","entry","name","includes","description","marketplaceName","selectedIndex","setSelectedIndex","selectedForInstall","setSelectedForInstall","Set","installingPlugins","setInstallingPlugins","pagination","totalItems","length","detailsMenuIndex","setDetailsMenuIndex","isInstalling","setIsInstalling","installError","setInstallError","warning","setWarning","emptyReason","setEmptyReason","loadAllPlugins","config","marketplaces","failures","allPlugins","data","marketplace","plugins","push","isInstalled","uninstalledPlugins","p","counts","sort","a","b","countA","get","countB","localeCompare","configuredCount","Object","keys","reason","configuredMarketplaceCount","failedMarketplaceCount","successCount","m","errorResult","message","Error","foundPlugin","find","err","installSelectedPlugins","size","pluginsToInstall","has","map","failureCount","newFailedPlugins","Array","scope","success","handleSinglePluginInstall","loaded","context","input","_key","keyIsNotCtrlOrMeta","ctrl","meta","test","select:previous","handleSelectionChange","select:next","select:accept","targetMarketplace","plugin:toggle","newSelection","delete","add","plugin:install","detailsMenuOptions","hasHomepage","homepage","githubRepo","action","finish","msg","outcome","detail","menuOptions","version","author","option","index","startsWith","label","visiblePlugins","getVisibleItems","needsPagination","scrollPosition","current","total","canScrollUp","arrowUp","visibleIndex","actualIndex","toActualIndex","isSelected","isSelectedForInstall","isInstallingThis","isLast","startIndex","undefined","pointer","ellipsis","radioOn","radioOff","tags","canScrollDown","arrowDown","cross","DiscoverPluginsKeyHint","t0","$","_c","hasSelection","canToggle","t1","t2","Symbol","for","t3","t4","t5","t6","EmptyStateMessage"],"sources":["DiscoverPlugins.tsx"],"sourcesContent":["import figures from 'figures'\nimport * as React from 'react'\nimport { useCallback, useEffect, useMemo, useState } from 'react'\nimport { ConfigurableShortcutHint } from '../../components/ConfigurableShortcutHint.js'\nimport { Byline } from '../../components/design-system/Byline.js'\nimport { SearchBox } from '../../components/SearchBox.js'\nimport { useSearchInput } from '../../hooks/useSearchInput.js'\nimport { useTerminalSize } from '../../hooks/useTerminalSize.js'\n// eslint-disable-next-line custom-rules/prefer-use-keybindings -- useInput needed for raw search mode text input\nimport { Box, Text, useInput, useTerminalFocus } from '../../ink.js'\nimport {\n  useKeybinding,\n  useKeybindings,\n} from '../../keybindings/useKeybinding.js'\nimport type { LoadedPlugin } from '../../types/plugin.js'\nimport { count } from '../../utils/array.js'\nimport { openBrowser } from '../../utils/browser.js'\nimport { logForDebugging } from '../../utils/debug.js'\nimport { errorMessage } from '../../utils/errors.js'\nimport { clearAllCaches } from '../../utils/plugins/cacheUtils.js'\nimport {\n  formatInstallCount,\n  getInstallCounts,\n} from '../../utils/plugins/installCounts.js'\nimport { isPluginGloballyInstalled } from '../../utils/plugins/installedPluginsManager.js'\nimport {\n  createPluginId,\n  detectEmptyMarketplaceReason,\n  type EmptyMarketplaceReason,\n  formatFailureDetails,\n  formatMarketplaceLoadingErrors,\n  loadMarketplacesWithGracefulDegradation,\n} from '../../utils/plugins/marketplaceHelpers.js'\nimport { loadKnownMarketplacesConfig } from '../../utils/plugins/marketplaceManager.js'\nimport { OFFICIAL_MARKETPLACE_NAME } from '../../utils/plugins/officialMarketplace.js'\nimport { installPluginFromMarketplace } from '../../utils/plugins/pluginInstallationHelpers.js'\nimport { isPluginBlockedByPolicy } from '../../utils/plugins/pluginPolicy.js'\nimport { plural } from '../../utils/stringUtils.js'\nimport { truncateToWidth } from '../../utils/truncate.js'\nimport {\n  findPluginOptionsTarget,\n  PluginOptionsFlow,\n} from './PluginOptionsFlow.js'\nimport { PluginTrustWarning } from './PluginTrustWarning.js'\nimport {\n  buildPluginDetailsMenuOptions,\n  extractGitHubRepo,\n  type InstallablePlugin,\n} from './pluginDetailsHelpers.js'\nimport type { ViewState as ParentViewState } from './types.js'\nimport { usePagination } from './usePagination.js'\n\ntype Props = {\n  error: string | null\n  setError: (error: string | null) => void\n  result: string | null\n  setResult: (result: string | null) => void\n  setViewState: (state: ParentViewState) => void\n  onInstallComplete?: () => void | Promise<void>\n  onSearchModeChange?: (isActive: boolean) => void\n  targetPlugin?: string\n}\n\ntype ViewState =\n  | 'plugin-list'\n  | 'plugin-details'\n  | { type: 'plugin-options'; plugin: LoadedPlugin; pluginId: string }\n\nexport function DiscoverPlugins({\n  error,\n  setError,\n  result: _result,\n  setResult,\n  setViewState: setParentViewState,\n  onInstallComplete,\n  onSearchModeChange,\n  targetPlugin,\n}: Props): React.ReactNode {\n  // View state\n  const [viewState, setViewState] = useState<ViewState>('plugin-list')\n  const [selectedPlugin, setSelectedPlugin] =\n    useState<InstallablePlugin | null>(null)\n\n  // Data state\n  const [availablePlugins, setAvailablePlugins] = useState<InstallablePlugin[]>(\n    [],\n  )\n  const [loading, setLoading] = useState(true)\n  const [installCounts, setInstallCounts] = useState<Map<\n    string,\n    number\n  > | null>(null)\n\n  // Search state\n  const [isSearchMode, setIsSearchModeRaw] = useState(false)\n  const setIsSearchMode = useCallback(\n    (active: boolean) => {\n      setIsSearchModeRaw(active)\n      onSearchModeChange?.(active)\n    },\n    [onSearchModeChange],\n  )\n  const {\n    query: searchQuery,\n    setQuery: setSearchQuery,\n    cursorOffset: searchCursorOffset,\n  } = useSearchInput({\n    isActive: viewState === 'plugin-list' && isSearchMode && !loading,\n    onExit: () => {\n      setIsSearchMode(false)\n    },\n  })\n  const isTerminalFocused = useTerminalFocus()\n  const { columns: terminalWidth } = useTerminalSize()\n\n  // Filter plugins based on search query\n  const filteredPlugins = useMemo(() => {\n    if (!searchQuery) return availablePlugins\n    const lowerQuery = searchQuery.toLowerCase()\n    return availablePlugins.filter(\n      plugin =>\n        plugin.entry.name.toLowerCase().includes(lowerQuery) ||\n        plugin.entry.description?.toLowerCase().includes(lowerQuery) ||\n        plugin.marketplaceName.toLowerCase().includes(lowerQuery),\n    )\n  }, [availablePlugins, searchQuery])\n\n  // Selection state\n  const [selectedIndex, setSelectedIndex] = useState(0)\n  const [selectedForInstall, setSelectedForInstall] = useState<Set<string>>(\n    new Set(),\n  )\n  const [installingPlugins, setInstallingPlugins] = useState<Set<string>>(\n    new Set(),\n  )\n\n  // Pagination for plugin list (continuous scrolling)\n  const pagination = usePagination<InstallablePlugin>({\n    totalItems: filteredPlugins.length,\n    selectedIndex,\n  })\n\n  // Reset selection when search query changes\n  useEffect(() => {\n    setSelectedIndex(0)\n  }, [searchQuery])\n\n  // Details view state\n  const [detailsMenuIndex, setDetailsMenuIndex] = useState(0)\n  const [isInstalling, setIsInstalling] = useState(false)\n  const [installError, setInstallError] = useState<string | null>(null)\n\n  // Warning state for non-critical errors\n  const [warning, setWarning] = useState<string | null>(null)\n\n  // Empty state reason\n  const [emptyReason, setEmptyReason] = useState<EmptyMarketplaceReason | null>(\n    null,\n  )\n\n  // Load all plugins from all marketplaces\n  useEffect(() => {\n    async function loadAllPlugins() {\n      try {\n        const config = await loadKnownMarketplacesConfig()\n\n        // Load marketplaces with graceful degradation\n        const { marketplaces, failures } =\n          await loadMarketplacesWithGracefulDegradation(config)\n\n        // Collect all plugins from all marketplaces\n        const allPlugins: InstallablePlugin[] = []\n\n        for (const { name, data: marketplace } of marketplaces) {\n          if (marketplace) {\n            for (const entry of marketplace.plugins) {\n              const pluginId = createPluginId(entry.name, name)\n              allPlugins.push({\n                entry,\n                marketplaceName: name,\n                pluginId,\n                // Only block when globally installed (user/managed scope).\n                // Project/local-scope installs don't block — user may want to\n                // promote to user scope so it's available everywhere (gh-29997).\n                isInstalled: isPluginGloballyInstalled(pluginId),\n              })\n            }\n          }\n        }\n\n        // Filter out installed and policy-blocked plugins\n        const uninstalledPlugins = allPlugins.filter(\n          p => !p.isInstalled && !isPluginBlockedByPolicy(p.pluginId),\n        )\n\n        // Fetch install counts and sort by popularity\n        try {\n          const counts = await getInstallCounts()\n          setInstallCounts(counts)\n\n          if (counts) {\n            // Sort by install count (descending), then alphabetically\n            uninstalledPlugins.sort((a, b) => {\n              const countA = counts.get(a.pluginId) ?? 0\n              const countB = counts.get(b.pluginId) ?? 0\n              if (countA !== countB) return countB - countA\n              return a.entry.name.localeCompare(b.entry.name)\n            })\n          } else {\n            // No counts available - sort alphabetically\n            uninstalledPlugins.sort((a, b) =>\n              a.entry.name.localeCompare(b.entry.name),\n            )\n          }\n        } catch (error) {\n          // Log the error, then gracefully degrade to alphabetical sort\n          logForDebugging(\n            `Failed to fetch install counts: ${errorMessage(error)}`,\n          )\n          uninstalledPlugins.sort((a, b) =>\n            a.entry.name.localeCompare(b.entry.name),\n          )\n        }\n\n        setAvailablePlugins(uninstalledPlugins)\n\n        // Detect empty reason if no plugins available\n        const configuredCount = Object.keys(config).length\n        if (uninstalledPlugins.length === 0) {\n          const reason = await detectEmptyMarketplaceReason({\n            configuredMarketplaceCount: configuredCount,\n            failedMarketplaceCount: failures.length,\n          })\n          setEmptyReason(reason)\n        }\n\n        // Handle marketplace loading errors/warnings\n        const successCount = count(marketplaces, m => m.data !== null)\n        const errorResult = formatMarketplaceLoadingErrors(\n          failures,\n          successCount,\n        )\n        if (errorResult) {\n          if (errorResult.type === 'warning') {\n            setWarning(errorResult.message + '. Showing available plugins.')\n          } else {\n            throw new Error(errorResult.message)\n          }\n        }\n\n        // Handle targetPlugin - navigate directly to plugin details\n        // Search in allPlugins (before filtering) to handle installed plugins gracefully\n        if (targetPlugin) {\n          const foundPlugin = allPlugins.find(\n            p => p.entry.name === targetPlugin,\n          )\n\n          if (foundPlugin) {\n            if (foundPlugin.isInstalled) {\n              setError(\n                `Plugin '${foundPlugin.pluginId}' is already installed. Use '/plugin' to manage existing plugins.`,\n              )\n            } else {\n              setSelectedPlugin(foundPlugin)\n              setViewState('plugin-details')\n            }\n          } else {\n            setError(`Plugin \"${targetPlugin}\" not found in any marketplace`)\n          }\n        }\n      } catch (err) {\n        setError(err instanceof Error ? err.message : 'Failed to load plugins')\n      } finally {\n        setLoading(false)\n      }\n    }\n    void loadAllPlugins()\n  }, [setError, targetPlugin])\n\n  // Install selected plugins\n  const installSelectedPlugins = async () => {\n    if (selectedForInstall.size === 0) return\n\n    const pluginsToInstall = availablePlugins.filter(p =>\n      selectedForInstall.has(p.pluginId),\n    )\n\n    setInstallingPlugins(new Set(pluginsToInstall.map(p => p.pluginId)))\n\n    let successCount = 0\n    let failureCount = 0\n    const newFailedPlugins: Array<{ name: string; reason: string }> = []\n\n    for (const plugin of pluginsToInstall) {\n      const result = await installPluginFromMarketplace({\n        pluginId: plugin.pluginId,\n        entry: plugin.entry,\n        marketplaceName: plugin.marketplaceName,\n        scope: 'user',\n      })\n\n      if (result.success) {\n        successCount++\n      } else {\n        failureCount++\n        newFailedPlugins.push({\n          name: plugin.entry.name,\n          reason: result.error,\n        })\n      }\n    }\n\n    setInstallingPlugins(new Set())\n    setSelectedForInstall(new Set())\n    clearAllCaches()\n\n    // Handle installation results\n    if (failureCount === 0) {\n      const message =\n        `✓ Installed ${successCount} ${plural(successCount, 'plugin')}. ` +\n        `Run /reload-plugins to activate.`\n      setResult(message)\n    } else if (successCount === 0) {\n      setError(\n        `Failed to install: ${formatFailureDetails(newFailedPlugins, true)}`,\n      )\n    } else {\n      const message =\n        `✓ Installed ${successCount} of ${successCount + failureCount} plugins. ` +\n        `Failed: ${formatFailureDetails(newFailedPlugins, false)}. ` +\n        `Run /reload-plugins to activate successfully installed plugins.`\n      setResult(message)\n    }\n\n    if (successCount > 0) {\n      if (onInstallComplete) {\n        await onInstallComplete()\n      }\n    }\n\n    setParentViewState({ type: 'menu' })\n  }\n\n  // Install single plugin from details view\n  const handleSinglePluginInstall = async (\n    plugin: InstallablePlugin,\n    scope: 'user' | 'project' | 'local' = 'user',\n  ) => {\n    setIsInstalling(true)\n    setInstallError(null)\n\n    const result = await installPluginFromMarketplace({\n      pluginId: plugin.pluginId,\n      entry: plugin.entry,\n      marketplaceName: plugin.marketplaceName,\n      scope,\n    })\n\n    if (result.success) {\n      const loaded = await findPluginOptionsTarget(plugin.pluginId)\n      if (loaded) {\n        setIsInstalling(false)\n        setViewState({\n          type: 'plugin-options',\n          plugin: loaded,\n          pluginId: plugin.pluginId,\n        })\n        return\n      }\n      setResult(result.message)\n      if (onInstallComplete) {\n        await onInstallComplete()\n      }\n      setParentViewState({ type: 'menu' })\n    } else {\n      setIsInstalling(false)\n      setInstallError(result.error)\n    }\n  }\n\n  // Handle error state\n  useEffect(() => {\n    if (error) {\n      setResult(error)\n    }\n  }, [error, setResult])\n\n  // Escape in plugin-details view - go back to plugin-list\n  useKeybinding(\n    'confirm:no',\n    () => {\n      setViewState('plugin-list')\n      setSelectedPlugin(null)\n    },\n    {\n      context: 'Confirmation',\n      isActive: viewState === 'plugin-details',\n    },\n  )\n\n  // Escape in plugin-list view (not search mode) - exit to parent menu\n  useKeybinding(\n    'confirm:no',\n    () => {\n      setParentViewState({ type: 'menu' })\n    },\n    {\n      context: 'Confirmation',\n      isActive: viewState === 'plugin-list' && !isSearchMode,\n    },\n  )\n\n  // Handle entering search mode (non-escape keys)\n  useInput(\n    (input, _key) => {\n      const keyIsNotCtrlOrMeta = !_key.ctrl && !_key.meta\n      if (!isSearchMode) {\n        // Enter search mode with '/' or any printable character\n        if (input === '/' && keyIsNotCtrlOrMeta) {\n          setIsSearchMode(true)\n          setSearchQuery('')\n        } else if (\n          keyIsNotCtrlOrMeta &&\n          input.length > 0 &&\n          !/^\\s+$/.test(input) &&\n          // Don't enter search mode for navigation keys\n          input !== 'j' &&\n          input !== 'k' &&\n          input !== 'i'\n        ) {\n          setIsSearchMode(true)\n          setSearchQuery(input)\n        }\n      }\n    },\n    { isActive: viewState === 'plugin-list' && !loading },\n  )\n\n  // Plugin-list navigation (non-search mode)\n  useKeybindings(\n    {\n      'select:previous': () => {\n        if (selectedIndex === 0) {\n          setIsSearchMode(true)\n        } else {\n          pagination.handleSelectionChange(selectedIndex - 1, setSelectedIndex)\n        }\n      },\n      'select:next': () => {\n        if (selectedIndex < filteredPlugins.length - 1) {\n          pagination.handleSelectionChange(selectedIndex + 1, setSelectedIndex)\n        }\n      },\n      'select:accept': () => {\n        if (\n          selectedIndex === filteredPlugins.length &&\n          selectedForInstall.size > 0\n        ) {\n          void installSelectedPlugins()\n        } else if (selectedIndex < filteredPlugins.length) {\n          const plugin = filteredPlugins[selectedIndex]\n          if (plugin) {\n            if (plugin.isInstalled) {\n              setParentViewState({\n                type: 'manage-plugins',\n                targetPlugin: plugin.entry.name,\n                targetMarketplace: plugin.marketplaceName,\n              })\n            } else {\n              setSelectedPlugin(plugin)\n              setViewState('plugin-details')\n              setDetailsMenuIndex(0)\n              setInstallError(null)\n            }\n          }\n        }\n      },\n    },\n    {\n      context: 'Select',\n      isActive: viewState === 'plugin-list' && !isSearchMode,\n    },\n  )\n\n  useKeybindings(\n    {\n      'plugin:toggle': () => {\n        if (selectedIndex < filteredPlugins.length) {\n          const plugin = filteredPlugins[selectedIndex]\n          if (plugin && !plugin.isInstalled) {\n            const newSelection = new Set(selectedForInstall)\n            if (newSelection.has(plugin.pluginId)) {\n              newSelection.delete(plugin.pluginId)\n            } else {\n              newSelection.add(plugin.pluginId)\n            }\n            setSelectedForInstall(newSelection)\n          }\n        }\n      },\n      'plugin:install': () => {\n        if (selectedForInstall.size > 0) {\n          void installSelectedPlugins()\n        }\n      },\n    },\n    {\n      context: 'Plugin',\n      isActive: viewState === 'plugin-list' && !isSearchMode,\n    },\n  )\n\n  // Plugin-details navigation\n  const detailsMenuOptions = React.useMemo(() => {\n    if (!selectedPlugin) return []\n    const hasHomepage = selectedPlugin.entry.homepage\n    const githubRepo = extractGitHubRepo(selectedPlugin)\n    return buildPluginDetailsMenuOptions(hasHomepage, githubRepo)\n  }, [selectedPlugin])\n\n  useKeybindings(\n    {\n      'select:previous': () => {\n        if (detailsMenuIndex > 0) {\n          setDetailsMenuIndex(detailsMenuIndex - 1)\n        }\n      },\n      'select:next': () => {\n        if (detailsMenuIndex < detailsMenuOptions.length - 1) {\n          setDetailsMenuIndex(detailsMenuIndex + 1)\n        }\n      },\n      'select:accept': () => {\n        if (!selectedPlugin) return\n        const action = detailsMenuOptions[detailsMenuIndex]?.action\n        const hasHomepage = selectedPlugin.entry.homepage\n        const githubRepo = extractGitHubRepo(selectedPlugin)\n        if (action === 'install-user') {\n          void handleSinglePluginInstall(selectedPlugin, 'user')\n        } else if (action === 'install-project') {\n          void handleSinglePluginInstall(selectedPlugin, 'project')\n        } else if (action === 'install-local') {\n          void handleSinglePluginInstall(selectedPlugin, 'local')\n        } else if (action === 'homepage' && hasHomepage) {\n          void openBrowser(hasHomepage)\n        } else if (action === 'github' && githubRepo) {\n          void openBrowser(`https://github.com/${githubRepo}`)\n        } else if (action === 'back') {\n          setViewState('plugin-list')\n          setSelectedPlugin(null)\n        }\n      },\n    },\n    {\n      context: 'Select',\n      isActive: viewState === 'plugin-details' && !!selectedPlugin,\n    },\n  )\n\n  if (typeof viewState === 'object' && viewState.type === 'plugin-options') {\n    const { plugin, pluginId } = viewState\n    function finish(msg: string): void {\n      setResult(msg)\n      if (onInstallComplete) {\n        void onInstallComplete()\n      }\n      setParentViewState({ type: 'menu' })\n    }\n    return (\n      <PluginOptionsFlow\n        plugin={plugin}\n        pluginId={pluginId}\n        onDone={(outcome, detail) => {\n          switch (outcome) {\n            case 'configured':\n              finish(\n                `✓ Installed and configured ${plugin.name}. Run /reload-plugins to apply.`,\n              )\n              break\n            case 'skipped':\n              finish(\n                `✓ Installed ${plugin.name}. Run /reload-plugins to apply.`,\n              )\n              break\n            case 'error':\n              finish(`Installed but failed to save config: ${detail}`)\n              break\n          }\n        }}\n      />\n    )\n  }\n\n  // Loading state\n  if (loading) {\n    return <Text>Loading…</Text>\n  }\n\n  // Error state\n  if (error) {\n    return <Text color=\"error\">{error}</Text>\n  }\n\n  // Plugin details view\n  if (viewState === 'plugin-details' && selectedPlugin) {\n    const hasHomepage = selectedPlugin.entry.homepage\n    const githubRepo = extractGitHubRepo(selectedPlugin)\n\n    const menuOptions = buildPluginDetailsMenuOptions(hasHomepage, githubRepo)\n\n    return (\n      <Box flexDirection=\"column\">\n        <Box marginBottom={1}>\n          <Text bold>Plugin details</Text>\n        </Box>\n\n        <Box flexDirection=\"column\" marginBottom={1}>\n          <Text bold>{selectedPlugin.entry.name}</Text>\n          <Text dimColor>from {selectedPlugin.marketplaceName}</Text>\n          {selectedPlugin.entry.version && (\n            <Text dimColor>Version: {selectedPlugin.entry.version}</Text>\n          )}\n          {selectedPlugin.entry.description && (\n            <Box marginTop={1}>\n              <Text>{selectedPlugin.entry.description}</Text>\n            </Box>\n          )}\n          {selectedPlugin.entry.author && (\n            <Box marginTop={1}>\n              <Text dimColor>\n                By:{' '}\n                {typeof selectedPlugin.entry.author === 'string'\n                  ? selectedPlugin.entry.author\n                  : selectedPlugin.entry.author.name}\n              </Text>\n            </Box>\n          )}\n        </Box>\n\n        <PluginTrustWarning />\n\n        {installError && (\n          <Box marginBottom={1}>\n            <Text color=\"error\">Error: {installError}</Text>\n          </Box>\n        )}\n\n        <Box flexDirection=\"column\">\n          {menuOptions.map((option, index) => (\n            <Box key={option.action}>\n              {detailsMenuIndex === index && <Text>{'> '}</Text>}\n              {detailsMenuIndex !== index && <Text>{'  '}</Text>}\n              <Text bold={detailsMenuIndex === index}>\n                {isInstalling && option.action.startsWith('install-')\n                  ? 'Installing…'\n                  : option.label}\n              </Text>\n            </Box>\n          ))}\n        </Box>\n\n        <Box marginTop={1}>\n          <Text dimColor>\n            <Byline>\n              <ConfigurableShortcutHint\n                action=\"select:accept\"\n                context=\"Select\"\n                fallback=\"Enter\"\n                description=\"select\"\n              />\n              <ConfigurableShortcutHint\n                action=\"confirm:no\"\n                context=\"Confirmation\"\n                fallback=\"Esc\"\n                description=\"back\"\n              />\n            </Byline>\n          </Text>\n        </Box>\n      </Box>\n    )\n  }\n\n  // Empty state\n  if (availablePlugins.length === 0) {\n    return (\n      <Box flexDirection=\"column\">\n        <Box marginBottom={1}>\n          <Text bold>Discover plugins</Text>\n        </Box>\n        <EmptyStateMessage reason={emptyReason} />\n        <Box marginTop={1}>\n          <Text dimColor italic>\n            Esc to go back\n          </Text>\n        </Box>\n      </Box>\n    )\n  }\n\n  // Get visible plugins from pagination\n  const visiblePlugins = pagination.getVisibleItems(filteredPlugins)\n\n  return (\n    <Box flexDirection=\"column\">\n      <Box>\n        <Text bold>Discover plugins</Text>\n        {pagination.needsPagination && (\n          <Text dimColor>\n            {' '}\n            ({pagination.scrollPosition.current}/\n            {pagination.scrollPosition.total})\n          </Text>\n        )}\n      </Box>\n\n      {/* Search box */}\n      <Box marginBottom={1}>\n        <SearchBox\n          query={searchQuery}\n          isFocused={isSearchMode}\n          isTerminalFocused={isTerminalFocused}\n          width={terminalWidth - 4}\n          cursorOffset={searchCursorOffset}\n        />\n      </Box>\n\n      {/* Warning banner */}\n      {warning && (\n        <Box marginBottom={1}>\n          <Text color=\"warning\">\n            {figures.warning} {warning}\n          </Text>\n        </Box>\n      )}\n\n      {/* No search results */}\n      {filteredPlugins.length === 0 && searchQuery && (\n        <Box marginBottom={1}>\n          <Text dimColor>No plugins match &quot;{searchQuery}&quot;</Text>\n        </Box>\n      )}\n\n      {/* Scroll up indicator */}\n      {pagination.scrollPosition.canScrollUp && (\n        <Box>\n          <Text dimColor> {figures.arrowUp} more above</Text>\n        </Box>\n      )}\n\n      {/* Plugin list - use startIndex in key to force re-render on scroll */}\n      {visiblePlugins.map((plugin, visibleIndex) => {\n        const actualIndex = pagination.toActualIndex(visibleIndex)\n        const isSelected = selectedIndex === actualIndex\n        const isSelectedForInstall = selectedForInstall.has(plugin.pluginId)\n        const isInstallingThis = installingPlugins.has(plugin.pluginId)\n        const isLast = visibleIndex === visiblePlugins.length - 1\n\n        return (\n          <Box\n            key={`${pagination.startIndex}-${plugin.pluginId}`}\n            flexDirection=\"column\"\n            marginBottom={isLast && !error ? 0 : 1}\n          >\n            <Box>\n              <Text\n                color={isSelected && !isSearchMode ? 'suggestion' : undefined}\n              >\n                {isSelected && !isSearchMode ? figures.pointer : ' '}{' '}\n              </Text>\n              <Text>\n                {isInstallingThis\n                  ? figures.ellipsis\n                  : isSelectedForInstall\n                    ? figures.radioOn\n                    : figures.radioOff}{' '}\n                {plugin.entry.name}\n                <Text dimColor> · {plugin.marketplaceName}</Text>\n                {plugin.entry.tags?.includes('community-managed') && (\n                  <Text dimColor> [Community Managed]</Text>\n                )}\n                {installCounts &&\n                  plugin.marketplaceName === OFFICIAL_MARKETPLACE_NAME && (\n                    <Text dimColor>\n                      {' · '}\n                      {formatInstallCount(\n                        installCounts.get(plugin.pluginId) ?? 0,\n                      )}{' '}\n                      installs\n                    </Text>\n                  )}\n              </Text>\n            </Box>\n            {plugin.entry.description && (\n              <Box marginLeft={4}>\n                <Text dimColor>\n                  {truncateToWidth(plugin.entry.description, 60)}\n                </Text>\n              </Box>\n            )}\n          </Box>\n        )\n      })}\n\n      {/* Scroll down indicator */}\n      {pagination.scrollPosition.canScrollDown && (\n        <Box>\n          <Text dimColor> {figures.arrowDown} more below</Text>\n        </Box>\n      )}\n\n      {/* Error messages */}\n      {error && (\n        <Box marginTop={1}>\n          <Text color=\"error\">\n            {figures.cross} {error}\n          </Text>\n        </Box>\n      )}\n\n      <DiscoverPluginsKeyHint\n        hasSelection={selectedForInstall.size > 0}\n        canToggle={\n          selectedIndex < filteredPlugins.length &&\n          !filteredPlugins[selectedIndex]?.isInstalled\n        }\n      />\n    </Box>\n  )\n}\n\nfunction DiscoverPluginsKeyHint({\n  hasSelection,\n  canToggle,\n}: {\n  hasSelection: boolean\n  canToggle: boolean\n}): React.ReactNode {\n  return (\n    <Box marginTop={1}>\n      <Text dimColor italic>\n        <Byline>\n          {hasSelection && (\n            <ConfigurableShortcutHint\n              action=\"plugin:install\"\n              context=\"Plugin\"\n              fallback=\"i\"\n              description=\"install\"\n              bold\n            />\n          )}\n          <Text>type to search</Text>\n          {canToggle && (\n            <ConfigurableShortcutHint\n              action=\"plugin:toggle\"\n              context=\"Plugin\"\n              fallback=\"Space\"\n              description=\"toggle\"\n            />\n          )}\n          <ConfigurableShortcutHint\n            action=\"select:accept\"\n            context=\"Select\"\n            fallback=\"Enter\"\n            description=\"details\"\n          />\n          <ConfigurableShortcutHint\n            action=\"confirm:no\"\n            context=\"Confirmation\"\n            fallback=\"Esc\"\n            description=\"back\"\n          />\n        </Byline>\n      </Text>\n    </Box>\n  )\n}\n\n/**\n * Context-aware empty state message for the Discover screen\n */\nfunction EmptyStateMessage({\n  reason,\n}: {\n  reason: EmptyMarketplaceReason | null\n}): React.ReactNode {\n  switch (reason) {\n    case 'git-not-installed':\n      return (\n        <>\n          <Text dimColor>Git is required to install marketplaces.</Text>\n          <Text dimColor>Please install git and restart Claude Code.</Text>\n        </>\n      )\n    case 'all-blocked-by-policy':\n      return (\n        <>\n          <Text dimColor>\n            Your organization policy does not allow any external marketplaces.\n          </Text>\n          <Text dimColor>Contact your administrator.</Text>\n        </>\n      )\n    case 'policy-restricts-sources':\n      return (\n        <>\n          <Text dimColor>\n            Your organization restricts which marketplaces can be added.\n          </Text>\n          <Text dimColor>\n            Switch to the Marketplaces tab to view allowed sources.\n          </Text>\n        </>\n      )\n    case 'all-marketplaces-failed':\n      return (\n        <>\n          <Text dimColor>Failed to load marketplace data.</Text>\n          <Text dimColor>Check your network connection.</Text>\n        </>\n      )\n    case 'all-plugins-installed':\n      return (\n        <>\n          <Text dimColor>All available plugins are already installed.</Text>\n          <Text dimColor>\n            Check for new plugins later or add more marketplaces.\n          </Text>\n        </>\n      )\n    case 'no-marketplaces-configured':\n    default:\n      return (\n        <>\n          <Text dimColor>No plugins available.</Text>\n          <Text dimColor>\n            Add a marketplace first using the Marketplaces tab.\n          </Text>\n        </>\n      )\n  }\n}\n"],"mappings":"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","ignoreList":[]}