Compare commits
	
		
			68 Commits
		
	
	
		
			v0.51.0
			...
			safe-reset
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 10c51896be | |||
| 96e12b5aba | |||
| bec3ba71cd | |||
| c6766d2a73 | |||
| befac6ad87 | |||
| 60bc38559b | |||
| 2a56155587 | |||
| 87c3673a71 | |||
| 21889162ff | |||
| f1cccc22cd | |||
| 34f05b3cf2 | |||
| 3c89ac8421 | |||
| d3f788050c | |||
| 10c1f3a849 | |||
| d168ef94e9 | |||
| 29f8b05f56 | |||
| af482c30bd | |||
| 4e36e398ee | |||
| b5f81b9384 | |||
| 879b471aed | |||
| 964d81dc0e | |||
| 443f7cd5fd | |||
| 2fc8cb5376 | |||
| ee20a09e7e | |||
| f4e801351c | |||
| 7329753211 | |||
| 7d46e7e271 | |||
| 0583eb07d5 | |||
| 73694563cf | |||
| bb4ed59191 | |||
| 566143757f | |||
| 822f2ffc73 | |||
| eb3ceba497 | |||
| 2c6d0621c9 | |||
| d8e84cb5e3 | |||
| 0b1e79871f | |||
| 954fddf578 | |||
| f94f748339 | |||
| e2b7b22ca9 | |||
| efc8c82d8b | |||
| eac5abba79 | |||
| 1956c14b8a | |||
| 017fac7041 | |||
| 0bdc50c78f | |||
| 57d78b6094 | |||
| db5ce7ba85 | |||
| 51c16d0048 | |||
| 6532b23f1c | |||
| 49304b9ecd | |||
| 358b34de4c | |||
| 9973e5fde3 | |||
| 36875e05fd | |||
| 42123383bb | |||
| ef4c606ed1 | |||
| 1ebb73b935 | |||
| b57d31c0e7 | |||
| 678ebbc310 | |||
| cc2efd316c | |||
| d1f811f91d | |||
| 7ca3afff9f | |||
| 71b9e40bd9 | |||
| 4f35197a96 | |||
| 40b0cf5fd3 | |||
| 355e6acf0d | |||
| 4ff38e7f44 | |||
| 1dcd3b84b7 | |||
| 2957216bd3 | |||
| 11160f0b40 | 
| @ -1,5 +1,6 @@ | ||||
| NODE_ENV=development | ||||
| DEV=true | ||||
|  | ||||
| VITE_KC_API_WS_MODELING_URL=wss://api.dev.zoo.dev/ws/modeling/commands | ||||
| VITE_KC_API_BASE_URL=https://api.dev.zoo.dev | ||||
| VITE_KC_SITE_BASE_URL=https://dev.zoo.dev | ||||
| @ -8,3 +9,5 @@ VITE_KC_SKIP_AUTH=false | ||||
| VITE_KC_CONNECTION_TIMEOUT_MS=5000 | ||||
| # ONLY add your token in .env.development.local if you want to skip auth, otherwise this token takes precedence! | ||||
| #VITE_KC_DEV_TOKEN="your token from dev.zoo.dev should go in .env.development.local" | ||||
|  | ||||
| FAIL_ON_CONSOLE_ERRORS=true | ||||
|  | ||||
							
								
								
									
										10
									
								
								.envrc
									
									
									
									
									
								
							
							
						
						| @ -1,3 +1,13 @@ | ||||
| # Load optional shared environment variables | ||||
| source_up_if_exists | ||||
|  | ||||
| # Load default development environment variables | ||||
| dotenv .env.development | ||||
|  | ||||
| # Load optional environment variables overrides | ||||
| dotenv_if_exists .env.development.local | ||||
|  | ||||
| # Load optional testing environment variables | ||||
| dotenv_if_exists e2e/playwright/playwright-secrets.env | ||||
|  | ||||
| use flake . | ||||
|  | ||||
							
								
								
									
										78
									
								
								.eslintrc
									
									
									
									
									
								
							
							
						
						| @ -1,78 +0,0 @@ | ||||
| { | ||||
|     "parser": "@typescript-eslint/parser", | ||||
|     "parserOptions": { | ||||
|       "project": "./tsconfig.json" | ||||
|     }, | ||||
|     "plugins": [ | ||||
|       "react-perf", | ||||
|       "css-modules", | ||||
|       "jest", | ||||
|       "jsx-a11y", | ||||
|       "react", | ||||
|       "react-hooks", | ||||
|       "suggest-no-throw", | ||||
|       "testing-library", | ||||
|       "@typescript-eslint" | ||||
|     ], | ||||
|     "extends": [ | ||||
|       "plugin:css-modules/recommended", | ||||
|       "plugin:jsx-a11y/recommended", | ||||
|       "plugin:react-hooks/recommended" | ||||
|     ], | ||||
|     "rules": { | ||||
|       "@typescript-eslint/no-floating-promises": "error", | ||||
|       "@typescript-eslint/no-misused-promises": "error", | ||||
|       "@typescript-eslint/no-unused-vars": ["error", { | ||||
|         "varsIgnorePattern": "^_", | ||||
|         "argsIgnorePattern": "^_", | ||||
|         "ignoreRestSiblings": true, | ||||
|         "vars": "all", | ||||
|         "args": "none" | ||||
|       }], | ||||
|       "jsx-a11y/click-events-have-key-events": "off", | ||||
|       "jsx-a11y/no-autofocus": "off", | ||||
|       "jsx-a11y/no-noninteractive-element-interactions": "off", | ||||
|       "no-restricted-globals": [ | ||||
|         "error", | ||||
|         { | ||||
|           "name": "isNaN", | ||||
|           "message": "Use Number.isNaN() instead." | ||||
|         }, | ||||
|       ], | ||||
|       "no-restricted-syntax": [ | ||||
|         "error", | ||||
|         { | ||||
|           "selector": "CallExpression[callee.object.name='Array'][callee.property.name='isArray']", | ||||
|           "message": "Use isArray() in lib/utils.ts instead of Array.isArray()." | ||||
|         } | ||||
|       ], | ||||
|       "semi": [ | ||||
|         "error", | ||||
|         "never" | ||||
|       ], | ||||
|       "react-hooks/exhaustive-deps": "off", | ||||
|       "suggest-no-throw/suggest-no-throw": "warn", | ||||
|     }, | ||||
|     "overrides": [ | ||||
|       { | ||||
|         "files": ["e2e/**/*.ts"], // Update the pattern based on your file structure | ||||
|         "extends": [ | ||||
|           "plugin:testing-library/react" | ||||
|         ], | ||||
|         "rules": { | ||||
|           "suggest-no-throw/suggest-no-throw": "off", | ||||
|           "testing-library/prefer-screen-queries": "off", | ||||
|           "jest/valid-expect": "off" | ||||
|         } | ||||
|       }, | ||||
|       { | ||||
|         "files": ["src/**/*.test.ts"], | ||||
|         "extends": [ | ||||
|           "plugin:testing-library/react" | ||||
|         ], | ||||
|         "rules": { | ||||
|           "suggest-no-throw/suggest-no-throw": "off", | ||||
|         } | ||||
|       } | ||||
|     ] | ||||
| } | ||||
							
								
								
									
										127
									
								
								.eslintrc.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,127 @@ | ||||
