mirror of
https://github.com/aljazceru/goose.git
synced 2025-12-24 01:24:28 +01:00
feat/fix(ui): Mac keyboard shortcuts (#2430)
This commit is contained in:
@@ -425,7 +425,8 @@ export default function App() {
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
console.log('Setting up keyboard shortcuts');
|
console.log('Setting up keyboard shortcuts');
|
||||||
const handleKeyDown = (event: KeyboardEvent) => {
|
const handleKeyDown = (event: KeyboardEvent) => {
|
||||||
if ((event.metaKey || event.ctrlKey) && event.key === 'n') {
|
const isMac = window.electron.platform === 'darwin';
|
||||||
|
if ((isMac ? event.metaKey : event.ctrlKey) && event.key === 'n') {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
try {
|
try {
|
||||||
const workingDir = window.appConfig.get('GOOSE_WORKING_DIR');
|
const workingDir = window.appConfig.get('GOOSE_WORKING_DIR');
|
||||||
|
|||||||
@@ -18,6 +18,10 @@ interface SearchBarProps {
|
|||||||
count: number;
|
count: number;
|
||||||
currentIndex: number;
|
currentIndex: number;
|
||||||
};
|
};
|
||||||
|
/** Optional ref for the search input element */
|
||||||
|
inputRef?: React.RefObject<HTMLInputElement>;
|
||||||
|
/** Initial search term */
|
||||||
|
initialSearchTerm?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -35,12 +39,15 @@ export const SearchBar: React.FC<SearchBarProps> = ({
|
|||||||
onClose,
|
onClose,
|
||||||
onNavigate,
|
onNavigate,
|
||||||
searchResults,
|
searchResults,
|
||||||
|
inputRef: externalInputRef,
|
||||||
|
initialSearchTerm = '',
|
||||||
}) => {
|
}) => {
|
||||||
const [searchTerm, setSearchTerm] = useState('');
|
const [searchTerm, setSearchTerm] = useState(initialSearchTerm);
|
||||||
const [displayTerm, setDisplayTerm] = useState(''); // For immediate visual feedback
|
const [displayTerm, setDisplayTerm] = useState(initialSearchTerm); // For immediate visual feedback
|
||||||
const [caseSensitive, setCaseSensitive] = useState(false);
|
const [caseSensitive, setCaseSensitive] = useState(false);
|
||||||
const [isExiting, setIsExiting] = useState(false);
|
const [isExiting, setIsExiting] = useState(false);
|
||||||
const inputRef = React.useRef<HTMLInputElement>(null);
|
const internalInputRef = React.useRef<HTMLInputElement>(null);
|
||||||
|
const inputRef = externalInputRef || internalInputRef;
|
||||||
|
|
||||||
// Create debounced search function
|
// Create debounced search function
|
||||||
const debouncedSearch = useCallback(
|
const debouncedSearch = useCallback(
|
||||||
@@ -54,7 +61,16 @@ export const SearchBar: React.FC<SearchBarProps> = ({
|
|||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
inputRef.current?.focus();
|
inputRef.current?.focus();
|
||||||
}, []);
|
}, [inputRef]);
|
||||||
|
|
||||||
|
// Handle changes to initialSearchTerm
|
||||||
|
useEffect(() => {
|
||||||
|
if (initialSearchTerm) {
|
||||||
|
setSearchTerm(initialSearchTerm);
|
||||||
|
setDisplayTerm(initialSearchTerm);
|
||||||
|
debouncedSearch(initialSearchTerm, caseSensitive);
|
||||||
|
}
|
||||||
|
}, [initialSearchTerm, caseSensitive, debouncedSearch]);
|
||||||
|
|
||||||
// Cleanup debounced function on unmount
|
// Cleanup debounced function on unmount
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
|||||||
@@ -22,11 +22,13 @@ export const SearchView: React.FC<PropsWithChildren<SearchViewProps>> = ({
|
|||||||
children,
|
children,
|
||||||
}) => {
|
}) => {
|
||||||
const [isSearchVisible, setIsSearchVisible] = useState(false);
|
const [isSearchVisible, setIsSearchVisible] = useState(false);
|
||||||
|
const [initialSearchTerm, setInitialSearchTerm] = useState('');
|
||||||
const [searchResults, setSearchResults] = useState<{
|
const [searchResults, setSearchResults] = useState<{
|
||||||
currentIndex: number;
|
currentIndex: number;
|
||||||
count: number;
|
count: number;
|
||||||
} | null>(null);
|
} | null>(null);
|
||||||
|
|
||||||
|
const searchInputRef = React.useRef<HTMLInputElement>(null);
|
||||||
const highlighterRef = React.useRef<SearchHighlighter | null>(null);
|
const highlighterRef = React.useRef<SearchHighlighter | null>(null);
|
||||||
const containerRef = React.useRef<HTMLDivElement | null>(null);
|
const containerRef = React.useRef<HTMLDivElement | null>(null);
|
||||||
const lastSearchRef = React.useRef<{ term: string; caseSensitive: boolean }>({
|
const lastSearchRef = React.useRef<{ term: string; caseSensitive: boolean }>({
|
||||||
@@ -58,38 +60,14 @@ export const SearchView: React.FC<PropsWithChildren<SearchViewProps>> = ({
|
|||||||
[]
|
[]
|
||||||
);
|
);
|
||||||
|
|
||||||
// Clean up highlighter and debounced functions on unmount
|
|
||||||
useEffect(() => {
|
|
||||||
return () => {
|
|
||||||
if (highlighterRef.current) {
|
|
||||||
highlighterRef.current.destroy();
|
|
||||||
highlighterRef.current = null;
|
|
||||||
}
|
|
||||||
debouncedHighlight.cancel?.();
|
|
||||||
};
|
|
||||||
}, [debouncedHighlight]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
const handleKeyDown = (e: KeyboardEvent) => {
|
|
||||||
if ((e.metaKey || e.ctrlKey) && e.key === 'f') {
|
|
||||||
e.preventDefault();
|
|
||||||
setIsSearchVisible(true);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
window.addEventListener('keydown', handleKeyDown);
|
|
||||||
return () => {
|
|
||||||
window.removeEventListener('keydown', handleKeyDown);
|
|
||||||
};
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handles the search operation when a user enters a search term.
|
* Handles the search operation when a user enters a search term.
|
||||||
* Uses debouncing to prevent excessive highlighting operations.
|
* Uses debouncing to prevent excessive highlighting operations.
|
||||||
* @param term - The text to search for
|
* @param term - The text to search for
|
||||||
* @param caseSensitive - Whether to perform a case-sensitive search
|
* @param caseSensitive - Whether to perform a case-sensitive search
|
||||||
*/
|
*/
|
||||||
const handleSearch = (term: string, caseSensitive: boolean) => {
|
const handleSearch = useCallback(
|
||||||
|
(term: string, caseSensitive: boolean) => {
|
||||||
// Store the latest search parameters
|
// Store the latest search parameters
|
||||||
lastSearchRef.current = { term, caseSensitive };
|
lastSearchRef.current = { term, caseSensitive };
|
||||||
|
|
||||||
@@ -125,13 +103,16 @@ export const SearchView: React.FC<PropsWithChildren<SearchViewProps>> = ({
|
|||||||
|
|
||||||
// Debounce the highlight operation
|
// Debounce the highlight operation
|
||||||
debouncedHighlight(term, caseSensitive, highlighterRef.current);
|
debouncedHighlight(term, caseSensitive, highlighterRef.current);
|
||||||
};
|
},
|
||||||
|
[debouncedHighlight]
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Navigates between search results in the specified direction.
|
* Navigates between search results in the specified direction.
|
||||||
* @param direction - Direction to navigate ('next' or 'prev')
|
* @param direction - Direction to navigate ('next' or 'prev')
|
||||||
*/
|
*/
|
||||||
const navigateResults = (direction: 'next' | 'prev') => {
|
const navigateResults = useCallback(
|
||||||
|
(direction: 'next' | 'prev') => {
|
||||||
if (!searchResults || searchResults.count === 0 || !highlighterRef.current) return;
|
if (!searchResults || searchResults.count === 0 || !highlighterRef.current) return;
|
||||||
|
|
||||||
let newIndex: number;
|
let newIndex: number;
|
||||||
@@ -155,12 +136,42 @@ export const SearchView: React.FC<PropsWithChildren<SearchViewProps>> = ({
|
|||||||
});
|
});
|
||||||
|
|
||||||
highlighterRef.current.setCurrentMatch(newIndex, true); // Explicitly scroll when navigating
|
highlighterRef.current.setCurrentMatch(newIndex, true); // Explicitly scroll when navigating
|
||||||
};
|
},
|
||||||
|
[searchResults]
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleFindCommand = useCallback(() => {
|
||||||
|
if (isSearchVisible && searchInputRef.current) {
|
||||||
|
searchInputRef.current.focus();
|
||||||
|
searchInputRef.current.select();
|
||||||
|
} else {
|
||||||
|
setIsSearchVisible(true);
|
||||||
|
}
|
||||||
|
}, [isSearchVisible]);
|
||||||
|
|
||||||
|
const handleFindNext = useCallback(() => {
|
||||||
|
if (isSearchVisible) {
|
||||||
|
navigateResults('next');
|
||||||
|
}
|
||||||
|
}, [isSearchVisible, navigateResults]);
|
||||||
|
|
||||||
|
const handleFindPrevious = useCallback(() => {
|
||||||
|
if (isSearchVisible) {
|
||||||
|
navigateResults('prev');
|
||||||
|
}
|
||||||
|
}, [isSearchVisible, navigateResults]);
|
||||||
|
|
||||||
|
const handleUseSelectionFind = useCallback(() => {
|
||||||
|
const selection = window.getSelection()?.toString().trim();
|
||||||
|
if (selection) {
|
||||||
|
setInitialSearchTerm(selection);
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Closes the search interface and cleans up highlights.
|
* Closes the search interface and cleans up highlights.
|
||||||
*/
|
*/
|
||||||
const handleCloseSearch = () => {
|
const handleCloseSearch = useCallback(() => {
|
||||||
setIsSearchVisible(false);
|
setIsSearchVisible(false);
|
||||||
setSearchResults(null);
|
setSearchResults(null);
|
||||||
if (highlighterRef.current) {
|
if (highlighterRef.current) {
|
||||||
@@ -168,7 +179,83 @@ export const SearchView: React.FC<PropsWithChildren<SearchViewProps>> = ({
|
|||||||
}
|
}
|
||||||
// Cancel any pending highlight operations
|
// Cancel any pending highlight operations
|
||||||
debouncedHighlight.cancel?.();
|
debouncedHighlight.cancel?.();
|
||||||
|
}, [debouncedHighlight]);
|
||||||
|
|
||||||
|
// Clean up highlighter and debounced functions on unmount
|
||||||
|
useEffect(() => {
|
||||||
|
return () => {
|
||||||
|
if (highlighterRef.current) {
|
||||||
|
highlighterRef.current.destroy();
|
||||||
|
highlighterRef.current = null;
|
||||||
|
}
|
||||||
|
debouncedHighlight.cancel?.();
|
||||||
};
|
};
|
||||||
|
}, [debouncedHighlight]);
|
||||||
|
|
||||||
|
// Listen for keyboard events
|
||||||
|
useEffect(() => {
|
||||||
|
const handleKeyDown = (e: KeyboardEvent) => {
|
||||||
|
const isMac = window.electron.platform === 'darwin';
|
||||||
|
|
||||||
|
// Handle ⌘F/Ctrl+F to show/focus search
|
||||||
|
if ((isMac ? e.metaKey : e.ctrlKey) && !e.shiftKey && e.key === 'f') {
|
||||||
|
e.preventDefault();
|
||||||
|
if (isSearchVisible && searchInputRef.current) {
|
||||||
|
// If search is already visible, focus and select the input
|
||||||
|
searchInputRef.current.focus();
|
||||||
|
searchInputRef.current.select();
|
||||||
|
} else {
|
||||||
|
// Otherwise show the search UI
|
||||||
|
setIsSearchVisible(true);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle ⌘E to use selection for find (Mac only)
|
||||||
|
if (isMac && e.metaKey && !e.shiftKey && e.key === 'e') {
|
||||||
|
// Don't handle ⌘E if we're in the search input - let the native behavior work
|
||||||
|
if (e.target instanceof HTMLInputElement && e.target.id === 'search-input') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
e.preventDefault();
|
||||||
|
handleUseSelectionFind();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Only handle ⌘G and ⇧⌘G if search is visible (Mac only)
|
||||||
|
if (isSearchVisible && isMac && e.metaKey && e.key === 'g') {
|
||||||
|
e.preventDefault();
|
||||||
|
if (e.shiftKey) {
|
||||||
|
// ⇧⌘G - Find Previous
|
||||||
|
navigateResults('prev');
|
||||||
|
} else {
|
||||||
|
// ⌘G - Find Next
|
||||||
|
navigateResults('next');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
window.addEventListener('keydown', handleKeyDown);
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener('keydown', handleKeyDown);
|
||||||
|
};
|
||||||
|
}, [isSearchVisible, navigateResults, handleSearch, handleUseSelectionFind]);
|
||||||
|
|
||||||
|
// Listen for Find menu commands
|
||||||
|
useEffect(() => {
|
||||||
|
window.electron.on('find-command', handleFindCommand);
|
||||||
|
window.electron.on('find-next', handleFindNext);
|
||||||
|
window.electron.on('find-previous', handleFindPrevious);
|
||||||
|
window.electron.on('use-selection-find', handleUseSelectionFind);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
window.electron.off('find-command', handleFindCommand);
|
||||||
|
window.electron.off('find-next', handleFindNext);
|
||||||
|
window.electron.off('find-previous', handleFindPrevious);
|
||||||
|
window.electron.off('use-selection-find', handleUseSelectionFind);
|
||||||
|
};
|
||||||
|
}, [handleFindCommand, handleFindNext, handleFindPrevious, handleUseSelectionFind]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div ref={containerRef} className={`search-container ${className}`}>
|
<div ref={containerRef} className={`search-container ${className}`}>
|
||||||
@@ -178,6 +265,8 @@ export const SearchView: React.FC<PropsWithChildren<SearchViewProps>> = ({
|
|||||||
onClose={handleCloseSearch}
|
onClose={handleCloseSearch}
|
||||||
onNavigate={navigateResults}
|
onNavigate={navigateResults}
|
||||||
searchResults={searchResults}
|
searchResults={searchResults}
|
||||||
|
inputRef={searchInputRef}
|
||||||
|
initialSearchTerm={initialSearchTerm}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{children}
|
{children}
|
||||||
|
|||||||
@@ -805,6 +805,59 @@ app.whenReady().then(async () => {
|
|||||||
appMenu.submenu.insert(1, new MenuItem({ type: 'separator' }));
|
appMenu.submenu.insert(1, new MenuItem({ type: 'separator' }));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Add Find submenu to Edit menu
|
||||||
|
const editMenu = menu?.items.find((item) => item.label === 'Edit');
|
||||||
|
if (editMenu?.submenu) {
|
||||||
|
// Find the index of Select All to insert after it
|
||||||
|
const selectAllIndex = editMenu.submenu.items.findIndex((item) => item.label === 'Select All');
|
||||||
|
|
||||||
|
// Create Find submenu
|
||||||
|
const findSubmenu = Menu.buildFromTemplate([
|
||||||
|
{
|
||||||
|
label: 'Find…',
|
||||||
|
accelerator: process.platform === 'darwin' ? 'Command+F' : 'Control+F',
|
||||||
|
click() {
|
||||||
|
const focusedWindow = BrowserWindow.getFocusedWindow();
|
||||||
|
if (focusedWindow) focusedWindow.webContents.send('find-command');
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Find Next',
|
||||||
|
accelerator: process.platform === 'darwin' ? 'Command+G' : 'Control+G',
|
||||||
|
click() {
|
||||||
|
const focusedWindow = BrowserWindow.getFocusedWindow();
|
||||||
|
if (focusedWindow) focusedWindow.webContents.send('find-next');
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Find Previous',
|
||||||
|
accelerator: process.platform === 'darwin' ? 'Shift+Command+G' : 'Shift+Control+G',
|
||||||
|
click() {
|
||||||
|
const focusedWindow = BrowserWindow.getFocusedWindow();
|
||||||
|
if (focusedWindow) focusedWindow.webContents.send('find-previous');
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Use Selection for Find',
|
||||||
|
accelerator: process.platform === 'darwin' ? 'Command+E' : null,
|
||||||
|
click() {
|
||||||
|
const focusedWindow = BrowserWindow.getFocusedWindow();
|
||||||
|
if (focusedWindow) focusedWindow.webContents.send('use-selection-find');
|
||||||
|
},
|
||||||
|
visible: process.platform === 'darwin', // Only show on Mac
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Add Find submenu to Edit menu
|
||||||
|
editMenu.submenu.insert(
|
||||||
|
selectAllIndex + 1,
|
||||||
|
new MenuItem({
|
||||||
|
label: 'Find',
|
||||||
|
submenu: findSubmenu,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
// Add Environment menu items to View menu
|
// Add Environment menu items to View menu
|
||||||
const viewMenu = menu?.items.find((item) => item.label === 'View');
|
const viewMenu = menu?.items.find((item) => item.label === 'View');
|
||||||
if (viewMenu?.submenu) {
|
if (viewMenu?.submenu) {
|
||||||
@@ -826,8 +879,20 @@ app.whenReady().then(async () => {
|
|||||||
const fileMenu = menu?.items.find((item) => item.label === 'File');
|
const fileMenu = menu?.items.find((item) => item.label === 'File');
|
||||||
|
|
||||||
if (fileMenu?.submenu) {
|
if (fileMenu?.submenu) {
|
||||||
// open goose to specific dir and set that as its working space
|
fileMenu.submenu.insert(
|
||||||
fileMenu.submenu.append(
|
0,
|
||||||
|
new MenuItem({
|
||||||
|
label: 'New Chat Window',
|
||||||
|
accelerator: process.platform === 'darwin' ? 'Cmd+N' : 'Ctrl+N',
|
||||||
|
click() {
|
||||||
|
ipcMain.emit('create-chat-window');
|
||||||
|
},
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
// Open goose to specific dir and set that as its working space
|
||||||
|
fileMenu.submenu.insert(
|
||||||
|
1,
|
||||||
new MenuItem({
|
new MenuItem({
|
||||||
label: 'Open Directory...',
|
label: 'Open Directory...',
|
||||||
accelerator: 'CmdOrCtrl+O',
|
accelerator: 'CmdOrCtrl+O',
|
||||||
@@ -838,8 +903,8 @@ app.whenReady().then(async () => {
|
|||||||
// Add Recent Files submenu
|
// Add Recent Files submenu
|
||||||
const recentFilesSubmenu = buildRecentFilesMenu();
|
const recentFilesSubmenu = buildRecentFilesMenu();
|
||||||
if (recentFilesSubmenu.length > 0) {
|
if (recentFilesSubmenu.length > 0) {
|
||||||
fileMenu.submenu.append(new MenuItem({ type: 'separator' }));
|
fileMenu.submenu.insert(
|
||||||
fileMenu.submenu.append(
|
2,
|
||||||
new MenuItem({
|
new MenuItem({
|
||||||
label: 'Recent Directories',
|
label: 'Recent Directories',
|
||||||
submenu: recentFilesSubmenu,
|
submenu: recentFilesSubmenu,
|
||||||
@@ -847,19 +912,12 @@ app.whenReady().then(async () => {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add menu items to File menu
|
fileMenu.submenu.insert(3, new MenuItem({ type: 'separator' }));
|
||||||
fileMenu.submenu.append(
|
|
||||||
new MenuItem({
|
|
||||||
label: 'New Chat Window',
|
|
||||||
accelerator: 'CmdOrCtrl+N',
|
|
||||||
click() {
|
|
||||||
ipcMain.emit('create-chat-window');
|
|
||||||
},
|
|
||||||
})
|
|
||||||
);
|
|
||||||
|
|
||||||
// Add menu item for hotkey
|
// The Close Window item is here.
|
||||||
fileMenu?.submenu.append(
|
|
||||||
|
// Add menu item to tell the user about the keyboard shortcut
|
||||||
|
fileMenu.submenu.append(
|
||||||
new MenuItem({
|
new MenuItem({
|
||||||
label: 'Focus Goose Window',
|
label: 'Focus Goose Window',
|
||||||
accelerator: 'CmdOrCtrl+Alt+Shift+G',
|
accelerator: 'CmdOrCtrl+Alt+Shift+G',
|
||||||
|
|||||||
@@ -25,6 +25,7 @@ const config = JSON.parse(process.argv.find((arg) => arg.startsWith('{')) || '{}
|
|||||||
|
|
||||||
// Define the API types in a single place
|
// Define the API types in a single place
|
||||||
type ElectronAPI = {
|
type ElectronAPI = {
|
||||||
|
platform: string;
|
||||||
reactReady: () => void;
|
reactReady: () => void;
|
||||||
getConfig: () => Record<string, unknown>;
|
getConfig: () => Record<string, unknown>;
|
||||||
hideWindow: () => void;
|
hideWindow: () => void;
|
||||||
@@ -68,6 +69,7 @@ type AppConfigAPI = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const electronAPI: ElectronAPI = {
|
const electronAPI: ElectronAPI = {
|
||||||
|
platform: process.platform,
|
||||||
reactReady: () => ipcRenderer.send('react-ready'),
|
reactReady: () => ipcRenderer.send('react-ready'),
|
||||||
getConfig: () => config,
|
getConfig: () => config,
|
||||||
hideWindow: () => ipcRenderer.send('hide-window'),
|
hideWindow: () => ipcRenderer.send('hide-window'),
|
||||||
|
|||||||
Reference in New Issue
Block a user