* Tweak toaster look and feel * Add icons, tweak plus icon names * Rename commandBarMeta to commandBarConfig * Refactor command bar, add support for icons * Create a tailwind plugin for aria-pressed button state * Remove overlay from behind command bar * Clean up toolbar * Button and other style tweaks * Icon tweaks follow-up: make old icons work with new sizing * Delete unused static icons * More CSS tweaks * Small CSS tweak to project sidebar * Add command bar E2E test * fumpt * A snapshot a day keeps the bugs away! 📷🐛 (OS: ubuntu) * fix typo in a comment * Fix icon padding (built version only) * Update onboarding and warning banner icons padding * Misc minor style fixes * Get Extrude opening and canceling from command bar * Iconography tweaks * Get extrude kind of working * Refactor command bar config types and organization * Move command bar configs to be co-located with each other * Start building a state machine for the command bar * Start converting command bar to state machine * Add support for multiple args, confirmation step * Submission behavior, hotkeys, code organization * Add new test for extruding from command bar * Polish step back and selection hotkeys, CSS tweaks * Loading style tweaks * Validate selection inputs, polish UX of args re-editing * Prevent submission with multiple selection on singlular arg * Remove stray console logs * Tweak test, CSS nit, remove extrude "result" argument * Fix linting warnings * Show Ctrl+/ instead of ⌘K on all platforms but Mac * A snapshot a day keeps the bugs away! 📷🐛 (OS: ubuntu) * Add "Enter sketch" to command bar * fix command bar test * Fix flaky cmd bar extrude test by waiting for engine select response * Cover both button labels '⌘K' and 'Ctrl+/' in test --------- Co-authored-by: github-actions[bot] <github-actions[bot]@users.noreply.github.com>
		
			
				
	
	
		
			222 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			222 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { WheelEvent, useRef, useMemo } from 'react'
 | |
| import { isCursorInSketchCommandRange } from 'lang/util'
 | |
| import { engineCommandManager } from './lang/std/engineConnection'
 | |
| import { useModelingContext } from 'hooks/useModelingContext'
 | |
| import { useCommandsContext } from 'hooks/useCommandsContext'
 | |
| import { ActionButton } from 'components/ActionButton'
 | |
| import usePlatform from 'hooks/usePlatform'
 | |
| 
 | |
