import { useRef, useMemo, memo } from 'react' import { isCursorInSketchCommandRange } from 'lang/util' import { engineCommandManager, kclManager } from 'lib/singletons' import { useModelingContext } from 'hooks/useModelingContext' import { useCommandsContext } from 'hooks/useCommandsContext' import { useNetworkContext } from 'hooks/useNetworkContext' import { NetworkHealthState } from 'hooks/useNetworkStatus' import { ActionButton } from 'components/ActionButton' import { isSingleCursorInPipe } from 'lang/queryAst' import { useKclContext } from 'lang/KclProvider' import { ActionButtonDropdown } from 'components/ActionButtonDropdown' import { useHotkeys } from 'react-hotkeys-hook' import Tooltip from 'components/Tooltip' import { useAppState } from 'AppState' import { CustomIcon } from 'components/CustomIcon' import { toolbarConfig, ToolbarItem, ToolbarItemCallbackProps, ToolbarItemResolved, ToolbarModeName, } from 'lib/toolbar' import { isDesktop } from 'lib/isDesktop' import { openExternalBrowserIfDesktop } from 'lib/openWindow' export function Toolbar({ className = '', ...props }: React.HTMLAttributes) { const { state, send, context } = useModelingContext() const { commandBarSend } = useCommandsContext() const iconClassName = 'group-disabled:text-chalkboard-50 !text-inherit dark:group-enabled:group-hover:!text-inherit' const bgClassName = '!bg-transparent' const buttonBgClassName = 'bg-chalkboard-transparent dark:bg-transparent disabled:bg-transparent dark:disabled:bg-transparent enabled:hover:bg-chalkboard-10 dark:enabled:hover:bg-chalkboard-100 pressed:!bg-primary pressed:enabled:hover:!text-chalkboard-10' const buttonBorderClassName = '!border-transparent hover:!border-chalkboard-20 dark:enabled:hover:!border-primary pressed:!border-primary ui-open:!border-primary' const sketchPathId = useMemo(() => { if (!isSingleCursorInPipe(context.selectionRanges, kclManager.ast)) { return false } return isCursorInSketchCommandRange( engineCommandManager.artifactGraph, context.selectionRanges ) }, [engineCommandManager.artifactGraph, context.selectionRanges]) const toolbarButtonsRef = useRef(null) const { overallState } = useNetworkContext() const { isExecuting } = useKclContext() const { isStreamReady } = useAppState() const disableAllButtons = (overallState !== NetworkHealthState.Ok && overallState !== NetworkHealthState.Weak) || isExecuting || !isStreamReady const currentMode = (Object.entries(toolbarConfig).find(([_, mode]) => mode.check(state) )?.[0] as ToolbarModeName) || 'modeling' /** These are the props that will be passed to the callbacks in the toolbar config * They are memoized to prevent unnecessary re-renders, * but they still get a lot of churn from the state machine * so I think there's a lot of room for improvement here */ const configCallbackProps: ToolbarItemCallbackProps = useMemo( () => ({ modelingState: state, modelingSend: send, commandBarSend, sketchPathId, }), [state, send, commandBarSend, sketchPathId] ) /** * Resolve all the callbacks and values for the current mode, * so we don't need to worry about the other modes */ const currentModeItems: ( | ToolbarItemResolved | ToolbarItemResolved[] | 'break' )[] = useMemo(() => { return toolbarConfig[currentMode].items.map((maybeIconConfig) => { if (maybeIconConfig === 'break') { return 'break' } else if (Array.isArray(maybeIconConfig)) { return maybeIconConfig.map(resolveItemConfig) } else { return resolveItemConfig(maybeIconConfig) } }) function resolveItemConfig( maybeIconConfig: ToolbarItem ): ToolbarItemResolved { const isDisabled = disableAllButtons || maybeIconConfig.status !== 'available' || maybeIconConfig.disabled?.(state) === true return { ...maybeIconConfig, title: typeof maybeIconConfig.title === 'string' ? maybeIconConfig.title : maybeIconConfig.title(configCallbackProps), description: maybeIconConfig.description, links: maybeIconConfig.links || [], isActive: maybeIconConfig.isActive?.(state), hotkey: typeof maybeIconConfig.hotkey === 'string' ? maybeIconConfig.hotkey : maybeIconConfig.hotkey?.(state), disabled: isDisabled, disabledReason: typeof maybeIconConfig.disabledReason === 'function' ? maybeIconConfig.disabledReason(state) : maybeIconConfig.disabledReason, disableHotkey: maybeIconConfig.disableHotkey?.(state), status: maybeIconConfig.status, } } }, [currentMode, disableAllButtons, configCallbackProps]) return (
    {/* A menu item will either be a vertical line break, a button with a dropdown, or a single button */} {currentModeItems.map((maybeIconConfig, i) => { // Vertical Line Break if (maybeIconConfig === 'break') { return (
    ) } else if (Array.isArray(maybeIconConfig)) { // A button with a dropdown return ( ({ id: itemConfig.id, label: itemConfig.title, hotkey: itemConfig.hotkey, onClick: () => itemConfig.onClick(configCallbackProps), disabled: disableAllButtons || itemConfig.status !== 'available' || itemConfig.disabled === true, status: itemConfig.status, }))} > maybeIconConfig[0].onClick(configCallbackProps) } > {maybeIconConfig[0].title} ) } const itemConfig = maybeIconConfig // A single button return (
    itemConfig.onClick(configCallbackProps)} > {itemConfig.title}
    ) })}
{state.matches('Sketch no face') && (

Select a plane or face to start sketching

)}
) } /** * The single button and dropdown button share content, so we extract it here * It contains a tooltip with the title, description, and links * and a hotkey listener */ const ToolbarItemTooltip = memo(function ToolbarItemContents({ itemConfig, configCallbackProps, }: { itemConfig: ToolbarItemResolved configCallbackProps: ToolbarItemCallbackProps }) { const { state } = useModelingContext() useHotkeys( itemConfig.hotkey || '', () => { itemConfig.onClick(configCallbackProps) }, { enabled: itemConfig.status === 'available' && !!itemConfig.hotkey && !itemConfig.disabled && !itemConfig.disableHotkey, } ) return (
{itemConfig.title} {itemConfig.status === 'available' && itemConfig.hotkey ? ( {itemConfig.hotkey} ) : itemConfig.status === 'kcl-only' ? ( <> KCL code only ) : ( itemConfig.status === 'unavailable' && ( <> In development ) )}

{itemConfig.description}

{/* Add disabled reason if item is disabled */} {itemConfig.disabled && itemConfig.disabledReason && ( <>

{typeof itemConfig.disabledReason === 'function' ? itemConfig.disabledReason(state) : itemConfig.disabledReason}

)} {itemConfig.links.length > 0 && ( <>
)}
) })