diff --git a/Makefile b/Makefile index 4cc05b788..2f7788cee 100644 --- a/Makefile +++ b/Makefile @@ -5,33 +5,40 @@ all: install build check # INSTALL ifeq ($(OS),Windows_NT) - CARGO ?= ~/.cargo/bin/cargo.exe - WASM_PACK ?= ~/.cargo/bin/wasm-pack.exe -else - CARGO ?= ~/.cargo/bin/cargo - WASM_PACK ?= ~/.cargo/bin/wasm-pack +export WINDOWS := true +ifndef MSYSTEM +export POWERSHELL := true endif +endif + +ifdef WINDOWS +CARGO ?= $(USERPROFILE)/.cargo/bin/cargo.exe +WASM_PACK ?= $(USERPROFILE)/.cargo/bin/wasm-pack.exe +else +CARGO ?= ~/.cargo/bin/cargo +WASM_PACK ?= ~/.cargo/bin/wasm-pack +endif .PHONY: install install: node_modules/.yarn-integrity $(CARGO) $(WASM_PACK) ## Install dependencies node_modules/.yarn-integrity: package.json yarn.lock yarn install -ifeq ($(OS),Windows_NT) +ifdef POWERSHELL @ type nul > $@ else @ touch $@ endif $(CARGO): -ifeq ($(OS),Windows_NT) +ifdef WINDOWS yarn install:rust:windows else yarn install:rust endif $(WASM_PACK): -ifeq ($(OS),Windows_NT) +ifdef WINDOWS yarn install:wasm-pack:cargo else yarn install:wasm-pack:sh @@ -57,7 +64,7 @@ build-web: install public/kcl_wasm_lib_bg.wasm build/index.html build-desktop: install public/kcl_wasm_lib_bg.wasm .vite/build/main.js public/kcl_wasm_lib_bg.wasm: $(CARGO_SOURCES) $(RUST_SOURCES) -ifeq ($(OS),Windows_NT) +ifdef WINDOWS yarn build:wasm:dev:windows else yarn build:wasm:dev @@ -140,8 +147,8 @@ endif .PHONY: clean clean: ## Delete all artifacts -ifeq ($(OS),Windows_NT) - git clean --force -d -X +ifdef POWERSHELL + git clean --force -d -x --exclude=.env* --exclude=**/*.env else rm -rf .vite/ build/ rm -rf trace.zip playwright-report/ test-results/ @@ -152,7 +159,7 @@ endif .PHONY: help help: install -ifeq ($(OS),Windows_NT) +ifdef POWERSHELL @ powershell -Command "Get-Content $(MAKEFILE_LIST) | Select-String -Pattern '^[^\s]+:.*##\s.*$$' | ForEach-Object { $$line = $$_.Line -split ':.*?##\s+'; Write-Host -NoNewline $$line[0].PadRight(30) -ForegroundColor Cyan; Write-Host $$line[1] }" else @ grep -E '^[^[:space:]]+:.*## .*$$' $(MAKEFILE_LIST) | awk 'BEGIN {FS = ":.*?## "}; {printf "\033[36m%-30s\033[0m %s\n", $$1, $$2}' diff --git a/docs/kcl/reduce.md b/docs/kcl/reduce.md index 7a371710f..da46f47a0 100644 --- a/docs/kcl/reduce.md +++ b/docs/kcl/reduce.md @@ -78,7 +78,7 @@ assertEqual(sum, 6, 0.00001, "1 + 2 + 3 summed is 6") ```js // Declare a function that sketches a decagon. fn decagon(radius) { - // Each side of the decagon is turned this many degrees from the previous angle. + // Each side of the decagon is turned this many radians from the previous angle. stepAngle = 1 / 10 * TAU // Start the decagon sketch at this point. diff --git a/docs/kcl/std-helix.md b/docs/kcl/std-helix.md index 490ae9223..8861ee450 100644 --- a/docs/kcl/std-helix.md +++ b/docs/kcl/std-helix.md @@ -9,7 +9,15 @@ Create a helix. ```js -helix(revolutions: number(_), angleStart: number(deg), ccw?: bool, radius?: number(mm), axis?: Axis3d | Edge, length?: number(mm), cylinder?: Solid): Helix +helix( + revolutions: number(_), + angleStart: number(deg), + ccw?: bool, + radius?: number(mm), + axis?: Axis3d | Edge, + length?: number(mm), + cylinder?: Solid, +): Helix ``` diff --git a/docs/kcl/std-math-polar.md b/docs/kcl/std-math-polar.md index 935149324..bac891600 100644 --- a/docs/kcl/std-math-polar.md +++ b/docs/kcl/std-math-polar.md @@ -6,10 +6,14 @@ layout: manual -Convert polar/sphere (azimuth, elevation, distance) coordinates tocartesian (x/y/z grid) coordinates. +Convert polar/sphere (azimuth, elevation, distance) coordinates to +cartesian (x/y/z grid) coordinates. ```js -polar(angle: number(deg), length: number(mm)): [number(mm); 2] +polar( + angle: number(deg), + length: number(mm), +): [number(mm); 2] ``` diff --git a/docs/kcl/std-math-sin.md b/docs/kcl/std-math-sin.md index b17f4d4df..0e1927fd2 100644 --- a/docs/kcl/std-math-sin.md +++ b/docs/kcl/std-math-sin.md @@ -31,7 +31,7 @@ exampleSketch = startSketchOn(XZ) |> startProfileAt([0, 0], %) |> angledLine({ angle = 50, - length = 15 / sin(toDegrees(135)), + length = 15 / sin(toRadians(135)), }, %) |> yLine(endAbsolute = 0) |> close() @@ -39,6 +39,6 @@ exampleSketch = startSketchOn(XZ) example = extrude(exampleSketch, length = 5) ``` -![Rendered example of std::math::sin 0]() +![Rendered example of std::math::sin 0]() diff --git a/docs/kcl/std-revolve.md b/docs/kcl/std-revolve.md index 03c7f5296..9df9fafc4 100644 --- a/docs/kcl/std-revolve.md +++ b/docs/kcl/std-revolve.md @@ -18,7 +18,14 @@ You can provide more than one sketch to revolve, and they will all be revolved around the same axis. ```js -revolve(@sketches: [Sketch; 1+], axis: Axis2d | Edge, angle?: number(deg), tolerance?: number(mm), tagStart?: tag, tagEnd?: tag): Solid +revolve( + @sketches: [Sketch; 1+], + axis: Axis2d | Edge, + angle?: number(deg), + tolerance?: number(mm), + tagStart?: tag, + tagEnd?: tag, +): Solid ``` diff --git a/docs/kcl/std-sketch-circle.md b/docs/kcl/std-sketch-circle.md index a327a43be..71287b3bc 100644 --- a/docs/kcl/std-sketch-circle.md +++ b/docs/kcl/std-sketch-circle.md @@ -6,10 +6,16 @@ layout: manual -Construct a 2-dimensional circle, of the specified radius, centered atthe provided (x, y) origin point. +Construct a 2-dimensional circle, of the specified radius, centered at +the provided (x, y) origin point. ```js -circle(@sketch_or_surface: Sketch | Plane | Face, center: Point2d, radius: number, tag?: tag): Sketch +circle( + @sketch_or_surface: Sketch | Plane | Face, + center: Point2d, + radius: number, + tag?: tag, +): Sketch ``` diff --git a/docs/kcl/std-sketch-mirror2d.md b/docs/kcl/std-sketch-mirror2d.md index 41959fd8b..435ecf868 100644 --- a/docs/kcl/std-sketch-mirror2d.md +++ b/docs/kcl/std-sketch-mirror2d.md @@ -11,7 +11,10 @@ Only works on unclosed sketches for now. Mirror occurs around a local sketch axis rather than a global axis. ```js -mirror2d(@sketches: [Sketch; 1+], axis: Axis2d | Edge): Sketch +mirror2d( + @sketches: [Sketch; 1+], + axis: Axis2d | Edge, +): Sketch ``` diff --git a/docs/kcl/std.json b/docs/kcl/std.json index ad50bc2e8..c72ed79d3 100644 --- a/docs/kcl/std.json +++ b/docs/kcl/std.json @@ -235702,7 +235702,7 @@ "examples": [ "// This function adds two numbers.\nfn add(a, b) {\n return a + b\n}\n\n// This function adds an array of numbers.\n// It uses the `reduce` function, to call the `add` function on every\n// element of the `arr` parameter. The starting value is 0.\nfn sum(arr) {\n return reduce(arr, 0, add)\n}\n\n/* The above is basically like this pseudo-code:\nfn sum(arr):\n sumSoFar = 0\n for i in arr:\n sumSoFar = add(sumSoFar, i)\n return sumSoFar */\n\n// We use `assertEqual` to check that our `sum` function gives the\n// expected result. It's good to check your work!\nassertEqual(sum([1, 2, 3]), 6, 0.00001, \"1 + 2 + 3 summed is 6\")", "// This example works just like the previous example above, but it uses\n// an anonymous `add` function as its parameter, instead of declaring a\n// named function outside.\narr = [1, 2, 3]\nsum = reduce(arr, 0, fn(i, result_so_far) {\n return i + result_so_far\n})\n\n// We use `assertEqual` to check that our `sum` function gives the\n// expected result. It's good to check your work!\nassertEqual(sum, 6, 0.00001, \"1 + 2 + 3 summed is 6\")", - "// Declare a function that sketches a decagon.\nfn decagon(radius) {\n // Each side of the decagon is turned this many degrees from the previous angle.\n stepAngle = 1 / 10 * TAU\n\n // Start the decagon sketch at this point.\n startOfDecagonSketch = startSketchOn(XY)\n |> startProfileAt([cos(0) * radius, sin(0) * radius], %)\n\n // Use a `reduce` to draw the remaining decagon sides.\n // For each number in the array 1..10, run the given function,\n // which takes a partially-sketched decagon and adds one more edge to it.\n fullDecagon = reduce([1..10], startOfDecagonSketch, fn(i, partialDecagon) {\n // Draw one edge of the decagon.\n x = cos(stepAngle * i) * radius\n y = sin(stepAngle * i) * radius\n return line(partialDecagon, end = [x, y])\n })\n\n return fullDecagon\n}\n\n/* The `decagon` above is basically like this pseudo-code:\nfn decagon(radius):\n stepAngle = (1/10) * TAU\n plane = startSketchOn('XY')\n startOfDecagonSketch = startProfileAt([(cos(0)*radius), (sin(0) * radius)], plane)\n\n // Here's the reduce part.\n partialDecagon = startOfDecagonSketch\n for i in [1..10]:\n x = cos(stepAngle * i) * radius\n y = sin(stepAngle * i) * radius\n partialDecagon = line(partialDecagon, end = [x, y])\n fullDecagon = partialDecagon // it's now full\n return fullDecagon */\n\n// Use the `decagon` function declared above, to sketch a decagon with radius 5.\ndecagon(5.0)\n |> close()" + "// Declare a function that sketches a decagon.\nfn decagon(radius) {\n // Each side of the decagon is turned this many radians from the previous angle.\n stepAngle = 1 / 10 * TAU\n\n // Start the decagon sketch at this point.\n startOfDecagonSketch = startSketchOn(XY)\n |> startProfileAt([cos(0) * radius, sin(0) * radius], %)\n\n // Use a `reduce` to draw the remaining decagon sides.\n // For each number in the array 1..10, run the given function,\n // which takes a partially-sketched decagon and adds one more edge to it.\n fullDecagon = reduce([1..10], startOfDecagonSketch, fn(i, partialDecagon) {\n // Draw one edge of the decagon.\n x = cos(stepAngle * i) * radius\n y = sin(stepAngle * i) * radius\n return line(partialDecagon, end = [x, y])\n })\n\n return fullDecagon\n}\n\n/* The `decagon` above is basically like this pseudo-code:\nfn decagon(radius):\n stepAngle = (1/10) * TAU\n plane = startSketchOn('XY')\n startOfDecagonSketch = startProfileAt([(cos(0)*radius), (sin(0) * radius)], plane)\n\n // Here's the reduce part.\n partialDecagon = startOfDecagonSketch\n for i in [1..10]:\n x = cos(stepAngle * i) * radius\n y = sin(stepAngle * i) * radius\n partialDecagon = line(partialDecagon, end = [x, y])\n fullDecagon = partialDecagon // it's now full\n return fullDecagon */\n\n// Use the `decagon` function declared above, to sketch a decagon with radius 5.\ndecagon(5.0)\n |> close()" ] }, { diff --git a/e2e/playwright/basic-sketch.spec.ts b/e2e/playwright/basic-sketch.spec.ts index adf2270bf..d386bda7e 100644 --- a/e2e/playwright/basic-sketch.spec.ts +++ b/e2e/playwright/basic-sketch.spec.ts @@ -137,7 +137,7 @@ async function doBasicSketch( await page.waitForTimeout(100) } - await page.getByRole('button', { name: 'Length: open menu' }).click() + await page.getByRole('button', { name: 'constraints: open menu' }).click() await page.getByRole('button', { name: 'Equal Length' }).click() // Open the code pane. diff --git a/e2e/playwright/editor-tests.spec.ts b/e2e/playwright/editor-tests.spec.ts index de0c4bdcf..cfd34c12d 100644 --- a/e2e/playwright/editor-tests.spec.ts +++ b/e2e/playwright/editor-tests.spec.ts @@ -1353,4 +1353,51 @@ sketch001 = startSketchOn(XZ) 15 ) }) + + test(`test-toolbar-buttons`, async ({ + page, + homePage, + toolbar, + scene, + cmdBar, + }) => { + await test.step('Load an empty file', async () => { + await page.addInitScript(async () => { + localStorage.setItem('persistCode', '') + }) + await page.setBodyDimensions({ width: 1200, height: 500 }) + await homePage.goToModelingScene() + + // wait until scene is ready to be interacted with + await scene.connectionEstablished() + await scene.settled(cmdBar) + }) + + await test.step('Test toolbar button correct selection', async () => { + await toolbar.expectToolbarMode.toBe('modeling') + + await toolbar.startSketchPlaneSelection() + + // Click on a default plane + await page.mouse.click(700, 200) + + // tools cannot be selected immediately, couldn't find an event to await instead. + await page.waitForTimeout(1000) + + await toolbar.selectCenterRectangle() + + await expect(page.getByTestId('center-rectangle')).toHaveAttribute( + 'aria-pressed', + 'true' + ) + }) + + await test.step('Test Toolbar dropdown remembering last selection', async () => { + // Select another tool + await page.getByTestId('circle-center').click() + + // center-rectangle should still be the active option in the rectangle dropdown + await expect(page.getByTestId('center-rectangle')).toBeVisible() + }) + }) }) diff --git a/e2e/playwright/fixtures/sceneFixture.ts b/e2e/playwright/fixtures/sceneFixture.ts index 81d2c064e..29daa9b3a 100644 --- a/e2e/playwright/fixtures/sceneFixture.ts +++ b/e2e/playwright/fixtures/sceneFixture.ts @@ -49,7 +49,9 @@ export class SceneFixture { constructor(page: Page) { this.page = page this.streamWrapper = page.getByTestId('stream') - this.networkToggleConnected = page.getByTestId('network-toggle-ok') + this.networkToggleConnected = page + .getByTestId('network-toggle-ok') + .or(page.getByTestId('network-toggle-other')) this.startEditSketchBtn = page .getByRole('button', { name: 'Start Sketch' }) .or(page.getByRole('button', { name: 'Edit Sketch' })) diff --git a/e2e/playwright/fixtures/toolbarFixture.ts b/e2e/playwright/fixtures/toolbarFixture.ts index 282ad87eb..8b98af630 100644 --- a/e2e/playwright/fixtures/toolbarFixture.ts +++ b/e2e/playwright/fixtures/toolbarFixture.ts @@ -169,7 +169,7 @@ export class ToolbarFixture { } selectCenterRectangle = async () => { await this.page - .getByRole('button', { name: 'caret down Corner rectangle:' }) + .getByRole('button', { name: 'caret down rectangles:' }) .click() await expect( this.page.getByTestId('dropdown-center-rectangle') @@ -178,7 +178,7 @@ export class ToolbarFixture { } selectBoolean = async (operation: 'union' | 'subtract' | 'intersect') => { await this.page - .getByRole('button', { name: 'caret down Union: open menu' }) + .getByRole('button', { name: 'caret down booleans: open menu' }) .click() const operationTestId = `dropdown-boolean-${operation}` await expect(this.page.getByTestId(operationTestId)).toBeVisible() @@ -186,25 +186,19 @@ export class ToolbarFixture { } selectCircleThreePoint = async () => { - await this.page - .getByRole('button', { name: 'caret down Center circle:' }) - .click() + await this.page.getByRole('button', { name: 'caret down circles:' }).click() await expect( this.page.getByTestId('dropdown-circle-three-points') ).toBeVisible() await this.page.getByTestId('dropdown-circle-three-points').click() } selectArc = async () => { - await this.page - .getByRole('button', { name: 'caret down Tangential Arc:' }) - .click() + await this.page.getByRole('button', { name: 'caret down arcs:' }).click() await expect(this.page.getByTestId('dropdown-arc')).toBeVisible() await this.page.getByTestId('dropdown-arc').click() } selectThreePointArc = async () => { - await this.page - .getByRole('button', { name: 'caret down Tangential Arc:' }) - .click() + await this.page.getByRole('button', { name: 'caret down arcs:' }).click() await expect( this.page.getByTestId('dropdown-three-point-arc') ).toBeVisible() diff --git a/e2e/playwright/named-views.spec.ts b/e2e/playwright/named-views.spec.ts index 1203d54d7..28e5b192c 100644 --- a/e2e/playwright/named-views.spec.ts +++ b/e2e/playwright/named-views.spec.ts @@ -61,6 +61,7 @@ function tomlStringOverWriteNamedViewUuids(toml: string): string { } test.describe('Named view tests', () => { + test.skip() // TODO: Jace is working on these test('Verify project.toml is not created', async ({ page }, testInfo) => { // Create project and load it const projectName = 'named-views' diff --git a/e2e/playwright/native-file-menu.spec.ts b/e2e/playwright/native-file-menu.spec.ts index b4c6535a8..3b753794b 100644 --- a/e2e/playwright/native-file-menu.spec.ts +++ b/e2e/playwright/native-file-menu.spec.ts @@ -7,6 +7,7 @@ import { expect, test } from '@e2e/playwright/zoo-test' * Test file menu actions that trigger something in the frontend */ test.describe('Native file menu', { tag: ['@electron'] }, () => { + test.skip() // TODO: Reimplement native file menu tests test.describe('Home page', () => { test.describe('File role', () => { test('Home.File.Create project', async ({ tronApp, cmdBar, page }) => { diff --git a/e2e/playwright/point-click-assemblies.spec.ts b/e2e/playwright/point-click-assemblies.spec.ts index fab574404..0a61fe9c9 100644 --- a/e2e/playwright/point-click-assemblies.spec.ts +++ b/e2e/playwright/point-click-assemblies.spec.ts @@ -1,13 +1,47 @@ import * as fsp from 'fs/promises' import path from 'path' -import { executorInputPath } from '@e2e/playwright/test-utils' -import { test } from '@e2e/playwright/zoo-test' +import type { CmdBarFixture } from '@e2e/playwright/fixtures/cmdBarFixture' +import type { ToolbarFixture } from '@e2e/playwright/fixtures/toolbarFixture' +import { + executorInputPath, + getUtils, + testsInputPath, +} from '@e2e/playwright/test-utils' +import { expect, test } from '@e2e/playwright/zoo-test' +import type { Page } from '@playwright/test' + +async function insertPartIntoAssembly( + path: string, + alias: string, + toolbar: ToolbarFixture, + cmdBar: CmdBarFixture, + page: Page +) { + await toolbar.insertButton.click() + await cmdBar.selectOption({ name: path }).click() + await cmdBar.expectState({ + stage: 'arguments', + currentArgKey: 'localName', + currentArgValue: '', + headerArguments: { Path: path, LocalName: '' }, + highlightedHeaderArg: 'localName', + commandName: 'Insert', + }) + await page.keyboard.insertText(alias) + await cmdBar.progressCmdBar() + await cmdBar.expectState({ + stage: 'review', + headerArguments: { Path: path, LocalName: alias }, + commandName: 'Insert', + }) + await cmdBar.progressCmdBar() +} // test file is for testing point an click code gen functionality that's assemblies related test.describe('Point-and-click assemblies tests', () => { test( - `Insert kcl part into assembly as whole module import`, + `Insert kcl parts into assembly as whole module import`, { tag: ['@electron'] }, async ({ context, @@ -23,11 +57,14 @@ test.describe('Point-and-click assemblies tests', () => { fail() } - // One dumb hardcoded screen pixel value - const testPoint = { x: 575, y: 200 } - const initialColor: [number, number, number] = [50, 50, 50] - const partColor: [number, number, number] = [150, 150, 150] + const midPoint = { x: 500, y: 250 } + const partPoint = { x: midPoint.x + 30, y: midPoint.y - 30 } // mid point, just off top right + const defaultPlanesColor: [number, number, number] = [180, 220, 180] + const partColor: [number, number, number] = [100, 100, 100] const tolerance = 50 + const u = await getUtils(page) + const gizmo = page.locator('[aria-label*=gizmo]') + const resetCameraButton = page.getByRole('button', { name: 'Reset view' }) await test.step('Setup parts and expect empty assembly scene', async () => { const projectName = 'assembly' @@ -36,41 +73,36 @@ test.describe('Point-and-click assemblies tests', () => { await fsp.mkdir(bracketDir, { recursive: true }) await Promise.all([ fsp.copyFile( - executorInputPath('cylinder-inches.kcl'), + executorInputPath('cylinder.kcl'), path.join(bracketDir, 'cylinder.kcl') ), fsp.copyFile( executorInputPath('e2e-can-sketch-on-chamfer.kcl'), path.join(bracketDir, 'bracket.kcl') ), + fsp.copyFile( + testsInputPath('cube.step'), + path.join(bracketDir, 'cube.step') + ), fsp.writeFile(path.join(bracketDir, 'main.kcl'), ''), ]) }) await page.setBodyDimensions({ width: 1000, height: 500 }) await homePage.openProject(projectName) await scene.settled(cmdBar) - await scene.expectPixelColor(initialColor, testPoint, tolerance) + await toolbar.closePane('code') + await scene.expectPixelColor(defaultPlanesColor, midPoint, tolerance) }) - await test.step('Insert first part into the assembly', async () => { - await toolbar.insertButton.click() - await cmdBar.selectOption({ name: 'cylinder.kcl' }).click() - await cmdBar.expectState({ - stage: 'arguments', - currentArgKey: 'localName', - currentArgValue: '', - headerArguments: { Path: 'cylinder.kcl', LocalName: '' }, - highlightedHeaderArg: 'localName', - commandName: 'Insert', - }) - await page.keyboard.insertText('cylinder') - await cmdBar.progressCmdBar() - await cmdBar.expectState({ - stage: 'review', - headerArguments: { Path: 'cylinder.kcl', LocalName: 'cylinder' }, - commandName: 'Insert', - }) - await cmdBar.progressCmdBar() + await test.step('Insert kcl as first part as module', async () => { + await insertPartIntoAssembly( + 'cylinder.kcl', + 'cylinder', + toolbar, + cmdBar, + page + ) + await toolbar.openPane('code') await editor.expectEditor.toContain( ` import "cylinder.kcl" as cylinder @@ -78,28 +110,27 @@ test.describe('Point-and-click assemblies tests', () => { `, { shouldNormalise: true } ) - await scene.expectPixelColor(partColor, testPoint, tolerance) + await scene.settled(cmdBar) + + // Check scene for changes + await toolbar.closePane('code') + await u.doAndWaitForCmd(async () => { + await gizmo.click({ button: 'right' }) + await resetCameraButton.click() + }, 'zoom_to_fit') + await toolbar.closePane('debug') + await scene.expectPixelColor(partColor, partPoint, tolerance) + await toolbar.openPane('code') }) - await test.step('Insert second part into the assembly', async () => { - await toolbar.insertButton.click() - await cmdBar.selectOption({ name: 'bracket.kcl' }).click() - await cmdBar.expectState({ - stage: 'arguments', - currentArgKey: 'localName', - currentArgValue: '', - headerArguments: { Path: 'bracket.kcl', LocalName: '' }, - highlightedHeaderArg: 'localName', - commandName: 'Insert', - }) - await page.keyboard.insertText('bracket') - await cmdBar.progressCmdBar() - await cmdBar.expectState({ - stage: 'review', - headerArguments: { Path: 'bracket.kcl', LocalName: 'bracket' }, - commandName: 'Insert', - }) - await cmdBar.progressCmdBar() + await test.step('Insert kcl second part as module', async () => { + await insertPartIntoAssembly( + 'bracket.kcl', + 'bracket', + toolbar, + cmdBar, + page + ) await editor.expectEditor.toContain( ` import "cylinder.kcl" as cylinder @@ -109,6 +140,152 @@ test.describe('Point-and-click assemblies tests', () => { `, { shouldNormalise: true } ) + await scene.settled(cmdBar) + }) + + await test.step('Insert a second time and expect error', async () => { + // TODO: revisit once we have clone with #6209 + await insertPartIntoAssembly( + 'bracket.kcl', + 'bracket', + toolbar, + cmdBar, + page + ) + await editor.expectEditor.toContain( + ` + import "cylinder.kcl" as cylinder + import "bracket.kcl" as bracket + import "bracket.kcl" as bracket + cylinder + bracket + bracket + `, + { shouldNormalise: true } + ) + await scene.settled(cmdBar) + await expect(page.locator('.cm-lint-marker-error')).toBeVisible() + }) + } + ) + + test( + `Insert foreign parts into assembly as whole module import`, + { tag: ['@electron'] }, + async ({ + context, + page, + homePage, + scene, + editor, + toolbar, + cmdBar, + tronApp, + }) => { + if (!tronApp) { + fail() + } + + const midPoint = { x: 500, y: 250 } + const partPoint = { x: midPoint.x + 30, y: midPoint.y - 30 } // mid point, just off top right + const defaultPlanesColor: [number, number, number] = [180, 220, 180] + const partColor: [number, number, number] = [150, 150, 150] + const tolerance = 50 + + const complexPlmFileName = 'cube_Complex-PLM_Name_-001.sldprt' + const camelCasedSolidworksFileName = 'cubeComplexPLMName001' + + await test.step('Setup parts and expect empty assembly scene', async () => { + const projectName = 'assembly' + await context.folderSetupFn(async (dir) => { + const bracketDir = path.join(dir, projectName) + await fsp.mkdir(bracketDir, { recursive: true }) + await Promise.all([ + fsp.copyFile( + testsInputPath('cube.step'), + path.join(bracketDir, 'cube.step') + ), + fsp.copyFile( + testsInputPath('cube.sldprt'), + path.join(bracketDir, complexPlmFileName) + ), + fsp.writeFile(path.join(bracketDir, 'main.kcl'), ''), + ]) + }) + await page.setBodyDimensions({ width: 1000, height: 500 }) + await homePage.openProject(projectName) + await scene.settled(cmdBar) + await toolbar.closePane('code') + await scene.expectPixelColor(defaultPlanesColor, midPoint, tolerance) + }) + + await test.step('Insert step part as module', async () => { + await insertPartIntoAssembly('cube.step', 'cube', toolbar, cmdBar, page) + await toolbar.openPane('code') + await editor.expectEditor.toContain( + ` + import "cube.step" as cube + cube + `, + { shouldNormalise: true } + ) + await scene.settled(cmdBar) + + // TODO: remove this once #5780 is fixed + await page.reload() + + await scene.settled(cmdBar) + await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() + await toolbar.closePane('code') + await scene.expectPixelColor(partColor, partPoint, tolerance) + }) + + await test.step('Insert second step part by clicking', async () => { + await toolbar.openPane('files') + await toolbar.expectFileTreeState([ + complexPlmFileName, + 'cube.step', + 'main.kcl', + ]) + await toolbar.openFile(complexPlmFileName) + + // Go through the ToastInsert prompt + await page.getByText('Insert into my current file').click() + + // Check getPathFilenameInVariableCase output + const parsedValueFromFile = + await cmdBar.currentArgumentInput.inputValue() + expect(parsedValueFromFile).toEqual(camelCasedSolidworksFileName) + + // Continue on with the flow + await page.keyboard.insertText('cubeSw') + await cmdBar.progressCmdBar() + await cmdBar.expectState({ + stage: 'review', + headerArguments: { Path: complexPlmFileName, LocalName: 'cubeSw' }, + commandName: 'Insert', + }) + await cmdBar.progressCmdBar() + await toolbar.closePane('files') + await toolbar.openPane('code') + await editor.expectEditor.toContain( + ` + import "cube.step" as cube + import "${complexPlmFileName}" as cubeSw + cube + cubeSw + `, + { shouldNormalise: true } + ) + await scene.settled(cmdBar) + + // TODO: remove this once #5780 is fixed + await page.reload() + await scene.settled(cmdBar) + + await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() + await toolbar.closePane('code') + await scene.expectPixelColor(partColor, partPoint, tolerance) }) } ) diff --git a/e2e/playwright/test-utils.ts b/e2e/playwright/test-utils.ts index 3396e38b1..524dda15d 100644 --- a/e2e/playwright/test-utils.ts +++ b/e2e/playwright/test-utils.ts @@ -1021,6 +1021,10 @@ export function executorInputPath(fileName: string): string { return path.join('rust', 'kcl-lib', 'e2e', 'executor', 'inputs', fileName) } +export function testsInputPath(fileName: string): string { + return path.join('rust', 'kcl-lib', 'tests', 'inputs', fileName) +} + export async function doAndWaitForImageDiff( page: Page, fn: () => Promise, diff --git a/e2e/playwright/testing-constraints.spec.ts b/e2e/playwright/testing-constraints.spec.ts index 12755f752..3882e3e20 100644 --- a/e2e/playwright/testing-constraints.spec.ts +++ b/e2e/playwright/testing-constraints.spec.ts @@ -115,7 +115,7 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { await page.waitForTimeout(100) // this wait is needed for webkit - not sure why await page .getByRole('button', { - name: 'Length: open menu', + name: 'constraints: open menu', }) .click() await page.getByRole('button', { name: 'remove constraints' }).click() @@ -189,7 +189,7 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { await page.waitForTimeout(100) await page .getByRole('button', { - name: 'Length: open menu', + name: 'constraints: open menu', }) .click() await page @@ -299,7 +299,7 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { await page.keyboard.up('Shift') await page .getByRole('button', { - name: 'Length: open menu', + name: 'constraints: open menu', }) .click() await page.getByRole('button', { name: constraint }).click() @@ -420,7 +420,7 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { await page.waitForTimeout(100) await page .getByRole('button', { - name: 'Length: open menu', + name: 'constraints: open menu', }) .click() await page @@ -533,7 +533,7 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { await page.keyboard.up('Shift') await page .getByRole('button', { - name: 'Length: open menu', + name: 'constraints: open menu', }) .click() await page.getByTestId('dropdown-constraint-angle').click() @@ -627,7 +627,7 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { await page.mouse.click(line3.x, line3.y) await page .getByRole('button', { - name: 'Length: open menu', + name: 'constraints: open menu', }) .click() await page.getByTestId('dropdown-constraint-' + constraint).click() @@ -719,7 +719,7 @@ part002 = startSketchOn(XZ) await page.mouse.click(line3.x, line3.y) await page .getByRole('button', { - name: 'Length: open menu', + name: 'constraints: open menu', }) .click() await page.getByTestId('dropdown-constraint-' + constraint).click() @@ -817,7 +817,7 @@ part002 = startSketchOn(XZ) const activeLinesContent = await page.locator('.cm-activeLine').all() const constraintMenuButton = page.getByRole('button', { - name: 'Length: open menu', + name: 'constraints: open menu', }) const constraintButton = page .getByRole('button', { @@ -905,7 +905,7 @@ part002 = startSketchOn(XZ) await page.mouse.click(line3.x - 3, line3.y + 20) await page.keyboard.up('Shift') const constraintMenuButton = page.getByRole('button', { - name: 'Length: open menu', + name: 'constraints: open menu', }) const constraintButton = page.getByRole('button', { name: constraintName, @@ -990,7 +990,7 @@ part002 = startSketchOn(XZ) await page.keyboard.up('Shift') await page.waitForTimeout(100) const constraintMenuButton = page.getByRole('button', { - name: 'Length: open menu', + name: 'constraints: open menu', }) const constraintButton = page.getByRole('button', { name: constraintName, @@ -1057,7 +1057,7 @@ part002 = startSketchOn(XZ) await page .getByRole('button', { - name: 'Length: open menu', + name: 'constraints: open menu', }) .click() await page.waitForTimeout(500) diff --git a/e2e/playwright/testing-selections.spec.ts b/e2e/playwright/testing-selections.spec.ts index 1b2485dae..72572f602 100644 --- a/e2e/playwright/testing-selections.spec.ts +++ b/e2e/playwright/testing-selections.spec.ts @@ -124,7 +124,7 @@ test.describe('Testing selections', { tag: ['@skipWin'] }, () => { // click a segment hold shift and click an axis, see that a relevant constraint is enabled const constrainButton = page.getByRole('button', { - name: 'Length: open menu', + name: 'constraints: open menu', }) const absXButton = page.getByRole('button', { name: 'Absolute X' }) diff --git a/rust/kcl-lib/src/docs/kcl_doc.rs b/rust/kcl-lib/src/docs/kcl_doc.rs index 1d0d25f4d..419727f41 100644 --- a/rust/kcl-lib/src/docs/kcl_doc.rs +++ b/rust/kcl-lib/src/docs/kcl_doc.rs @@ -1,4 +1,4 @@ -use std::{collections::HashSet, str::FromStr}; +use std::{collections::HashSet, fmt, str::FromStr}; use regex::Regex; use tower_lsp::lsp_types::{ @@ -389,21 +389,23 @@ impl FnData { pub fn fn_signature(&self) -> String { let mut signature = String::new(); - signature.push('('); - for (i, arg) in self.args.iter().enumerate() { - if i > 0 { - signature.push_str(", "); - } - match &arg.kind { - ArgKind::Special => signature.push_str(&format!("@{}", arg.name)), - ArgKind::Labelled(false) => signature.push_str(&arg.name), - ArgKind::Labelled(true) => signature.push_str(&format!("{}?", arg.name)), - } - if let Some(ty) = &arg.ty { - signature.push_str(&format!(": {ty}")); + if self.args.is_empty() { + signature.push_str("()"); + } else if self.args.len() == 1 { + signature.push('('); + signature.push_str(&self.args[0].to_string()); + signature.push(')'); + } else { + signature.push('('); + for a in &self.args { + signature.push_str("\n "); + signature.push_str(&a.to_string()); + signature.push(','); } + signature.push('\n'); + signature.push(')'); } - signature.push(')'); + if let Some(ty) = &self.return_type { signature.push_str(&format!(": {ty}")); } @@ -515,6 +517,20 @@ pub struct ArgData { pub docs: Option, } +impl fmt::Display for ArgData { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + match &self.kind { + ArgKind::Special => write!(f, "@{}", self.name)?, + ArgKind::Labelled(false) => f.write_str(&self.name)?, + ArgKind::Labelled(true) => write!(f, "{}?", self.name)?, + } + if let Some(ty) = &self.ty { + write!(f, ": {ty}")?; + } + Ok(()) + } +} + #[derive(Debug, Clone, Copy, PartialEq)] pub enum ArgKind { Special, @@ -766,8 +782,8 @@ trait ApplyMeta { description = summary; summary = None; let d = description.as_mut().unwrap(); - d.push_str(l); d.push('\n'); + d.push_str(l); } continue; } diff --git a/rust/kcl-lib/src/execution/kcl_value.rs b/rust/kcl-lib/src/execution/kcl_value.rs index 7b3c9b9b9..da0fb6411 100644 --- a/rust/kcl-lib/src/execution/kcl_value.rs +++ b/rust/kcl-lib/src/execution/kcl_value.rs @@ -360,15 +360,6 @@ impl KclValue { result } - /// Put the number into a KCL value. - pub const fn from_number(f: f64, meta: Vec) -> Self { - Self::Number { - value: f, - meta, - ty: NumericType::Unknown, - } - } - pub const fn from_number_with_type(f: f64, ty: NumericType, meta: Vec) -> Self { Self::Number { value: f, meta, ty } } diff --git a/rust/kcl-lib/src/execution/types.rs b/rust/kcl-lib/src/execution/types.rs index 05f0363bf..d3bb69250 100644 --- a/rust/kcl-lib/src/execution/types.rs +++ b/rust/kcl-lib/src/execution/types.rs @@ -19,7 +19,7 @@ use crate::{ }; lazy_static::lazy_static! { - pub(super) static ref CHECK_NUMERIC_TYPES: bool = { + pub(crate) static ref CHECK_NUMERIC_TYPES: bool = { let env_var = std::env::var("ZOO_NUM_TYS"); let Ok(env_var) = env_var else { return false; @@ -416,6 +416,80 @@ impl NumericType { } } + pub fn combine_eq_array(input: &[TyF64]) -> (Vec, NumericType) { + use NumericType::*; + + let mut result = input.iter().map(|t| t.n).collect(); + + let mut ty = Any; + // Invariant mismatch is true => ty is Known + let mut mismatch = false; + for i in input { + if i.ty == Any || ty == i.ty { + continue; + } + + match (&ty, &i.ty) { + (Any, t) => { + ty = t.clone(); + } + (_, Unknown) | (Default { .. }, Default { .. }) => return (result, Unknown), + + // Known types and compatible, but needs adjustment. + (Known(UnitType::Length(_)), Known(UnitType::Length(_))) + | (Known(UnitType::Angle(_)), Known(UnitType::Angle(_))) => { + mismatch = true; + } + + // Known but incompatible. + (Known(_), Known(_)) => return (result, Unknown), + + // Known and unknown, no adjustment for counting numbers. + (Known(UnitType::Count), Default { .. }) | (Default { .. }, Known(UnitType::Count)) => { + ty = Known(UnitType::Count); + } + + (Known(UnitType::Length(l1)), Default { len: l2, .. }) => { + mismatch |= l1 != l2; + } + (Known(UnitType::Angle(a1)), Default { angle: a2, .. }) => { + mismatch |= a1 != a2; + } + + (Default { len: l1, .. }, Known(UnitType::Length(l2))) => { + mismatch |= l1 != l2; + ty = Known(UnitType::Length(*l2)); + } + (Default { angle: a1, .. }, Known(UnitType::Angle(a2))) => { + mismatch |= a1 != a2; + ty = Known(UnitType::Angle(*a2)); + } + + (Unknown, _) | (_, Any) => unreachable!(), + } + } + + if !mismatch { + return (result, ty); + } + + result = result + .into_iter() + .zip(input) + .map(|(n, i)| match (&ty, &i.ty) { + (Known(UnitType::Length(l1)), Known(UnitType::Length(l2)) | Default { len: l2, .. }) => { + l2.adjust_to(n, *l1) + } + (Known(UnitType::Angle(a1)), Known(UnitType::Angle(a2)) | Default { angle: a2, .. }) => { + a2.adjust_to(n, *a1) + } + _ => unreachable!(), + }) + .collect(); + + (result, ty) + } + /// Combine two types for multiplication-like operations. pub fn combine_mul(a: TyF64, b: TyF64) -> (f64, f64, NumericType) { use NumericType::*; @@ -621,7 +695,7 @@ pub enum UnitLen { impl UnitLen { fn adjust_to(self, value: f64, to: UnitLen) -> f64 { - if self == to { + if !*CHECK_NUMERIC_TYPES || self == to { return value; } @@ -734,6 +808,11 @@ impl UnitAngle { fn adjust_to(self, value: f64, to: UnitAngle) -> f64 { use std::f64::consts::PI; use UnitAngle::*; + + if !*CHECK_NUMERIC_TYPES { + return value; + } + match (self, to) { (Degrees, Degrees) => value, (Degrees, Radians) => (value / 180.0) * PI, @@ -1847,11 +1926,16 @@ n = 10inch / 2mm o = 3mm / 3 p = 3_ / 4 q = 4inch / 2_ + +r = min(0, 3, 42) +s = min(0, 3mm, -42) +t = min(100, 3in, 142mm) +u = min(3rad, 4in) "#; let result = parse_execute(program).await.unwrap(); if *CHECK_NUMERIC_TYPES { - assert_eq!(result.exec_state.errors().len(), 2); + assert_eq!(result.exec_state.errors().len(), 3); } else { assert!(result.exec_state.errors().is_empty()); } @@ -1861,7 +1945,9 @@ q = 4inch / 2_ assert_value_and_type("c", &result, 13.0, NumericType::mm()); assert_value_and_type("d", &result, 13.0, NumericType::mm()); assert_value_and_type("e", &result, 13.0, NumericType::mm()); - assert_value_and_type("f", &result, 5.0, NumericType::mm()); + if *CHECK_NUMERIC_TYPES { + assert_value_and_type("f", &result, 5.0, NumericType::mm()); + } assert_value_and_type("g", &result, 20.0, NumericType::default()); assert_value_and_type("h", &result, 20.0, NumericType::mm()); @@ -1871,9 +1957,30 @@ q = 4inch / 2_ assert_value_and_type("l", &result, 0.0, NumericType::count()); assert_value_and_type("m", &result, 2.0, NumericType::count()); - assert_value_and_type("n", &result, 127.0, NumericType::count()); + if *CHECK_NUMERIC_TYPES { + assert_value_and_type("n", &result, 127.0, NumericType::count()); + } assert_value_and_type("o", &result, 1.0, NumericType::mm()); assert_value_and_type("p", &result, 1.0, NumericType::count()); assert_value_and_type("q", &result, 2.0, NumericType::Known(UnitType::Length(UnitLen::Inches))); + + assert_value_and_type("r", &result, 0.0, NumericType::default()); + assert_value_and_type("s", &result, -42.0, NumericType::mm()); + assert_value_and_type("t", &result, 3.0, NumericType::Known(UnitType::Length(UnitLen::Inches))); + assert_value_and_type("u", &result, 3.0, NumericType::Unknown); + } + + #[tokio::test(flavor = "multi_thread")] + async fn bad_typed_arithmetic() { + let program = r#" +a = 1rad +b = 180 / PI * a + 360 +"#; + + let result = parse_execute(program).await.unwrap(); + + assert_value_and_type("a", &result, 1.0, NumericType::radians()); + // TODO type is not ideal + assert_value_and_type("b", &result, 417.0, NumericType::radians()); } } diff --git a/rust/kcl-lib/src/std/args.rs b/rust/kcl-lib/src/std/args.rs index 9b3d40cc6..13b022faf 100644 --- a/rust/kcl-lib/src/std/args.rs +++ b/rust/kcl-lib/src/std/args.rs @@ -523,15 +523,6 @@ impl Args { }) } - pub(super) fn make_user_val_from_f64(&self, f: f64) -> KclValue { - KclValue::from_number( - f, - vec![Metadata { - source_range: self.source_range, - }], - ) - } - pub(super) fn make_user_val_from_f64_with_type(&self, f: TyF64) -> KclValue { KclValue::from_number_with_type( f.n, diff --git a/rust/kcl-lib/src/std/array.rs b/rust/kcl-lib/src/std/array.rs index 05e30deb8..7d86033b4 100644 --- a/rust/kcl-lib/src/std/array.rs +++ b/rust/kcl-lib/src/std/array.rs @@ -133,7 +133,7 @@ pub async fn reduce(exec_state: &mut ExecState, args: Args) -> Result Result { - let n = args.get_unlabeled_kw_arg("number to divide")?; - let d = args.get_kw_arg("divisor")?; +pub async fn rem(exec_state: &mut ExecState, args: Args) -> Result { + let n: TyF64 = args.get_unlabeled_kw_arg("number to divide")?; + let d: TyF64 = args.get_kw_arg("divisor")?; + + let (n, d, ty) = NumericType::combine_div(n, d); + if *types::CHECK_NUMERIC_TYPES && ty == NumericType::Unknown { + // TODO suggest how to fix this + exec_state.warn(CompilationError::err( + args.source_range, + "Remainder of numbers which have unknown or incompatible units.", + )); + } let remainder = inner_rem(n, d); - Ok(args.make_user_val_from_f64(remainder)) + Ok(args.make_user_val_from_f64_with_type(TyF64::new(remainder, ty))) } /// Compute the remainder after dividing `num` by `div`. @@ -243,11 +256,19 @@ fn inner_ceil(num: f64) -> Result { } /// Compute the minimum of the given arguments. -pub async fn min(_exec_state: &mut ExecState, args: Args) -> Result { - let nums = args.get_number_array()?; +pub async fn min(exec_state: &mut ExecState, args: Args) -> Result { + let nums = args.get_number_array_with_types()?; + let (nums, ty) = NumericType::combine_eq_array(&nums); + if *types::CHECK_NUMERIC_TYPES && ty == NumericType::Unknown { + // TODO suggest how to fix this + exec_state.warn(CompilationError::err( + args.source_range, + "Calling `min` on numbers which have unknown or incompatible units.", + )); + } let result = inner_min(nums); - Ok(args.make_user_val_from_f64(result)) + Ok(args.make_user_val_from_f64_with_type(TyF64::new(result, ty))) } /// Compute the minimum of the given arguments. @@ -280,11 +301,19 @@ fn inner_min(args: Vec) -> f64 { } /// Compute the maximum of the given arguments. -pub async fn max(_exec_state: &mut ExecState, args: Args) -> Result { - let nums = args.get_number_array()?; +pub async fn max(exec_state: &mut ExecState, args: Args) -> Result { + let nums = args.get_number_array_with_types()?; + let (nums, ty) = NumericType::combine_eq_array(&nums); + if *types::CHECK_NUMERIC_TYPES && ty == NumericType::Unknown { + // TODO suggest how to fix this + exec_state.warn(CompilationError::err( + args.source_range, + "Calling `max` on numbers which have unknown or incompatible units.", + )); + } let result = inner_max(nums); - Ok(args.make_user_val_from_f64(result)) + Ok(args.make_user_val_from_f64_with_type(TyF64::new(result, ty))) } /// Compute the maximum of the given arguments. diff --git a/rust/kcl-lib/std/math.kcl b/rust/kcl-lib/std/math.kcl index 3af03f8ff..55f75a52d 100644 --- a/rust/kcl-lib/std/math.kcl +++ b/rust/kcl-lib/std/math.kcl @@ -69,7 +69,7 @@ export fn cos(@num: number(rad)): number(_) {} /// |> startProfileAt([0, 0], %) /// |> angledLine({ /// angle = 50, -/// length = 15 / sin(toDegrees(135)), +/// length = 15 / sin(toRadians(135)), /// }, %) /// |> yLine(endAbsolute = 0) /// |> close() diff --git a/rust/kcl-lib/tests/outputs/serial_test_example_std-math-sin0.png b/rust/kcl-lib/tests/outputs/serial_test_example_std-math-sin0.png index 61d468c2f..cc39c2753 100644 Binary files a/rust/kcl-lib/tests/outputs/serial_test_example_std-math-sin0.png and b/rust/kcl-lib/tests/outputs/serial_test_example_std-math-sin0.png differ diff --git a/src/Toolbar.tsx b/src/Toolbar.tsx index 93a438652..fc9aa65d2 100644 --- a/src/Toolbar.tsx +++ b/src/Toolbar.tsx @@ -17,12 +17,14 @@ import { isDesktop } from '@src/lib/isDesktop' import { openExternalBrowserIfDesktop } from '@src/lib/openWindow' import { editorManager, kclManager } from '@src/lib/singletons' import type { + ToolbarDropdown, ToolbarItem, ToolbarItemCallbackProps, ToolbarItemResolved, + ToolbarItemResolvedDropdown, ToolbarModeName, } from '@src/lib/toolbar' -import { toolbarConfig } from '@src/lib/toolbar' +import { isToolbarItemResolvedDropdown, toolbarConfig } from '@src/lib/toolbar' import { isArray } from '@src/lib/utils' import { commandBarActor } from '@src/machines/commandBarMachine' @@ -131,21 +133,27 @@ export function Toolbar({ */ const currentModeItems: ( | ToolbarItemResolved - | ToolbarItemResolved[] + | ToolbarItemResolvedDropdown | 'break' )[] = useMemo(() => { return toolbarConfig[currentMode].items.map((maybeIconConfig) => { if (maybeIconConfig === 'break') { return 'break' - } else if (isArray(maybeIconConfig)) { - return maybeIconConfig.map(resolveItemConfig) + } else if (isToolbarDropdown(maybeIconConfig)) { + return { + id: maybeIconConfig.id, + array: maybeIconConfig.array.map((item) => + resolveItemConfig(item, maybeIconConfig.id) + ), + } } else { return resolveItemConfig(maybeIconConfig) } }) function resolveItemConfig( - maybeIconConfig: ToolbarItem + maybeIconConfig: ToolbarItem, + dropdownId?: string ): ToolbarItemResolved { const isDisabled = disableAllButtons || @@ -176,6 +184,14 @@ export function Toolbar({ } }, [currentMode, disableAllButtons, configCallbackProps]) + // To remember the last selected item in an ActionButtonDropdown + const [lastSelectedMultiActionItem, _] = useState( + new Map< + number /* index in currentModeItems */, + number /* index in maybeIconConfig */ + >() + ) + return ( ) - } else if (isArray(maybeIconConfig)) { + } else if (isToolbarItemResolvedDropdown(maybeIconConfig)) { // A button with a dropdown + const selectedIcon = + maybeIconConfig.array.find((c) => c.isActive) || + maybeIconConfig.array[lastSelectedMultiActionItem.get(i) ?? 0] + + // Save the last selected item in the dropdown + lastSelectedMultiActionItem.set( + i, + maybeIconConfig.array.indexOf(selectedIcon) + ) return ( ({ + splitMenuItems={maybeIconConfig.array.map((itemConfig) => ({ id: itemConfig.id, label: itemConfig.title, hotkey: itemConfig.hotkey, @@ -236,11 +261,11 @@ export function Toolbar({ > - maybeIconConfig[0].onClick(configCallbackProps) - } + aria-description={selectedIcon.description} + onClick={() => selectedIcon.onClick(configCallbackProps)} > - - {maybeIconConfig[0].title} + + {selectedIcon.title} {showRichContent ? ( ) : ( )} @@ -430,7 +451,9 @@ const ToolbarItemTooltipShortContent = ({ > {title} {hotkey && ( - {hotkey} + + {displayHotkeys(hotkey)} + )} ) @@ -461,7 +484,9 @@ const ToolbarItemTooltipRichContent = ({ {itemConfig.title} {itemConfig.status === 'available' && itemConfig.hotkey ? ( - {itemConfig.hotkey} + + {displayHotkeys(itemConfig.hotkey)} + ) : itemConfig.status === 'kcl-only' ? ( <> @@ -522,3 +547,14 @@ const ToolbarItemTooltipRichContent = ({ ) } + +// We don't want to display Esc hotkeys to avoid confusion in the Toolbar UI (eg. "EscR") +function displayHotkeys(hotkey: string | string[]) { + return (isArray(hotkey) ? hotkey : [hotkey]).filter((h) => h !== 'Esc') +} + +function isToolbarDropdown( + item: ToolbarItem | ToolbarDropdown +): item is ToolbarDropdown { + return 'array' in item +} diff --git a/src/components/ActionButton.tsx b/src/components/ActionButton.tsx index 766d19d93..3da3e0b65 100644 --- a/src/components/ActionButton.tsx +++ b/src/components/ActionButton.tsx @@ -71,6 +71,7 @@ export const ActionButton = forwardRef((props: ActionButtonProps, ref) => {