| export const Toolbar = () => {
 | |
|   const platform = usePlatform()
 | |
|   const { commandBarSend } = useCommandsContext()
 | |
|   const { state, send, context } = useModelingContext()
 | |
|   const toolbarButtonsRef = useRef<HTMLUListElement>(null)
 | |
|   const bgClassName =
 | |
|     'group-enabled:group-hover:bg-energy-10 group-pressed:bg-energy-10 dark:group-enabled:group-hover:bg-chalkboard-80 dark:group-pressed:bg-chalkboard-80'
 | |
|   const pathId = useMemo(
 | |
|     () =>
 | |
|       isCursorInSketchCommandRange(
 | |
|         engineCommandManager.artifactMap,
 | |
|         context.selectionRanges
 | |
|       ),
 | |
|     [engineCommandManager.artifactMap, context.selectionRanges]
 | |
|   )
 | |
| 
 | |
|   function handleToolbarButtonsWheelEvent(ev: WheelEvent<HTMLSpanElement>) {
 | |
|     const span = toolbarButtonsRef.current
 | |
|     if (!span) {
 | |
|       return
 | |
|     }
 | |
| 
 | |
|     span.scrollLeft = span.scrollLeft += ev.deltaY
 | |
|   }
 | |
| 
 | |
|   function ToolbarButtons({
 | |
|     className = '',
 | |
|     ...props
 | |
|   }: React.HTMLAttributes<HTMLElement>) {
 | |
|     return (
 | |
|       <ul
 | |
|         {...props}
 | |
|         ref={toolbarButtonsRef}
 | |
|         onWheel={handleToolbarButtonsWheelEvent}
 | |
|         className={
 | |
|           'm-0 py-1 rounded-l-sm flex gap-2 items-center overflow-x-auto ' +
 | |
|           className
 | |
|         }
 | |
|         style={{ scrollbarWidth: 'thin' }}
 | |
|       >
 | |
|         {state.nextEvents.includes('Enter sketch') && (
 | |
|           <li className="contents">
 | |
|             <ActionButton
 | |
|               Element="button"
 | |
|               onClick={() => send({ type: 'Enter sketch' })}
 | |
|               icon={{
 | |
|                 icon: 'sketch',
 | |
|                 bgClassName,
 | |
|               }}
 | |
|             >
 | |
|               <span data-testid="start-sketch">Start Sketch</span>
 | |
|             </ActionButton>
 | |
|           </li>
 | |
|         )}
 | |
|         {state.nextEvents.includes('Enter sketch') && pathId && (
 | |
|           <li className="contents">
 | |
|             <ActionButton
 | |
|               Element="button"
 | |
|               onClick={() => send({ type: 'Enter sketch' })}
 | |
|               icon={{
 | |
|                 icon: 'sketch',
 | |
|                 bgClassName,
 | |
|               }}
 | |
|             >
 | |
|               Edit Sketch
 | |
|             </ActionButton>
 | |
|           </li>
 | |
|         )}
 | |
|         {state.nextEvents.includes('Cancel') && !state.matches('idle') && (
 | |
|           <li className="contents">
 | |
|             <ActionButton
 | |
|               Element="button"
 | |
|               onClick={() => send({ type: 'Cancel' })}
 | |
|               icon={{
 | |
|                 icon: 'arrowLeft',
 | |
|                 bgClassName,
 | |
|               }}
 | |
|             >
 | |
|               Exit Sketch
 | |
|             </ActionButton>
 | |
|           </li>
 | |
|         )}
 | |
|         {state.matches('Sketch') && !state.matches('idle') && (
 | |
|           <li className="contents">
 | |
|             <ActionButton
 | |
|               Element="button"
 | |
|               onClick={() =>
 | |
|                 state.matches('Sketch.Line Tool')
 | |
|                   ? send('CancelSketch')
 | |
|                   : send('Equip tool')
 | |
|               }
 | |
|               aria-pressed={state.matches('Sketch.Line Tool')}
 | |
|               className="pressed:bg-energy-10/20 dark:pressed:bg-energy-80"
 | |
|               icon={{
 | |
|                 icon: 'line',
 | |
|                 bgClassName,
 | |
|               }}
 | |
|             >
 | |
|               Line
 | |
|             </ActionButton>
 | |
|           </li>
 | |
|         )}
 | |
|         {state.matches('Sketch') && (
 | |
|           <li className="contents">
 | |
|             <ActionButton
 | |
|               Element="button"
 | |
|               onClick={() =>
 | |
|                 state.matches('Sketch.Move Tool')
 | |
|                   ? send('CancelSketch')
 | |
|                   : send('Equip move tool')
 | |
|               }
 | |
|               aria-pressed={state.matches('Sketch.Move Tool')}
 | |
|               className="pressed:bg-energy-10/20 dark:pressed:bg-energy-80"
 | |
|               icon={{
 | |
|                 icon: 'move',
 | |
|                 bgClassName,
 | |
|               }}
 | |
|             >
 | |
|               Move
 | |
|             </ActionButton>
 | |
|           </li>
 | |
|         )}
 | |
|         {state.matches('Sketch.SketchIdle') &&
 | |
|           state.nextEvents
 | |
|             .filter(
 | |
|               (eventName) =>
 | |
|                 eventName.includes('Make segment') ||
 | |
|                 eventName.includes('Constrain')
 | |
|             )
 | |
|             .sort((a, b) => {
 | |
|               const aisEnabled = state.nextEvents
 | |
|                 .filter((event) => state.can(event as any))
 | |
|                 .includes(a)
 | |
|               const bIsEnabled = state.nextEvents
 | |
|                 .filter((event) => state.can(event as any))
 | |
|                 .includes(b)
 | |
|               if (aisEnabled && !bIsEnabled) {
 | |
|                 return -1
 | |
|               }
 | |
|               if (!aisEnabled && bIsEnabled) {
 | |
|                 return 1
 | |
|               }
 | |
|               return 0
 | |
|             })
 | |
|             .map((eventName) => (
 | |
|               <li className="contents">
 | |
|                 <ActionButton
 | |
|                   Element="button"
 | |
|                   className="text-sm"
 | |
|                   key={eventName}
 | |
|                   onClick={() => send(eventName)}
 | |
|                   disabled={
 | |
|                     !state.nextEvents
 | |
|                       .filter((event) => state.can(event as any))
 | |
|                       .includes(eventName)
 | |
|                   }
 | |
|                   title={eventName}
 | |
|                   icon={{
 | |
|                     icon: 'line',
 | |
|                     bgClassName,
 | |
|                   }}
 | |
|                 >
 | |
|                   {eventName
 | |
|                     .replace('Make segment ', '')
 | |
|                     .replace('Constrain ', '')}
 | |
|                 </ActionButton>
 | |
|               </li>
 | |
|             ))}
 | |
|         {state.matches('idle') && (
 | |
|           <li className="contents">
 | |
|             <ActionButton
 | |
|               Element="button"
 | |
|               className="text-sm"
 | |
|               onClick={() =>
 | |
|                 commandBarSend({
 | |
|                   type: 'Find and select command',
 | |
|                   data: { name: 'Extrude', ownerMachine: 'modeling' },
 | |
|                 })
 | |
|               }
 | |
|               disabled={!state.can('Extrude')}
 | |
|               title={
 | |
|                 state.can('Extrude')
 | |
|                   ? 'extrude'
 | |
|                   : 'sketches need to be closed, or not already extruded'
 | |
|               }
 | |
|               icon={{
 | |
|                 icon: 'extrude',
 | |
|                 bgClassName,
 | |
|               }}
 | |
|             >
 | |
|               Extrude
 | |
|             </ActionButton>
 | |
|           </li>
 | |
|         )}
 | |
|       </ul>
 | |
|     )
 | |
|   }
 | |
| 
 | |
|   return (
 | |
|     <div className="max-w-full flex items-stretch rounded-l-sm rounded-r-full bg-chalkboard-10 dark:bg-chalkboard-100 relative">
 | |
|       <menu className="flex-1 pl-1 pr-2 py-0 overflow-hidden rounded-l-sm whitespace-nowrap bg-chalkboard-10 dark:bg-chalkboard-100 border-solid border border-energy-10 dark:border-chalkboard-90 border-r-0">
 | |
|         <ToolbarButtons />
 | |
|       </menu>
 | |
|       <ActionButton
 | |
|         Element="button"
 | |
|         onClick={() => commandBarSend({ type: 'Open' })}
 | |
|         className="rounded-r-full pr-4 self-stretch border-energy-10 hover:border-energy-10 dark:border-chalkboard-80 bg-energy-10/50 hover:bg-energy-10 dark:bg-chalkboard-80 dark:text-energy-10"
 | |
|       >
 | |
|         {platform === 'darwin' ? '⌘K' : 'Ctrl+/'}
 | |
|       </ActionButton>
 | |
|     </div>
 | |
|   )
 | |
| }
 |