import { useOpenCopilotRightDrawer } from '@/activities/copilot/right-drawer/hooks/useOpenCopilotRightDrawer'; import { copilotQueryState } from '@/activities/copilot/right-drawer/states/copilotQueryState'; import { useOpenActivityRightDrawer } from '@/activities/hooks/useOpenActivityRightDrawer'; import { Note } from '@/activities/types/Note'; import { Task } from '@/activities/types/Task'; import { CommandGroup } from '@/command-menu/components/CommandGroup'; import { CommandMenuItem } from '@/command-menu/components/CommandMenuItem'; import { CommandMenuTopBar } from '@/command-menu/components/CommandMenuTopBar'; import { COMMAND_MENU_SEARCH_BAR_HEIGHT } from '@/command-menu/constants/CommandMenuSearchBarHeight'; import { COMMAND_MENU_SEARCH_BAR_PADDING } from '@/command-menu/constants/CommandMenuSearchBarPadding'; import { useCommandMenu } from '@/command-menu/hooks/useCommandMenu'; import { commandMenuCommandsComponentSelector } from '@/command-menu/states/commandMenuCommandsSelector'; import { commandMenuSearchState } from '@/command-menu/states/commandMenuSearchState'; import { isCommandMenuOpenedState } from '@/command-menu/states/isCommandMenuOpenedState'; import { Command, CommandScope, CommandType, } from '@/command-menu/types/Command'; import { Company } from '@/companies/types/Company'; import { contextStoreNumberOfSelectedRecordsComponentState } from '@/context-store/states/contextStoreNumberOfSelectedRecordsComponentState'; import { contextStoreTargetedRecordsRuleComponentState } from '@/context-store/states/contextStoreTargetedRecordsRuleComponentState'; import { useKeyboardShortcutMenu } from '@/keyboard-shortcut-menu/hooks/useKeyboardShortcutMenu'; import { CoreObjectNamePlural } from '@/object-metadata/types/CoreObjectNamePlural'; import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; import { getCompanyDomainName } from '@/object-metadata/utils/getCompanyDomainName'; import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; import { useMultiObjectSearch } from '@/object-record/relation-picker/hooks/useMultiObjectSearch'; import { useMultiObjectSearchQueryResultFormattedAsObjectRecordsMap } from '@/object-record/relation-picker/hooks/useMultiObjectSearchQueryResultFormattedAsObjectRecordsMap'; import { makeOrFilterVariables } from '@/object-record/utils/makeOrFilterVariables'; import { SelectableItem } from '@/ui/layout/selectable-list/components/SelectableItem'; import { SelectableList } from '@/ui/layout/selectable-list/components/SelectableList'; import { useScopedHotkeys } from '@/ui/utilities/hotkey/hooks/useScopedHotkeys'; import { AppHotkeyScope } from '@/ui/utilities/hotkey/types/AppHotkeyScope'; import { useListenClickOutsideV2 } from '@/ui/utilities/pointer-event/hooks/useListenClickOutsideV2'; import { useIsMobile } from '@/ui/utilities/responsive/hooks/useIsMobile'; import { ScrollWrapper } from '@/ui/utilities/scroll/components/ScrollWrapper'; import { useRecoilComponentValueV2 } from '@/ui/utilities/state/component-state/hooks/useRecoilComponentValueV2'; import { useSetRecoilComponentStateV2 } from '@/ui/utilities/state/component-state/hooks/useSetRecoilComponentStateV2'; import { useIsFeatureEnabled } from '@/workspace/hooks/useIsFeatureEnabled'; import styled from '@emotion/styled'; import { isNonEmptyString } from '@sniptt/guards'; import isEmpty from 'lodash.isempty'; import { useMemo, useRef } from 'react'; import { useRecoilState, useRecoilValue, useSetRecoilState } from 'recoil'; import { Key } from 'ts-key-enum'; import { Avatar, IconCheckbox, IconComponent, IconNotes, IconSparkles, isDefined, } from 'twenty-ui'; import { useDebounce } from 'use-debounce'; import { getLogoUrlFromDomainName } from '~/utils'; import { capitalize } from '~/utils/string/capitalize'; const MOBILE_NAVIGATION_BAR_HEIGHT = 64; type CommandGroupConfig = { heading: string; items?: any[]; renderItem: (item: any) => { id: string; Icon?: IconComponent; label: string; to?: string; onClick?: () => void; key?: string; firstHotKey?: string; secondHotKey?: string; }; }; const StyledCommandMenu = styled.div` background: ${({ theme }) => theme.background.secondary}; border-left: 1px solid ${({ theme }) => theme.border.color.medium}; box-shadow: ${({ theme }) => theme.boxShadow.strong}; font-family: ${({ theme }) => theme.font.family}; height: 100%; overflow: hidden; padding: 0; position: fixed; right: 0%; top: 0%; width: ${() => (useIsMobile() ? '100%' : '500px')}; z-index: 1000; `; const StyledList = styled.div` background: ${({ theme }) => theme.background.secondary}; overscroll-behavior: contain; transition: 100ms ease; transition-property: height; `; const StyledInnerList = styled.div<{ isMobile: boolean }>` max-height: ${({ isMobile }) => isMobile ? `calc(100dvh - ${COMMAND_MENU_SEARCH_BAR_HEIGHT}px - ${ COMMAND_MENU_SEARCH_BAR_PADDING * 2 }px - ${MOBILE_NAVIGATION_BAR_HEIGHT}px)` : `calc(100dvh - ${COMMAND_MENU_SEARCH_BAR_HEIGHT}px - ${ COMMAND_MENU_SEARCH_BAR_PADDING * 2 }px)`}; padding-left: ${({ theme }) => theme.spacing(2)}; padding-right: ${({ theme }) => theme.spacing(2)}; padding-top: ${({ theme }) => theme.spacing(1)}; width: calc(100% - ${({ theme }) => theme.spacing(4)}); `; const StyledEmpty = styled.div` align-items: center; color: ${({ theme }) => theme.font.color.light}; display: flex; font-size: ${({ theme }) => theme.font.size.md}; height: 64px; justify-content: center; white-space: pre-wrap; `; export const CommandMenu = () => { const { toggleCommandMenu, onItemClick, closeCommandMenu } = useCommandMenu(); const commandMenuRef = useRef(null); const openActivityRightDrawer = useOpenActivityRightDrawer({ objectNameSingular: CoreObjectNameSingular.Note, }); const isCommandMenuOpened = useRecoilValue(isCommandMenuOpenedState); const [commandMenuSearch, setCommandMenuSearch] = useRecoilState( commandMenuSearchState, ); const [deferredCommandMenuSearch] = useDebounce(commandMenuSearch, 300); // 200ms - 500ms const { closeKeyboardShortcutMenu } = useKeyboardShortcutMenu(); const setContextStoreTargetedRecordsRule = useSetRecoilComponentStateV2( contextStoreTargetedRecordsRuleComponentState, ); const setContextStoreNumberOfSelectedRecords = useSetRecoilComponentStateV2( contextStoreNumberOfSelectedRecordsComponentState, ); const isMobile = useIsMobile(); const commandMenuCommands = useRecoilComponentValueV2( commandMenuCommandsComponentSelector, ); useScopedHotkeys( 'ctrl+k,meta+k', () => { closeKeyboardShortcutMenu(); toggleCommandMenu(); }, AppHotkeyScope.CommandMenu, [toggleCommandMenu], ); useScopedHotkeys( [Key.Escape], () => { closeCommandMenu(); }, AppHotkeyScope.CommandMenuOpen, [closeCommandMenu], ); useScopedHotkeys( [Key.Backspace, Key.Delete], () => { if (!isNonEmptyString(commandMenuSearch)) { setContextStoreTargetedRecordsRule({ mode: 'selection', selectedRecordIds: [], }); setContextStoreNumberOfSelectedRecords(0); } }, AppHotkeyScope.CommandMenuOpen, [closeCommandMenu], { preventDefault: false, }, ); const { matchesSearchFilterObjectRecordsQueryResult, matchesSearchFilterObjectRecordsLoading: loading, } = useMultiObjectSearch({ excludedObjects: [CoreObjectNameSingular.Task, CoreObjectNameSingular.Note], searchFilterValue: deferredCommandMenuSearch ?? undefined, limit: 3, }); const { objectRecordsMap: matchesSearchFilterObjectRecords } = useMultiObjectSearchQueryResultFormattedAsObjectRecordsMap({ multiObjectRecordsQueryResult: matchesSearchFilterObjectRecordsQueryResult, }); const { loading: isNotesLoading, records: notes } = useFindManyRecords({ skip: !isCommandMenuOpened, objectNameSingular: CoreObjectNameSingular.Note, filter: deferredCommandMenuSearch ? makeOrFilterVariables([ { title: { ilike: `%${deferredCommandMenuSearch}%` } }, { body: { ilike: `%${deferredCommandMenuSearch}%` } }, ]) : undefined, limit: 3, }); const { loading: isTasksLoading, records: tasks } = useFindManyRecords({ skip: !isCommandMenuOpened, objectNameSingular: CoreObjectNameSingular.Task, filter: deferredCommandMenuSearch ? makeOrFilterVariables([ { title: { ilike: `%${deferredCommandMenuSearch}%` } }, { body: { ilike: `%${deferredCommandMenuSearch}%` } }, ]) : undefined, limit: 3, }); const people = matchesSearchFilterObjectRecords.people?.map( (people) => people.record, ); const companies = matchesSearchFilterObjectRecords.companies?.map( (companies) => companies.record, ); const opportunities = matchesSearchFilterObjectRecords.opportunities?.map( (opportunities) => opportunities.record, ); const customObjectRecordsMap = useMemo(() => { return Object.fromEntries( Object.entries(matchesSearchFilterObjectRecords).filter( ([namePlural, records]) => ![ CoreObjectNamePlural.Person, CoreObjectNamePlural.Opportunity, CoreObjectNamePlural.Company, ].includes(namePlural as CoreObjectNamePlural) && !isEmpty(records), ), ); }, [matchesSearchFilterObjectRecords]); const peopleCommands = useMemo( () => people?.map(({ id, name: { firstName, lastName } }) => ({ id, label: `${firstName} ${lastName}`, to: `object/person/${id}`, })), [people], ); const companyCommands = useMemo( () => companies?.map(({ id, name }) => ({ id, label: name ?? '', to: `object/company/${id}`, })), [companies], ); const opportunityCommands = useMemo( () => opportunities?.map(({ id, name }) => ({ id, label: name ?? '', to: `object/opportunity/${id}`, })), [opportunities], ); const noteCommands = useMemo( () => notes?.map((note) => ({ id: note.id, label: note.title ?? '', to: '', onCommandClick: () => openActivityRightDrawer(note.id), })), [notes, openActivityRightDrawer], ); const tasksCommands = useMemo( () => tasks?.map((task) => ({ id: task.id, label: task.title ?? '', to: '', onCommandClick: () => openActivityRightDrawer(task.id), })), [tasks, openActivityRightDrawer], ); const customObjectCommands = useMemo(() => { const customObjectCommandsArray: Command[] = []; Object.values(customObjectRecordsMap).forEach((objectRecords) => { customObjectCommandsArray.push( ...objectRecords.map((objectRecord) => ({ id: objectRecord.record.id, label: objectRecord.recordIdentifier.name, to: `object/${objectRecord.objectMetadataItem.nameSingular}/${objectRecord.record.id}`, })), ); }); return customObjectCommandsArray; }, [customObjectRecordsMap]); const otherCommands = useMemo(() => { const commandsArray: Command[] = []; if (peopleCommands?.length > 0) { commandsArray.push(...(peopleCommands as Command[])); } if (companyCommands?.length > 0) { commandsArray.push(...(companyCommands as Command[])); } if (opportunityCommands?.length > 0) { commandsArray.push(...(opportunityCommands as Command[])); } if (noteCommands?.length > 0) { commandsArray.push(...(noteCommands as Command[])); } if (tasksCommands?.length > 0) { commandsArray.push(...(tasksCommands as Command[])); } if (customObjectCommands?.length > 0) { commandsArray.push(...(customObjectCommands as Command[])); } return commandsArray; }, [ peopleCommands, companyCommands, opportunityCommands, noteCommands, customObjectCommands, tasksCommands, ]); const checkInShortcuts = (cmd: Command, search: string) => { return (cmd.firstHotKey + (cmd.secondHotKey ?? '')) .toLowerCase() .includes(search.toLowerCase()); }; const checkInLabels = (cmd: Command, search: string) => { if (isNonEmptyString(cmd.label)) { return cmd.label.toLowerCase().includes(search.toLowerCase()); } return false; }; const matchingNavigateCommand = commandMenuCommands.filter( (cmd) => (deferredCommandMenuSearch.length > 0 ? checkInShortcuts(cmd, deferredCommandMenuSearch) || checkInLabels(cmd, deferredCommandMenuSearch) : true) && cmd.type === CommandType.Navigate, ); const matchingCreateCommand = commandMenuCommands.filter( (cmd) => (deferredCommandMenuSearch.length > 0 ? checkInShortcuts(cmd, deferredCommandMenuSearch) || checkInLabels(cmd, deferredCommandMenuSearch) : true) && cmd.type === CommandType.Create, ); const matchingStandardActionRecordSelectionCommands = commandMenuCommands.filter( (cmd) => (deferredCommandMenuSearch.length > 0 ? checkInShortcuts(cmd, deferredCommandMenuSearch) || checkInLabels(cmd, deferredCommandMenuSearch) : true) && cmd.type === CommandType.StandardAction && cmd.scope === CommandScope.RecordSelection, ); const matchingStandardActionGlobalCommands = commandMenuCommands.filter( (cmd) => (deferredCommandMenuSearch.length > 0 ? checkInShortcuts(cmd, deferredCommandMenuSearch) || checkInLabels(cmd, deferredCommandMenuSearch) : true) && cmd.type === CommandType.StandardAction && cmd.scope === CommandScope.Global, ); const matchingWorkflowRunRecordSelectionCommands = commandMenuCommands.filter( (cmd) => (deferredCommandMenuSearch.length > 0 ? checkInShortcuts(cmd, deferredCommandMenuSearch) || checkInLabels(cmd, deferredCommandMenuSearch) : true) && cmd.type === CommandType.WorkflowRun && cmd.scope === CommandScope.RecordSelection, ); const matchingWorkflowRunGlobalCommands = commandMenuCommands.filter( (cmd) => (deferredCommandMenuSearch.length > 0 ? checkInShortcuts(cmd, deferredCommandMenuSearch) || checkInLabels(cmd, deferredCommandMenuSearch) : true) && cmd.type === CommandType.WorkflowRun && cmd.scope === CommandScope.Global, ); useListenClickOutsideV2({ refs: [commandMenuRef], callback: closeCommandMenu, listenerId: 'COMMAND_MENU_LISTENER_ID', hotkeyScope: AppHotkeyScope.CommandMenuOpen, }); const isCopilotEnabled = useIsFeatureEnabled('IS_COPILOT_ENABLED'); const setCopilotQuery = useSetRecoilState(copilotQueryState); const openCopilotRightDrawer = useOpenCopilotRightDrawer(); const copilotCommand: Command = { id: 'copilot', to: '', // TODO Icon: IconSparkles, label: 'Open Copilot', type: CommandType.Navigate, onCommandClick: () => { setCopilotQuery(deferredCommandMenuSearch); openCopilotRightDrawer(); }, }; const copilotCommands: Command[] = isCopilotEnabled ? [copilotCommand] : []; const selectableItemIds = copilotCommands .map((cmd) => cmd.id) .concat(matchingStandardActionRecordSelectionCommands.map((cmd) => cmd.id)) .concat(matchingWorkflowRunRecordSelectionCommands.map((cmd) => cmd.id)) .concat(matchingStandardActionGlobalCommands.map((cmd) => cmd.id)) .concat(matchingWorkflowRunGlobalCommands.map((cmd) => cmd.id)) .concat(matchingCreateCommand.map((cmd) => cmd.id)) .concat(matchingNavigateCommand.map((cmd) => cmd.id)) .concat(people?.map((person) => person.id)) .concat(companies?.map((company) => company.id)) .concat(opportunities?.map((opportunity) => opportunity.id)) .concat(notes?.map((note) => note.id)) .concat(tasks?.map((task) => task.id)) .concat( Object.values(customObjectRecordsMap) ?.map((objectRecords) => objectRecords.map((objectRecord) => objectRecord.record.id), ) .flat() ?? [], ); const isNoResults = !matchingStandardActionRecordSelectionCommands.length && !matchingWorkflowRunRecordSelectionCommands.length && !matchingStandardActionGlobalCommands.length && !matchingWorkflowRunGlobalCommands.length && !matchingCreateCommand.length && !matchingNavigateCommand.length && !people?.length && !companies?.length && !notes?.length && !tasks?.length && !opportunities?.length && isEmpty(customObjectRecordsMap); const isLoading = loading || isNotesLoading || isTasksLoading; const commandGroups: CommandGroupConfig[] = [ { heading: 'Navigate', items: matchingNavigateCommand, renderItem: (command) => ({ id: command.id, Icon: command.Icon, label: command.label, to: command.to, onClick: command.onCommandClick, firstHotKey: command.firstHotKey, secondHotKey: command.secondHotKey, }), }, { heading: 'Other', items: matchingCreateCommand, renderItem: (command) => ({ id: command.id, Icon: command.Icon, label: command.label, to: command.to, onClick: command.onCommandClick, firstHotKey: command.firstHotKey, secondHotKey: command.secondHotKey, }), }, { heading: 'People', items: people, renderItem: (person) => ({ id: person.id, label: `${person.name.firstName} ${person.name.lastName}`, to: `object/person/${person.id}`, Icon: () => ( ), firstHotKey: person.firstHotKey, secondHotKey: person.secondHotKey, }), }, { heading: 'Companies', items: companies, renderItem: (company) => ({ id: company.id, label: company.name, to: `object/company/${company.id}`, Icon: () => ( ), firstHotKey: company.firstHotKey, secondHotKey: company.secondHotKey, }), }, { heading: 'Opportunities', items: opportunities, renderItem: (opportunity) => ({ id: opportunity.id, label: opportunity.name ?? '', to: `object/opportunity/${opportunity.id}`, Icon: () => ( ), }), }, { heading: 'Notes', items: notes, renderItem: (note) => ({ id: note.id, Icon: IconNotes, label: note.title ?? '', onClick: () => openActivityRightDrawer(note.id), }), }, { heading: 'Tasks', items: tasks, renderItem: (task) => ({ id: task.id, Icon: IconCheckbox, label: task.title ?? '', onClick: () => openActivityRightDrawer(task.id), }), }, ...Object.entries(customObjectRecordsMap).map( ([customObjectNamePlural, objectRecords]): CommandGroupConfig => ({ heading: capitalize(customObjectNamePlural), items: objectRecords, renderItem: (objectRecord) => ({ key: objectRecord.record.id, id: objectRecord.record.id, label: objectRecord.recordIdentifier.name, to: `object/${objectRecord.objectMetadataItem.nameSingular}/${objectRecord.record.id}`, Icon: () => ( ), }), }), ), ]; return ( <> {isCommandMenuOpened && ( { const command = [ ...copilotCommands, ...commandMenuCommands, ...otherCommands, ].find((cmd) => cmd.id === itemId); if (isDefined(command)) { const { to, onCommandClick } = command; onItemClick(onCommandClick, to); } }} > {isNoResults && !isLoading && ( No results found )} {isCopilotEnabled && ( 2 ? `"${deferredCommandMenuSearch}"` : '' }`} onClick={copilotCommand.onCommandClick} firstHotKey={copilotCommand.firstHotKey} secondHotKey={copilotCommand.secondHotKey} /> )} {matchingStandardActionRecordSelectionCommands?.map( (standardActionrecordSelectionCommand) => ( ), )} {matchingWorkflowRunRecordSelectionCommands?.map( (workflowRunRecordSelectionCommand) => ( ), )} {matchingStandardActionGlobalCommands?.length > 0 && ( {matchingStandardActionGlobalCommands?.map( (standardActionGlobalCommand) => ( ), )} )} {matchingWorkflowRunGlobalCommands?.length > 0 && ( {matchingWorkflowRunGlobalCommands?.map( (workflowRunGlobalCommand) => ( ), )} )} {commandGroups.map(({ heading, items, renderItem }) => items?.length ? ( {items.map((item) => { const { id, Icon, label, to, onClick, key, firstHotKey, secondHotKey, } = renderItem(item); return ( ); })} ) : null, )} )} ); };