| { | ||||
|   "parser": "@typescript-eslint/parser", | ||||
|   "parserOptions": { | ||||
|     "project": "./tsconfig.json" | ||||
|   }, | ||||
|   "plugins": [ | ||||
|     "react-perf", | ||||
|     "css-modules", | ||||
|     "jest", | ||||
|     "jsx-a11y", | ||||
|     "react", | ||||
|     "react-hooks", | ||||
|     "suggest-no-throw", | ||||
|     "testing-library", | ||||
|     "@typescript-eslint" | ||||
|   ], | ||||
|   "extends": [ | ||||
|     "plugin:css-modules/recommended", | ||||
|     "plugin:jsx-a11y/recommended", | ||||
|     "plugin:react-hooks/recommended" | ||||
|   ], | ||||
|   "rules": { | ||||
|     "no-array-constructor": "off", // This is wrong; use the @typescript-eslint one instead. | ||||
|     "@typescript-eslint/no-array-constructor": "error", | ||||
|     "@typescript-eslint/no-array-delete": "error", | ||||
|     "@typescript-eslint/no-duplicate-enum-values": "error", | ||||
|     "@typescript-eslint/no-duplicate-type-constituents": "error", | ||||
|     "@typescript-eslint/no-empty-object-type": "error", | ||||
|     "@typescript-eslint/no-extra-non-null-assertion": "error", | ||||
|     "@typescript-eslint/no-floating-promises": "error", | ||||
|     "@typescript-eslint/no-for-in-array": "error", | ||||
|     "no-implied-eval": "off", // This is wrong; use the @typescript-eslint one instead. | ||||
|     "@typescript-eslint/no-implied-eval": "error", | ||||
|     "@typescript-eslint/no-misused-new": "error", | ||||
|     "@typescript-eslint/no-misused-promises": "error", | ||||
|     "@typescript-eslint/no-namespace": "error", | ||||
|     "@typescript-eslint/no-non-null-asserted-optional-chain": "error", | ||||
|     "@typescript-eslint/no-redundant-type-constituents": "error", | ||||
|     "@typescript-eslint/no-this-alias": "warn", | ||||
|     "@typescript-eslint/no-unnecessary-type-assertion": "error", | ||||
|     "@typescript-eslint/no-unnecessary-type-constraint": "error", | ||||
|     "no-unused-vars": "off", // This is wrong; use the @typescript-eslint one instead. | ||||
|     "@typescript-eslint/no-unused-vars": [ | ||||
|       "error", | ||||
|       { | ||||
|         "varsIgnorePattern": "^_", | ||||
|         "argsIgnorePattern": "^_", | ||||
|         "ignoreRestSiblings": true, | ||||
|         "vars": "all", | ||||
|         "args": "none" | ||||
|       } | ||||
|     ], | ||||
|     "@typescript-eslint/no-unsafe-unary-minus": "error", | ||||
|     "@typescript-eslint/no-wrapper-object-types": "error", | ||||
|     "no-throw-literal": "off", // Use @typescript-eslint/only-throw-error instead. | ||||
|     "@typescript-eslint/only-throw-error": "error", | ||||
|     "@typescript-eslint/prefer-as-const": "warn", | ||||
|     "@typescript-eslint/prefer-namespace-keyword": "error", | ||||
|     "@typescript-eslint/consistent-type-imports": "error", | ||||
|     "@typescript-eslint/restrict-plus-operands": "error", | ||||
|     "jsx-a11y/click-events-have-key-events": "off", | ||||
|     "jsx-a11y/no-autofocus": "off", | ||||
|     "jsx-a11y/no-noninteractive-element-interactions": "off", | ||||
|     "no-restricted-globals": [ | ||||
|       "error", | ||||
|       { | ||||
|         "name": "isNaN", | ||||
|         "message": "Use Number.isNaN() instead." | ||||
|       } | ||||
|     ], | ||||
|     "no-restricted-syntax": [ | ||||
|       "error", | ||||
|       { | ||||
|         "selector": "CallExpression[callee.object.name='Array'][callee.property.name='isArray']", | ||||
|         "message": "Use isArray() in lib/utils.ts instead of Array.isArray()." | ||||
|       }, | ||||
|       { | ||||
|         "selector": "CallExpression[callee.object.name='TOML'][callee.property.name='stringify']", | ||||
|         "message": "Do not use TOML.stringify directly. Use the wrappers in test-utils instead like settingsToToml." | ||||
|       }, | ||||
|       { | ||||
|         "selector": "CallExpression[callee.object.name='TOML'][callee.property.name='parse']", | ||||
|         "message": "Do not use TOML.parse directly. Use the wrappers in test-utils instead like tomlToSettings." | ||||
|       } | ||||
|     ], | ||||
|     "no-restricted-imports": [ | ||||
|       "error", | ||||
|       { | ||||
|         "patterns": [ | ||||
|           // Restrict all relative imports except for .css files. | ||||
|           { | ||||
|             "group": ["./*", "../*", "!./*.css", "!../*.css"], | ||||
|             "message": "Use absolute imports instead." | ||||
|           } | ||||
|         ] | ||||
|       } | ||||
|     ], | ||||
|     "semi": ["error", "never"], | ||||
|     "react-hooks/exhaustive-deps": "off", | ||||
|     "suggest-no-throw/suggest-no-throw": "error" | ||||
|   }, | ||||
|   "overrides": [ | ||||
|     { | ||||
|       "files": ["e2e/**/*.ts"], // Update the pattern based on your file structure | ||||
|       "extends": ["plugin:testing-library/react"], | ||||
|       "rules": { | ||||
|         "suggest-no-throw/suggest-no-throw": "off", | ||||
|         "testing-library/prefer-screen-queries": "off", | ||||
|         "jest/valid-expect": "off" | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "files": ["src/**/*.test.ts"], | ||||
|       "extends": ["plugin:testing-library/react"], | ||||
|       "rules": { | ||||
|         "suggest-no-throw/suggest-no-throw": "off" | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "files": ["packages/**/*.ts", "rust/**/*.ts"], | ||||
|       "extends": [], | ||||
|       "rules": { | ||||
|         "no-restricted-imports": "off" | ||||
|       } | ||||
|     } | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										8
									
								
								.github/workflows/e2e-tests.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -229,10 +229,6 @@ jobs: | ||||
|           timeout_minutes: 30 | ||||
|           max_attempts: 3 | ||||
|         env: | ||||
|           CI: true | ||||
|           NODE_ENV: development | ||||
|           VITE_KC_DEV_TOKEN: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|           VITE_KC_SKIP_AUTH: true | ||||
|           token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|           snapshottoken: ${{ secrets.KITTYCAD_API_TOKEN }} | ||||
|  | ||||
| @ -377,11 +373,7 @@ jobs: | ||||
|           timeout_minutes: 45 | ||||
|           max_attempts: 15 | ||||
|         env: | ||||
|           CI: true | ||||
|           FAIL_ON_CONSOLE_ERRORS: true | ||||
|           NODE_ENV: development | ||||
|           VITE_KC_DEV_TOKEN: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|           VITE_KC_SKIP_AUTH: true | ||||
|           token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|  | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|  | ||||
							
								
								
									
										174
									
								
								.github/workflows/static-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -28,43 +28,57 @@ jobs: | ||||
|       - run: yarn fmt-check | ||||
|  | ||||
|   yarn-build-wasm: | ||||
|     runs-on: ubuntu-22.04 | ||||
|  | ||||
|     # Build the wasm blob once on the fastest runner. | ||||
|     runs-on: runs-on=${{ github.run_id }}/family=i7ie.2xlarge/image=ubuntu22-full-x64 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version-file: '.nvmrc' | ||||
|           cache: 'yarn' | ||||
|       - run: yarn install | ||||
|  | ||||
|       - name: Install dependencies | ||||
|         run: yarn install | ||||
|  | ||||
|       - name: Use correct Rust toolchain | ||||
|         shell: bash | ||||
|         run: | | ||||
|           [ -e rust-toolchain.toml ] || cp rust/rust-toolchain.toml ./ | ||||
|  | ||||
|       - name: Install rust | ||||
|         uses: actions-rust-lang/setup-rust-toolchain@v1 | ||||
|         with: | ||||
|           cache: false # Configured below. | ||||
|  | ||||
|       - uses: taiki-e/install-action@37bdc826eaedac215f638a96472df572feab0f9b | ||||
|         with: | ||||
|           tool: wasm-pack | ||||
|       - run: yarn build:wasm | ||||
|  | ||||
|   yarn-tsc: | ||||
|     runs-on: ubuntu-22.04 | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version-file: '.nvmrc' | ||||
|           cache: 'yarn' | ||||
|       - run: yarn install | ||||
|       - run: yarn --cwd ./rust/kcl-language-server --modules-folder node_modules install | ||||
|       - uses: Swatinem/rust-cache@v2 | ||||
|       - name: Rust Cache | ||||
|         uses: Swatinem/rust-cache@v2 | ||||
|         with: | ||||
|           workspaces: './rust' | ||||
|  | ||||
|       - uses: taiki-e/install-action@37bdc826eaedac215f638a96472df572feab0f9b | ||||
|         with: | ||||
|           tool: wasm-pack | ||||
|       - run: yarn build:wasm | ||||
|       - run: yarn tsc | ||||
|       - name: Build Wasm | ||||
|         shell: bash | ||||
|         run: yarn build:wasm | ||||
|  | ||||
|   yarn-lint: | ||||
|     runs-on: ubuntu-22.04 | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: prepared-wasm | ||||
|           path: | | ||||
|             rust/kcl-wasm-lib/pkg/kcl_wasm_lib* | ||||
|  | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: prepared-ts-rs-bindings | ||||
|           path: | | ||||
|             rust/kcl-lib/bindings/* | ||||
|  | ||||
|   yarn-tsc: | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: yarn-build-wasm | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
| @ -73,9 +87,85 @@ jobs: | ||||
|           node-version-file: '.nvmrc' | ||||
|           cache: 'yarn' | ||||
|       - run: yarn install | ||||
|       - run: yarn --cwd ./rust/kcl-language-server --modules-folder node_modules install | ||||
|  | ||||
|       - name: Download all artifacts | ||||
|         uses: actions/download-artifact@v4 | ||||
|  | ||||
|       - name: Copy prepared wasm | ||||
|         run: | | ||||
|           ls -R prepared-wasm | ||||
|           cp prepared-wasm/kcl_wasm_lib_bg.wasm public | ||||
|           mkdir rust/kcl-wasm-lib/pkg | ||||
|           cp prepared-wasm/kcl_wasm_lib* rust/kcl-wasm-lib/pkg | ||||
|  | ||||
|       - name: Copy prepared ts-rs bindings | ||||
|         run: | | ||||
|           ls -R prepared-ts-rs-bindings | ||||
|           mkdir rust/kcl-lib/bindings | ||||
|           cp -r prepared-ts-rs-bindings/* rust/kcl-lib/bindings/ | ||||
|  | ||||
|       - run: yarn tsc | ||||
|  | ||||
|   yarn-lint: | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: yarn-build-wasm | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version-file: '.nvmrc' | ||||
|           cache: 'yarn' | ||||
|       - run: yarn install | ||||
|  | ||||
|       - name: Download all artifacts | ||||
|         uses: actions/download-artifact@v4 | ||||
|  | ||||
|       - name: Copy prepared wasm | ||||
|         run: | | ||||
|           ls -R prepared-wasm | ||||
|           cp prepared-wasm/kcl_wasm_lib_bg.wasm public | ||||
|           mkdir rust/kcl-wasm-lib/pkg | ||||
|           cp prepared-wasm/kcl_wasm_lib* rust/kcl-wasm-lib/pkg | ||||
|  | ||||
|       - name: Copy prepared ts-rs bindings | ||||
|         run: | | ||||
|           ls -R prepared-ts-rs-bindings | ||||
|           mkdir rust/kcl-lib/bindings | ||||
|           cp -r prepared-ts-rs-bindings/* rust/kcl-lib/bindings/ | ||||
|  | ||||
|       - run: yarn lint | ||||
|  | ||||
|   yarn-circular-dependencies: | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: yarn-build-wasm | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version-file: '.nvmrc' | ||||
|           cache: 'yarn' | ||||
|       - run: yarn install | ||||
|  | ||||
|       - name: Download all artifacts | ||||
|         uses: actions/download-artifact@v4 | ||||
|  | ||||
|       - name: Copy prepared wasm | ||||
|         run: | | ||||
|           ls -R prepared-wasm | ||||
|           cp prepared-wasm/kcl_wasm_lib_bg.wasm public | ||||
|           mkdir rust/kcl-wasm-lib/pkg | ||||
|           cp prepared-wasm/kcl_wasm_lib* rust/kcl-wasm-lib/pkg | ||||
|  | ||||
|       - name: Copy prepared ts-rs bindings | ||||
|         run: | | ||||
|           ls -R prepared-ts-rs-bindings | ||||
|           mkdir rust/kcl-lib/bindings | ||||
|           cp -r prepared-ts-rs-bindings/* rust/kcl-lib/bindings/ | ||||
|  | ||||
|       - run: yarn circular-deps:diff | ||||
|  | ||||
|   python-codespell: | ||||
|     runs-on: ubuntu-22.04 | ||||
|     steps: | ||||
| @ -91,6 +181,7 @@ jobs: | ||||
|  | ||||
|   yarn-unit-test-kcl-samples: | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: yarn-build-wasm | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
| @ -103,7 +194,22 @@ jobs: | ||||
|       - uses: taiki-e/install-action@37bdc826eaedac215f638a96472df572feab0f9b | ||||
|         with: | ||||
|           tool: wasm-pack | ||||
|       - run: yarn build:wasm | ||||
|  | ||||
|       - name: Download all artifacts | ||||
|         uses: actions/download-artifact@v4 | ||||
|  | ||||
|       - name: Copy prepared wasm | ||||
|         run: | | ||||
|           ls -R prepared-wasm | ||||
|           cp prepared-wasm/kcl_wasm_lib_bg.wasm public | ||||
|           mkdir rust/kcl-wasm-lib/pkg | ||||
|           cp prepared-wasm/kcl_wasm_lib* rust/kcl-wasm-lib/pkg | ||||
|  | ||||
|       - name: Copy prepared ts-rs bindings | ||||
|         run: | | ||||
|           ls -R prepared-ts-rs-bindings | ||||
|           mkdir rust/kcl-lib/bindings | ||||
|           cp -r prepared-ts-rs-bindings/* rust/kcl-lib/bindings/ | ||||
|  | ||||
|       - run: yarn simpleserver:bg | ||||
|         if: ${{ github.event_name != 'release' && github.event_name != 'schedule' }} | ||||
| @ -120,6 +226,7 @@ jobs: | ||||
|  | ||||
|   yarn-unit-test: | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: yarn-build-wasm | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
| @ -132,7 +239,22 @@ jobs: | ||||
|       - uses: taiki-e/install-action@37bdc826eaedac215f638a96472df572feab0f9b | ||||
|         with: | ||||
|           tool: wasm-pack | ||||
|       - run: yarn build:wasm | ||||
|  | ||||
|       - name: Download all artifacts | ||||
|         uses: actions/download-artifact@v4 | ||||
|  | ||||
|       - name: Copy prepared wasm | ||||
|         run: | | ||||
|           ls -R prepared-wasm | ||||
|           cp prepared-wasm/kcl_wasm_lib_bg.wasm public | ||||
|           mkdir rust/kcl-wasm-lib/pkg | ||||
|           cp prepared-wasm/kcl_wasm_lib* rust/kcl-wasm-lib/pkg | ||||
|  | ||||
|       - name: Copy prepared ts-rs bindings | ||||
|         run: | | ||||
|           ls -R prepared-ts-rs-bindings | ||||
|           mkdir rust/kcl-lib/bindings | ||||
|           cp -r prepared-ts-rs-bindings/* rust/kcl-lib/bindings/ | ||||
|  | ||||
|       - run: yarn simpleserver:bg | ||||
|         if: ${{ github.event_name != 'release' && github.event_name != 'schedule' }} | ||||
|  | ||||
							
								
								
									
										2
									
								
								.github/workflows/update-e2e-branch.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -38,7 +38,7 @@ jobs: | ||||
|           # Get a new SHA to prevent overwriting the commit status on main | ||||
|           git config --local user.email "github-actions[bot]@users.noreply.github.com" | ||||
|           git config --local user.name "github-actions[bot]" | ||||
|           git commit --amend --message="[all-e2e] $(git log --max-count=1 --pretty=%B)" | ||||
|           git commit --allow-empty --message="[all-e2e] $(git log --max-count=1 --pretty=%B)" | ||||
|  | ||||
|           # Overwrite the branch | ||||
|           git remote set-url origin https://x-access-token:${{ steps.app-token.outputs.token }}@github.com/${{ github.repository }}.git | ||||
|  | ||||
							
								
								
									
										54
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						| @ -4,18 +4,38 @@ all: install build check | ||||
| ############################################################################### | ||||
| # INSTALL | ||||
|  | ||||
| WASM_PACK ?= ~/.cargo/bin/wasm-pack | ||||
| 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 | ||||
| endif | ||||
|  | ||||
| .PHONY: install | ||||
| install: node_modules/.yarn-integrity $(WASM_PACK) ## Install dependencies | ||||
| install: node_modules/.yarn-integrity $(CARGO) $(WASM_PACK) ## Install dependencies | ||||
|  | ||||
| node_modules/.yarn-integrity: package.json yarn.lock | ||||
| 	yarn install | ||||
| ifeq ($(OS),Windows_NT) | ||||
| 	@ type nul > $@ | ||||
| else | ||||
| 	@ touch $@ | ||||
| endif | ||||
|  | ||||
| $(CARGO): | ||||
| ifeq ($(OS),Windows_NT) | ||||
| 	yarn install:rust:windows | ||||
| else | ||||
| 	yarn install:rust | ||||
| endif | ||||
|  | ||||
| $(WASM_PACK): | ||||
| 	yarn install:rust | ||||
| ifeq ($(OS),Windows_NT) | ||||
| 	yarn install:wasm-pack:cargo | ||||
| else | ||||
| 	yarn install:wasm-pack:sh | ||||
| endif | ||||
|  | ||||
| ############################################################################### | ||||
| # BUILD | ||||
| @ -31,13 +51,17 @@ VITE_SOURCES := $(wildcard vite.*) $(wildcard vite/**/*.tsx) | ||||
| build: build-web build-desktop | ||||
|  | ||||
| .PHONY: build-web | ||||
| build-web: public/kcl_wasm_lib_bg.wasm build/index.html | ||||
| build-web: install public/kcl_wasm_lib_bg.wasm build/index.html | ||||
|  | ||||
| .PHONY: build-desktop | ||||
| build-desktop: public/kcl_wasm_lib_bg.wasm .vite/build/main.js | ||||
| build-desktop: install public/kcl_wasm_lib_bg.wasm .vite/build/main.js | ||||
|  | ||||
| public/kcl_wasm_lib_bg.wasm: $(CARGO_SOURCES)$(RUST_SOURCES) | ||||
| 	yarn build:wasm | ||||
| public/kcl_wasm_lib_bg.wasm: $(CARGO_SOURCES) $(RUST_SOURCES) | ||||
| ifeq ($(OS),Windows_NT) | ||||
| 	yarn build:wasm:dev:windows | ||||
| else | ||||
| 	yarn build:wasm:dev | ||||
| endif | ||||
|  | ||||
| build/index.html: $(REACT_SOURCES) $(TYPESCRIPT_SOURCES) $(VITE_SOURCES) | ||||
| 	yarn build:local | ||||
| @ -63,8 +87,10 @@ lint: install ## Lint the code | ||||
| ############################################################################### | ||||
| # RUN | ||||
|  | ||||
| TARGET ?= web | ||||
|  | ||||
| .PHONY: run | ||||
| run: run-web | ||||
| run: run-$(TARGET) | ||||
|  | ||||
| .PHONY: run-web | ||||
| run-web: install build-web ## Start the web app | ||||
| @ -90,7 +116,7 @@ test-unit: install ## Run the unit tests | ||||
| 	yarn test:unit | ||||
|  | ||||
| .PHONY: test-e2e | ||||
| test-e2e: test-e2e-desktop | ||||
| test-e2e: test-e2e-$(TARGET) | ||||
|  | ||||
| .PHONY: test-e2e-web | ||||
| test-e2e-web: install build-web ## Run the web e2e tests | ||||
| @ -99,22 +125,30 @@ test-e2e-web: install build-web ## Run the web e2e tests | ||||
|  | ||||
| .PHONY: test-e2e-desktop | ||||
| test-e2e-desktop: install build-desktop ## Run the desktop e2e tests | ||||
| 	yarn test:playwright:electron --workers=$(E2E_WORKERS) --max-failures=$(E2E_FAILURES) --grep=$(E2E_GREP) | ||||
| 	yarn test:playwright:electron --workers=$(E2E_WORKERS) --max-failures=$(E2E_FAILURES) --grep="$(E2E_GREP)" | ||||
|  | ||||
| ############################################################################### | ||||
| # CLEAN | ||||
|  | ||||
| .PHONY: clean | ||||
| clean: ## Delete all artifacts | ||||
| ifeq ($(OS),Windows_NT) | ||||
| 	git clean --force -d -X | ||||
| else | ||||
| 	rm -rf .vite/ build/ | ||||
| 	rm -rf trace.zip playwright-report/ test-results/ | ||||
| 	rm -rf public/kcl_wasm_lib_bg.wasm | ||||
| 	rm -rf rust/*/bindings/ rust/*/pkg/ rust/target/ | ||||
| 	rm -rf node_modules/ rust/*/node_modules/ | ||||
| endif | ||||
|  | ||||
| .PHONY: help | ||||
| help: install | ||||
| ifeq ($(OS),Windows_NT) | ||||
| 	@ 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}' | ||||
| endif | ||||
|  | ||||
| .DEFAULT_GOAL := help | ||||
|  | ||||
|  | ||||
| @ -9,13 +9,9 @@ layout: manual | ||||
|  | ||||
| ### `std` | ||||
|  | ||||
| - [`HALF_TURN`](/docs/kcl/consts/std-HALF_TURN) | ||||
| - [`QUARTER_TURN`](/docs/kcl/consts/std-QUARTER_TURN) | ||||
| - [`THREE_QUARTER_TURN`](/docs/kcl/consts/std-THREE_QUARTER_TURN) | ||||
| - [`XY`](/docs/kcl/consts/std-XY) | ||||
| - [`XZ`](/docs/kcl/consts/std-XZ) | ||||
| - [`YZ`](/docs/kcl/consts/std-YZ) | ||||
| - [`ZERO`](/docs/kcl/consts/std-ZERO) | ||||
|  | ||||
| ### `std::math` | ||||
|  | ||||
| @ -23,3 +19,10 @@ layout: manual | ||||
| - [`PI`](/docs/kcl/consts/std-math-PI) | ||||
| - [`TAU`](/docs/kcl/consts/std-math-TAU) | ||||
|  | ||||
| ### `std::turns` | ||||
|  | ||||
| - [`HALF_TURN`](/docs/kcl/consts/std-turns-HALF_TURN) | ||||
| - [`QUARTER_TURN`](/docs/kcl/consts/std-turns-QUARTER_TURN) | ||||
| - [`THREE_QUARTER_TURN`](/docs/kcl/consts/std-turns-THREE_QUARTER_TURN) | ||||
| - [`ZERO`](/docs/kcl/consts/std-turns-ZERO) | ||||
|  | ||||
|  | ||||
| @ -1,15 +0,0 @@ | ||||
| --- | ||||
| title: "std::HALF_TURN" | ||||
| excerpt: "" | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ```js | ||||
| std::HALF_TURN: number(deg) = 180deg | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -1,15 +0,0 @@ | ||||
| --- | ||||
| title: "std::QUARTER_TURN" | ||||
| excerpt: "" | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ```js | ||||
| std::QUARTER_TURN: number(deg) = 90deg | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -1,15 +0,0 @@ | ||||
| --- | ||||
| title: "std::THREE_QUARTER_TURN" | ||||
| excerpt: "" | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ```js | ||||
| std::THREE_QUARTER_TURN: number(deg) = 270deg | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -1,15 +0,0 @@ | ||||
| --- | ||||
| title: "std::ZERO" | ||||
| excerpt: "" | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ```js | ||||
| std::ZERO: number = 0 | ||||
| ``` | ||||
|  | ||||
|  | ||||
							
								
								
									
										15
									
								
								docs/kcl/consts/std-turns-HALF_TURN.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,15 @@ | ||||
| --- | ||||
| title: "std::turns::HALF_TURN" | ||||
| excerpt: "" | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ```js | ||||
| std::turns::HALF_TURN: number(deg) = 180deg | ||||
| ``` | ||||
|  | ||||
|  | ||||
							
								
								
									
										15
									
								
								docs/kcl/consts/std-turns-QUARTER_TURN.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,15 @@ | ||||
| --- | ||||
| title: "std::turns::QUARTER_TURN" | ||||
| excerpt: "" | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ```js | ||||
| std::turns::QUARTER_TURN: number(deg) = 90deg | ||||
| ``` | ||||
|  | ||||
|  | ||||
							
								
								
									
										15
									
								
								docs/kcl/consts/std-turns-THREE_QUARTER_TURN.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,15 @@ | ||||
| --- | ||||
| title: "std::turns::THREE_QUARTER_TURN" | ||||
| excerpt: "" | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ```js | ||||
| std::turns::THREE_QUARTER_TURN: number(deg) = 270deg | ||||
| ``` | ||||
|  | ||||
|  | ||||
							
								
								
									
										15
									
								
								docs/kcl/consts/std-turns-ZERO.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,15 @@ | ||||
| --- | ||||
| title: "std::turns::ZERO" | ||||
| excerpt: "" | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ```js | ||||
| std::turns::ZERO: number = 0 | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -23,19 +23,15 @@ layout: manual | ||||
|   * [`tag`](kcl/types/tag) | ||||
| * **std** | ||||
|   * [`Face`](kcl/types/Face) | ||||
|   * [`HALF_TURN`](kcl/consts/std-HALF_TURN) | ||||
|   * [`Helix`](kcl/types/Helix) | ||||
|   * [`Plane`](kcl/types/Plane) | ||||
|   * [`Point2d`](kcl/types/Point2d) | ||||
|   * [`Point3d`](kcl/types/Point3d) | ||||
|   * [`QUARTER_TURN`](kcl/consts/std-QUARTER_TURN) | ||||
|   * [`Sketch`](kcl/types/Sketch) | ||||
|   * [`Solid`](kcl/types/Solid) | ||||
|   * [`THREE_QUARTER_TURN`](kcl/consts/std-THREE_QUARTER_TURN) | ||||
|   * [`XY`](kcl/consts/std-XY) | ||||
|   * [`XZ`](kcl/consts/std-XZ) | ||||
|   * [`YZ`](kcl/consts/std-YZ) | ||||
|   * [`ZERO`](kcl/consts/std-ZERO) | ||||
|   * [`abs`](kcl/abs) | ||||
|   * [`acos`](kcl/acos) | ||||
|   * [`angleToMatchLengthX`](kcl/angleToMatchLengthX) | ||||
| @ -146,3 +142,8 @@ layout: manual | ||||
|   * [`tan`](kcl/std-math-tan) | ||||
| * **std::sketch** | ||||
|   * [`circle`](kcl/std-sketch-circle) | ||||
| * **std::turns** | ||||
|   * [`turns::HALF_TURN`](kcl/consts/std-turns-HALF_TURN) | ||||
|   * [`turns::QUARTER_TURN`](kcl/consts/std-turns-QUARTER_TURN) | ||||
|   * [`turns::THREE_QUARTER_TURN`](kcl/consts/std-turns-THREE_QUARTER_TURN) | ||||
|   * [`turns::ZERO`](kcl/consts/std-turns-ZERO) | ||||
|  | ||||
| @ -17,7 +17,7 @@ circle(@sketch_or_surface: Sketch | Plane | Face, center: Point2d, radius: numbe | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `sketch_or_surface` | [`Sketch`](/docs/kcl/types/Sketch) `|` [`Plane`](/docs/kcl/types/Face) `|` [`Plane`](/docs/kcl/types/Face) | Sketch to extend, or plane or surface to sketch on. | Yes | | ||||
| | `sketch_or_surface` | [`Sketch`](/docs/kcl/types/Sketch) OR [`Plane`](/docs/kcl/types/Plane) OR [`Face`](/docs/kcl/types/Face) | Sketch to extend, or plane or surface to sketch on. | Yes | | ||||
| | `center` | [`Point2d`](/docs/kcl/types/Point2d) | The center of the circle. | Yes | | ||||
| | `radius` | [`number`](/docs/kcl/types/number) | The radius of the circle. | Yes | | ||||
| | [`tag`](/docs/kcl/types/tag) | [`tag`](/docs/kcl/types/tag) | Create a new tag which refers to this circle. | No | | ||||
|  | ||||
							
								
								
									
										6833
									
								
								docs/kcl/std.json
									
									
									
									
									
								
							
							
						
						| @ -28,7 +28,7 @@ An extrude plane. | ||||
| | `faceId` |[`string`](/docs/kcl/types/string)| The face id for the extrude plane. | No | | ||||
| | [`tag`](/docs/kcl/types/tag) |[`TagDeclarator`](/docs/kcl/types#tag-declaration)| The tag. | No | | ||||
| | `id` |[`string`](/docs/kcl/types/string)| The id of the geometry. | No | | ||||
| | `sourceRange` |[`SourceRange`](/docs/kcl/types/SourceRange)| The source range. | No | | ||||
| | `sourceRange` |`[integer, integer, integer]`| The source range. | No | | ||||
|  | ||||
|  | ||||
| ---- | ||||
| @ -48,7 +48,7 @@ An extruded arc. | ||||
| | `faceId` |[`string`](/docs/kcl/types/string)| The face id for the extrude plane. | No | | ||||
| | [`tag`](/docs/kcl/types/tag) |[`TagDeclarator`](/docs/kcl/types#tag-declaration)| The tag. | No | | ||||
| | `id` |[`string`](/docs/kcl/types/string)| The id of the geometry. | No | | ||||
| | `sourceRange` |[`SourceRange`](/docs/kcl/types/SourceRange)| The source range. | No | | ||||
| | `sourceRange` |`[integer, integer, integer]`| The source range. | No | | ||||
|  | ||||
|  | ||||
| ---- | ||||
| @ -68,7 +68,7 @@ Geometry metadata. | ||||
| | `faceId` |[`string`](/docs/kcl/types/string)| The id for the chamfer surface. | No | | ||||
| | [`tag`](/docs/kcl/types/tag) |[`TagDeclarator`](/docs/kcl/types#tag-declaration)| The tag. | No | | ||||
| | `id` |[`string`](/docs/kcl/types/string)| The id of the geometry. | No | | ||||
| | `sourceRange` |[`SourceRange`](/docs/kcl/types/SourceRange)| The source range. | No | | ||||
| | `sourceRange` |`[integer, integer, integer]`| The source range. | No | | ||||
|  | ||||
|  | ||||
| ---- | ||||
| @ -88,7 +88,7 @@ Geometry metadata. | ||||
| | `faceId` |[`string`](/docs/kcl/types/string)| The id for the fillet surface. | No | | ||||
| | [`tag`](/docs/kcl/types/tag) |[`TagDeclarator`](/docs/kcl/types#tag-declaration)| The tag. | No | | ||||
| | `id` |[`string`](/docs/kcl/types/string)| The id of the geometry. | No | | ||||
| | `sourceRange` |[`SourceRange`](/docs/kcl/types/SourceRange)| The source range. | No | | ||||
| | `sourceRange` |`[integer, integer, integer]`| The source range. | No | | ||||
|  | ||||
|  | ||||
| ---- | ||||
|  | ||||
| @ -17,6 +17,6 @@ Geometry metadata. | ||||
| | Property | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `id` |[`string`](/docs/kcl/types/string)| The id of the geometry. | No | | ||||
| | `sourceRange` |[`SourceRange`](/docs/kcl/types/SourceRange)| The source range. | No | | ||||
| | `sourceRange` |`[integer, integer, integer]`| The source range. | No | | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ A helix. | ||||
| | Property | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `value` |[`string`](/docs/kcl/types/string)| The id of the helix. | No | | ||||
| | `artifactId` |[`ArtifactId`](/docs/kcl/types/ArtifactId)| The artifact ID. | No | | ||||
| | `artifactId` |[`string`](/docs/kcl/types/string)| The artifact ID. | No | | ||||
| | `revolutions` |[`number`](/docs/kcl/types/number)| Number of revolutions. | No | | ||||
| | `angleStart` |[`number`](/docs/kcl/types/number)| Start angle (in degrees). | No | | ||||
| | `ccw` |`boolean`| Is the helix rotation counter clockwise? | No | | ||||
|  | ||||
| @ -285,7 +285,7 @@ Data for an imported geometry. | ||||
| | Property | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `type` |enum: `Module`|  | No | | ||||
| | `value` |[`ModuleId`](/docs/kcl/types/ModuleId)| Identifier of a source file.  Uses a u32 to keep the size small. | No | | ||||
| | `value` |`integer`| Identifier of a source file.  Uses a u32 to keep the size small. | No | | ||||
|  | ||||
|  | ||||
| ---- | ||||
|  | ||||
| @ -17,6 +17,6 @@ Data for polar coordinates. | ||||
| | Property | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `angle` |[`number`](/docs/kcl/types/number)| The angle of the line (in degrees). | No | | ||||
| | `length` |[`TyF64`](/docs/kcl/types/TyF64)| The length of the line. | No | | ||||
| | `length` |[`number`](/docs/kcl/types/number)| The length of the line. | No | | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -25,7 +25,7 @@ A sketch type. | ||||
| |----------|------|-------------|----------| | ||||
| | `type` |enum: `plane`|  | No | | ||||
| | `id` |[`string`](/docs/kcl/types/string)| The id of the plane. | No | | ||||
| | `artifactId` |[`ArtifactId`](/docs/kcl/types/ArtifactId)| The artifact ID. | No | | ||||
| | `artifactId` |[`string`](/docs/kcl/types/string)| The artifact ID. | No | | ||||
| | `value` |[`PlaneType`](/docs/kcl/types/PlaneType)| Type for a plane. | No | | ||||
| | `origin` |[`Point3d`](/docs/kcl/types/Point3d)| Origin of the plane. | No | | ||||
| | `xAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the plane's X axis be? | No | | ||||
| @ -49,7 +49,7 @@ A face. | ||||
| |----------|------|-------------|----------| | ||||
| | `type` |enum: `face`|  | No | | ||||
| | `id` |[`string`](/docs/kcl/types/string)| The id of the face. | No | | ||||
| | `artifactId` |[`ArtifactId`](/docs/kcl/types/ArtifactId)| The artifact ID. | No | | ||||
| | `artifactId` |[`string`](/docs/kcl/types/string)| The artifact ID. | No | | ||||
| | `value` |[`string`](/docs/kcl/types/string)| The tag of the face. | No | | ||||
| | `xAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the face's X axis be? | No | | ||||
| | `yAxis` |[`Point3d`](/docs/kcl/types/Point3d)| What should the face's Y axis be? | No | | ||||
|  | ||||
| @ -1,15 +0,0 @@ | ||||
| --- | ||||
| title: "SourceRange" | ||||
| excerpt: "" | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
|  | ||||
| **Type:** `integer` (`uint`) | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -5,17 +5,11 @@ layout: manual | ||||
| --- | ||||
|  | ||||
|  | ||||
| **Type:** `object` | ||||
| **Type:** [`number`](/docs/kcl/types/number) (`double`) | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Properties | ||||
|  | ||||
| | Property | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `n` |[`number`](/docs/kcl/types/number)|  | No | | ||||
| | `ty` |[`NumericType`](/docs/kcl/types/NumericType)|  | No | | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Electron app header tests', () => { | ||||
|   test( | ||||
|  | ||||
| @ -1,13 +1,14 @@ | ||||
| import { Page } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
| import type { Page } from '@playwright/test' | ||||
|  | ||||
| import type { HomePageFixture } from '@e2e/playwright/fixtures/homePageFixture' | ||||
| import { | ||||
|   getUtils, | ||||
|   PERSIST_MODELING_CONTEXT, | ||||
|   TEST_COLORS, | ||||
|   commonPoints, | ||||
|   PERSIST_MODELING_CONTEXT, | ||||
|   getUtils, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
| } from './test-utils' | ||||
| import { HomePageFixture } from './fixtures/homePageFixture' | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.setTimeout(120000) | ||||
|  | ||||
| @ -85,7 +86,7 @@ async function doBasicSketch( | ||||
|   await page.mouse.click(startXPx, 500 - PUR * 20) | ||||
|   if (openPanes.includes('code')) { | ||||
|     await expect(u.codeLocator) | ||||
|       .toHaveText(`sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${ | ||||
|       .toHaveText(`@settings(defaultLengthUnit = in)sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${ | ||||
|       commonPoints.startAt | ||||
|     }, sketch001) | ||||
|   |> xLine(length = ${commonPoints.num1}) | ||||
| @ -119,10 +120,7 @@ async function doBasicSketch( | ||||
|   await page.waitForTimeout(100) | ||||
|  | ||||
|   if (openPanes.includes('code')) { | ||||
|     await expect( | ||||
|       await u.getGreatestPixDiff(line1, TEST_COLORS.BLUE) | ||||
|     ).toBeLessThan(3) | ||||
|     await expect(await u.getGreatestPixDiff(line1, [0, 0, 255])).toBeLessThan(3) | ||||
|     expect(await u.getGreatestPixDiff(line1, TEST_COLORS.BLUE)).toBeLessThan(3) | ||||
|   } | ||||
|  | ||||
|   // hold down shift | ||||
| @ -145,7 +143,7 @@ async function doBasicSketch( | ||||
|   // Open the code pane. | ||||
|   await u.openKclCodePanel() | ||||
|   await expect(u.codeLocator) | ||||
|     .toHaveText(`sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${ | ||||
|     .toHaveText(`@settings(defaultLengthUnit = in)sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${ | ||||
|     commonPoints.startAt | ||||
|   }, sketch001) | ||||
|   |> xLine(length = ${commonPoints.num1}, tag = $seg01) | ||||
|  | ||||
							
								
								
									
										118
									
								
								e2e/playwright/boolean.spec.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,118 @@ | ||||
| import fs from 'node:fs/promises' | ||||
| import path from 'node:path' | ||||
|  | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Point and click for boolean workflows', () => { | ||||
|   // Boolean operations to test | ||||
|   const booleanOperations = [ | ||||
|     { | ||||
|       name: 'union', | ||||
|       code: 'union([extrude001, extrude006])', | ||||
|     }, | ||||
|     { | ||||
|       name: 'subtract', | ||||
|       code: 'subtract([extrude001], tools = [extrude006])', | ||||
|     }, | ||||
|     { | ||||
|       name: 'intersect', | ||||
|       code: 'intersect([extrude001, extrude006])', | ||||
|     }, | ||||
|   ] as const | ||||
|   for (let i = 0; i < booleanOperations.length; i++) { | ||||
|     const operation = booleanOperations[i] | ||||
|     const operationName = operation.name | ||||
|     const commandName = `Boolean ${ | ||||
|       operationName.charAt(0).toUpperCase() + operationName.slice(1) | ||||
|     }` | ||||
|     test(`Create boolean operation -- ${operationName}`, async ({ | ||||
|       context, | ||||
|       homePage, | ||||
|       cmdBar, | ||||
|       editor, | ||||
|       toolbar, | ||||
|       scene, | ||||
|       page, | ||||
|     }) => { | ||||
|       const file = await fs.readFile( | ||||
|         path.resolve( | ||||
|           __dirname, | ||||
|           '../../', | ||||
|           './rust/kcl-lib/e2e/executor/inputs/boolean-setup-with' | ||||
|         ), | ||||
|         'utf-8' | ||||
|       ) | ||||
|       await context.addInitScript((file) => { | ||||
|         localStorage.setItem('persistCode', file) | ||||
|       }, file) | ||||
|       await homePage.goToModelingScene() | ||||
|       await scene.waitForExecutionDone() | ||||
|  | ||||
|       await scene.settled(cmdBar) | ||||
|  | ||||
|       // Test coordinates for selection - these might need adjustment based on actual scene layout | ||||
|       const cylinderPoint = { x: 592, y: 174 } | ||||
|       const secondObjectPoint = { x: 683, y: 273 } | ||||
|  | ||||
|       // Create mouse helpers for selecting objects | ||||
|       const [clickFirstObject] = scene.makeMouseHelpers( | ||||
|         cylinderPoint.x, | ||||
|         cylinderPoint.y, | ||||
|         { steps: 10 } | ||||
|       ) | ||||
|       const [clickSecondObject] = scene.makeMouseHelpers( | ||||
|         secondObjectPoint.x, | ||||
|         secondObjectPoint.y, | ||||
|         { steps: 10 } | ||||
|       ) | ||||
|  | ||||
|       await test.step(`Test ${operationName} operation`, async () => { | ||||
|         // Click the boolean operation button in the toolbar | ||||
|         await toolbar.selectBoolean(operationName) | ||||
|  | ||||
|         // Verify command bar is showing the right command | ||||
|         await expect(cmdBar.page.getByTestId('command-name')).toContainText( | ||||
|           commandName | ||||
|         ) | ||||
|  | ||||
|         // Select first object in the scene, expect there to be a pixel diff from the selection color change | ||||
|         await clickFirstObject({ pixelDiff: 50 }) | ||||
|  | ||||
|         // For subtract, we need to proceed to the next step before selecting the second object | ||||
|         if (operationName !== 'subtract') { | ||||
|           // should down shift key to select multiple objects | ||||
|           await page.keyboard.down('Shift') | ||||
|         } | ||||
|  | ||||
|         // Select second object | ||||
|         await clickSecondObject({ pixelDiff: 50 }) | ||||
|  | ||||
|         // Confirm the operation in the command bar | ||||
|         await cmdBar.progressCmdBar() | ||||
|  | ||||
|         if (operationName === 'union' || operationName === 'intersect') { | ||||
|           await cmdBar.expectState({ | ||||
|             stage: 'review', | ||||
|             headerArguments: { | ||||
|               Solids: '2 paths', | ||||
|             }, | ||||
|             commandName, | ||||
|           }) | ||||
|         } else if (operationName === 'subtract') { | ||||
|           await cmdBar.expectState({ | ||||
|             stage: 'review', | ||||
|             headerArguments: { | ||||
|               Tool: '1 path', | ||||
|               Target: '1 path', | ||||
|             }, | ||||
|             commandName, | ||||
|           }) | ||||
|         } | ||||
|  | ||||
|         await cmdBar.submit() | ||||
|  | ||||
|         await editor.expectEditor.toContain(operation.code) | ||||
|       }) | ||||
|     }) | ||||
|   } | ||||
| }) | ||||
| @ -1,10 +1,11 @@ | ||||
| import { Page } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
| import { HomePageFixture } from './fixtures/homePageFixture' | ||||
| import { getUtils } from './test-utils' | ||||
| import { EngineCommand } from 'lang/std/artifactGraph' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
| import { SceneFixture } from './fixtures/sceneFixture' | ||||
| import type { Page } from '@playwright/test' | ||||
| import type { EngineCommand } from '@src/lang/std/artifactGraph' | ||||
| import { uuidv4 } from '@src/lib/utils' | ||||
|  | ||||
| import type { HomePageFixture } from '@e2e/playwright/fixtures/homePageFixture' | ||||
| import type { SceneFixture } from '@e2e/playwright/fixtures/sceneFixture' | ||||
| import { getUtils } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe( | ||||
|   'Can create sketches on all planes and their back sides', | ||||
| @ -46,7 +47,7 @@ test.describe( | ||||
|         }, | ||||
|       } | ||||
|  | ||||
|       const code = `sketch001 = startSketchOn(${plane})profile001 = startProfileAt([0.91, -1.22], sketch001)` | ||||
|       const code = `@settings(defaultLengthUnit = in)sketch001 = startSketchOn(${plane})profile001 = startProfileAt([0.91, -1.22], sketch001)` | ||||
|  | ||||
|       await u.openDebugPanel() | ||||
|  | ||||
|  | ||||
| @ -1,13 +1,14 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { | ||||
|   orRunWhenFullSuiteEnabled, | ||||
|   getUtils, | ||||
|   executorInputPath, | ||||
| } from './test-utils' | ||||
| import { join } from 'path' | ||||
| import { bracket } from 'lib/exampleKcl' | ||||
| import { TEST_CODE_LONG_WITH_ERROR_OUT_OF_VIEW } from './storageStates' | ||||
| import { bracket } from '@src/lib/exampleKcl' | ||||
| import fsp from 'fs/promises' | ||||
| import { join } from 'path' | ||||
|  | ||||
| import { TEST_CODE_LONG_WITH_ERROR_OUT_OF_VIEW } from '@e2e/playwright/storageStates' | ||||
| import { | ||||
|   executorInputPath, | ||||
|   getUtils, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Code pane and errors', { tag: ['@skipWin'] }, () => { | ||||
|   test('Typing KCL errors induces a badge on the code pane button', async ({ | ||||
| @ -251,11 +252,11 @@ test( | ||||
|       ]) | ||||
|       await Promise.all([ | ||||
|         fsp.copyFile( | ||||
|           executorInputPath('router-template-slate.kcl'), | ||||
|           executorInputPath('cylinder-inches.kcl'), | ||||
|           join(routerTemplateDir, 'main.kcl') | ||||
|         ), | ||||
|         fsp.copyFile( | ||||
|           executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|           executorInputPath('e2e-can-sketch-on-chamfer.kcl'), | ||||
|           join(bracketDir, 'main.kcl') | ||||
|         ), | ||||
|       ]) | ||||
|  | ||||
| @ -1,12 +1,13 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { KCL_DEFAULT_LENGTH } from '@src/lib/constants' | ||||
| import * as fsp from 'fs/promises' | ||||
| import path, { join } from 'path' | ||||
|  | ||||
| import { | ||||
|   executorInputPath, | ||||
|   getUtils, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
| } from './test-utils' | ||||
| import { KCL_DEFAULT_LENGTH } from 'lib/constants' | ||||
| import path, { join } from 'path' | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Command bar tests', { tag: ['@skipWin'] }, () => { | ||||
|   test('Extrude from command bar selects extrude line after', async ({ | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { getUtils } from './test-utils' | ||||
| import { getUtils } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Copilot ghost text', () => { | ||||
|   // eslint-disable-next-line jest/valid-title | ||||
|  | ||||
| @ -1,6 +1,5 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
|  | ||||
| import { getUtils } from './test-utils' | ||||
| import { getUtils } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| function countNewlines(input: string): number { | ||||
|   let count = 0 | ||||
|  | ||||
| @ -1,11 +1,12 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import fsp from 'fs/promises' | ||||
| import path from 'path' | ||||
|  | ||||
| import { | ||||
|   getUtils, | ||||
|   executorInputPath, | ||||
|   getPlaywrightDownloadDir, | ||||
| } from './test-utils' | ||||
| import fsp from 'fs/promises' | ||||
|   getUtils, | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test( | ||||
|   'export works on the first try', | ||||
| @ -20,11 +21,11 @@ test( | ||||
|       await Promise.all([fsp.mkdir(bracketDir, { recursive: true })]) | ||||
|       await Promise.all([ | ||||
|         fsp.copyFile( | ||||
|           executorInputPath('router-template-slate.kcl'), | ||||
|           executorInputPath('cylinder-inches.kcl'), | ||||
|           path.join(bracketDir, 'other.kcl') | ||||
|         ), | ||||
|         fsp.copyFile( | ||||
|           executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|           executorInputPath('e2e-can-sketch-on-chamfer.kcl'), | ||||
|           path.join(bracketDir, 'main.kcl') | ||||
|         ), | ||||
|       ]) | ||||
| @ -107,7 +108,7 @@ test( | ||||
|             }, | ||||
|             { timeout: 15_000 } | ||||
|           ) | ||||
|           .toBeGreaterThan(300_000) | ||||
|           .toBeGreaterThan(30_000) | ||||
|       }) | ||||
|     }) | ||||
|  | ||||
| @ -187,7 +188,7 @@ test( | ||||
|             }, | ||||
|             { timeout: 15_000 } | ||||
|           ) | ||||
|           .toBeGreaterThan(70_000) | ||||
|           .toBeGreaterThan(50_000) | ||||
|       }) | ||||
|     }) | ||||
|   } | ||||
|  | ||||
| @ -1,14 +1,14 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { uuidv4 } from '@src/lib/utils' | ||||
| import fsp from 'fs/promises' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
| import { join } from 'path' | ||||
|  | ||||
| import { | ||||
|   TEST_COLORS, | ||||
|   executorInputPath, | ||||
|   getUtils, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
|   TEST_COLORS, | ||||
| } from './test-utils' | ||||
|  | ||||
| import { join } from 'path' | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Editor tests', { tag: ['@skipWin'] }, () => { | ||||
|   test('can comment out code with ctrl+/', async ({ page, homePage }) => { | ||||
| @ -32,26 +32,30 @@ test.describe('Editor tests', { tag: ['@skipWin'] }, () => { | ||||
|     await page.keyboard.press('/') | ||||
|     await page.keyboard.up('ControlOrMeta') | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn(XY) | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XY) | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line(end = [20, 0]) | ||||
|   |> line(end = [0, 20]) | ||||
|   |> line(end = [-20, 0]) | ||||
|   // |> close()`) | ||||
|   // |> close()`.replaceAll('\n', '') | ||||
|     ) | ||||
|  | ||||
|     // uncomment the code | ||||
|     await page.keyboard.down('ControlOrMeta') | ||||
|     await page.keyboard.press('/') | ||||
|     await page.keyboard.up('ControlOrMeta') | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn(XY) | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XY) | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line(end = [20, 0]) | ||||
|   |> line(end = [0, 20]) | ||||
|   |> line(end = [-20, 0]) | ||||
|   |> close()`) | ||||
|   |> close()`.replaceAll('\n', '') | ||||
|     ) | ||||
|   }) | ||||
|  | ||||
|   test('ensure we use the cache, and do not re-execute', async ({ | ||||
| @ -178,13 +182,15 @@ test.describe('Editor tests', { tag: ['@skipWin'] }, () => { | ||||
|     await page.locator('#code-pane button:first-child').click() | ||||
|     await page.locator('button:has-text("Format code")').click() | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn(XY) | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XY) | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line(end = [20, 0]) | ||||
|   |> line(end = [0, 20]) | ||||
|   |> line(end = [-20, 0]) | ||||
|   |> close()`) | ||||
|   |> close()`.replaceAll('\n', '') | ||||
|     ) | ||||
|   }) | ||||
|  | ||||
|   test('if you click the format button it formats your code and executes so lints are still there', async ({ | ||||
| @ -227,13 +233,15 @@ test.describe('Editor tests', { tag: ['@skipWin'] }, () => { | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch_001 = startSketchOn(XY) | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       `@settings(defaultLengthUnit = in) | ||||
| sketch_001 = startSketchOn(XY) | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line(end = [20, 0]) | ||||
|   |> line(end = [0, 20]) | ||||
|   |> line(end = [-20, 0]) | ||||
|   |> close()`) | ||||
|   |> close()`.replaceAll('\n', '') | ||||
|     ) | ||||
|  | ||||
|     // error in guter | ||||
|     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible() | ||||
| @ -471,6 +479,7 @@ test.describe('Editor tests', { tag: ['@skipWin'] }, () => { | ||||
|   test('if you write kcl with lint errors you get lints', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|     scene, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
| @ -490,10 +499,7 @@ test.describe('Editor tests', { tag: ['@skipWin'] }, () => { | ||||
|     await page.keyboard.press('ArrowLeft') | ||||
|     await page.keyboard.press('ArrowRight') | ||||
|  | ||||
|     // FIXME: lsp errors do not propagate to the frontend until engine is connected and code is executed | ||||
|     // This timeout is to wait for engine connection. LSP and code execution errors should be handled differently | ||||
|     // LSP can emit errors as fast as it waits and show them in the editor | ||||
|     await page.waitForTimeout(10000) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // error in guter | ||||
|     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible() | ||||
| @ -815,10 +821,12 @@ test.describe('Editor tests', { tag: ['@skipWin'] }, () => { | ||||
|       // there shouldn't be any auto complete options for 'lin' in the comment | ||||
|       await expect(page.locator('.cm-completionLabel')).not.toBeVisible() | ||||
|  | ||||
|       await expect(page.locator('.cm-content')) | ||||
|         .toHaveText(`sketch001 = startSketchOn(XZ) | ||||
|       await expect(page.locator('.cm-content')).toHaveText( | ||||
|         `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([3.14, 12], %) | ||||
|         |> xLine(%, length = 5) // lin`) | ||||
|         |> xLine(%, length = 5) // lin`.replaceAll('\n', '') | ||||
|       ) | ||||
|  | ||||
|       // expect there to be no KCL errors | ||||
|       await expect(page.locator('.cm-lint-marker-error')).toHaveCount(0) | ||||
| @ -888,10 +896,12 @@ test.describe('Editor tests', { tag: ['@skipWin'] }, () => { | ||||
|       // there shouldn't be any auto complete options for 'lin' in the comment | ||||
|       await expect(page.locator('.cm-completionLabel')).not.toBeVisible() | ||||
|  | ||||
|       await expect(page.locator('.cm-content')) | ||||
|         .toHaveText(`sketch001 = startSketchOn(XZ) | ||||
|       await expect(page.locator('.cm-content')).toHaveText( | ||||
|         `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([3.14, 12], %) | ||||
|         |> xLine(%, length = 5) // lin`) | ||||
|         |> xLine(%, length = 5) // lin`.replaceAll('\n', '') | ||||
|       ) | ||||
|     }) | ||||
|   }) | ||||
|   test('Can undo a click and point extrude with ctrl+z', async ({ | ||||
| @ -975,12 +985,13 @@ test.describe('Editor tests', { tag: ['@skipWin'] }, () => { | ||||
|   test( | ||||
|     'Can undo a sketch modification with ctrl+z', | ||||
|     { tag: ['@skipWin'] }, | ||||
|     async ({ page, homePage }) => { | ||||
|     async ({ page, homePage, editor }) => { | ||||
|       const u = await getUtils(page) | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `sketch001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit=in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([4.61, -10.01], %) | ||||
|   |> line(end = [12.73, -0.09]) | ||||
|   |> tangentialArcTo([24.95, -0.38], %) | ||||
| @ -1070,41 +1081,45 @@ test.describe('Editor tests', { tag: ['@skipWin'] }, () => { | ||||
|       await expect(page.locator('.cm-content')).not.toHaveText(prevContent) | ||||
|  | ||||
|       // expect the code to have changed | ||||
|       await expect(page.locator('.cm-content')) | ||||
|         .toHaveText(`sketch001 = startSketchOn(XZ) | ||||
|       await editor.expectEditor.toContain( | ||||
|         `sketch001 = startSketchOn(XZ) | ||||
|     |> startProfileAt([2.71, -2.71], %) | ||||
|     |> line(end = [15.4, -2.78]) | ||||
|     |> tangentialArcTo([27.6, -3.05], %) | ||||
|     |> close() | ||||
|     |> extrude(length = 5) | ||||
|   `) | ||||
|     |> extrude(length = 5)`, | ||||
|         { shouldNormalise: true } | ||||
|       ) | ||||
|  | ||||
|       // Hit undo | ||||
|       await page.keyboard.down('Control') | ||||
|       await page.keyboard.press('KeyZ') | ||||
|       await page.keyboard.up('Control') | ||||
|  | ||||
|       await expect(page.locator('.cm-content')) | ||||
|         .toHaveText(`sketch001 = startSketchOn(XZ) | ||||
|       await editor.expectEditor.toContain( | ||||
|         `sketch001 = startSketchOn(XZ) | ||||
|     |> startProfileAt([2.71, -2.71], %) | ||||
|     |> line(end = [15.4, -2.78]) | ||||
|     |> tangentialArcTo([24.95, -0.38], %) | ||||
|     |> close() | ||||
|     |> extrude(length = 5)`) | ||||
|     |> extrude(length = 5)`, | ||||
|         { shouldNormalise: true } | ||||
|       ) | ||||
|  | ||||
|       // Hit undo again. | ||||
|       await page.keyboard.down('Control') | ||||
|       await page.keyboard.press('KeyZ') | ||||
|       await page.keyboard.up('Control') | ||||
|  | ||||
|       await expect(page.locator('.cm-content')) | ||||
|         .toHaveText(`sketch001 = startSketchOn(XZ) | ||||
|       await editor.expectEditor.toContain( | ||||
|         `sketch001 = startSketchOn(XZ) | ||||
|     |> startProfileAt([2.71, -2.71], %) | ||||
|     |> line(end = [12.73, -0.09]) | ||||
|     |> tangentialArcTo([24.95, -0.38], %) | ||||
|     |> close() | ||||
|     |> extrude(length = 5) | ||||
|   `) | ||||
|     |> extrude(length = 5)`, | ||||
|         { shouldNormalise: true } | ||||
|       ) | ||||
|  | ||||
|       // Hit undo again. | ||||
|       await page.keyboard.down('Control') | ||||
| @ -1112,13 +1127,15 @@ test.describe('Editor tests', { tag: ['@skipWin'] }, () => { | ||||
|       await page.keyboard.up('Control') | ||||
|  | ||||
|       await page.waitForTimeout(100) | ||||
|       await expect(page.locator('.cm-content')) | ||||
|         .toHaveText(`sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([4.61, -10.01], %) | ||||
|   |> line(end = [12.73, -0.09]) | ||||
|   |> tangentialArcTo([24.95, -0.38], %) | ||||
|   |> close() | ||||
|   |> extrude(length = 5)`) | ||||
|       await editor.expectEditor.toContain( | ||||
|         `sketch001 = startSketchOn(XZ) | ||||
|     |> startProfileAt([4.61, -10.01], %) | ||||
|     |> line(end = [12.73, -0.09]) | ||||
|     |> tangentialArcTo([24.95, -0.38], %) | ||||
|     |> close() | ||||
|     |> extrude(length = 5)`, | ||||
|         { shouldNormalise: true } | ||||
|       ) | ||||
|     } | ||||
|   ) | ||||
|  | ||||
| @ -1206,4 +1223,55 @@ test.describe('Editor tests', { tag: ['@skipWin'] }, () => { | ||||
|       }) | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test('Rectangle tool panning with middle click', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|     toolbar, | ||||
|     scene, | ||||
|     cmdBar, | ||||
|     editor, | ||||
|   }) => { | ||||
|     await page.setBodyDimensions({ width: 1200, height: 900 }) | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // wait until scene is ready to be interacted with | ||||
|     await scene.connectionEstablished() | ||||
|     await scene.settled(cmdBar) | ||||
|  | ||||
|     await page.getByRole('button', { name: 'Start Sketch' }).click() | ||||
|  | ||||
|     // select an axis plane | ||||
|     await page.mouse.click(700, 200) | ||||
|  | ||||
|     // Needed as we don't yet have a way to get a signal from the engine that the camera has animated to the sketch plane | ||||
|     await page.waitForTimeout(1000) | ||||
|  | ||||
|     const middleMousePan = async ( | ||||
|       startX: number, | ||||
|       startY: number, | ||||
|       endX: number, | ||||
|       endY: number | ||||
|     ) => { | ||||
|       const initialCode = await editor.getCurrentCode() | ||||
|  | ||||
|       await page.mouse.click(startX, startY, { button: 'middle' }) | ||||
|       await page.mouse.move(endX, endY, { | ||||
|         steps: 10, | ||||
|       }) | ||||
|  | ||||
|       // We expect the code to be the same, middle mouse click should not modify the code, only do panning | ||||
|       await editor.expectEditor.toBe(initialCode) | ||||
|     } | ||||
|  | ||||
|     await test.step(`Verify corner rectangle panning`, async () => { | ||||
|       await page.getByTestId('corner-rectangle').click() | ||||
|       await middleMousePan(800, 500, 900, 600) | ||||
|     }) | ||||
|  | ||||
|     await test.step(`Verify center rectangle panning`, async () => { | ||||
|       await toolbar.selectCenterRectangle() | ||||
|       await middleMousePan(800, 200, 900, 300) | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -1,7 +1,8 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import * as fsp from 'fs/promises' | ||||
| import { join } from 'path' | ||||
|  | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| const FEATURE_TREE_EXAMPLE_CODE = `export fn timesFive(x) { | ||||
|   return 5 * x | ||||
| } | ||||
|  | ||||
| @ -1,15 +1,16 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import * as fsp from 'fs/promises' | ||||
| import { FILE_EXT } from '@src/lib/constants' | ||||
| import * as fs from 'fs' | ||||
| import * as fsp from 'fs/promises' | ||||
| import { join } from 'path' | ||||
|  | ||||
| import { | ||||
|   createProject, | ||||
|   executorInputPath, | ||||
|   getUtils, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
|   runningOnWindows, | ||||
| } from './test-utils' | ||||
| import { join } from 'path' | ||||
| import { FILE_EXT } from 'lib/constants' | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('integrations tests', () => { | ||||
|   test( | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| import type { Page, Locator, Route, Request } from '@playwright/test' | ||||
| import { expect, TestInfo } from '@playwright/test' | ||||
| import type { Locator, Page, Request, Route, TestInfo } from '@playwright/test' | ||||
| import { expect } from '@playwright/test' | ||||
| import * as fs from 'fs' | ||||
| import * as path from 'path' | ||||
|  | ||||
|  | ||||
| @ -1,11 +1,12 @@ | ||||
| import type { Page, Locator } from '@playwright/test' | ||||
| import type { Locator, Page } from '@playwright/test' | ||||
| import { expect } from '@playwright/test' | ||||
|  | ||||
| import { | ||||
|   closePane, | ||||
|   checkIfPaneIsOpen, | ||||
|   closePane, | ||||
|   openPane, | ||||
|   sansWhitespace, | ||||
| } from '../test-utils' | ||||
| } from '@e2e/playwright/test-utils' | ||||
|  | ||||
| interface EditorState { | ||||
|   activeLines: Array<string> | ||||
| @ -81,6 +82,13 @@ export class EditorFixture { | ||||
|   expectEditor = { | ||||
|     toContain: this._expectEditorToContain(), | ||||
|     not: { toContain: this._expectEditorToContain(true) }, | ||||
|     toBe: async (code: string) => { | ||||
|       const currentCode = await this.getCurrentCode() | ||||
|       return expect(currentCode).toBe(code) | ||||
|     }, | ||||
|   } | ||||
|   getCurrentCode = async () => { | ||||
|     return await this.codeContent.innerText() | ||||
|   } | ||||
|   snapshot = async (options?: { timeout?: number; name?: string }) => { | ||||
|     const wasPaneOpen = await this.checkIfPaneIsOpen() | ||||
|  | ||||
| @ -1,28 +1,28 @@ | ||||
| /* eslint-disable react-hooks/rules-of-hooks */ | ||||
|  | ||||
| import type { | ||||
|   BrowserContext, | ||||
|   ElectronApplication, | ||||
|   TestInfo, | ||||
|   Page, | ||||
|   TestInfo, | ||||
| } from '@playwright/test' | ||||
|  | ||||
| import { _electron as electron } from '@playwright/test' | ||||
|  | ||||
| import * as TOML from '@iarna/toml' | ||||
| import { TEST_SETTINGS } from '../storageStates' | ||||
| import { SETTINGS_FILE_NAME } from 'lib/constants' | ||||
| import { getUtils, setup } from '../test-utils' | ||||
| import { SETTINGS_FILE_NAME } from '@src/lib/constants' | ||||
| import type { DeepPartial } from '@src/lib/types' | ||||
| import fsp from 'fs/promises' | ||||
| import fs from 'node:fs' | ||||
| import path from 'path' | ||||
| import { CmdBarFixture } from './cmdBarFixture' | ||||
| import { EditorFixture } from './editorFixture' | ||||
| import { ToolbarFixture } from './toolbarFixture' | ||||
| import { SceneFixture } from './sceneFixture' | ||||
| import { HomePageFixture } from './homePageFixture' | ||||
| import { DeepPartial } from 'lib/types' | ||||
| import { Settings } from '@rust/kcl-lib/bindings/Settings' | ||||
|  | ||||
| import type { Settings } from '@rust/kcl-lib/bindings/Settings' | ||||
|  | ||||
| import { CmdBarFixture } from '@e2e/playwright/fixtures/cmdBarFixture' | ||||
| import { EditorFixture } from '@e2e/playwright/fixtures/editorFixture' | ||||
| import { HomePageFixture } from '@e2e/playwright/fixtures/homePageFixture' | ||||
| import { SceneFixture } from '@e2e/playwright/fixtures/sceneFixture' | ||||
| import { ToolbarFixture } from '@e2e/playwright/fixtures/toolbarFixture' | ||||
|  | ||||
| import { TEST_SETTINGS } from '@e2e/playwright/storageStates' | ||||
| import { getUtils, settingsToToml, setup } from '@e2e/playwright/test-utils' | ||||
|  | ||||
| export class AuthenticatedApp { | ||||
|   public readonly page: Page | ||||
| @ -124,6 +124,7 @@ export class ElectronZoo { | ||||
|  | ||||
|     // We need to expose this in order for some tests that require folder | ||||
|     // creation and some code below. | ||||
|     // eslint-disable-next-line @typescript-eslint/no-this-alias | ||||
|     const that = this | ||||
|  | ||||
|     const options = { | ||||
| @ -286,26 +287,30 @@ export class ElectronZoo { | ||||
|     let settingsOverridesToml = '' | ||||
|  | ||||
|     if (appSettings) { | ||||
|       settingsOverridesToml = TOML.stringify({ | ||||
|         // @ts-expect-error | ||||
|       settingsOverridesToml = settingsToToml({ | ||||
|         settings: { | ||||
|           ...TEST_SETTINGS, | ||||
|           ...appSettings, | ||||
|           app: { | ||||
|             ...TEST_SETTINGS.app, | ||||
|             project_directory: this.projectDirName, | ||||
|             ...appSettings.app, | ||||
|           }, | ||||
|           project: { | ||||
|             ...TEST_SETTINGS.project, | ||||
|             directory: this.projectDirName, | ||||
|           }, | ||||
|         }, | ||||
|       }) | ||||
|     } else { | ||||
|       settingsOverridesToml = TOML.stringify({ | ||||
|         // @ts-expect-error | ||||
|       settingsOverridesToml = settingsToToml({ | ||||
|         settings: { | ||||
|           ...TEST_SETTINGS, | ||||
|           app: { | ||||
|             ...TEST_SETTINGS.app, | ||||
|             project_directory: this.projectDirName, | ||||
|           }, | ||||
|           project: { | ||||
|             ...TEST_SETTINGS.project, | ||||
|             directory: this.projectDirName, | ||||
|           }, | ||||
|         }, | ||||
|       }) | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import type { Page, Locator } from '@playwright/test' | ||||
| import type { Locator, Page } from '@playwright/test' | ||||
| import { expect } from '@playwright/test' | ||||
|  | ||||
| interface ProjectCardState { | ||||
|  | ||||
| @ -1,15 +1,17 @@ | ||||
| import type { Page, Locator } from '@playwright/test' | ||||
| import { expect } from '../zoo-test' | ||||
| import { isArray, uuidv4 } from 'lib/utils' | ||||
| import { CmdBarFixture } from './cmdBarFixture' | ||||
| import type { Locator, Page } from '@playwright/test' | ||||
| import { isArray, uuidv4 } from '@src/lib/utils' | ||||
|  | ||||
| import type { CmdBarFixture } from '@e2e/playwright/fixtures/cmdBarFixture' | ||||
|  | ||||
| import { | ||||
|   closeDebugPanel, | ||||
|   doAndWaitForImageDiff, | ||||
|   getPixelRGBs, | ||||
|   getUtils, | ||||
|   openAndClearDebugPanel, | ||||
|   sendCustomCmd, | ||||
|   getUtils, | ||||
| } from '../test-utils' | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| type MouseParams = { | ||||
|   pixelDiff?: number | ||||
| @ -310,7 +312,9 @@ export async function expectPixelColor( | ||||
|     .toBeTruthy() | ||||
|     .catch((cause) => { | ||||
|       throw new Error( | ||||
|         `ExpectPixelColor: expecting ${colour} got ${finalValue}`, | ||||
|         `ExpectPixelColor: point ${JSON.stringify( | ||||
|           coords | ||||
|         )} was expecting ${colour} but got ${finalValue}`, | ||||
|         { cause } | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
| @ -1,14 +1,18 @@ | ||||
| import { type Page, type Locator, test } from '@playwright/test' | ||||
| import { expect } from '../zoo-test' | ||||
| import { type Locator, type Page, test } from '@playwright/test' | ||||
| import type { SidebarType } from '@src/components/ModelingSidebar/ModelingPanes' | ||||
| import { SIDEBAR_BUTTON_SUFFIX } from '@src/lib/constants' | ||||
| import type { ToolbarModeName } from '@src/lib/toolbar' | ||||
|  | ||||
| import { | ||||
|   checkIfPaneIsOpen, | ||||
|   closePane, | ||||
|   doAndWaitForImageDiff, | ||||
|   openPane, | ||||
| } from '../test-utils' | ||||
| import { SidebarType } from 'components/ModelingSidebar/ModelingPanes' | ||||
| import { SIDEBAR_BUTTON_SUFFIX } from 'lib/constants' | ||||
| import { ToolbarModeName } from 'lib/toolbar' | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect } from '@e2e/playwright/zoo-test' | ||||
| import { type baseUnitLabels } from '@src/lib/settings/settingsTypes' | ||||
|  | ||||
| type LengthUnitLabel = (typeof baseUnitLabels)[keyof typeof baseUnitLabels] | ||||
|  | ||||
| export class ToolbarFixture { | ||||
|   public page: Page | ||||
| @ -181,6 +185,14 @@ export class ToolbarFixture { | ||||
|     ).toBeVisible() | ||||
|     await this.page.getByTestId('dropdown-center-rectangle').click() | ||||
|   } | ||||
|   selectBoolean = async (operation: 'union' | 'subtract' | 'intersect') => { | ||||
|     await this.page | ||||
|       .getByRole('button', { name: 'caret down Union: open menu' }) | ||||
|       .click() | ||||
|     const operationTestId = `dropdown-boolean-${operation}` | ||||
|     await expect(this.page.getByTestId(operationTestId)).toBeVisible() | ||||
|     await this.page.getByTestId(operationTestId).click() | ||||
|   } | ||||
|  | ||||
|   selectCircleThreePoint = async () => { | ||||
|     await this.page | ||||
| @ -227,6 +239,12 @@ export class ToolbarFixture { | ||||
|   async checkIfFeatureTreePaneIsOpen() { | ||||
|     return this.checkIfPaneIsOpen(this.featureTreeId) | ||||
|   } | ||||
|   async selectUnit(unit: LengthUnitLabel) { | ||||
|     await this.page.getByTestId('units-menu').click() | ||||
|     const optionLocator = this.page.getByRole('button', { name: unit }) | ||||
|     await expect(optionLocator).toBeVisible() | ||||
|     await optionLocator.click() | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * Get a specific operation button from the Feature Tree pane. | ||||
|  | ||||
| @ -257,6 +257,29 @@ export const isErrorWhitelisted = (exception: Error) => { | ||||
|       project: 'Google Chrome', | ||||
|       foundInSpec: 'e2e/playwright/testing-settings.spec.ts', | ||||
|     }, | ||||
|     // TODO: fix this error in the code | ||||
|     { | ||||
|       name: 'ReferenceError', | ||||
|       message: '_testUtils is not defined', | ||||
|       stack: '', | ||||
|       project: 'Google Chrome', | ||||
|       foundInSpec: 'e2e/playwright/snapshot-tests.spec.ts', | ||||
|     }, | ||||
|     { | ||||
|       name: 'TypeError', | ||||
|       message: 'Failed to fetch', | ||||
|       stack: '', | ||||
|       project: 'Google Chrome', | ||||
|       foundInSpec: 'e2e/playwright/snapshot-tests.spec.ts', | ||||
|     }, | ||||
|     { | ||||
|       name: 'Error', | ||||
|       message: 'The "path" argument must be of type string. Received undefined', | ||||
|       stack: | ||||
|         'Error: The "path" argument must be of type string. Received undefined', | ||||
|       project: 'Google Chrome', | ||||
|       foundInSpec: '', // many tests are impacted by this error | ||||
|     }, | ||||
|   ] | ||||
|  | ||||
|   const cleanString = (str: string) => str.replace(/[`"]/g, '') | ||||
|  | ||||
| @ -1,7 +1,8 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { executorInputPath } from './test-utils' | ||||
| import { join } from 'path' | ||||
| import fsp from 'fs/promises' | ||||
| import { join } from 'path' | ||||
|  | ||||
| import { executorInputPath } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test( | ||||
|   'When machine-api server not found butt is disabled and shows the reason', | ||||
| @ -11,7 +12,7 @@ test( | ||||
|       const bracketDir = join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|       await fsp.copyFile( | ||||
|         executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|         executorInputPath('cylinder-inches.kcl'), | ||||
|         join(bracketDir, 'main.kcl') | ||||
|       ) | ||||
|     }) | ||||
| @ -51,7 +52,7 @@ test( | ||||
|       const bracketDir = join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|       await fsp.copyFile( | ||||
|         executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|         executorInputPath('cylinder-inches.kcl'), | ||||
|         join(bracketDir, 'main.kcl') | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
| @ -1,13 +1,15 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { PROJECT_SETTINGS_FILE_NAME } from 'lib/constants' | ||||
| import { PROJECT_SETTINGS_FILE_NAME } from '@src/lib/constants' | ||||
| import * as fsp from 'fs/promises' | ||||
| import { join } from 'path' | ||||
|  | ||||
| import type { NamedView } from '@rust/kcl-lib/bindings/NamedView' | ||||
|  | ||||
| import { | ||||
|   createProject, | ||||
|   tomlToPerProjectSettings, | ||||
|   perProjectsettingsToToml, | ||||
| } from './test-utils' | ||||
| import { NamedView } from '@rust/kcl-lib/bindings/NamedView' | ||||
|   tomlToPerProjectSettings, | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| // Helper function to determine if the file path on disk exists | ||||
| // Specifically this is used to check if project.toml exists on disk | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| /** | ||||
|  * Not all menu actions are tested. Some are default electron menu actions. | ||||
| @ -10,6 +10,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       test('File.Create project', async ({ tronApp, cmdBar, page }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const newProject = | ||||
| @ -29,6 +30,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       test('File.Open project', async ({ tronApp, cmdBar, page }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const openProject = | ||||
| @ -52,6 +54,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const userSettings = app.applicationMenu.getMenuItemById( | ||||
| @ -75,6 +78,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const keybindings = app.applicationMenu.getMenuItemById( | ||||
| @ -96,6 +100,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const menu = app.applicationMenu.getMenuItemById( | ||||
| @ -112,6 +117,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       test('File.Preferences.Theme', async ({ tronApp, cmdBar, page }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const menu = app.applicationMenu.getMenuItemById( | ||||
| @ -136,6 +142,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const menu = app.applicationMenu.getMenuItemById( | ||||
| @ -152,6 +159,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       test('File.Preferences.Sign out', async ({ tronApp, cmdBar, page }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const menu = app.applicationMenu.getMenuItemById('File.Sign out') | ||||
| @ -170,6 +178,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       test('Edit.Rename project', async ({ tronApp, cmdBar, page }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const menu = app.applicationMenu.getMenuItemById( | ||||
| @ -188,6 +197,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       test('Edit.Delete project', async ({ tronApp, cmdBar, page }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const menu = app.applicationMenu.getMenuItemById( | ||||
| @ -210,6 +220,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const menu = app.applicationMenu.getMenuItemById( | ||||
| @ -228,6 +239,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       test('View.Command Palette...', async ({ tronApp, cmdBar, page }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const menu = app.applicationMenu.getMenuItemById( | ||||
| @ -245,6 +257,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       test('Help.Show all commands', async ({ tronApp, cmdBar, page }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const menu = app.applicationMenu.getMenuItemById( | ||||
| @ -260,6 +273,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       test('Help.KCL code samples', async ({ tronApp, cmdBar, page }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const menu = app.applicationMenu.getMenuItemById( | ||||
| @ -275,6 +289,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const menu = app.applicationMenu.getMenuItemById( | ||||
| @ -293,6 +308,7 @@ test.describe('Native file menu', { tag: ['@electron'] }, () => { | ||||
|       test('Help.Reset onboarding', async ({ tronApp, cmdBar, page }) => { | ||||
|         if (!tronApp) fail() | ||||
|         // Run electron snippet to find the Menu! | ||||
|         await page.waitForTimeout(100) // wait for createModelingPageMenu() to run | ||||
|         await tronApp.electron.evaluate(async ({ app }) => { | ||||
|           if (!app || !app.applicationMenu) fail() | ||||
|           const menu = app.applicationMenu.getMenuItemById( | ||||
|  | ||||
| @ -2,8 +2,7 @@ | ||||
| // application, check it can make it to the project pane, and nothing more. | ||||
| // It also tests our test wrappers are working. | ||||
| // Additionally this serves as a nice minimal example. | ||||
|  | ||||
| import { test, expect } from './zoo-test' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Open the application', () => { | ||||
|   test('see the project view', async ({ page, context }) => { | ||||
|  | ||||
| @ -1,22 +1,23 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { join } from 'path' | ||||
| import { bracket } from '@src/lib/exampleKcl' | ||||
| import { onboardingPaths } from '@src/routes/Onboarding/paths' | ||||
| import fsp from 'fs/promises' | ||||
| import { | ||||
|   getUtils, | ||||
|   executorInputPath, | ||||
|   createProject, | ||||
|   settingsToToml, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
| } from './test-utils' | ||||
| import { bracket } from 'lib/exampleKcl' | ||||
| import { onboardingPaths } from 'routes/Onboarding/paths' | ||||
| import { join } from 'path' | ||||
|  | ||||
| import { expectPixelColor } from '@e2e/playwright/fixtures/sceneFixture' | ||||
| import { | ||||
|   TEST_SETTINGS_KEY, | ||||
|   TEST_SETTINGS_ONBOARDING_START, | ||||
|   TEST_SETTINGS_ONBOARDING_EXPORT, | ||||
|   TEST_SETTINGS_ONBOARDING_START, | ||||
|   TEST_SETTINGS_ONBOARDING_USER_MENU, | ||||
| } from './storageStates' | ||||
| import { expectPixelColor } from './fixtures/sceneFixture' | ||||
| } from '@e2e/playwright/storageStates' | ||||
| import { | ||||
|   createProject, | ||||
|   executorInputPath, | ||||
|   getUtils, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
|   settingsToToml, | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| // Because our default test settings have the onboardingStatus set to 'dismissed', | ||||
| // we must set it to empty for the tests where we want to see the onboarding immediately. | ||||
| @ -230,9 +231,9 @@ test.describe('Onboarding tests', () => { | ||||
|  | ||||
|     // Override beforeEach test setup | ||||
|     await context.addInitScript( | ||||
|       async ({ settingsKey, settings }) => { | ||||
|       async ({ settingsKey, settings, code }) => { | ||||
|         // Give some initial code, so we can test that it's cleared | ||||
|         localStorage.setItem('persistCode', originalCode) | ||||
|         localStorage.setItem('persistCode', code) | ||||
|         localStorage.setItem(settingsKey, settings) | ||||
|       }, | ||||
|       { | ||||
| @ -240,6 +241,7 @@ test.describe('Onboarding tests', () => { | ||||
|         settings: settingsToToml({ | ||||
|           settings: TEST_SETTINGS_ONBOARDING_EXPORT, | ||||
|         }), | ||||
|         code: originalCode, | ||||
|       } | ||||
|     ) | ||||
|  | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { expect, test } from '@playwright/test' | ||||
|  | ||||
| /** @deprecated, import from ./fixtureSetup.ts instead */ | ||||
| export const _test = test | ||||
|  | ||||
| @ -1,12 +1,12 @@ | ||||
| import { Page } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
| import { EditorFixture } from './fixtures/editorFixture' | ||||
| import { SceneFixture } from './fixtures/sceneFixture' | ||||
| import { ToolbarFixture } from './fixtures/toolbarFixture' | ||||
| import type { Locator, Page } from '@playwright/test' | ||||
| import fs from 'node:fs/promises' | ||||
| import path from 'node:path' | ||||
| import { getUtils, orRunWhenFullSuiteEnabled } from './test-utils' | ||||
| import { Locator } from '@playwright/test' | ||||
|  | ||||
| import type { EditorFixture } from '@e2e/playwright/fixtures/editorFixture' | ||||
| import type { SceneFixture } from '@e2e/playwright/fixtures/sceneFixture' | ||||
| import type { ToolbarFixture } from '@e2e/playwright/fixtures/toolbarFixture' | ||||
| import { getUtils, orRunWhenFullSuiteEnabled } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| // test file is for testing point an click code gen functionality that's not sketch mode related | ||||
|  | ||||
| @ -137,7 +137,7 @@ test.describe('Point-and-click tests', () => { | ||||
|  | ||||
|         await scene.moveCameraTo(cameraPos, cameraTarget) | ||||
|  | ||||
|         await test.step('check chamfer selection changes cursor positon', async () => { | ||||
|         await test.step('check chamfer selection changes cursor position', async () => { | ||||
|           await expect(async () => { | ||||
|             // sometimes initial click doesn't register | ||||
|             await clickChamfer() | ||||
| @ -173,7 +173,7 @@ test.describe('Point-and-click tests', () => { | ||||
|         }) | ||||
|         await test.step('Check there is no errors after code created in previous steps executes', async () => { | ||||
|           await editor.expectState({ | ||||
|             activeLines: ['sketch001 = startSketchOn(XZ)'], | ||||
|             activeLines: ['@settings(defaultLengthUnit = in)'], | ||||
|             highlightedCode: '', | ||||
|             diagnostics: [], | ||||
|           }) | ||||
| @ -299,7 +299,8 @@ test.describe('Point-and-click tests', () => { | ||||
|  | ||||
|       await test.step('verify at the end of the test that final code is what is expected', async () => { | ||||
|         await editor.expectEditor.toContain( | ||||
|           `sketch001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([75.8, 317.2], %) // [$startCapTag, $EndCapTag] | ||||
|   |> angledLine([0, 268.43], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
| @ -369,7 +370,7 @@ profile001 = startProfileAt([205.96, 254.59], sketch002) | ||||
|       }) | ||||
|     }) | ||||
|  | ||||
|     test('Works on chamfers that are non in a pipeExpression can break up multi edges in a chamfer array', async ({ | ||||
|     test('Works on chamfers that are not in a pipeExpression can break up multi edges in a chamfer array', async ({ | ||||
|       context, | ||||
|       page, | ||||
|       homePage, | ||||
| @ -418,7 +419,8 @@ profile001 = startProfileAt([205.96, 254.59], sketch002) | ||||
|         |>close()`, | ||||
|       }) | ||||
|       await editor.expectEditor.toContain( | ||||
|         `sketch001 = startSketchOn(XZ) | ||||
|         `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([75.8, 317.2], %) | ||||
|   |> angledLine([0, 268.43], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
| @ -1082,8 +1084,8 @@ openSketch = startSketchOn(XY) | ||||
|   }) => { | ||||
|     // One dumb hardcoded screen pixel value | ||||
|     const testPoint = { x: 620, y: 257 } | ||||
|     const expectedOutput = `helix001 = helix(  revolutions = 1,  angleStart = 360,  ccw = false,  radius = 5,  axis = 'X',  length = 5,)` | ||||
|     const expectedLine = `revolutions=1,` | ||||
|     const expectedOutput = `helix001 = helix(  axis = 'X',  radius = 5,  length = 5,  revolutions = 1,  angleStart = 360,  ccw = false,)` | ||||
|     const expectedLine = `axis='X',` | ||||
|  | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
| @ -1091,17 +1093,16 @@ openSketch = startSketchOn(XY) | ||||
|       await toolbar.helixButton.click() | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'arguments', | ||||
|         currentArgKey: 'axisOrEdge', | ||||
|         currentArgKey: 'mode', | ||||
|         currentArgValue: '', | ||||
|         headerArguments: { | ||||
|           Mode: '', | ||||
|           AngleStart: '', | ||||
|           AxisOrEdge: '', | ||||
|           CounterClockWise: '', | ||||
|           Length: '', | ||||
|           Radius: '', | ||||
|           Revolutions: '', | ||||
|           Radius: '', | ||||
|           CounterClockWise: '', | ||||
|         }, | ||||
|         highlightedHeaderArg: 'axisOrEdge', | ||||
|         highlightedHeaderArg: 'mode', | ||||
|         commandName: 'Helix', | ||||
|       }) | ||||
|       await cmdBar.progressCmdBar() | ||||
| @ -1110,7 +1111,19 @@ openSketch = startSketchOn(XY) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'review', | ||||
|         headerArguments: { | ||||
|           Mode: 'Axis', | ||||
|           Axis: 'X', | ||||
|           AngleStart: '360', | ||||
|           Revolutions: '1', | ||||
|           Length: '5', | ||||
|           Radius: '5', | ||||
|           CounterClockWise: '', | ||||
|         }, | ||||
|         commandName: 'Helix', | ||||
|       }) | ||||
|       await cmdBar.progressCmdBar() | ||||
|     }) | ||||
|  | ||||
| @ -1134,30 +1147,31 @@ openSketch = startSketchOn(XY) | ||||
|       await cmdBar.expectState({ | ||||
|         commandName: 'Helix', | ||||
|         stage: 'arguments', | ||||
|         currentArgKey: 'length', | ||||
|         currentArgValue: initialInput, | ||||
|         currentArgKey: 'CounterClockWise', | ||||
|         currentArgValue: '', | ||||
|         headerArguments: { | ||||
|           AngleStart: '360', | ||||
|           Axis: 'X', | ||||
|           CounterClockWise: '', | ||||
|           Length: initialInput, | ||||
|           Radius: '5', | ||||
|           AngleStart: '360', | ||||
|           Revolutions: '1', | ||||
|           Radius: '5', | ||||
|           Length: initialInput, | ||||
|           CounterClockWise: '', | ||||
|         }, | ||||
|         highlightedHeaderArg: 'length', | ||||
|         highlightedHeaderArg: 'CounterClockWise', | ||||
|       }) | ||||
|       await page.keyboard.press('Shift+Backspace') | ||||
|       await expect(cmdBar.currentArgumentInput).toBeVisible() | ||||
|       await cmdBar.currentArgumentInput.locator('.cm-content').fill(newInput) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'review', | ||||
|         headerArguments: { | ||||
|           AngleStart: '360', | ||||
|           Axis: 'X', | ||||
|           CounterClockWise: '', | ||||
|           Length: newInput, | ||||
|           Radius: '5', | ||||
|           AngleStart: '360', | ||||
|           Revolutions: '1', | ||||
|           Radius: '5', | ||||
|           Length: newInput, | ||||
|           CounterClockWise: '', | ||||
|         }, | ||||
|         commandName: 'Helix', | ||||
|       }) | ||||
| @ -1181,14 +1195,14 @@ openSketch = startSketchOn(XY) | ||||
|     { | ||||
|       selectionType: 'segment', | ||||
|       testPoint: { x: 513, y: 221 }, | ||||
|       expectedOutput: `helix001 = helix(  revolutions = 20,  angleStart = 0,  ccw = true,  radius = 1,  axis = seg01,  length = 100,)`, | ||||
|       expectedEditedOutput: `helix001 = helix(  revolutions = 20,  angleStart = 0,  ccw = true,  radius = 1,  axis = seg01,  length = 50,)`, | ||||
|       expectedOutput: `helix001 = helix(  axis = seg01,  radius = 1,  revolutions = 20,  angleStart = 0,  ccw = false,)`, | ||||
|       expectedEditedOutput: `helix001 = helix(  axis = seg01,  radius = 5,  revolutions = 20,  angleStart = 0,  ccw = false,)`, | ||||
|     }, | ||||
|     { | ||||
|       selectionType: 'sweepEdge', | ||||
|       testPoint: { x: 564, y: 364 }, | ||||
|       expectedOutput: `helix001 = helix(  revolutions = 20,  angleStart = 0,  ccw = true,  radius = 1,  axis =   getOppositeEdge(seg01),  length = 100,)`, | ||||
|       expectedEditedOutput: `helix001 = helix(  revolutions = 20,  angleStart = 0,  ccw = true,  radius = 1,  axis =   getOppositeEdge(seg01),  length = 50,)`, | ||||
|       expectedOutput: `helix001 = helix(  axis =   getOppositeEdge(seg01),  radius = 1,  revolutions = 20,  angleStart = 0,  ccw = false,)`, | ||||
|       expectedEditedOutput: `helix001 = helix(  axis =   getOppositeEdge(seg01),  radius = 5,  revolutions = 20,  angleStart = 0,  ccw = false,)`, | ||||
|     }, | ||||
|   ] | ||||
|   helixCases.map( | ||||
| @ -1225,17 +1239,16 @@ openSketch = startSketchOn(XY) | ||||
|           await toolbar.helixButton.click() | ||||
|           await cmdBar.expectState({ | ||||
|             stage: 'arguments', | ||||
|             currentArgKey: 'axisOrEdge', | ||||
|             currentArgKey: 'mode', | ||||
|             currentArgValue: '', | ||||
|             headerArguments: { | ||||
|               AngleStart: '', | ||||
|               AxisOrEdge: '', | ||||
|               Mode: '', | ||||
|               CounterClockWise: '', | ||||
|               Length: '', | ||||
|               Radius: '', | ||||
|               Revolutions: '', | ||||
|             }, | ||||
|             highlightedHeaderArg: 'axisOrEdge', | ||||
|             highlightedHeaderArg: 'mode', | ||||
|             commandName: 'Helix', | ||||
|           }) | ||||
|           await cmdBar.selectOption({ name: 'Edge' }).click() | ||||
| @ -1246,21 +1259,17 @@ openSketch = startSketchOn(XY) | ||||
|           await cmdBar.progressCmdBar() | ||||
|           await page.keyboard.insertText('0') | ||||
|           await cmdBar.progressCmdBar() | ||||
|           await cmdBar.selectOption({ name: 'True' }).click() | ||||
|           await page.keyboard.insertText('1') | ||||
|           await cmdBar.progressCmdBar() | ||||
|           await page.keyboard.insertText('100') | ||||
|           await cmdBar.progressCmdBar() | ||||
|           await cmdBar.expectState({ | ||||
|             stage: 'review', | ||||
|             headerArguments: { | ||||
|               AngleStart: '0', | ||||
|               AxisOrEdge: 'Edge', | ||||
|               Mode: 'Edge', | ||||
|               Edge: `1 ${selectionType}`, | ||||
|               CounterClockWise: '', | ||||
|               Length: '100', | ||||
|               Radius: '1', | ||||
|               AngleStart: '0', | ||||
|               Revolutions: '20', | ||||
|               Radius: '1', | ||||
|               CounterClockWise: '', | ||||
|             }, | ||||
|             commandName: 'Helix', | ||||
|           }) | ||||
| @ -1280,22 +1289,24 @@ openSketch = startSketchOn(XY) | ||||
|             0 | ||||
|           ) | ||||
|           await operationButton.dblclick() | ||||
|           const initialInput = '100' | ||||
|           const newInput = '50' | ||||
|           const initialInput = '1' | ||||
|           const newInput = '5' | ||||
|           await cmdBar.expectState({ | ||||
|             commandName: 'Helix', | ||||
|             stage: 'arguments', | ||||
|             currentArgKey: 'length', | ||||
|             currentArgValue: initialInput, | ||||
|             currentArgKey: 'CounterClockWise', | ||||
|             currentArgValue: '', | ||||
|             headerArguments: { | ||||
|               AngleStart: '0', | ||||
|               CounterClockWise: '', | ||||
|               Length: initialInput, | ||||
|               Radius: '1', | ||||
|               Revolutions: '20', | ||||
|               Radius: initialInput, | ||||
|               CounterClockWise: '', | ||||
|             }, | ||||
|             highlightedHeaderArg: 'length', | ||||
|             highlightedHeaderArg: 'CounterClockWise', | ||||
|           }) | ||||
|           await page | ||||
|             .getByRole('button', { name: 'radius', exact: false }) | ||||
|             .click() | ||||
|           await expect(cmdBar.currentArgumentInput).toBeVisible() | ||||
|           await cmdBar.currentArgumentInput | ||||
|             .locator('.cm-content') | ||||
| @ -1305,10 +1316,9 @@ openSketch = startSketchOn(XY) | ||||
|             stage: 'review', | ||||
|             headerArguments: { | ||||
|               AngleStart: '0', | ||||
|               CounterClockWise: '', | ||||
|               Length: newInput, | ||||
|               Radius: '1', | ||||
|               Revolutions: '20', | ||||
|               Radius: newInput, | ||||
|               CounterClockWise: '', | ||||
|             }, | ||||
|             commandName: 'Helix', | ||||
|           }) | ||||
| @ -1336,6 +1346,140 @@ openSketch = startSketchOn(XY) | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test('Helix point-and-click on cylinder', async ({ | ||||
|     context, | ||||
|     page, | ||||
|     homePage, | ||||
|     scene, | ||||
|     editor, | ||||
|     toolbar, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     const initialCode = `sketch001 = startSketchOn(XY) | ||||
| profile001 = circle( | ||||
|   sketch001, | ||||
|   center = [0, 0], | ||||
|   radius = 100, | ||||
|   tag = $seg01, | ||||
| ) | ||||
| extrude001 = extrude(profile001, length = 100) | ||||
|     ` | ||||
|     await context.addInitScript((initialCode) => { | ||||
|       localStorage.setItem('persistCode', initialCode) | ||||
|     }, initialCode) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // One dumb hardcoded screen pixel value | ||||
|     const testPoint = { x: 620, y: 257 } | ||||
|     const [clickOnWall] = scene.makeMouseHelpers(testPoint.x, testPoint.y) | ||||
|     const expectedOutput = `helix001 = helix(  cylinder = extrude001,  revolutions = 1,  angleStart = 360,  ccw = false,)` | ||||
|     const expectedLine = `cylinder = extrude001,` | ||||
|     const expectedEditedOutput = `helix001 = helix(  cylinder = extrude001,  revolutions = 1,  angleStart = 360,  ccw = true,)` | ||||
|  | ||||
|     await test.step(`Go through the command bar flow`, async () => { | ||||
|       await toolbar.helixButton.click() | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'arguments', | ||||
|         currentArgKey: 'mode', | ||||
|         currentArgValue: '', | ||||
|         headerArguments: { | ||||
|           Mode: '', | ||||
|           AngleStart: '', | ||||
|           Revolutions: '', | ||||
|           Radius: '', | ||||
|           CounterClockWise: '', | ||||
|         }, | ||||
|         highlightedHeaderArg: 'mode', | ||||
|         commandName: 'Helix', | ||||
|       }) | ||||
|       await cmdBar.selectOption({ name: 'Cylinder' }).click() | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'arguments', | ||||
|         currentArgKey: 'cylinder', | ||||
|         currentArgValue: '', | ||||
|         headerArguments: { | ||||
|           Mode: 'Cylinder', | ||||
|           Cylinder: '', | ||||
|           AngleStart: '', | ||||
|           Revolutions: '', | ||||
|           CounterClockWise: '', | ||||
|         }, | ||||
|         highlightedHeaderArg: 'cylinder', | ||||
|         commandName: 'Helix', | ||||
|       }) | ||||
|       await clickOnWall() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'review', | ||||
|         headerArguments: { | ||||
|           Mode: 'Cylinder', | ||||
|           Cylinder: '1 face', | ||||
|           AngleStart: '360', | ||||
|           Revolutions: '1', | ||||
|           CounterClockWise: '', | ||||
|         }, | ||||
|         commandName: 'Helix', | ||||
|       }) | ||||
|       await cmdBar.progressCmdBar() | ||||
|     }) | ||||
|  | ||||
|     await test.step(`Confirm code is added to the editor, scene has changed`, async () => { | ||||
|       await editor.expectEditor.toContain(expectedOutput) | ||||
|       await editor.expectState({ | ||||
|         diagnostics: [], | ||||
|         activeLines: [expectedLine], | ||||
|         highlightedCode: '', | ||||
|       }) | ||||
|     }) | ||||
|  | ||||
|     await test.step(`Edit helix through the feature tree`, async () => { | ||||
|       await editor.closePane() | ||||
|       const operationButton = await toolbar.getFeatureTreeOperation('Helix', 0) | ||||
|       await operationButton.dblclick() | ||||
|       await cmdBar.expectState({ | ||||
|         commandName: 'Helix', | ||||
|         stage: 'arguments', | ||||
|         currentArgKey: 'CounterClockWise', | ||||
|         currentArgValue: '', | ||||
|         headerArguments: { | ||||
|           AngleStart: '360', | ||||
|           Revolutions: '1', | ||||
|           CounterClockWise: '', | ||||
|         }, | ||||
|         highlightedHeaderArg: 'CounterClockWise', | ||||
|       }) | ||||
|       await cmdBar.selectOption({ name: 'True' }).click() | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'review', | ||||
|         headerArguments: { | ||||
|           AngleStart: '360', | ||||
|           Revolutions: '1', | ||||
|           CounterClockWise: 'true', | ||||
|         }, | ||||
|         commandName: 'Helix', | ||||
|       }) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await toolbar.closePane('feature-tree') | ||||
|       await toolbar.openPane('code') | ||||
|       await editor.expectEditor.toContain(expectedEditedOutput) | ||||
|       await editor.closePane() | ||||
|     }) | ||||
|  | ||||
|     await test.step('Delete helix via feature tree selection', async () => { | ||||
|       await toolbar.openPane('feature-tree') | ||||
|       const operationButton = await toolbar.getFeatureTreeOperation('Helix', 0) | ||||
|       await operationButton.click({ button: 'left' }) | ||||
|       await page.keyboard.press('Delete') | ||||
|       await toolbar.closePane('feature-tree') | ||||
|       await toolbar.openPane('code') | ||||
|       await editor.expectEditor.not.toContain(expectedEditedOutput) | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   const loftPointAndClickCases = [ | ||||
|     { shouldPreselect: true }, | ||||
|     { shouldPreselect: false }, | ||||
| @ -1497,9 +1641,10 @@ loft001 = loft([sketch001, sketch002]) | ||||
|     { | ||||
|       targetType: 'circle', | ||||
|       testPoint: { x: 700, y: 250 }, | ||||
|       initialCode: `sketch001 = startSketchOn('YZ') | ||||
|       initialCode: `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(YZ) | ||||
| profile001 = circle(sketch001, center = [0, 0], radius = 500) | ||||
| sketch002 = startSketchOn('XZ') | ||||
| sketch002 = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> xLine(length = -500) | ||||
|   |> tangentialArcTo([-2000, 500], %)`, | ||||
| @ -1507,7 +1652,8 @@ sketch002 = startSketchOn('XZ') | ||||
|     { | ||||
|       targetType: 'rectangle', | ||||
|       testPoint: { x: 710, y: 255 }, | ||||
|       initialCode: `sketch001 = startSketchOn('YZ') | ||||
|       initialCode: `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(YZ) | ||||
| profile001 = startProfileAt([-400, -400], sketch001) | ||||
|   |> angledLine([0, 800], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
| @ -1520,7 +1666,7 @@ profile001 = startProfileAt([-400, -400], sketch001) | ||||
|      ], %) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| sketch002 = startSketchOn('XZ') | ||||
| sketch002 = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> xLine(length = -500) | ||||
|   |> tangentialArcTo([-2000, 500], %)`, | ||||
| @ -1664,7 +1810,8 @@ sketch002 = startSketchOn('XZ') | ||||
|     toolbar, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     const initialCode = `sketch001 = startSketchOn(YZ) | ||||
|     const initialCode = `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(YZ) | ||||
|   |> circle( | ||||
|        center = [0, 0], | ||||
|        radius = 500 | ||||
| @ -2333,7 +2480,8 @@ extrude001 = extrude(profile001, length = 5) | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     // Code samples | ||||
|     const initialCode = `sketch001 = startSketchOn(XY) | ||||
|     const initialCode = `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XY) | ||||
|   |> startProfileAt([-12, -6], %) | ||||
|   |> line(end = [0, 12]) | ||||
|   |> line(end = [24, 0]) | ||||
| @ -2625,7 +2773,8 @@ extrude001 = extrude(sketch001, length = -12) | ||||
|     toolbar, | ||||
|   }) => { | ||||
|     // Code samples | ||||
|     const initialCode = `sketch001 = startSketchOn(XY) | ||||
|     const initialCode = `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XY) | ||||
|   |> startProfileAt([-12, -6], %) | ||||
|   |> line(end = [0, 12]) | ||||
|   |> line(end = [24, 0], tag = $seg02) | ||||
| @ -2779,7 +2928,8 @@ chamfer04 = chamfer(extrude001, length = 5, tags = [getOppositeEdge(seg02)]) | ||||
|       toolbar, | ||||
|       cmdBar, | ||||
|     }) => { | ||||
|       const initialCode = `sketch001 = startSketchOn(XZ) | ||||
|       const initialCode = `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> circle(center = [0, 0], radius = 30) | ||||
| extrude001 = extrude(sketch001, length = 30) | ||||
|     ` | ||||
| @ -2914,7 +3064,8 @@ extrude001 = extrude(sketch001, length = 30) | ||||
|     toolbar, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     const initialCode = `sketch001 = startSketchOn(XY) | ||||
|     const initialCode = `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XY) | ||||
|   |> startProfileAt([-20, 20], %) | ||||
|   |> xLine(length = 40) | ||||
|   |> yLine(length = -60) | ||||
| @ -3032,7 +3183,8 @@ extrude001 = extrude(sketch001, length = 40) | ||||
|   }) | ||||
|  | ||||
|   const shellSketchOnFacesCases = [ | ||||
|     `sketch001 = startSketchOn(XZ) | ||||
|     `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> circle(center = [0, 0], radius = 100) | ||||
|   |> extrude(length = 100) | ||||
|  | ||||
| @ -3040,7 +3192,8 @@ sketch002 = startSketchOn(sketch001, 'END') | ||||
|   |> circle(center = [0, 0], radius = 50) | ||||
|   |> extrude(length = 50) | ||||
|   `, | ||||
|     `sketch001 = startSketchOn(XZ) | ||||
|     `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> circle(center = [0, 0], radius = 100) | ||||
| extrude001 = extrude(sketch001, length = 100) | ||||
|  | ||||
| @ -3323,6 +3476,39 @@ segAng(rectangleSegmentA002), | ||||
|  | ||||
|       const newCodeToFind = `revolve001 = revolve(sketch002, angle = 360, axis = 'X')` | ||||
|       expect(editor.expectEditor.toContain(newCodeToFind)).toBeTruthy() | ||||
|  | ||||
|       // Edit flow | ||||
|       const newAngle = '90' | ||||
|       await toolbar.openPane('feature-tree') | ||||
|       const operationButton = await toolbar.getFeatureTreeOperation( | ||||
|         'Revolve', | ||||
|         0 | ||||
|       ) | ||||
|       await operationButton.dblclick({ button: 'left' }) | ||||
|       await cmdBar.expectState({ | ||||
|         commandName: 'Revolve', | ||||
|         currentArgKey: 'angle', | ||||
|         currentArgValue: '360', | ||||
|         headerArguments: { | ||||
|           Angle: '360', | ||||
|         }, | ||||
|         highlightedHeaderArg: 'angle', | ||||
|         stage: 'arguments', | ||||
|       }) | ||||
|       await page.keyboard.insertText(newAngle) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'review', | ||||
|         headerArguments: { | ||||
|           Angle: newAngle, | ||||
|         }, | ||||
|         commandName: 'Revolve', | ||||
|       }) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await toolbar.closePane('feature-tree') | ||||
|       await editor.expectEditor.toContain( | ||||
|         newCodeToFind.replace('angle = 360', 'angle = ' + newAngle) | ||||
|       ) | ||||
|     }) | ||||
|     test('revolve surface around edge from an extruded solid2d', async ({ | ||||
|       context, | ||||
| @ -3333,26 +3519,22 @@ segAng(rectangleSegmentA002), | ||||
|       toolbar, | ||||
|       cmdBar, | ||||
|     }) => { | ||||
|       const initialCode = ` | ||||
| sketch001 = startSketchOn(XZ) | ||||
| |> startProfileAt([-102.57, 101.72], %) | ||||
| |> angledLine([0, 202.6], %, $rectangleSegmentA001) | ||||
| |> angledLine([ | ||||
| segAng(rectangleSegmentA001) - 90, | ||||
| 202.6 | ||||
| ], %, $rectangleSegmentB001) | ||||
| |> angledLine([ | ||||
| segAng(rectangleSegmentA001), | ||||
| -segLen(rectangleSegmentA001) | ||||
| ], %, $rectangleSegmentC001) | ||||
| |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
| |> close() | ||||
|       const initialCode = `sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([-102.57, 101.72], %) | ||||
|   |> angledLine([0, 202.6], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        202.6 | ||||
|      ], %, $rectangleSegmentB001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %, $rectangleSegmentC001) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| extrude001 = extrude(sketch001, length = 50) | ||||
| sketch002 = startSketchOn(extrude001, rectangleSegmentA001) | ||||
| |> circle( | ||||
| center = [-11.34, 10.0], | ||||
| radius = 8.69 | ||||
| ) | ||||
|   |> circle(center = [-11.34, 10.0], radius = 8.69) | ||||
| ` | ||||
|       await context.addInitScript((initialCode) => { | ||||
|         localStorage.setItem('persistCode', initialCode) | ||||
| @ -3370,9 +3552,49 @@ radius = 8.69 | ||||
|       const lineCodeToSelection = `|> angledLine([0, 202.6], %, $rectangleSegmentA001)` | ||||
|       await page.getByText(lineCodeToSelection).click() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
|  | ||||
|       const newCodeToFind = `revolve001 = revolve(sketch002, angle = 360, axis = getOppositeEdge(rectangleSegmentA001)) ` | ||||
|       expect(editor.expectEditor.toContain(newCodeToFind)).toBeTruthy() | ||||
|       const newCodeToFind = `revolve001 = revolve(sketch002, angle = 360, axis = rectangleSegmentA001)` | ||||
|       await editor.expectEditor.toContain(newCodeToFind) | ||||
|  | ||||
|       // Edit flow | ||||
|       const newAngle = '180' | ||||
|       await toolbar.openPane('feature-tree') | ||||
|       const operationButton = await toolbar.getFeatureTreeOperation( | ||||
|         'Revolve', | ||||
|         0 | ||||
|       ) | ||||
|       await operationButton.dblclick({ button: 'left' }) | ||||
|       await cmdBar.expectState({ | ||||
|         commandName: 'Revolve', | ||||
|         currentArgKey: 'angle', | ||||
|         currentArgValue: '360', | ||||
|         headerArguments: { | ||||
|           Angle: '360', | ||||
|         }, | ||||
|         highlightedHeaderArg: 'angle', | ||||
|         stage: 'arguments', | ||||
|       }) | ||||
|       await page.keyboard.insertText(newAngle) | ||||
|       await page.getByRole('button', { name: 'Create new variable' }).click() | ||||
|       await expect(page.getByPlaceholder('Variable name')).toHaveValue( | ||||
|         'angle001' | ||||
|       ) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'review', | ||||
|         headerArguments: { | ||||
|           Angle: newAngle, | ||||
|         }, | ||||
|         commandName: 'Revolve', | ||||
|       }) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await toolbar.closePane('feature-tree') | ||||
|       await editor.expectEditor.toContain('angle001 = ' + newAngle) | ||||
|       await editor.expectEditor.toContain( | ||||
|         newCodeToFind.replace('angle = 360', 'angle = angle001') | ||||
|       ) | ||||
|     }) | ||||
|     test('revolve sketch circle around line segment from startProfileAt sketch', async ({ | ||||
|       context, | ||||
| @ -3383,26 +3605,22 @@ radius = 8.69 | ||||
|       toolbar, | ||||
|       cmdBar, | ||||
|     }) => { | ||||
|       const initialCode = ` | ||||
|     sketch002 = startSketchOn(XY) | ||||
|       |> startProfileAt([-2.02, 1.79], %) | ||||
|       |> xLine(length = 2.6) | ||||
|     sketch001 = startSketchOn('-XY') | ||||
|       |> startProfileAt([-0.48, 1.25], %) | ||||
|       |> angledLine([0, 2.38], %, $rectangleSegmentA001) | ||||
|       |> angledLine([segAng(rectangleSegmentA001) - 90, 2.4], %, $rectangleSegmentB001) | ||||
|       |> angledLine([ | ||||
|         segAng(rectangleSegmentA001), | ||||
|           -segLen(rectangleSegmentA001) | ||||
|       ], %, $rectangleSegmentC001) | ||||
|       |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|       |> close() | ||||
|     extrude001 = extrude(sketch001, length = 5) | ||||
|     sketch003 = startSketchOn(extrude001, 'START') | ||||
|       |> circle( | ||||
|         center = [-0.69, 0.56], | ||||
|         radius = 0.28 | ||||
|       ) | ||||
|       const initialCode = `sketch002 = startSketchOn(XY) | ||||
|   |> startProfileAt([-2.02, 1.79], %) | ||||
|   |> xLine(length = 2.6) | ||||
| sketch001 = startSketchOn(-XY) | ||||
|   |> startProfileAt([-0.48, 1.25], %) | ||||
|   |> angledLine([0, 2.38], %, $rectangleSegmentA001) | ||||
|   |> angledLine([segAng(rectangleSegmentA001) - 90, 2.4], %, $rectangleSegmentB001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %, $rectangleSegmentC001) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| extrude001 = extrude(sketch001, length = 5) | ||||
| sketch003 = startSketchOn(extrude001, 'START') | ||||
|   |> circle(center = [-0.69, 0.56], radius = 0.28) | ||||
| ` | ||||
|  | ||||
|       await context.addInitScript((initialCode) => { | ||||
| @ -3421,9 +3639,44 @@ radius = 8.69 | ||||
|       const lineCodeToSelection = `|> xLine(length = 2.6)` | ||||
|       await page.getByText(lineCodeToSelection).click() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
|  | ||||
|       const newCodeToFind = `revolve001 = revolve(sketch003, angle = 360, axis = seg01)` | ||||
|       expect(editor.expectEditor.toContain(newCodeToFind)).toBeTruthy() | ||||
|  | ||||
|       // Edit flow | ||||
|       const newAngle = '270' | ||||
|       await toolbar.openPane('feature-tree') | ||||
|       const operationButton = await toolbar.getFeatureTreeOperation( | ||||
|         'Revolve', | ||||
|         0 | ||||
|       ) | ||||
|       await operationButton.dblclick({ button: 'left' }) | ||||
|       await cmdBar.expectState({ | ||||
|         commandName: 'Revolve', | ||||
|         currentArgKey: 'angle', | ||||
|         currentArgValue: '360', | ||||
|         headerArguments: { | ||||
|           Angle: '360', | ||||
|         }, | ||||
|         highlightedHeaderArg: 'angle', | ||||
|         stage: 'arguments', | ||||
|       }) | ||||
|       await page.keyboard.insertText(newAngle) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'review', | ||||
|         headerArguments: { | ||||
|           Angle: newAngle, | ||||
|         }, | ||||
|         commandName: 'Revolve', | ||||
|       }) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await toolbar.closePane('feature-tree') | ||||
|       await editor.expectEditor.toContain( | ||||
|         newCodeToFind.replace('angle = 360', 'angle = ' + newAngle) | ||||
|       ) | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
| @ -3436,7 +3689,8 @@ radius = 8.69 | ||||
|     toolbar, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     const initialCode = `sketch001 = startSketchOn(XZ) | ||||
|     const initialCode = `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
| profile001 = circle( | ||||
|   sketch001, | ||||
|   center = [0, 0], | ||||
|  | ||||
| @ -1,19 +1,20 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { DEFAULT_PROJECT_KCL_FILE } from '@src/lib/constants' | ||||
| import fs from 'fs' | ||||
| import fsp from 'fs/promises' | ||||
| import path from 'path' | ||||
|  | ||||
| import type { Paths } from '@e2e/playwright/test-utils' | ||||
| import { | ||||
|   createProject, | ||||
|   doExport, | ||||
|   executorInputPath, | ||||
|   getPlaywrightDownloadDir, | ||||
|   getUtils, | ||||
|   isOutOfViewInScrollContainer, | ||||
|   Paths, | ||||
|   createProject, | ||||
|   getPlaywrightDownloadDir, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
|   runningOnWindows, | ||||
| } from './test-utils' | ||||
| import fsp from 'fs/promises' | ||||
| import fs from 'fs' | ||||
| import path from 'path' | ||||
| import { DEFAULT_PROJECT_KCL_FILE } from 'lib/constants' | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test( | ||||
|   'projects reload if a new one is created, deleted, or renamed externally', | ||||
| @ -87,7 +88,7 @@ test( | ||||
|       const bracketDir = path.join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|       await fsp.copyFile( | ||||
|         executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|         executorInputPath('cylinder-inches.kcl'), | ||||
|         path.join(bracketDir, 'main.kcl') | ||||
|       ) | ||||
|     }) | ||||
| @ -124,7 +125,7 @@ test( | ||||
|       const bracketDir = path.join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|       await fsp.copyFile( | ||||
|         executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|         executorInputPath('cylinder-inches.kcl'), | ||||
|         path.join(bracketDir, 'main.kcl') | ||||
|       ) | ||||
|       const errorDir = path.join(dir, 'broken-code') | ||||
| @ -162,7 +163,7 @@ test( | ||||
|       // gray at this pixel means the stream has loaded in the most | ||||
|       // user way we can verify it (pixel color) | ||||
|       await expect | ||||
|         .poll(() => u.getGreatestPixDiff(pointOnModel, [85, 85, 85]), { | ||||
|         .poll(() => u.getGreatestPixDiff(pointOnModel, [110, 110, 110]), { | ||||
|           timeout: 10_000, | ||||
|         }) | ||||
|         .toBeLessThan(20) | ||||
| @ -213,7 +214,7 @@ test( | ||||
|       const bracketDir = path.join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|       await fsp.copyFile( | ||||
|         executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|         executorInputPath('cylinder-inches.kcl'), | ||||
|         path.join(bracketDir, 'main.kcl') | ||||
|       ) | ||||
|       const emptyDir = path.join(dir, 'empty') | ||||
| @ -248,7 +249,7 @@ test( | ||||
|       // gray at this pixel means the stream has loaded in the most | ||||
|       // user way we can verify it (pixel color) | ||||
|       await expect | ||||
|         .poll(() => u.getGreatestPixDiff(pointOnModel, [85, 85, 85]), { | ||||
|         .poll(() => u.getGreatestPixDiff(pointOnModel, [125, 125, 125]), { | ||||
|           timeout: 10_000, | ||||
|         }) | ||||
|         .toBeLessThan(15) | ||||
| @ -290,7 +291,7 @@ test( | ||||
|       const bracketDir = path.join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|       await fsp.copyFile( | ||||
|         executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|         executorInputPath('cylinder-inches.kcl'), | ||||
|         path.join(bracketDir, 'main.kcl') | ||||
|       ) | ||||
|  | ||||
| @ -319,7 +320,7 @@ test( | ||||
|       // gray at this pixel means the stream has loaded in the most | ||||
|       // user way we can verify it (pixel color) | ||||
|       await expect | ||||
|         .poll(() => u.getGreatestPixDiff(pointOnModel, [85, 85, 85]), { | ||||
|         .poll(() => u.getGreatestPixDiff(pointOnModel, [125, 125, 125]), { | ||||
|           timeout: 10_000, | ||||
|         }) | ||||
|         .toBeLessThan(15) | ||||
| @ -359,7 +360,7 @@ test( | ||||
|       const bracketDir = path.join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|       await fsp.copyFile( | ||||
|         executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|         executorInputPath('cylinder-inches.kcl'), | ||||
|         path.join(bracketDir, 'main.kcl') | ||||
|       ) | ||||
|       await fsp.copyFile( | ||||
| @ -393,7 +394,7 @@ test( | ||||
|       // gray at this pixel means the stream has loaded in the most | ||||
|       // user way we can verify it (pixel color) | ||||
|       await expect | ||||
|         .poll(() => u.getGreatestPixDiff(pointOnModel, [85, 85, 85]), { | ||||
|         .poll(() => u.getGreatestPixDiff(pointOnModel, [125, 125, 125]), { | ||||
|           timeout: 10_000, | ||||
|         }) | ||||
|         .toBeLessThan(15) | ||||
| @ -443,7 +444,6 @@ test( | ||||
|     await page.getByText('broken-code').click() | ||||
|  | ||||
|     // Gotcha: You can not use scene.waitForExecutionDone() since the KCL code is going to fail | ||||
|     await expect(page.getByTestId('loading')).toBeAttached() | ||||
|     await expect(page.getByTestId('loading')).not.toBeAttached({ | ||||
|       timeout: 20_000, | ||||
|     }) | ||||
| @ -473,12 +473,15 @@ test.describe('Can export from electron app', () => { | ||||
|         if (!tronApp) { | ||||
|           fail() | ||||
|         } | ||||
|         if (runningOnWindows()) { | ||||
|           test.fixme(orRunWhenFullSuiteEnabled()) | ||||
|         } | ||||
|  | ||||
|         await context.folderSetupFn(async (dir) => { | ||||
|           const bracketDir = path.join(dir, 'bracket') | ||||
|           await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|           await fsp.copyFile( | ||||
|             executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|             executorInputPath('cylinder-inches.kcl'), | ||||
|             path.join(bracketDir, 'main.kcl') | ||||
|           ) | ||||
|         }) | ||||
| @ -510,7 +513,7 @@ test.describe('Can export from electron app', () => { | ||||
|           // gray at this pixel means the stream has loaded in the most | ||||
|           // user way we can verify it (pixel color) | ||||
|           await expect | ||||
|             .poll(() => u.getGreatestPixDiff(pointOnModel, [85, 85, 85]), { | ||||
|             .poll(() => u.getGreatestPixDiff(pointOnModel, [125, 125, 125]), { | ||||
|               timeout: 10_000, | ||||
|             }) | ||||
|             .toBeLessThan(15) | ||||
| @ -551,7 +554,7 @@ test.describe('Can export from electron app', () => { | ||||
|               }, | ||||
|               { timeout: 15_000 } | ||||
|             ) | ||||
|             .toBeGreaterThan(300_000) | ||||
|             .toBeGreaterThan(50_000) | ||||
|  | ||||
|           // clean up exported file | ||||
|           await fsp.rm(filepath) | ||||
| @ -1504,7 +1507,12 @@ test( | ||||
|  | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     await page.locator('.cm-content').fill(`sketch001 = startSketchOn(XZ) | ||||
|     // The file should be prepopulated with the user's unit settings. | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       '@settings(defaultLengthUnit = in)' | ||||
|     ) | ||||
|  | ||||
|     await page.locator('.cm-content').fill(`sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-87.4, 282.92], %) | ||||
|   |> line(end = [324.07, 27.199], tag = $seg01) | ||||
|   |> line(end = [118.328, -291.754]) | ||||
|  | ||||
| @ -1,4 +1,5 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| /* eslint-disable jest/no-conditional-expect */ | ||||
|  | ||||
| /** | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { orRunWhenFullSuiteEnabled } from './test-utils' | ||||
| import { orRunWhenFullSuiteEnabled } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| /* eslint-disable jest/no-conditional-expect */ | ||||
|  | ||||
|  | ||||
| @ -1,17 +1,18 @@ | ||||
| import { Page } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
| import path from 'path' | ||||
| import type { Page } from '@playwright/test' | ||||
| import { bracket } from '@src/lib/exampleKcl' | ||||
| import { reportRejection } from '@src/lib/trap' | ||||
| import * as fsp from 'fs/promises' | ||||
| import path from 'path' | ||||
|  | ||||
| import { TEST_CODE_TRIGGER_ENGINE_EXPORT_ERROR } from '@e2e/playwright/storageStates' | ||||
| import type { TestColor } from '@e2e/playwright/test-utils' | ||||
| import { | ||||
|   getUtils, | ||||
|   executorInputPath, | ||||
|   TEST_COLORS, | ||||
|   TestColor, | ||||
|   executorInputPath, | ||||
|   getUtils, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
| } from './test-utils' | ||||
| import { TEST_CODE_TRIGGER_ENGINE_EXPORT_ERROR } from './storageStates' | ||||
| import { bracket } from 'lib/exampleKcl' | ||||
| import { reportRejection } from 'lib/trap' | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Regression tests', { tag: ['@skipWin'] }, () => { | ||||
|   // bugs we found that don't fit neatly into other categories | ||||
| @ -331,7 +332,7 @@ extrude001 = extrude(sketch001, length = 50) | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `@settings(defaultLengthUnit = mm) | ||||
| sketch002 = startSketchOn('XY') | ||||
| sketch002 = startSketchOn(XY) | ||||
| profile002 = startProfileAt([72.24, -52.05], sketch002) | ||||
|   |> angledLine([0, 181.26], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
| @ -582,7 +583,7 @@ extrude002 = extrude(profile002, length = 150) | ||||
|         const bracketDir = path.join(dir, 'bracket') | ||||
|         await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|           executorInputPath('cylinder-inches.kcl'), | ||||
|           path.join(bracketDir, 'main.kcl') | ||||
|         ) | ||||
|       }) | ||||
| @ -619,6 +620,7 @@ extrude002 = extrude(profile002, length = 150) | ||||
|   test(`View gizmo stays visible even when zoomed out all the way`, async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|     scene, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
| @ -632,7 +634,7 @@ extrude002 = extrude(profile002, length = 150) | ||||
|  | ||||
|     await test.step(`Load an empty file`, async () => { | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem('persistCode', '') | ||||
|         localStorage.setItem('persistCode', '@settings(defaultLengthUnit = in)') | ||||
|       }) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
| @ -646,22 +648,31 @@ extrude002 = extrude(profile002, length = 150) | ||||
|           timeout: 5000, | ||||
|           message: 'Plane color is visible', | ||||
|         }) | ||||
|         .toBeLessThanOrEqual(15) | ||||
|         .toBeLessThanOrEqual(20) | ||||
|       await expect(scene.startEditSketchBtn).toBeEnabled() | ||||
|  | ||||
|       let maxZoomOuts = 10 | ||||
|       let middlePixelIsBackgroundColor = | ||||
|         (await middlePixelIsColor(bgColor)) < 10 | ||||
|  | ||||
|       console.time('pressing control') | ||||
|       await page.keyboard.down('Control') | ||||
|  | ||||
|       while (!middlePixelIsBackgroundColor && maxZoomOuts > 0) { | ||||
|         await page.keyboard.down('Control') | ||||
|         await page.mouse.move(600, 460) | ||||
|         await page.mouse.down({ button: 'right' }) | ||||
|         await page.mouse.move(600, 50, { steps: 20 }) | ||||
|         await page.mouse.up({ button: 'right' }) | ||||
|         await page.keyboard.up('Control') | ||||
|         await page.waitForTimeout(100) | ||||
|         await page.mouse.move(650, 460) | ||||
|         console.time('moved to start point') | ||||
|         await page.mouse.down({ button: 'right' }) | ||||
|         console.time('moused down') | ||||
|         await page.mouse.move(650, 50, { steps: 20 }) | ||||
|         console.time('moved to end point') | ||||
|         await page.waitForTimeout(100) | ||||
|         await page.mouse.up({ button: 'right' }) | ||||
|         console.time('moused up') | ||||
|         maxZoomOuts-- | ||||
|         middlePixelIsBackgroundColor = (await middlePixelIsColor(bgColor)) < 10 | ||||
|         middlePixelIsBackgroundColor = (await middlePixelIsColor(bgColor)) < 15 | ||||
|       } | ||||
|       await page.keyboard.up('Control') | ||||
|  | ||||
|       expect(middlePixelIsBackgroundColor, { | ||||
|         message: 'We should not see the default planes', | ||||
| @ -678,13 +689,12 @@ extrude002 = extrude(profile002, length = 150) | ||||
|     homePage, | ||||
|     scene, | ||||
|     toolbar, | ||||
|     viewport, | ||||
|   }) => { | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       const legoDir = path.join(dir, 'lego') | ||||
|       await fsp.mkdir(legoDir, { recursive: true }) | ||||
|       await fsp.copyFile( | ||||
|         executorInputPath('lego.kcl'), | ||||
|         executorInputPath('e2e-can-sketch-on-chamfer.kcl'), | ||||
|         path.join(legoDir, 'main.kcl') | ||||
|       ) | ||||
|     }) | ||||
| @ -697,11 +707,8 @@ extrude002 = extrude(profile002, length = 150) | ||||
|       await scene.loadingIndicator.waitFor({ state: 'detached' }) | ||||
|     }) | ||||
|     await test.step(`The part should start loading quickly, not waiting until execution is complete`, async () => { | ||||
|       await scene.expectPixelColor( | ||||
|         [143, 143, 143], | ||||
|         { x: (viewport?.width ?? 1200) / 2, y: (viewport?.height ?? 500) / 2 }, | ||||
|         15 | ||||
|       ) | ||||
|       // TODO: use the viewport size to pick the center point, but the `viewport` fixture's values were wrong. | ||||
|       await scene.expectPixelColor([116, 116, 116], { x: 500, y: 250 }, 15) | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
| @ -778,6 +785,87 @@ plane002 = offsetPlane(XZ, offset = -2 * x)` | ||||
|       await editor.expectEditor.not.toContain(`plane002`) | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   test.fail( | ||||
|     'Console errors cause tests to fail', | ||||
|     async ({ page, homePage }) => { | ||||
|       const u = await getUtils(page) | ||||
|       await homePage.goToModelingScene() | ||||
|       await u.openAndClearDebugPanel() | ||||
|  | ||||
|       await page.getByTestId('custom-cmd-input').fill('foobar') | ||||
|       await page.getByTestId('custom-cmd-send-button').scrollIntoViewIfNeeded() | ||||
|       await page.getByTestId('custom-cmd-send-button').click() | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test('scale other than default works with sketch mode', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|     toolbar, | ||||
|     editor, | ||||
|     scene, | ||||
|   }) => { | ||||
|     await test.step('Load the washer code', async () => { | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `@settings(defaultLengthUnit = in) | ||||
|  | ||||
| innerDiameter = 0.203 | ||||
| outerDiameter = 0.438 | ||||
| thicknessMax = 0.038 | ||||
| thicknessMin = 0.024 | ||||
| washerSketch = startSketchOn(XY) | ||||
|   |> circle(center = [0, 0], radius = outerDiameter / 2) | ||||
|  | ||||
| washer = extrude(washerSketch, length = thicknessMax)` | ||||
|         ) | ||||
|       }) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|     }) | ||||
|     const [circleCenterClick] = scene.makeMouseHelpers(650, 300) | ||||
|     const [circleRadiusClick] = scene.makeMouseHelpers(800, 320) | ||||
|     const [washerFaceClick] = scene.makeMouseHelpers(657, 286) | ||||
|  | ||||
|     await page.waitForTimeout(100) | ||||
|     await test.step('Start sketching on the washer face', async () => { | ||||
|       await toolbar.startSketchPlaneSelection() | ||||
|       await washerFaceClick() | ||||
|       await page.waitForTimeout(600) // engine animation | ||||
|       await toolbar.expectToolbarMode.toBe('sketching') | ||||
|     }) | ||||
|  | ||||
|     await test.step('Draw a circle and verify code', async () => { | ||||
|       // select circle tool | ||||
|       await expect | ||||
|         .poll(async () => { | ||||
|           await toolbar.circleBtn.click() | ||||
|           return toolbar.circleBtn.getAttribute('aria-pressed') | ||||
|         }) | ||||
|         .toBe('true') | ||||
|       await page.waitForTimeout(100) | ||||
|       await circleCenterClick() | ||||
|       // this number will be different if the scale is not set correctly for inches | ||||
|       await editor.expectEditor.toContain( | ||||
|         'circle(sketch001, center = [0.06, -0.06]' | ||||
|       ) | ||||
|       await circleRadiusClick() | ||||
|  | ||||
|       await editor.expectEditor.toContain( | ||||
|         'circle(sketch001, center = [0.06, -0.06], radius = 0.18' | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await test.step('Exit sketch mode', async () => { | ||||
|       await toolbar.exitSketch() | ||||
|       await toolbar.expectToolbarMode.toBe('modeling') | ||||
|  | ||||
|       await toolbar.selectUnit('Yards') | ||||
|       await editor.expectEditor.toContain('@settings(defaultLengthUnit = yd)') | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| async function clickExportButton(page: Page) { | ||||
|  | ||||
| @ -1,20 +1,20 @@ | ||||
| import { Page } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
| import type { Page } from '@playwright/test' | ||||
| import { roundOff, uuidv4 } from '@src/lib/utils' | ||||
| import fs from 'node:fs/promises' | ||||
| import path from 'node:path' | ||||
| import { HomePageFixture } from './fixtures/homePageFixture' | ||||
|  | ||||
| import type { CmdBarFixture } from '@e2e/playwright/fixtures/cmdBarFixture' | ||||
| import type { HomePageFixture } from '@e2e/playwright/fixtures/homePageFixture' | ||||
| import type { SceneFixture } from '@e2e/playwright/fixtures/sceneFixture' | ||||
| import type { ToolbarFixture } from '@e2e/playwright/fixtures/toolbarFixture' | ||||
| import { | ||||
|   getMovementUtils, | ||||
|   getUtils, | ||||
|   PERSIST_MODELING_CONTEXT, | ||||
|   TEST_COLORS, | ||||
|   getMovementUtils, | ||||
|   getUtils, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
| } from './test-utils' | ||||
| import { uuidv4, roundOff } from 'lib/utils' | ||||
| import { SceneFixture } from './fixtures/sceneFixture' | ||||
| import { ToolbarFixture } from './fixtures/toolbarFixture' | ||||
| import { CmdBarFixture } from './fixtures/cmdBarFixture' | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Sketch tests', { tag: ['@skipWin'] }, () => { | ||||
|   test('multi-sketch file shows multiple Edit Sketch buttons', async ({ | ||||
| @ -113,7 +113,8 @@ test.describe('Sketch tests', { tag: ['@skipWin'] }, () => { | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn(XZ) | ||||
|         `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([2.61, -4.01], %) | ||||
|   |> xLine(length = 8.73) | ||||
|   |> tangentialArcTo([8.33, -1.31], %)` | ||||
| @ -159,7 +160,10 @@ test.describe('Sketch tests', { tag: ['@skipWin'] }, () => { | ||||
|       await page.mouse.click(700, 200) | ||||
|  | ||||
|       await expect.poll(u.normalisedEditorCode, { timeout: 1000 }) | ||||
|         .toBe(`sketch002 = startSketchOn(XZ) | ||||
|         .toBe(`@settings(defaultLengthUnit = in) | ||||
|  | ||||
|  | ||||
| sketch002 = startSketchOn(XZ) | ||||
| sketch001 = startProfileAt([12.34, -12.34], sketch002) | ||||
|   |> yLine(length = 12.34) | ||||
|  | ||||
| @ -475,7 +479,8 @@ sketch001 = startProfileAt([12.34, -12.34], sketch002) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn(XZ) | ||||
|         `@settings(defaultLengthUnit=in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|     |> circle(center = [4.61, -5.01], radius = 8)` | ||||
|       ) | ||||
|     }) | ||||
| @ -560,12 +565,14 @@ sketch001 = startProfileAt([12.34, -12.34], sketch002) | ||||
|   test('Can edit a sketch that has been extruded in the same pipe', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|     editor, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn(XZ) | ||||
|         `@settings(defaultLengthUnit=in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([4.61, -10.01], %) | ||||
|   |> line(end = [12.73, -0.09]) | ||||
|   |> tangentialArcTo([24.95, -0.38], %) | ||||
| @ -650,26 +657,29 @@ sketch001 = startProfileAt([12.34, -12.34], sketch002) | ||||
|     await expect(page.locator('.cm-content')).not.toHaveText(prevContent) | ||||
|  | ||||
|     // expect the code to have changed | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn(XZ) | ||||
|     await editor.expectEditor.toContain( | ||||
|       `sketch001 = startSketchOn(XZ) | ||||
|     |> startProfileAt([7.12, -12.68], %) | ||||
|     |> line(end = [12.68, -1.09]) | ||||
|     |> tangentialArcTo([24.89, 0.68], %) | ||||
|     |> close() | ||||
|     |> extrude(length = 5) | ||||
|   `) | ||||
|     |> extrude(length = 5)`, | ||||
|       { shouldNormalise: true } | ||||
|     ) | ||||
|   }) | ||||
|  | ||||
|   test('Can edit a sketch that has been revolved in the same pipe', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|     scene, | ||||
|     editor, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn(XZ) | ||||
|         `@settings(defaultLengthUnit=in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([4.61, -14.01], %) | ||||
|   |> line(end = [12.73, -0.09]) | ||||
|   |> tangentialArcTo([24.95, -5.38], %) | ||||
| @ -754,14 +764,16 @@ sketch001 = startProfileAt([12.34, -12.34], sketch002) | ||||
|     await expect(page.locator('.cm-content')).not.toHaveText(prevContent) | ||||
|  | ||||
|     // expect the code to have changed | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn(XZ) | ||||
|     await editor.expectEditor.toContain( | ||||
|       `sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([6.44, -12.07], %) | ||||
|   |> line(end = [14.72, 1.97]) | ||||
|   |> tangentialArcTo([24.95, -5.38], %) | ||||
|   |> line(end = [1.97, 2.06]) | ||||
|   |> close() | ||||
|   |> revolve(axis = "X")`) | ||||
|   |> revolve(axis = "X")`, | ||||
|       { shouldNormalise: true } | ||||
|     ) | ||||
|   }) | ||||
|   test('Can add multiple sketches', async ({ page, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
| @ -789,7 +801,8 @@ sketch001 = startProfileAt([12.34, -12.34], sketch002) | ||||
|       200 | ||||
|     ) | ||||
|  | ||||
|     let codeStr = 'sketch001 = startSketchOn(XY)' | ||||
|     let codeStr = | ||||
|       '@settings(defaultLengthUnit = in)sketch001 = startSketchOn(XY)' | ||||
|  | ||||
|     await page.mouse.click(center.x, viewportSize.height * 0.55) | ||||
|     await expect(u.codeLocator).toHaveText(codeStr) | ||||
| @ -868,7 +881,8 @@ sketch001 = startProfileAt([12.34, -12.34], sketch002) | ||||
|  | ||||
|       await u.openDebugPanel() | ||||
|  | ||||
|       const code = `sketch001 = startSketchOn(-XZ) | ||||
|       const code = `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(-XZ) | ||||
| profile001 = startProfileAt([${roundOff(scale * 69.6)}, ${roundOff( | ||||
|         scale * 34.8 | ||||
|       )}], sketch001) | ||||
| @ -898,7 +912,7 @@ profile001 = startProfileAt([${roundOff(scale * 69.6)}, ${roundOff( | ||||
|       await page.mouse.move(700, 200, { steps: 10 }) | ||||
|       await page.mouse.click(700, 200, { delay: 200 }) | ||||
|       await expect(page.locator('.cm-content')).toHaveText( | ||||
|         `sketch001 = startSketchOn(-XZ)` | ||||
|         `@settings(defaultLengthUnit = in)sketch001 = startSketchOn(-XZ)` | ||||
|       ) | ||||
|  | ||||
|       let prevContent = await page.locator('.cm-content').innerText() | ||||
| @ -1426,7 +1440,8 @@ test.describe(`Sketching with offset planes`, () => { | ||||
|     await context.addInitScript(() => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `offsetPlane001 = offsetPlane(XY, offset = 10)` | ||||
|         `@settings(defaultLengthUnit = in) | ||||
| offsetPlane001 = offsetPlane(XY, offset = 10)` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
| @ -1440,7 +1455,7 @@ test.describe(`Sketching with offset planes`, () => { | ||||
|       await test.step(`Hovering should highlight code`, async () => { | ||||
|         await planeHover() | ||||
|         await editor.expectState({ | ||||
|           activeLines: [`offsetPlane001=offsetPlane(XY,offset=10)`], | ||||
|           activeLines: [`@settings(defaultLengthUnit = in)`], | ||||
|           diagnostics: [], | ||||
|           highlightedCode: 'offsetPlane(XY, offset = 10)', | ||||
|         }) | ||||
| @ -1453,7 +1468,7 @@ test.describe(`Sketching with offset planes`, () => { | ||||
|         await expect(toolbar.lineBtn).toBeEnabled() | ||||
|         await editor.expectEditor.toContain('startSketchOn(offsetPlane001)') | ||||
|         await editor.expectState({ | ||||
|           activeLines: [`offsetPlane001=offsetPlane(XY,offset=10)`], | ||||
|           activeLines: [`@settings(defaultLengthUnit = in)`], | ||||
|           diagnostics: [], | ||||
|           highlightedCode: '', | ||||
|         }) | ||||
| @ -1604,7 +1619,8 @@ profile002 = startProfileAt([117.2, 56.08], sketch001) | ||||
|       await context.addInitScript(() => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `sketch001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
| profile002 = startProfileAt([40.68, 87.67], sketch001) | ||||
|   |> xLine(length = 239.17) | ||||
| profile003 = startProfileAt([206.63, -56.73], sketch001) | ||||
| @ -2172,7 +2188,8 @@ profile003 = startProfileAt([206.63, -56.73], sketch001) | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `sketch001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
| profile001 = startProfileAt([6.24, 4.54], sketch001) | ||||
|   |> line(end = [-0.41, 6.99]) | ||||
|   |> line(end = [8.61, 0.74]) | ||||
| @ -2317,7 +2334,8 @@ profile004 = circleThreePoint(sketch001, p1 = [13.44, -6.8], p2 = [13.39, -2.07] | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `sketch001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
| profile001 = startProfileAt([6.24, 4.54], sketch001) | ||||
|   |> line(end = [-0.41, 6.99]) | ||||
|   |> line(end = [8.61, 0.74]) | ||||
| @ -2422,7 +2440,8 @@ profile003 = circle(sketch001, center = [6.92, -4.2], radius = 3.16) | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `sketch001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
| profile001 = startProfileAt([-63.43, 193.08], sketch001) | ||||
|   |> line(end = [168.52, 149.87]) | ||||
|   |> line(end = [190.29, -39.18]) | ||||
| @ -2486,7 +2505,11 @@ extrude001 = extrude(profile003, length = 5) | ||||
|     page, | ||||
|   }) => { | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem('persistCode', `myVar = 5`) | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `@settings(defaultLengthUnit = in) | ||||
|         myVar = 5` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
| @ -2533,7 +2556,8 @@ extrude001 = extrude(profile003, length = 5) | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `sketch001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
| profile001 = startProfileAt([85.19, 338.59], sketch001) | ||||
|   |> line(end = [213.3, -94.52]) | ||||
|   |> line(end = [-230.09, -55.34]) | ||||
| @ -2575,7 +2599,8 @@ profile002 = startProfileAt([85.81, 52.55], sketch002) | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `thePart = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
| thePart = startSketchOn(XZ) | ||||
|   |> startProfileAt([7.53, 10.51], %) | ||||
|   |> line(end = [12.54, 1.83]) | ||||
|   |> line(end = [6.65, -6.91]) | ||||
| @ -2636,7 +2661,8 @@ extrude001 = extrude(thePart, length = 75) | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `sketch001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
| profile001 = startProfileAt([6.71, -3.66], sketch001) | ||||
|   |> line(end = [2.65, 9.02], tag = $seg02) | ||||
|   |> line(end = [3.73, -9.36], tag = $seg01) | ||||
| @ -2809,7 +2835,8 @@ extrude003 = extrude(profile011, length = 2.5) | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `sketch001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
| profile001 = startProfileAt([34, 42.66], sketch001) | ||||
|   |> line(end = [102.65, 151.99]) | ||||
|   |> line(end = [76, -138.66]) | ||||
|  | ||||
| @ -1,21 +1,22 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { secrets } from './secrets' | ||||
| import { | ||||
|   Paths, | ||||
|   doExport, | ||||
|   getUtils, | ||||
|   settingsToToml, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
| } from './test-utils' | ||||
| import { Models } from '@kittycad/lib' | ||||
| import fsp from 'fs/promises' | ||||
| import type { Models } from '@kittycad/lib' | ||||
| import { KCL_DEFAULT_LENGTH } from '@src/lib/constants' | ||||
| import { spawn } from 'child_process' | ||||
| import { KCL_DEFAULT_LENGTH } from 'lib/constants' | ||||
| import fsp from 'fs/promises' | ||||
| import JSZip from 'jszip' | ||||
| import path from 'path' | ||||
| import { TEST_SETTINGS, TEST_SETTINGS_KEY } from './storageStates' | ||||
| import { SceneFixture } from './fixtures/sceneFixture' | ||||
| import { CmdBarFixture } from './fixtures/cmdBarFixture' | ||||
|  | ||||
| import type { CmdBarFixture } from '@e2e/playwright/fixtures/cmdBarFixture' | ||||
| import type { SceneFixture } from '@e2e/playwright/fixtures/sceneFixture' | ||||
| import { secrets } from '@e2e/playwright/secrets' | ||||
| import { TEST_SETTINGS, TEST_SETTINGS_KEY } from '@e2e/playwright/storageStates' | ||||
| import type { Paths } from '@e2e/playwright/test-utils' | ||||
| import { | ||||
|   doExport, | ||||
|   getUtils, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
|   settingsToToml, | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.beforeEach(async ({ page, context }) => { | ||||
|   // Make the user avatar image always 404 | ||||
| @ -76,11 +77,11 @@ part001 = startSketchOn(-XZ) | ||||
|   |> xLine(endAbsolute = totalLen, tag = $seg03) | ||||
|   |> yLine(length = -armThick, tag = $seg01) | ||||
|   |> angledLineThatIntersects({ | ||||
|         angle = HALF_TURN, | ||||
|         angle = turns::HALF_TURN, | ||||
|         offset = -armThick, | ||||
|         intersectTag = seg04 | ||||
|       }, %) | ||||
|   |> angledLineToY([segAng(seg04, %) + 180, ZERO], %) | ||||
|   |> angledLineToY([segAng(seg04, %) + 180, turns::ZERO], %) | ||||
|   |> angledLineToY({ | ||||
|         angle = -bottomAng, | ||||
|         to = -totalHeightHalf - armThick, | ||||
| @ -88,12 +89,12 @@ part001 = startSketchOn(-XZ) | ||||
|   |> xLine(length = endAbsolute = segEndX(seg03) + 0) | ||||
|   |> yLine(length = -segLen(seg01, %)) | ||||
|   |> angledLineThatIntersects({ | ||||
|         angle = HALF_TURN, | ||||
|         angle = turns::HALF_TURN, | ||||
|         offset = -armThick, | ||||
|         intersectTag = seg02 | ||||
|       }, %) | ||||
|   |> angledLineToY([segAng(seg02, %) + 180, -baseHeight], %) | ||||
|   |> xLine(endAbsolute = ZERO) | ||||
|   |> xLine(endAbsolute = turns::ZERO) | ||||
|   |> close() | ||||
|   |> extrude(length = 4)` | ||||
|       ) | ||||
| @ -345,7 +346,9 @@ const extrudeDefaultPlane = async ( | ||||
|           app: { | ||||
|             onboarding_status: 'dismissed', | ||||
|             show_debug_panel: true, | ||||
|             theme: 'dark', | ||||
|             appearance: { | ||||
|               theme: 'dark', | ||||
|             }, | ||||
|           }, | ||||
|           project: { | ||||
|             default_project_name: 'project-$nnn', | ||||
| @ -452,7 +455,7 @@ test( | ||||
|     await page.waitForTimeout(700) // TODO detect animation ending, or disable animation | ||||
|  | ||||
|     await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) | ||||
|     code += `profile001 = startProfileAt([7.19, -9.7], sketch001)` | ||||
|     code += `profile001 = startProfileAt([182.59, -246.32], sketch001)` | ||||
|     await expect(page.locator('.cm-content')).toHaveText(code) | ||||
|     await page.waitForTimeout(100) | ||||
|  | ||||
| @ -470,7 +473,7 @@ test( | ||||
|     await page.waitForTimeout(500) | ||||
|  | ||||
|     code += ` | ||||
|   |> xLine(length = 7.25)` | ||||
|   |> xLine(length = 184.3)` | ||||
|     await expect(page.locator('.cm-content')).toHaveText(code) | ||||
|  | ||||
|     await page | ||||
| @ -628,7 +631,7 @@ test( | ||||
|       mask: [page.getByTestId('model-state-indicator')], | ||||
|     }) | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       `sketch001 = startSketchOn(XZ)profile001 = circle(sketch001, center = [14.44, -2.44], radius = 1)` | ||||
|       `sketch001 = startSketchOn(XZ)profile001 = circle(sketch001, center = [366.89, -62.01], radius = 1)` | ||||
|     ) | ||||
|   } | ||||
| ) | ||||
| @ -665,7 +668,7 @@ test.describe( | ||||
|  | ||||
|       const startXPx = 600 | ||||
|       await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) | ||||
|       code += `profile001 = startProfileAt([7.19, -9.7], sketch001)` | ||||
|       code += `profile001 = startProfileAt([182.59, -246.32], sketch001)` | ||||
|       await expect(u.codeLocator).toHaveText(code) | ||||
|       await page.waitForTimeout(100) | ||||
|  | ||||
| @ -673,7 +676,7 @@ test.describe( | ||||
|       await page.waitForTimeout(100) | ||||
|  | ||||
|       code += ` | ||||
|   |> xLine(length = 7.25)` | ||||
|   |> xLine(length = 184.3)` | ||||
|       await expect(u.codeLocator).toHaveText(code) | ||||
|  | ||||
|       await page | ||||
| @ -688,7 +691,7 @@ test.describe( | ||||
|       await page.mouse.click(startXPx + PUR * 30, 500 - PUR * 20) | ||||
|  | ||||
|       code += ` | ||||
|   |> tangentialArcTo([21.7, -2.44], %)` | ||||
|   |> tangentialArcTo([551.2, -62.01], %)` | ||||
|       await expect(u.codeLocator).toHaveText(code) | ||||
|  | ||||
|       // click tangential arc tool again to unequip it | ||||
|  | ||||
| Before Width: | Height: | Size: 54 KiB After Width: | Height: | Size: 58 KiB | 
| Before Width: | Height: | Size: 53 KiB After Width: | Height: | Size: 52 KiB | 
| Before Width: | Height: | Size: 60 KiB After Width: | Height: | Size: 59 KiB | 
| Before Width: | Height: | Size: 53 KiB After Width: | Height: | Size: 52 KiB | 
| Before Width: | Height: | Size: 48 KiB After Width: | Height: | Size: 52 KiB | 
| Before Width: | Height: | Size: 45 KiB After Width: | Height: | Size: 50 KiB | 
| Before Width: | Height: | Size: 42 KiB After Width: | Height: | Size: 46 KiB | 
| Before Width: | Height: | Size: 45 KiB After Width: | Height: | Size: 49 KiB | 
| Before Width: | Height: | Size: 49 KiB After Width: | Height: | Size: 55 KiB | 
| Before Width: | Height: | Size: 55 KiB After Width: | Height: | Size: 60 KiB | 
| Before Width: | Height: | Size: 68 KiB After Width: | Height: | Size: 65 KiB | 
| Before Width: | Height: | Size: 34 KiB After Width: | Height: | Size: 34 KiB | 
| Before Width: | Height: | Size: 54 KiB After Width: | Height: | Size: 54 KiB | 
| Before Width: | Height: | Size: 75 KiB After Width: | Height: | Size: 78 KiB | 
| Before Width: | Height: | Size: 53 KiB After Width: | Height: | Size: 50 KiB | 
| Before Width: | Height: | Size: 65 KiB After Width: | Height: | Size: 60 KiB | 
| Before Width: | Height: | Size: 143 KiB After Width: | Height: | Size: 140 KiB | 
| Before Width: | Height: | Size: 131 KiB After Width: | Height: | Size: 124 KiB | 
| Before Width: | Height: | Size: 68 KiB After Width: | Height: | Size: 68 KiB | 
| Before Width: | Height: | Size: 76 KiB After Width: | Height: | Size: 71 KiB | 
| Before Width: | Height: | Size: 69 KiB After Width: | Height: | Size: 67 KiB | 
| Before Width: | Height: | Size: 72 KiB After Width: | Height: | Size: 70 KiB | 
| Before Width: | Height: | Size: 72 KiB After Width: | Height: | Size: 70 KiB | 
| Before Width: | Height: | Size: 68 KiB After Width: | Height: | Size: 66 KiB | 
| @ -1,5 +1,5 @@ | ||||
| { | ||||
|   "original_source_code": "sketch001 = startSketchOn('XZ')\nprofile001 = startProfileAt([57.81, 250.51], sketch001)\n  |> line(end = [121.13, 56.63], tag = $seg02)\n  |> line(end = [83.37, -34.61], tag = $seg01)\n  |> line(end = [19.66, -116.4])\n  |> line(end = [-221.8, -41.69])\n  |> line(endAbsolute = [profileStartX(%), profileStartY(%)])\n  |> close()\nextrude001 = extrude(profile001, length = 200)\nsketch002 = startSketchOn('XZ')\n  |> startProfileAt([-73.64, -42.89], %)\n  |> xLine(length = 173.71)\n  |> line(end = [-22.12, -94.4])\n  |> xLine(length = -156.98)\n  |> line(endAbsolute = [profileStartX(%), profileStartY(%)])\n  |> close()\nextrude002 = extrude(sketch002, length = 50)\nsketch003 = startSketchOn('XY')\n  |> startProfileAt([52.92, 157.81], %)\n  |> angledLine([0, 176.4], %, $rectangleSegmentA001)\n  |> angledLine([\n       segAng(rectangleSegmentA001) - 90,\n       53.4\n     ], %, $rectangleSegmentB001)\n  |> angledLine([\n       segAng(rectangleSegmentA001),\n       -segLen(rectangleSegmentA001)\n     ], %, $rectangleSegmentC001)\n  |> line(endAbsolute = [profileStartX(%), profileStartY(%)])\n  |> close()\nextrude003 = extrude(sketch003, length = 20)\n", | ||||
|   "original_source_code": "sketch001 = startSketchOn('XZ')\nprofile001 = startProfileAt([57.81, 250.51], sketch001)\n  |> line(end = [121.13, 56.63], tag = $seg02)\n  |> line(end = [83.37, -34.61], tag = $seg01)\n  |> line(end = [19.66, -116.4])\n  |> line(end = [-221.8, -41.69])\n  |> line(endAbsolute = [profileStartX(%), profileStartY(%)])\n  |> close()\nextrude001 = extrude(profile001, length = 200)\nsketch002 = startSketchOn('XZ')\n  |> startProfileAt([-73.64, -42.89], %)\n  |> xLine(length = 173.71)\n  |> line(end = [-22.12, -94.4])\n  |> xLine(length = -156.98)\n  |> line(endAbsolute = [profileStartX(%), profileStartY(%)])\n  |> close()\nextrude002 = extrude(sketch002, length = 50)\nsketch003 = startSketchOn(XY)\n  |> startProfileAt([52.92, 157.81], %)\n  |> angledLine([0, 176.4], %, $rectangleSegmentA001)\n  |> angledLine([\n       segAng(rectangleSegmentA001) - 90,\n       53.4\n     ], %, $rectangleSegmentB001)\n  |> angledLine([\n       segAng(rectangleSegmentA001),\n       -segLen(rectangleSegmentA001)\n     ], %, $rectangleSegmentC001)\n  |> line(endAbsolute = [profileStartX(%), profileStartY(%)])\n  |> close()\nextrude003 = extrude(sketch003, length = 20)\n", | ||||
|   "prompt": "make this neon green please, use #39FF14", | ||||
|   "source_ranges": [ | ||||
|     { | ||||
| @ -30,4 +30,4 @@ | ||||
|     } | ||||
|   ], | ||||
|   "kcl_version": "0.2.48" | ||||
| } | ||||
| } | ||||
|  | ||||
| @ -29,5 +29,5 @@ | ||||
|       } | ||||
|     } | ||||
|   ], | ||||
|   "kcl_version": "0.2.54" | ||||
|   "kcl_version": "0.2.57" | ||||
| } | ||||
| @ -1,17 +1,19 @@ | ||||
| import { Settings } from '@rust/kcl-lib/bindings/Settings' | ||||
| import { SaveSettingsPayload } from 'lib/settings/settingsTypes' | ||||
| import { Themes } from 'lib/theme' | ||||
| import { DeepPartial } from 'lib/types' | ||||
| import { onboardingPaths } from 'routes/Onboarding/paths' | ||||
| import type { SaveSettingsPayload } from '@src/lib/settings/settingsTypes' | ||||
| import { Themes } from '@src/lib/theme' | ||||
| import type { DeepPartial } from '@src/lib/types' | ||||
| import { onboardingPaths } from '@src/routes/Onboarding/paths' | ||||
|  | ||||
| import type { Settings } from '@rust/kcl-lib/bindings/Settings' | ||||
|  | ||||
| export const IS_PLAYWRIGHT_KEY = 'playwright' | ||||
|  | ||||
| export const TEST_SETTINGS_KEY = '/settings.toml' | ||||
| export const TEST_SETTINGS: DeepPartial<Settings> = { | ||||
|   app: { | ||||
|     theme: Themes.Dark, | ||||
|     appearance: { | ||||
|       theme: Themes.Dark, | ||||
|     }, | ||||
|     onboarding_status: 'dismissed', | ||||
|     project_directory: '', | ||||
|     show_debug_panel: true, | ||||
|   }, | ||||
|   modeling: { | ||||
| @ -22,6 +24,7 @@ export const TEST_SETTINGS: DeepPartial<Settings> = { | ||||
|   }, | ||||
|   project: { | ||||
|     default_project_name: 'project-$nnn', | ||||
|     directory: '', | ||||
|   }, | ||||
|   text_editor: { | ||||
|     text_wrapping: true, | ||||
| @ -54,7 +57,7 @@ export const TEST_SETTINGS_ONBOARDING_START: DeepPartial<Settings> = { | ||||
|  | ||||
| export const TEST_SETTINGS_DEFAULT_THEME: DeepPartial<Settings> = { | ||||
|   ...TEST_SETTINGS, | ||||
|   app: { ...TEST_SETTINGS.app, theme: Themes.System }, | ||||
|   app: { ...TEST_SETTINGS.app, appearance: { theme: Themes.System } }, | ||||
| } | ||||
|  | ||||
| export const TEST_SETTINGS_CORRUPTED = { | ||||
|  | ||||
| @ -1,7 +1,12 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { commonPoints, getUtils, orRunWhenFullSuiteEnabled } from './test-utils' | ||||
| import { EngineCommand } from 'lang/std/artifactGraph' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
| import type { EngineCommand } from '@src/lang/std/artifactGraph' | ||||
| import { uuidv4 } from '@src/lib/utils' | ||||
|  | ||||
| import { | ||||
|   commonPoints, | ||||
|   getUtils, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Test network and connection issues', () => { | ||||
|   test( | ||||
|  | ||||
| @ -1,9 +1,9 @@ | ||||
| import { | ||||
|   orRunWhenFullSuiteEnabled, | ||||
|   runningOnLinux, | ||||
|   runningOnMac, | ||||
|   runningOnWindows, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
| } from './test-utils' | ||||
| } from '@e2e/playwright/test-utils' | ||||
|  | ||||
| describe('platform detection utilities', () => { | ||||
|   const originalPlatform = process.platform | ||||
|  | ||||
| @ -1,32 +1,29 @@ | ||||
| import { | ||||
|   expect, | ||||
|   BrowserContext, | ||||
|   TestInfo, | ||||
|   Locator, | ||||
|   Page, | ||||
| } from '@playwright/test' | ||||
| import { test } from './zoo-test' | ||||
| import { EngineCommand } from 'lang/std/artifactGraph' | ||||
| import * as TOML from '@iarna/toml' | ||||
| import type { Models } from '@kittycad/lib' | ||||
| import type { BrowserContext, Locator, Page, TestInfo } from '@playwright/test' | ||||
| import { expect } from '@playwright/test' | ||||
| import type { EngineCommand } from '@src/lang/std/artifactGraph' | ||||
| import type { Configuration } from '@src/lang/wasm' | ||||
| import { COOKIE_NAME } from '@src/lib/constants' | ||||
| import { reportRejection } from '@src/lib/trap' | ||||
| import type { DeepPartial } from '@src/lib/types' | ||||
| import { isArray } from '@src/lib/utils' | ||||
| import fsp from 'fs/promises' | ||||
| import path from 'path' | ||||
| import pixelMatch from 'pixelmatch' | ||||
| import type { Protocol } from 'playwright-core/types/protocol' | ||||
| import { PNG } from 'pngjs' | ||||
| import { Protocol } from 'playwright-core/types/protocol' | ||||
| import type { Models } from '@kittycad/lib' | ||||
| import { COOKIE_NAME } from 'lib/constants' | ||||
| import { secrets } from './secrets' | ||||
|  | ||||
| import type { ProjectConfiguration } from '@rust/kcl-lib/bindings/ProjectConfiguration' | ||||
|  | ||||
| import { isErrorWhitelisted } from '@e2e/playwright/lib/console-error-whitelist' | ||||
| import { secrets } from '@e2e/playwright/secrets' | ||||
| import { | ||||
|   TEST_SETTINGS_KEY, | ||||
|   TEST_SETTINGS, | ||||
|   IS_PLAYWRIGHT_KEY, | ||||
| } from './storageStates' | ||||
| import * as TOML from '@iarna/toml' | ||||
| import { isErrorWhitelisted } from './lib/console-error-whitelist' | ||||
| import { isArray } from 'lib/utils' | ||||
| import { reportRejection } from 'lib/trap' | ||||
| import { DeepPartial } from 'lib/types' | ||||
| import { Configuration } from 'lang/wasm' | ||||
| import { ProjectConfiguration } from '@rust/kcl-lib/bindings/ProjectConfiguration' | ||||
|   TEST_SETTINGS, | ||||
|   TEST_SETTINGS_KEY, | ||||
| } from '@e2e/playwright/storageStates' | ||||
| import { test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| const toNormalizedCode = (text: string) => { | ||||
|   return text.replace(/\s+/g, '') | ||||
| @ -683,8 +680,8 @@ const _makeTemplate = ( | ||||
|           isArray(currentOptions) | ||||
|             ? currentOptions[i] | ||||
|             : typeof currentOptions === 'number' | ||||
|             ? currentOptions | ||||
|             : '' | ||||
|               ? currentOptions | ||||
|               : '' | ||||
|         ) | ||||
|       ) | ||||
|     }) | ||||
| @ -903,15 +900,21 @@ export async function setup( | ||||
|         settings: { | ||||
|           ...TEST_SETTINGS, | ||||
|           app: { | ||||
|             appearance: { | ||||
|               ...TEST_SETTINGS.app?.appearance, | ||||
|               theme: 'dark', | ||||
|             }, | ||||
|             ...TEST_SETTINGS.project, | ||||
|             project_directory: TEST_SETTINGS.app?.project_directory, | ||||
|             onboarding_status: 'dismissed', | ||||
|             theme: 'dark', | ||||
|           }, | ||||
|           project: { | ||||
|             ...TEST_SETTINGS.project, | ||||
|             directory: TEST_SETTINGS.project?.directory, | ||||
|           }, | ||||
|         }, | ||||
|       }), | ||||
|       IS_PLAYWRIGHT_KEY, | ||||
|       PLAYWRIGHT_TEST_DIR: TEST_SETTINGS.app?.project_directory || '', | ||||
|       PLAYWRIGHT_TEST_DIR: TEST_SETTINGS.project?.directory || '', | ||||
|       PERSIST_MODELING_CONTEXT, | ||||
|     } | ||||
|   ) | ||||
| @ -935,47 +938,39 @@ export async function setup( | ||||
| } | ||||
|  | ||||
| function failOnConsoleErrors(page: Page, testInfo?: TestInfo) { | ||||
|   // enabled for chrome for now | ||||
|   if (page.context().browser()?.browserType().name() === 'chromium') { | ||||
|     // No idea wtf exception is | ||||
|     page.on('pageerror', (exception: any) => { | ||||
|       if (isErrorWhitelisted(exception)) { | ||||
|         return | ||||
|       } | ||||
|   page.on('pageerror', (exception: any) => { | ||||
|     if (isErrorWhitelisted(exception)) { | ||||
|       return | ||||
|     } | ||||
|     // Only disable this environment variable if you want to collect console errors | ||||
|     if (process.env.FAIL_ON_CONSOLE_ERRORS !== 'false') { | ||||
|       // Use expect to prevent page from closing and not cleaning up | ||||
|       expect(`An error was detected in the console: \r\n message:${exception.message} \r\n name:${exception.name} \r\n stack:${exception.stack} | ||||
|  | ||||
|       // only set this env var to false if you want to collect console errors | ||||
|       // This can be configured in the GH workflow.  This should be set to true by default (we want tests to fail when | ||||
|       // unwhitelisted console errors are detected). | ||||
|       if (process.env.FAIL_ON_CONSOLE_ERRORS === 'true') { | ||||
|         // Fail when running on CI and FAIL_ON_CONSOLE_ERRORS is set | ||||
|         // use expect to prevent page from closing and not cleaning up | ||||
|         expect(`An error was detected in the console: \r\n message:${exception.message} \r\n name:${exception.name} \r\n stack:${exception.stack} | ||||
|  | ||||
|           *Either fix the console error or add it to the whitelist defined in ./lib/console-error-whitelist.ts (if the error can be safely ignored) | ||||
|           `).toEqual('Console error detected') | ||||
|       } else { | ||||
|         // the (test-results/exceptions.txt) file will be uploaded as part of an upload artifact in GH | ||||
|         fsp | ||||
|           .appendFile( | ||||
|             './test-results/exceptions.txt', | ||||
|             [ | ||||
|               '~~~', | ||||
|               `triggered_by_test:${ | ||||
|                 testInfo?.file + ' ' + (testInfo?.title || ' ') | ||||
|               }`, | ||||
|               `name:${exception.name}`, | ||||
|               `message:${exception.message}`, | ||||
|               `stack:${exception.stack}`, | ||||
|               `project:${testInfo?.project.name}`, | ||||
|               '~~~', | ||||
|             ].join('\n') | ||||
|           ) | ||||
|           .catch((err) => { | ||||
|             console.error(err) | ||||
|           }) | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
|         *Either fix the console error or add it to the whitelist defined in ./lib/console-error-whitelist.ts (if the error can be safely ignored) | ||||
|         `).toEqual('Console error detected') | ||||
|     } else { | ||||
|       // Add errors to `test-results/exceptions.txt` as a test artifact | ||||
|       fsp | ||||
|         .appendFile( | ||||
|           './test-results/exceptions.txt', | ||||
|           [ | ||||
|             '~~~', | ||||
|             `triggered_by_test:${ | ||||
|               testInfo?.file + ' ' + (testInfo?.title || ' ') | ||||
|             }`, | ||||
|             `name:${exception.name}`, | ||||
|             `message:${exception.message}`, | ||||
|             `stack:${exception.stack}`, | ||||
|             `project:${testInfo?.project.name}`, | ||||
|             '~~~', | ||||
|           ].join('\n') | ||||
|         ) | ||||
|         .catch((err) => { | ||||
|           console.error(err) | ||||
|         }) | ||||
|     } | ||||
|   }) | ||||
| } | ||||
| export async function isOutOfViewInScrollContainer( | ||||
|   element: Locator, | ||||
| @ -1120,21 +1115,25 @@ export async function pollEditorLinesSelectedLength(page: Page, lines: number) { | ||||
| } | ||||
|  | ||||
| export function settingsToToml(settings: DeepPartial<Configuration>) { | ||||
|   // eslint-disable-next-line no-restricted-syntax | ||||
|   return TOML.stringify(settings as any) | ||||
| } | ||||
|  | ||||
| export function tomlToSettings(toml: string): DeepPartial<Configuration> { | ||||
|   // eslint-disable-next-line no-restricted-syntax | ||||
|   return TOML.parse(toml) | ||||
| } | ||||
|  | ||||
| export function tomlToPerProjectSettings( | ||||
|   toml: string | ||||
| ): DeepPartial<ProjectConfiguration> { | ||||
|   // eslint-disable-next-line no-restricted-syntax | ||||
|   return TOML.parse(toml) | ||||
| } | ||||
|  | ||||
| export function perProjectsettingsToToml( | ||||
|   settings: DeepPartial<ProjectConfiguration> | ||||
| ) { | ||||
|   // eslint-disable-next-line no-restricted-syntax | ||||
|   return TOML.stringify(settings as any) | ||||
| } | ||||
|  | ||||
| @ -1,7 +1,8 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { EngineCommand } from 'lang/std/artifactGraph' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
| import { getUtils, orRunWhenFullSuiteEnabled } from './test-utils' | ||||
| import type { EngineCommand } from '@src/lang/std/artifactGraph' | ||||
| import { uuidv4 } from '@src/lib/utils' | ||||
|  | ||||
| import { getUtils, orRunWhenFullSuiteEnabled } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Testing Camera Movement', { tag: ['@skipWin'] }, () => { | ||||
|   test('Can move camera reliably', async ({ page, context, homePage }) => { | ||||
|  | ||||
| @ -1,15 +1,15 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { XOR } from '@src/lib/utils' | ||||
| import * as fsp from 'fs/promises' | ||||
| import { | ||||
|   getUtils, | ||||
|   TEST_COLORS, | ||||
|   pollEditorLinesSelectedLength, | ||||
|   executorInputPath, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
| } from './test-utils' | ||||
| import { XOR } from 'lib/utils' | ||||
| import path from 'node:path' | ||||
|  | ||||
| import { | ||||
|   TEST_COLORS, | ||||
|   getUtils, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
|   pollEditorLinesSelectedLength, | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { | ||||
|   test('Can constrain line length', async ({ page, homePage }) => { | ||||
|     await page.addInitScript(async () => { | ||||
| @ -81,7 +81,8 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `yo = 79 | ||||
|         `@settings(defaultLengthUnit = in) | ||||
|   yo = 79 | ||||
|   part001 = startSketchOn(XZ) | ||||
|     |> startProfileAt([-7.54, -26.74], %) | ||||
|     |> line(end = [74.36, 130.4], tag = $seg01) | ||||
| @ -145,7 +146,8 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
|             `@settings(defaultLengthUnit = in) | ||||
|       yo = 5 | ||||
|       part001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line(end = [74.36, 130.4], tag = $seg01) | ||||
| @ -159,31 +161,6 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { | ||||
|         |> xLine(length = segLen(seg_what)) | ||||
|         |> line(endAbsolute = [profileStartX(%), profileStartY(%)])` | ||||
|           ) | ||||
|  | ||||
|           const isChecked = await createNewVariableCheckbox.isChecked() | ||||
|           const addVariable = testName === 'Add variable' | ||||
|           XOR(isChecked, addVariable) && // XOR because no need to click the checkbox if the state is already correct | ||||
|             (await createNewVariableCheckbox.click()) | ||||
|  | ||||
|           await page | ||||
|             .getByRole('button', { name: 'Add constraining value' }) | ||||
|             .click() | ||||
|  | ||||
|           // Wait for the codemod to take effect | ||||
|           await expect(page.locator('.cm-content')).toContainText(`angle: -57,`) | ||||
|           await expect(page.locator('.cm-content')).toContainText( | ||||
|             `offset: ${offset},` | ||||
|           ) | ||||
|  | ||||
|           await pollEditorLinesSelectedLength(page, 2) | ||||
|           const activeLinesContent = await page.locator('.cm-activeLine').all() | ||||
|           await expect(activeLinesContent[0]).toHaveText( | ||||
|             `|> line(end = [74.36, 130.4], tag = $seg01)` | ||||
|           ) | ||||
|           await expect(activeLinesContent[1]).toHaveText(`}, %)`) | ||||
|  | ||||
|           // checking the count of the overlays is a good proxy check that the client sketch scene is in a good state | ||||
|           await expect(page.getByTestId('segment-overlay')).toHaveCount(4) | ||||
|         }) | ||||
|         const u = await getUtils(page) | ||||
|         await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
| @ -277,7 +254,8 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
|             `@settings(defaultLengthUnit = in) | ||||
|       yo = 5 | ||||
|       part001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line(end = [74.36, 130.4]) | ||||
| @ -387,7 +365,8 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
|             `@settings(defaultLengthUnit = in) | ||||
|       yo = 5 | ||||
|       part001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line(end = [74.36, 130.4]) | ||||
| @ -486,13 +465,13 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { | ||||
|         testName: 'Add variable, selecting axis', | ||||
|         addVariable: true, | ||||
|         axisSelect: true, | ||||
|         value: 'QUARTER_TURN - angle001', | ||||
|         value: 'turns::QUARTER_TURN - angle001', | ||||
|       }, | ||||
|       { | ||||
|         testName: 'No variable, selecting axis', | ||||
|         addVariable: false, | ||||
|         axisSelect: true, | ||||
|         value: 'QUARTER_TURN - 7', | ||||
|         value: 'turns::QUARTER_TURN - 7', | ||||
|       }, | ||||
|     ] as const | ||||
|     for (const { testName, addVariable, value, axisSelect } of cases) { | ||||
| @ -500,7 +479,8 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
|             `@settings(defaultLengthUnit = in) | ||||
|       yo = 5 | ||||
|       part001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line(end = [74.36, 130.4]) | ||||
| @ -602,7 +582,8 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
|             `@settings(defaultLengthUnit = in) | ||||
|       yo = 5 | ||||
|       part001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line(end = [74.36, 130.4]) | ||||
| @ -688,7 +669,8 @@ test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
|             `@settings(defaultLengthUnit = in) | ||||
| yo = 5 | ||||
| part001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([-7.54, -26.74], %) | ||||
|   |> line(end = [74.36, 130.4]) | ||||
| @ -768,7 +750,8 @@ part002 = startSketchOn(XZ) | ||||
|         await page.addInitScript(async (customCode) => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
|             `@settings(defaultLengthUnit = in) | ||||
|       yo = 5 | ||||
|       part001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line(end = [74.36, 130.4]) | ||||
| @ -869,7 +852,8 @@ part002 = startSketchOn(XZ) | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
|             `@settings(defaultLengthUnit = in) | ||||
|       yo = 5 | ||||
|       part001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line(end = [74.36, 130.4]) | ||||
| @ -935,12 +919,12 @@ part002 = startSketchOn(XZ) | ||||
|   test.describe('Axis & segment - no modal constraints', () => { | ||||
|     const cases = [ | ||||
|       { | ||||
|         codeAfter: `|> line(endAbsolute = [154.9, ZERO])`, | ||||
|         codeAfter: `|> line(endAbsolute = [154.9, turns::ZERO])`, | ||||
|         axisClick: { x: 950, y: 250 }, | ||||
|         constraintName: 'Snap To X', | ||||
|       }, | ||||
|       { | ||||
|         codeAfter: `|> line(endAbsolute = [ZERO, 61.34])`, | ||||
|         codeAfter: `|> line(endAbsolute = [turns::ZERO, 61.34])`, | ||||
|         axisClick: { x: 600, y: 150 }, | ||||
|         constraintName: 'Snap To Y', | ||||
|       }, | ||||
| @ -950,7 +934,8 @@ part002 = startSketchOn(XZ) | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
|             `@settings(defaultLengthUnit = in) | ||||
|       yo = 5 | ||||
|       part001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line(end = [74.36, 130.4]) | ||||
| @ -1113,13 +1098,23 @@ test.describe('Electron constraint tests', () => { | ||||
|   test( | ||||
|     'Able to double click label to set constraint', | ||||
|     { tag: '@electron' }, | ||||
|     async ({ page, context, homePage, scene, editor, toolbar }) => { | ||||
|     async ({ page, context, homePage, scene, editor, toolbar, cmdBar }) => { | ||||
|       await context.folderSetupFn(async (dir) => { | ||||
|         const bracketDir = path.join(dir, 'test-sample') | ||||
|         await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           executorInputPath('angled_line.kcl'), | ||||
|           path.join(bracketDir, 'main.kcl') | ||||
|         await fsp.writeFile( | ||||
|           path.join(bracketDir, 'main.kcl'), | ||||
|           `@settings(defaultLengthUnit = in) | ||||
|           const part001 = startSketchOn(XY) | ||||
|             |> startProfileAt([4.83, 12.56], %) | ||||
|             |> line(end = [15.1, 2.48]) | ||||
|             |> line(end = [3.15, -9.85], tag = $seg01) | ||||
|             |> line(end = [-15.17, -4.1]) | ||||
|             |> angledLine([segAng(seg01), 12.35], %) | ||||
|             |> line(end = [-13.02, 10.03]) | ||||
|             |> close() | ||||
|             |> extrude(length = 4)`, | ||||
|           'utf-8' | ||||
|         ) | ||||
|       }) | ||||
|  | ||||
| @ -1137,6 +1132,14 @@ test.describe('Electron constraint tests', () => { | ||||
|         await scene.waitForExecutionDone() | ||||
|       }) | ||||
|  | ||||
|       async function clickOnFirstSegmentLabel() { | ||||
|         const child = page | ||||
|           .locator('.segment-length-label-text') | ||||
|           .first() | ||||
|           .locator('xpath=..') | ||||
|         await child.dblclick() | ||||
|       } | ||||
|  | ||||
|       await test.step('Double click to constrain', async () => { | ||||
|         // Enter sketch edit mode via feature tree | ||||
|         await toolbar.openPane('feature-tree') | ||||
| @ -1144,21 +1147,19 @@ test.describe('Electron constraint tests', () => { | ||||
|         await op.dblclick() | ||||
|         await toolbar.closePane('feature-tree') | ||||
|  | ||||
|         const child = page | ||||
|           .locator('.segment-length-label-text') | ||||
|           .first() | ||||
|           .locator('xpath=..') | ||||
|         await child.dblclick() | ||||
|         const cmdBarSubmitButton = page.getByRole('button', { | ||||
|           name: 'arrow right Continue', | ||||
|         }) | ||||
|         await cmdBarSubmitButton.click() | ||||
|         await expect(page.locator('.cm-content')).toContainText( | ||||
|           'length001 = 15.3' | ||||
|         ) | ||||
|         await expect(page.locator('.cm-content')).toContainText( | ||||
|           '|> angledLine([9, length001], %)' | ||||
|         ) | ||||
|         await clickOnFirstSegmentLabel() | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await editor.expectEditor.toContain('length001 = 15.3') | ||||
|         await editor.expectEditor.toContain('|> angledLine([9, length001], %)') | ||||
|       }) | ||||
|  | ||||
|       await test.step('Double click again and expect failure', async () => { | ||||
|         await clickOnFirstSegmentLabel() | ||||
|  | ||||
|         await expect( | ||||
|           page.getByText('Unable to constrain the length of this segment') | ||||
|         ).toBeVisible() | ||||
|  | ||||
|         await page.getByRole('button', { name: 'Exit Sketch' }).click() | ||||
|       }) | ||||
|     } | ||||
|  | ||||
| @ -1,7 +1,8 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { getUtils } from './test-utils' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
| import { TEST_CODE_GIZMO } from './storageStates' | ||||
| import { uuidv4 } from '@src/lib/utils' | ||||
|  | ||||
| import { TEST_CODE_GIZMO } from '@e2e/playwright/storageStates' | ||||
| import { getUtils } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Testing Gizmo', { tag: ['@skipWin'] }, () => { | ||||
|   const cases = [ | ||||
| @ -255,7 +256,7 @@ test.describe(`Testing gizmo, fixture-based`, () => { | ||||
|     await context.addInitScript(() => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         ` | ||||
|         `@settings(defaultLengthUnit = in) | ||||
|         const sketch002 = startSketchOn(XZ) | ||||
|           |> startProfileAt([-108.83, -57.48], %) | ||||
|           |> angledLine([0, 105.13], %, $rectangleSegmentA001) | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { getUtils, orRunWhenFullSuiteEnabled } from './test-utils' | ||||
| import { getUtils, orRunWhenFullSuiteEnabled } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Test toggling perspective', () => { | ||||
|   test('via command palette and toggle', async ({ page, homePage }) => { | ||||
|  | ||||
| @ -1,9 +1,10 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import { getUtils } from './test-utils' | ||||
| import { bracket } from 'lib/exampleKcl' | ||||
| import { FILE_EXT } from '@src/lib/constants' | ||||
| import { bracket } from '@src/lib/exampleKcl' | ||||
| import * as fsp from 'fs/promises' | ||||
| import { join } from 'path' | ||||
| import { FILE_EXT } from 'lib/constants' | ||||
|  | ||||
| import { getUtils } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Testing in-app sample loading', () => { | ||||
|   /** | ||||
| @ -46,7 +47,7 @@ test.describe('Testing in-app sample loading', () => { | ||||
|       page.getByRole('option', { | ||||
|         name, | ||||
|       }) | ||||
|     const warningText = page.getByText('Overwrite current file and units?') | ||||
|     const warningText = page.getByText('Overwrite current file with sample?') | ||||
|     const confirmButton = page.getByRole('button', { name: 'Submit command' }) | ||||
|  | ||||
|     await test.step(`Precondition: check the initial code`, async () => { | ||||
| @ -110,11 +111,9 @@ test.describe('Testing in-app sample loading', () => { | ||||
|       const commandMethodOption = page.getByRole('option', { | ||||
|         name: 'Overwrite', | ||||
|       }) | ||||
|       const newFileWarning = page.getByText( | ||||
|         'Create a new file, overwrite project units?' | ||||
|       ) | ||||
|       const newFileWarning = page.getByText('Create a new file from sample?') | ||||
|       const overwriteWarning = page.getByText( | ||||
|         'Overwrite current file and units?' | ||||
|         'Overwrite current file with sample?' | ||||
|       ) | ||||
|       const confirmButton = page.getByRole('button', { name: 'Submit command' }) | ||||
|       const projectMenuButton = page.getByTestId('project-sidebar-toggle') | ||||
|  | ||||
| @ -1,15 +1,15 @@ | ||||
| import { Page } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
| import type { Page } from '@playwright/test' | ||||
| import type { LineInputsType } from '@src/lang/std/sketchcombos' | ||||
| import { uuidv4 } from '@src/lib/utils' | ||||
|  | ||||
| import type { EditorFixture } from '@e2e/playwright/fixtures/editorFixture' | ||||
| import { | ||||
|   deg, | ||||
|   getUtils, | ||||
|   wiggleMove, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
| } from './test-utils' | ||||
| import { LineInputsType } from 'lang/std/sketchcombos' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
| import { EditorFixture } from './fixtures/editorFixture' | ||||
|   wiggleMove, | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Testing segment overlays', { tag: ['@skipWin'] }, () => { | ||||
|   test('Hover over a segment should show its overlay, hovering over the input overlays should show its popover, clicking the input overlay should constrain/unconstrain it:\nfor the following segments', () => { | ||||
| @ -210,7 +210,8 @@ test.describe('Testing segment overlays', { tag: ['@skipWin'] }, () => { | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `part001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
|       part001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([5 + 0, 20 + 0], %) | ||||
|         |> line(end = [0.5, -14 + 0]) | ||||
|         |> angledLine({ angle = 3 + 0, length = 32 + 0 }, %) | ||||
| @ -380,7 +381,8 @@ test.describe('Testing segment overlays', { tag: ['@skipWin'] }, () => { | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `yRel001 = -14 | ||||
|           `@settings(defaultLengthUnit = in) | ||||
|     yRel001 = -14 | ||||
|     xRel001 = 0.5 | ||||
|     angle001 = 3 | ||||
|     len001 = 32 | ||||
| @ -459,7 +461,8 @@ test.describe('Testing segment overlays', { tag: ['@skipWin'] }, () => { | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `part001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
|       part001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([0, 0], %) | ||||
|         |> line(end = [0.5, -14 + 0]) | ||||
|         |> angledLine({ angle = 3 + 0, length = 32 + 0 }, %) | ||||
| @ -590,7 +593,8 @@ test.describe('Testing segment overlays', { tag: ['@skipWin'] }, () => { | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `part001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
|       part001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([0, 0], %) | ||||
|         |> line(end = [0.5, -14 + 0]) | ||||
|         |> angledLine({ angle = 3 + 0, length = 32 + 0 }, %) | ||||
| @ -751,7 +755,8 @@ test.describe('Testing segment overlays', { tag: ['@skipWin'] }, () => { | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `part001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
|       part001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([0, 0], %) | ||||
|         |> line(end = [0.5, -14 + 0]) | ||||
|         |> angledLine({ angle = 3 + 0, length = 32 + 0 }, %) | ||||
| @ -831,7 +836,8 @@ test.describe('Testing segment overlays', { tag: ['@skipWin'] }, () => { | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `sketch001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
| profile001 = startProfileAt([56.37, 120.33], sketch001) | ||||
|   |> line(end = [162.86, 106.48]) | ||||
|   |> arcTo({ | ||||
| @ -957,7 +963,8 @@ profile001 = startProfileAt([56.37, 120.33], sketch001) | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `part001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
| part001 = startSketchOn(XZ) | ||||
|       |> circle(center = [1 + 0, 0], radius = 8) | ||||
|     ` | ||||
|         ) | ||||
| @ -1077,7 +1084,8 @@ profile001 = startProfileAt([56.37, 120.33], sketch001) | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `part001 = startSketchOn(XZ) | ||||
|           `@settings(defaultLengthUnit = in) | ||||
| part001 = startSketchOn(XZ) | ||||
|   |>startProfileAt([0, 0], %) | ||||
|   |> line(end = [0.5, -14 + 0]) | ||||
|   |> angledLine({ angle = 3 + 0, length = 32 + 0 }, %) | ||||
| @ -1351,7 +1359,8 @@ profile001 = startProfileAt([56.37, 120.33], sketch001) | ||||
|             async ({ lineToBeDeleted, extraLine }) => { | ||||
|               localStorage.setItem( | ||||
|                 'persistCode', | ||||
|                 `part001 = startSketchOn(XZ) | ||||
|                 `@settings(defaultLengthUnit = in) | ||||
|         part001 = startSketchOn(XZ) | ||||
|           |> startProfileAt([5, 6], %) | ||||
|           |> ${lineToBeDeleted} | ||||
|           |> line(end = [-10, -15]) | ||||
| @ -1516,7 +1525,8 @@ profile001 = startProfileAt([56.37, 120.33], sketch001) | ||||
|           async ({ lineToBeDeleted }) => { | ||||
|             localStorage.setItem( | ||||
|               'persistCode', | ||||
|               `part001 = startSketchOn(XZ) | ||||
|               `@settings(defaultLengthUnit = in) | ||||
|       part001 = startSketchOn(XZ) | ||||
|         |> startProfileAt([5, 6], %) | ||||
|         |> ${lineToBeDeleted} | ||||
|         |> line(end = [-10, -15]) | ||||
|  | ||||
| @ -1,9 +1,13 @@ | ||||
| import { test, expect } from './zoo-test' | ||||
| import type { Coords2d } from '@src/lang/std/sketch' | ||||
| import { KCL_DEFAULT_LENGTH } from '@src/lib/constants' | ||||
| import { uuidv4 } from '@src/lib/utils' | ||||
|  | ||||
| import { commonPoints, getUtils, orRunWhenFullSuiteEnabled } from './test-utils' | ||||
| import { Coords2d } from 'lang/std/sketch' | ||||
| import { KCL_DEFAULT_LENGTH } from 'lib/constants' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
| import { | ||||
|   commonPoints, | ||||
|   getUtils, | ||||
|   orRunWhenFullSuiteEnabled, | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| test.describe('Testing selections', { tag: ['@skipWin'] }, () => { | ||||
|   test.setTimeout(90_000) | ||||
| @ -68,20 +72,20 @@ test.describe('Testing selections', { tag: ['@skipWin'] }, () => { | ||||
|     await u.closeDebugPanel() | ||||
|     await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       `sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${commonPoints.startAt}, sketch001)` | ||||
|       `@settings(defaultLengthUnit = in)sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${commonPoints.startAt}, sketch001)` | ||||
|     ) | ||||
|  | ||||
|     await page.waitForTimeout(100) | ||||
|     await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${commonPoints.startAt}, sketch001) | ||||
|       .toHaveText(`@settings(defaultLengthUnit = in)sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${commonPoints.startAt}, sketch001) | ||||
|     |> xLine(length = ${commonPoints.num1})`) | ||||
|  | ||||
|     await page.waitForTimeout(100) | ||||
|     await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 20) | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${ | ||||
|       .toHaveText(`@settings(defaultLengthUnit = in)sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${ | ||||
|       commonPoints.startAt | ||||
|     }, sketch001) | ||||
|     |> xLine(length = ${commonPoints.num1}) | ||||
| @ -89,7 +93,7 @@ test.describe('Testing selections', { tag: ['@skipWin'] }, () => { | ||||
|     await page.waitForTimeout(100) | ||||
|     await page.mouse.click(startXPx, 500 - PUR * 20) | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${ | ||||
|       .toHaveText(`@settings(defaultLengthUnit = in)sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${ | ||||
|       commonPoints.startAt | ||||
|     }, sketch001) | ||||
|     |> xLine(length = ${commonPoints.num1}) | ||||
| @ -260,7 +264,8 @@ test.describe('Testing selections', { tag: ['@skipWin'] }, () => { | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn(XZ) | ||||
|         `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([-79.26, 95.04], %) | ||||
|   |> line(end = [112.54, 127.64], tag = $seg02) | ||||
|   |> line(end = [170.36, -121.61], tag = $seg01) | ||||
| @ -528,7 +533,8 @@ profile001 = startProfileAt([7.49, 9.96], sketch001) | ||||
|     await page.addInitScript(async (KCL_DEFAULT_LENGTH) => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `part001 = startSketchOn(XZ) | ||||
|         `@settings(defaultLengthUnit = in) | ||||
| part001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([20, 0], %) | ||||
|   |> line(end = [7.13, 4 + 0]) | ||||
|   |> angledLine({ angle = 3 + 0, length = 3.14 + 0 }, %) | ||||
| @ -747,7 +753,8 @@ profile001 = startProfileAt([7.49, 9.96], sketch001) | ||||
|     await page.waitForTimeout(200) | ||||
|  | ||||
|     await u.removeCurrentCode() | ||||
|     await u.codeLocator.fill(`sketch001 = startSketchOn(XZ) | ||||
|     await u.codeLocator.fill(`@settings(defaultLengthUnit = in) | ||||
|     sketch001 = startSketchOn(XZ) | ||||
|     |> startProfileAt([75.8, 317.2], %) // [$startCapTag, $EndCapTag] | ||||
|     |> angledLine([0, 268.43], %, $rectangleSegmentA001) | ||||
|     |> angledLine([ | ||||
| @ -965,7 +972,8 @@ profile001 = startProfileAt([7.49, 9.96], sketch001) | ||||
|       async ({ cases }) => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `yo = 79 | ||||
|           `@settings(defaultLengthUnit = in) | ||||
|   yo = 79 | ||||
|   part001 = startSketchOn(XZ) | ||||
|     |> startProfileAt([-7.54, -26.74], %) | ||||
|     |> ${cases[0].expectedCode} | ||||
| @ -1020,7 +1028,8 @@ profile001 = startProfileAt([7.49, 9.96], sketch001) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn(XZ) | ||||
|         `@settings(defaultLengthUnit = in) | ||||
|   sketch001 = startSketchOn(XZ) | ||||
|     |> startProfileAt([-79.26, 95.04], %) | ||||
|     |> line(end = [112.54, 127.64]) | ||||
|     |> line(end = [170.36, -121.61], tag = $seg01) | ||||
| @ -1253,7 +1262,7 @@ profile001 = startProfileAt([7.49, 9.96], sketch001) | ||||
|     await page.mouse.click(700, 200) | ||||
|  | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       `sketch001 = startSketchOn(XZ)` | ||||
|       `@settings(defaultLengthUnit = in)sketch001 = startSketchOn(XZ)` | ||||
|     ) | ||||
|  | ||||
|     await page.waitForTimeout(600) | ||||
|  | ||||
