source dump of claude code
at main 194 lines 22 kB view raw
1import { c as _c } from "react/compiler-runtime"; 2/** 3 * Hook for LSP plugin recommendations 4 * 5 * Detects file edits and recommends LSP plugins when: 6 * - File extension matches an LSP plugin 7 * - LSP binary is already installed on the system 8 * - Plugin is not already installed 9 * - User hasn't disabled recommendations 10 * 11 * Only shows one recommendation per session. 12 */ 13 14import { extname, join } from 'path'; 15import * as React from 'react'; 16import { hasShownLspRecommendationThisSession, setLspRecommendationShownThisSession } from '../bootstrap/state.js'; 17import { useNotifications } from '../context/notifications.js'; 18import { useAppState } from '../state/AppState.js'; 19import { saveGlobalConfig } from '../utils/config.js'; 20import { logForDebugging } from '../utils/debug.js'; 21import { logError } from '../utils/log.js'; 22import { addToNeverSuggest, getMatchingLspPlugins, incrementIgnoredCount } from '../utils/plugins/lspRecommendation.js'; 23import { cacheAndRegisterPlugin } from '../utils/plugins/pluginInstallationHelpers.js'; 24import { getSettingsForSource, updateSettingsForSource } from '../utils/settings/settings.js'; 25import { installPluginAndNotify, usePluginRecommendationBase } from './usePluginRecommendationBase.js'; 26 27// Threshold for detecting timeout vs explicit dismiss (ms) 28// Menu auto-dismisses at 30s, so anything over 28s is likely timeout 29const TIMEOUT_THRESHOLD_MS = 28_000; 30export type LspRecommendationState = { 31 pluginId: string; 32 pluginName: string; 33 pluginDescription?: string; 34 fileExtension: string; 35 shownAt: number; // Timestamp for timeout detection 36} | null; 37type UseLspPluginRecommendationResult = { 38 recommendation: LspRecommendationState; 39 handleResponse: (response: 'yes' | 'no' | 'never' | 'disable') => void; 40}; 41export function useLspPluginRecommendation() { 42 const $ = _c(12); 43 const trackedFiles = useAppState(_temp); 44 const { 45 addNotification 46 } = useNotifications(); 47 let t0; 48 if ($[0] === Symbol.for("react.memo_cache_sentinel")) { 49 t0 = new Set(); 50 $[0] = t0; 51 } else { 52 t0 = $[0]; 53 } 54 const checkedFilesRef = React.useRef(t0); 55 const { 56 recommendation, 57 clearRecommendation, 58 tryResolve 59 } = usePluginRecommendationBase(); 60 let t1; 61 let t2; 62 if ($[1] !== trackedFiles || $[2] !== tryResolve) { 63 t1 = () => { 64 tryResolve(async () => { 65 if (hasShownLspRecommendationThisSession()) { 66 return null; 67 } 68 const newFiles = []; 69 for (const file of trackedFiles) { 70 if (!checkedFilesRef.current.has(file)) { 71 checkedFilesRef.current.add(file); 72 newFiles.push(file); 73 } 74 } 75 for (const filePath of newFiles) { 76 ; 77 try { 78 const matches = await getMatchingLspPlugins(filePath); 79 const match = matches[0]; 80 if (match) { 81 logForDebugging(`[useLspPluginRecommendation] Found match: ${match.pluginName} for ${filePath}`); 82 setLspRecommendationShownThisSession(true); 83 return { 84 pluginId: match.pluginId, 85 pluginName: match.pluginName, 86 pluginDescription: match.description, 87 fileExtension: extname(filePath), 88 shownAt: Date.now() 89 }; 90 } 91 } catch (t3) { 92 const error = t3; 93 logError(error); 94 } 95 } 96 return null; 97 }); 98 }; 99 t2 = [trackedFiles, tryResolve]; 100 $[1] = trackedFiles; 101 $[2] = tryResolve; 102 $[3] = t1; 103 $[4] = t2; 104 } else { 105 t1 = $[3]; 106 t2 = $[4]; 107 } 108 React.useEffect(t1, t2); 109 let t3; 110 if ($[5] !== addNotification || $[6] !== clearRecommendation || $[7] !== recommendation) { 111 t3 = response => { 112 if (!recommendation) { 113 return; 114 } 115 const { 116 pluginId, 117 pluginName, 118 shownAt 119 } = recommendation; 120 logForDebugging(`[useLspPluginRecommendation] User response: ${response} for ${pluginName}`); 121 bb60: switch (response) { 122 case "yes": 123 { 124 installPluginAndNotify(pluginId, pluginName, "lsp-plugin", addNotification, async pluginData => { 125 logForDebugging(`[useLspPluginRecommendation] Installing plugin: ${pluginId}`); 126 const localSourcePath = typeof pluginData.entry.source === "string" ? join(pluginData.marketplaceInstallLocation, pluginData.entry.source) : undefined; 127 await cacheAndRegisterPlugin(pluginId, pluginData.entry, "user", undefined, localSourcePath); 128 const settings = getSettingsForSource("userSettings"); 129 updateSettingsForSource("userSettings", { 130 enabledPlugins: { 131 ...settings?.enabledPlugins, 132 [pluginId]: true 133 } 134 }); 135 logForDebugging(`[useLspPluginRecommendation] Plugin installed: ${pluginId}`); 136 }); 137 break bb60; 138 } 139 case "no": 140 { 141 const elapsed = Date.now() - shownAt; 142 if (elapsed >= TIMEOUT_THRESHOLD_MS) { 143 logForDebugging(`[useLspPluginRecommendation] Timeout detected (${elapsed}ms), incrementing ignored count`); 144 incrementIgnoredCount(); 145 } 146 break bb60; 147 } 148 case "never": 149 { 150 addToNeverSuggest(pluginId); 151 break bb60; 152 } 153 case "disable": 154 { 155 saveGlobalConfig(_temp2); 156 } 157 } 158 clearRecommendation(); 159 }; 160 $[5] = addNotification; 161 $[6] = clearRecommendation; 162 $[7] = recommendation; 163 $[8] = t3; 164 } else { 165 t3 = $[8]; 166 } 167 const handleResponse = t3; 168 let t4; 169 if ($[9] !== handleResponse || $[10] !== recommendation) { 170 t4 = { 171 recommendation, 172 handleResponse 173 }; 174 $[9] = handleResponse; 175 $[10] = recommendation; 176 $[11] = t4; 177 } else { 178 t4 = $[11]; 179 } 180 return t4; 181} 182function _temp2(current) { 183 if (current.lspRecommendationDisabled) { 184 return current; 185 } 186 return { 187 ...current, 188 lspRecommendationDisabled: true 189 }; 190} 191function _temp(s) { 192 return s.fileHistory.trackedFiles; 193} 194//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["extname","join","React","hasShownLspRecommendationThisSession","setLspRecommendationShownThisSession","useNotifications","useAppState","saveGlobalConfig","logForDebugging","logError","addToNeverSuggest","getMatchingLspPlugins","incrementIgnoredCount","cacheAndRegisterPlugin","getSettingsForSource","updateSettingsForSource","installPluginAndNotify","usePluginRecommendationBase","TIMEOUT_THRESHOLD_MS","LspRecommendationState","pluginId","pluginName","pluginDescription","fileExtension","shownAt","UseLspPluginRecommendationResult","recommendation","handleResponse","response","useLspPluginRecommendation","$","_c","trackedFiles","_temp","addNotification","t0","Symbol","for","Set","checkedFilesRef","useRef","clearRecommendation","tryResolve","t1","t2","newFiles","file","current","has","add","push","filePath","matches","match","description","Date","now","t3","error","useEffect","bb60","pluginData","localSourcePath","entry","source","marketplaceInstallLocation","undefined","settings","enabledPlugins","elapsed","_temp2","t4","lspRecommendationDisabled","s","fileHistory"],"sources":["useLspPluginRecommendation.tsx"],"sourcesContent":["/**\n * Hook for LSP plugin recommendations\n *\n * Detects file edits and recommends LSP plugins when:\n * - File extension matches an LSP plugin\n * - LSP binary is already installed on the system\n * - Plugin is not already installed\n * - User hasn't disabled recommendations\n *\n * Only shows one recommendation per session.\n */\n\nimport { extname, join } from 'path'\nimport * as React from 'react'\nimport {\n  hasShownLspRecommendationThisSession,\n  setLspRecommendationShownThisSession,\n} from '../bootstrap/state.js'\nimport { useNotifications } from '../context/notifications.js'\nimport { useAppState } from '../state/AppState.js'\nimport { saveGlobalConfig } from '../utils/config.js'\nimport { logForDebugging } from '../utils/debug.js'\nimport { logError } from '../utils/log.js'\nimport {\n  addToNeverSuggest,\n  getMatchingLspPlugins,\n  incrementIgnoredCount,\n} from '../utils/plugins/lspRecommendation.js'\nimport { cacheAndRegisterPlugin } from '../utils/plugins/pluginInstallationHelpers.js'\nimport {\n  getSettingsForSource,\n  updateSettingsForSource,\n} from '../utils/settings/settings.js'\nimport {\n  installPluginAndNotify,\n  usePluginRecommendationBase,\n} from './usePluginRecommendationBase.js'\n\n// Threshold for detecting timeout vs explicit dismiss (ms)\n// Menu auto-dismisses at 30s, so anything over 28s is likely timeout\nconst TIMEOUT_THRESHOLD_MS = 28_000\n\nexport type LspRecommendationState = {\n  pluginId: string\n  pluginName: string\n  pluginDescription?: string\n  fileExtension: string\n  shownAt: number // Timestamp for timeout detection\n} | null\n\ntype UseLspPluginRecommendationResult = {\n  recommendation: LspRecommendationState\n  handleResponse: (response: 'yes' | 'no' | 'never' | 'disable') => void\n}\n\nexport function useLspPluginRecommendation(): UseLspPluginRecommendationResult {\n  const trackedFiles = useAppState(s => s.fileHistory.trackedFiles)\n  const { addNotification } = useNotifications()\n  const checkedFilesRef = React.useRef<Set<string>>(new Set())\n  const { recommendation, clearRecommendation, tryResolve } =\n    usePluginRecommendationBase<NonNullable<LspRecommendationState>>()\n\n  React.useEffect(() => {\n    tryResolve(async () => {\n      if (hasShownLspRecommendationThisSession()) return null\n\n      const newFiles: string[] = []\n      for (const file of trackedFiles) {\n        if (!checkedFilesRef.current.has(file)) {\n          checkedFilesRef.current.add(file)\n          newFiles.push(file)\n        }\n      }\n\n      for (const filePath of newFiles) {\n        try {\n          const matches = await getMatchingLspPlugins(filePath)\n          const match = matches[0] // official plugins prioritized\n          if (match) {\n            logForDebugging(\n              `[useLspPluginRecommendation] Found match: ${match.pluginName} for ${filePath}`,\n            )\n            setLspRecommendationShownThisSession(true)\n            return {\n              pluginId: match.pluginId,\n              pluginName: match.pluginName,\n              pluginDescription: match.description,\n              fileExtension: extname(filePath),\n              shownAt: Date.now(),\n            }\n          }\n        } catch (error) {\n          logError(error)\n        }\n      }\n      return null\n    })\n  }, [trackedFiles, tryResolve])\n\n  const handleResponse = React.useCallback(\n    (response: 'yes' | 'no' | 'never' | 'disable') => {\n      if (!recommendation) return\n\n      const { pluginId, pluginName, shownAt } = recommendation\n\n      logForDebugging(\n        `[useLspPluginRecommendation] User response: ${response} for ${pluginName}`,\n      )\n\n      switch (response) {\n        case 'yes':\n          void installPluginAndNotify(\n            pluginId,\n            pluginName,\n            'lsp-plugin',\n            addNotification,\n            async pluginData => {\n              logForDebugging(\n                `[useLspPluginRecommendation] Installing plugin: ${pluginId}`,\n              )\n              const localSourcePath =\n                typeof pluginData.entry.source === 'string'\n                  ? join(\n                      pluginData.marketplaceInstallLocation,\n                      pluginData.entry.source,\n                    )\n                  : undefined\n              await cacheAndRegisterPlugin(\n                pluginId,\n                pluginData.entry,\n                'user',\n                undefined, // projectPath - not needed for user scope\n                localSourcePath,\n              )\n              // Enable in user settings so it loads on restart\n              const settings = getSettingsForSource('userSettings')\n              updateSettingsForSource('userSettings', {\n                enabledPlugins: {\n                  ...settings?.enabledPlugins,\n                  [pluginId]: true,\n                },\n              })\n              logForDebugging(\n                `[useLspPluginRecommendation] Plugin installed: ${pluginId}`,\n              )\n            },\n          )\n          break\n\n        case 'no': {\n          const elapsed = Date.now() - shownAt\n          if (elapsed >= TIMEOUT_THRESHOLD_MS) {\n            logForDebugging(\n              `[useLspPluginRecommendation] Timeout detected (${elapsed}ms), incrementing ignored count`,\n            )\n            incrementIgnoredCount()\n          }\n          break\n        }\n\n        case 'never':\n          addToNeverSuggest(pluginId)\n          break\n\n        case 'disable':\n          saveGlobalConfig(current => {\n            if (current.lspRecommendationDisabled) return current\n            return { ...current, lspRecommendationDisabled: true }\n          })\n          break\n      }\n\n      clearRecommendation()\n    },\n    [recommendation, addNotification, clearRecommendation],\n  )\n\n  return { recommendation, handleResponse }\n}\n"],"mappings":"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","ignoreList":[]}