Compare commits
	
		
			467 Commits
		
	
	
		
			set-timeou
			...
			lee/native
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 17ddf52264 | |||
| d318f4ddb2 | |||
| 7823bff8d7 | |||
| 9ef114d1bd | |||
| f72eb0e8a7 | |||
| 40479d177f | |||
| b88359dee2 | |||
| f4c0347104 | |||
| ad36b5f5fa | |||
| b798cf19d3 | |||
| 7cfa897561 | |||
| 0d8804005a | |||
| cbd26d29fa | |||
| e501a542ac | |||
| 7cb4f4d101 | |||
| 1162f5f4c4 | |||
| 3975e6d8f5 | |||
| d68d7a7e00 | |||
| b135b97de6 | |||
| de5885ce0b | |||
| ad7c544754 | |||
| 4d77875bdc | |||
| 3377923dcb | |||
| c6005660c8 | |||
| 66e62c6037 | |||
| 0a4a517bb4 | |||
| 70f3ded7e2 | |||
| 095108252b | |||
| 20b1c93f12 | |||
| 3747a1b993 | |||
| 198feb7d44 | |||
| c7a8b8313e | |||
| 1576dc3256 | |||
| 341a3b7609 | |||
| ecb42b89a6 | |||
| f00ee3a44a | |||
| 900e3b96ad | |||
| 15fae05659 | |||
| 2730b6d152 | |||
| 602e7afef6 | |||
| d9bcadb062 | |||
| 19f669b94c | |||
| d9ef471385 | |||
| 39f8b306a2 | |||
| 19925d22c1 | |||
| e1af4b4219 | |||
| c699611f5b | |||
| 00ede7ec1a | |||
| f30601bd2c | |||
| cfbc77b62f | |||
| 808830d29e | |||
| e714103655 | |||
| fbcb96add5 | |||
| 7386ccf1bf | |||
| 6e73578933 | |||
| b88d5c8799 | |||
| 5430c1fa66 | |||
| c0d4bb6c9f | |||
| 25260a88c3 | |||
| b6d6f0f4c1 | |||
| b1276b2ed8 | |||
| 5f0f3f40d0 | |||
| f1ea9b6ece | |||
| b94c5be1af | |||
| 8378eb1e94 | |||
| 05f98a8c39 | |||
| 386571fa60 | |||
| b0abdf4f70 | |||
| 81e70e139f | |||
| d6bfc38d62 | |||
| ada66de92d | |||
| 8f133f9662 | |||
| b360dbb961 | |||
| eca3dc2967 | |||
| ae36ab6982 | |||
| 8cb6cf1b8a | |||
| 3c235c890a | |||
| b6dfd30840 | |||
| 65d128eecd | |||
| 77b7c602f2 | |||
| fa0e61a2be | |||
| 1cf35a611e | |||
| 952d0e4c7c | |||
| 0f85de9df8 | |||
| 0e8eed3f82 | |||
| 5b43a5075f | |||
| f5ed4e37b2 | |||
| 19c8da1a86 | |||
| a25f89aaba | |||
| aeebe5416f | |||
| 661788b8b0 | |||
| ac24563159 | |||
| d17342dfb8 | |||
| 2e93b58ae6 | |||
| 6593656b08 | |||
| 47be749ec7 | |||
| a03e7f5c41 | |||
| b78e9fa131 | |||
| c629233eaa | |||
| f640f7a5e0 | |||
| 64398381a9 | |||
| 0bc5534056 | |||
| 9fc1df7c1d | |||
| a5879ceeda | |||
| 379c30824e | |||
| a4d3263b88 | |||
| c1f661ab52 | |||
| 7d887a1497 | |||
| 4ca341e132 | |||
| c6249f36d2 | |||
| dcbe5d7f75 | |||
| 390cb2d51d | |||
| 98f7a564ea | |||
| 05f9e3c290 | |||
| 09760fc2e9 | |||
| 18ffc43e89 | |||
| de63e4f19f | |||
| b70b271e6b | |||
| 08b7cdc5f6 | |||
| 6efe6b54c0 | |||
| 69f72d62e0 | |||
| e04b09fcd8 | |||
| 4903f6b9fc | |||
| ef8149f03a | |||
| 1b75321bf1 | |||
| 3ed263da6b | |||
| d59c4a2258 | |||
| 9c8351ea40 | |||
| db98bcf2a0 | |||
| 15d96a072d | |||
| 088968c664 | |||
| 4bbf98bc34 | |||
| ca08f5b337 | |||
| a3649d09c0 | |||
| 635cb58036 | |||
| 7f050b114f | |||
| c999819450 | |||
| 82905caad6 | |||
| 519e6d74ac | |||
| edb7d68c05 | |||
| 345dd45caa | |||
| b6a5f133f3 | |||
| bc6407be6e | |||
| 038409124a | |||
| d5567f8602 | |||
| df8c17ac18 | |||
| 9d40f282a8 | |||
| a61d931826 | |||
| 418350ddbc | |||
| d43abe20d9 | |||
| 84380f3da9 | |||
| eea55ff2b1 | |||
| 10b6c1cfbc | |||
| d5570e5c62 | |||
| 0c9589f7ee | |||
| ddf66c1e0f | |||
| cf1f2bd235 | |||
| 0b5bb5f77d | |||
| 0825cb5a59 | |||
| 4ec94a721c | |||
| 16dd5aab96 | |||
| bf68a87897 | |||
| c6e97e729a | |||
| d2535bb8c2 | |||
| b01357b49e | |||
| 793e3510cc | |||
| 04ae8141c3 | |||
| 3ae5393dd7 | |||
| 38119d5a3b | |||
| b453b4b453 | |||
| 3972431cb4 | |||
| 884545fcde | |||
| 6deb242eb5 | |||
| 77fa9af71e | |||
| 6a9a0a8bd7 | |||
| 90e432b10e | |||
| 90499e086f | |||
| 8b398a8dd5 | |||
| 23d2dc8dc8 | |||
| 764a73ec8b | |||
| b69451d2fe | |||
| 173d50517c | |||
| 3b63632005 | |||
| 2bd3b06178 | |||
| 9c58cde35f | |||
| 3eb92bb0c4 | |||
| f3083eb59d | |||
| cef29013b8 | |||
| 58d1303468 | |||
| 7c11b7b739 | |||
| 9f27f3c1ce | |||
| f6cbc752d7 | |||
| 6df1ae7161 | |||
| 159ec08211 | |||
| 6aab9c6e23 | |||
| afd8daae15 | |||
| 1132779b4b | |||
| e3a65f5b3f | |||
| d53665a12a | |||
| 447f4f9f8f | |||
| 859927c06d | |||
| b88425efc8 | |||
| c37dfc61ef | |||
| b170ac739f | |||
| d712add4da | |||
| d8aad4bd4f | |||
| 1f1c44e598 | |||
| b20e685eea | |||
| 3690d986c1 | |||
| 9a7f434ede | |||
| 6afacd7427 | |||
| 957001ee88 | |||
| 8b4cc306af | |||
| 52d88171ca | |||
| 9142cf3af7 | |||
| 361500058c | |||
| 198479a71a | |||
| 905784c1e5 | |||
| c33aaad800 | |||
| d175c75780 | |||
| ba348d1222 | |||
| 1f49ddfc29 | |||
| 58659652c1 | |||
| 251971238d | |||
| 381d0b3bc8 | |||
| fa7943d06a | |||
| 7a384251d4 | |||
| 8e07ea32a6 | |||
| 23adf9d905 | |||
| 9f0ac5f6fd | |||
| 08dbd2e9c3 | |||
| 2e2ba5adbd | |||
| a21dbf1055 | |||
| 5ecb176467 | |||
| 66135636ec | |||
| 685a16545c | |||
| 9adb15ee93 | |||
| a8c4c97d79 | |||
| 39e8e1f259 | |||
| 1672c1fd1f | |||
| 6ec5881985 | |||
| 7272cc9fbd | |||
| b925ed9b65 | |||
| 0db5db2181 | |||
| 898e3db9d1 | |||
| d337ac2546 | |||
| 371d8e08f7 | |||
| 338c43a29d | |||
| 52bb5a2657 | |||
| 1b6a06d266 | |||
| c68d4778a5 | |||
| a8abea4fb5 | |||
| a0678d22a8 | |||
| acbfae2e65 | |||
| 1e1bec6a8a | |||
| 06462b5a65 | |||
| 2f292fb1be | |||
| 8184e7b376 | |||
| b1084cbf80 | |||
| 548b45905e | |||
| 141fd2f3f1 | |||
| 604d931962 | |||
| b1668410f8 | |||
| 13176cec38 | |||
| 3a59ae13b6 | |||
| 57c2481943 | |||
| a1c555c51e | |||
| 4d520541be | |||
| 82586f002b | |||
| 4bd08f7444 | |||
| 6b2603b1c4 | |||
| af49bebde3 | |||
| ca056996fd | |||
| 34163da361 | |||
| 7c22bac638 | |||
| 37a65b166b | |||
| 1189f272ba | |||
| ca5bc880dc | |||
| 828daba304 | |||
| 0b9ba55bb4 | |||
| 2d2a85ae7d | |||
| cc57a302cc | |||
| fdbfd0c4b6 | |||
| 2e419907e6 | |||
| 3d0c5c10b0 | |||
| 4d47c067b7 | |||
| 3b3b5371eb | |||
| 3ea77f8e1e | |||
| 4fa7c07e54 | |||
| c66a96a333 | |||
| 4196ff91ac | |||
| cf66b93963 | |||
| 0b0219b810 | |||
| 36c7fcf6d7 | |||
| 023c3cbb90 | |||
| 387f7e0912 | |||
| 9b55b1fd12 | |||
| 4b6662169c | |||
| d36abfcb3d | |||
| 9002ae9efb | |||
| 4deea25394 | |||
| b5940d2cb7 | |||
| 932b467c1e | |||
| 7c7f5c81c4 | |||
| 066b4f3e06 | |||
| c6067bfc7a | |||
| 2018f0d517 | |||
| 74aae3d15f | |||
| 812f419e75 | |||
| 5ec8cc69db | |||
| a5302b6e0e | |||
| 2114cc0d94 | |||
| 2471ce1aba | |||
| 35772475b9 | |||
| 86c592c0f6 | |||
| 0e98973cfa | |||
| 7dd16fe6de | |||
| 478b636049 | |||
| c779311a56 | |||
| ca02ec1151 | |||
| b271d5060e | |||
| 19f11fe55a | |||
| f6f1574982 | |||
| 6dc4fbc808 | |||
| 8843d02380 | |||
| 3578ec07e6 | |||
| db35f73e41 | |||
| 5cfc2b7941 | |||
| 318e4a0cc7 | |||
| 1e23be8f08 | |||
| ef547e7db8 | |||
| 71b48bbd89 | |||
| c825eac27e | |||
| 82e8a491c4 | |||
| 93e806fc99 | |||
| f1a14f1e3d | |||
| 57c01ec3a2 | |||
| ce951d7c12 | |||
| 0aa2a6cee7 | |||
| ba8f5d9785 | |||
| 50a133b2fa | |||
| 3b15bc12f7 | |||
| 8eedee328b | |||
| 49b321feb5 | |||
| 35b5ad7d9b | |||
| 8fad9ef3c2 | |||
| b257b202c3 | |||
| c6af62797d | |||
| 16a9acad56 | |||
| 8a80a88ad3 | |||
| 71d1bb70ef | |||
| 4853872614 | |||
| 1ca5204a1a | |||
| 7baed0b5bd | |||
| e4969857bd | |||
| 9b7cc7afa4 | |||
| 714917429e | |||
| 5af9c6b22d | |||
| 396a994fe6 | |||
| 872da51da5 | |||
| 05cd8cfec9 | |||
| 2a02f6e039 | |||
| 5b90686e5e | |||
| 298269d117 | |||
| b379f6518f | |||
| 6b22c8789d | |||
| cb4683e70b | |||
| 0a020d9959 | |||
| 7aae3dccdc | |||
| 818bf96d0b | |||
| 03bc2eaf22 | |||
| 8ad1476c13 | |||
| 6c15a743a2 | |||
| d0930477ad | |||
| e5e30d231b | |||
| 9822576077 | |||
| 629f326f4c | |||
| 89b880d9ae | |||
| f6de0de1bf | |||
| 65ebb86b67 | |||
| cce8274902 | |||
| c515bef8e4 | |||
| b17e61d963 | |||
| d31d07d9c8 | |||
| 7aa2d63c21 | |||
| e1081b0ee6 | |||
| 59223279b7 | |||
| 8a4e717565 | |||
| 80b542ca18 | |||
| e4bfc863ea | |||
| 77ef255de4 | |||
| 64c3841079 | |||
| c7bb6bc845 | |||
| 1af8a8c64f | |||
| eb4776826b | |||
| f3dd0469d5 | |||
| deea74754d | |||
| 3fd798c704 | |||
| cc9eaf2991 | |||
| 6f24031220 | |||
| 672bcd297f | |||
| 3bc182fe16 | |||
| 589cd39eec | |||
| 63feebef5c | |||
| 65037abd9a | |||
| 97bc339a62 | |||
| 4e9a6375a5 | |||
| 3d19dfb800 | |||
| d2a7b84292 | |||
| 9e02bab155 | |||
| 7352de5a70 | |||
| 9797d0cb81 | |||
| 83907fa9db | |||
| a367be4e2b | |||
| 056fa00adc | |||
| 4759fb2e6f | |||
| 45f497d9cd | |||
| dc61bdebdf | |||
| 61943055e5 | |||
| 416fe0f644 | |||
| 708465d818 | |||
| e706fb02d6 | |||
| 1bf7daa474 | |||
| ffc47f8f40 | |||
| 768aaa84f6 | |||
| f3a700eec8 | |||
| c853637a9a | |||
| 9af30d9ef6 | |||
| 6164714a6b | |||
| 64ceb98eba | |||
| 2cbf260900 | |||
| cfaaedf602 | |||
| 12b3717eb5 | |||
| 0bc685b0c4 | |||
| 9ee032771a | |||
| c307ddd1b1 | |||
| a30818ff2b | |||
| 53e763d938 | |||
| 8f74cd1d0c | |||
| c271942897 | |||
| a03d09b41d | |||
| 2971b7752b | |||
| 70e99eb00b | |||
| 5c66af59d2 | |||
| 6dda6daeef | |||
| b5387f1220 | |||
| fd5921b366 | |||
| 716ad938fc | |||
| 40136eb392 | |||
| 8d2b89fcd1 | |||
| ad9fba3390 | |||
| 911c43af50 | |||
| ab4e04f6c2 | |||
| 94aef05f74 | |||
| d820cf2446 | |||
| 0c724c4971 | |||
| b54ac4a694 | |||
| 27227092b1 | |||
| 04e1b92a5b | |||
| 0553cd4621 | |||
| 61a0c88af4 | |||
| d5b0544437 | |||
| 6cc8af5c23 | |||
| 888104080e | |||
| b6769889e3 | |||
| a32258dac4 | |||
| 18dbbad244 | 
							
								
								
									
										3
									
								
								.codespellrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,3 @@ | |||||||
|  | [codespell] | ||||||
|  | ignore-words-list: crate,everytime,inout,co-ordinate,ot,nwo | ||||||
|  | skip: **/target,node_modules,build,**/Cargo.lock | ||||||
| @ -1,6 +1,6 @@ | |||||||
| VITE_KC_API_WS_MODELING_URL=wss://api.dev.kittycad.io/ws/modeling/commands | VITE_KC_API_WS_MODELING_URL=wss://api.dev.zoo.dev/ws/modeling/commands | ||||||
| VITE_KC_API_BASE_URL=https://api.dev.kittycad.io | VITE_KC_API_BASE_URL=https://api.dev.zoo.dev | ||||||
| VITE_KC_SITE_BASE_URL=https://dev.kittycad.io | VITE_KC_SITE_BASE_URL=https://dev.zoo.dev | ||||||
| VITE_KC_SKIP_AUTH=false | VITE_KC_SKIP_AUTH=false | ||||||
| VITE_KC_CONNECTION_TIMEOUT_MS=5000 | VITE_KC_CONNECTION_TIMEOUT_MS=5000 | ||||||
| VITE_KC_SENTRY_DSN= | VITE_KC_SENTRY_DSN= | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| VITE_KC_API_WS_MODELING_URL=wss://api.kittycad.io/ws/modeling/commands | VITE_KC_API_WS_MODELING_URL=wss://api.zoo.dev/ws/modeling/commands | ||||||
| VITE_KC_API_BASE_URL=https://api.kittycad.io | VITE_KC_API_BASE_URL=https://api.zoo.dev | ||||||
| VITE_KC_SITE_BASE_URL=https://kittycad.io | VITE_KC_SITE_BASE_URL=https://zoo.dev | ||||||
| VITE_KC_SKIP_AUTH=false | VITE_KC_SKIP_AUTH=false | ||||||
| VITE_KC_CONNECTION_TIMEOUT_MS=15000 | VITE_KC_CONNECTION_TIMEOUT_MS=15000 | ||||||
| VITE_KC_SENTRY_DSN=https://a814f2f66734989a90367f48feee28ca@o1042111.ingest.sentry.io/4505789425844224 | VITE_KC_SENTRY_DSN=https://a814f2f66734989a90367f48feee28ca@o1042111.ingest.sentry.io/4505789425844224 | ||||||
|  | |||||||
| @ -1 +1,2 @@ | |||||||
| src/wasm-lib/* | src/wasm-lib/* | ||||||
|  | *.typegen.ts | ||||||
|  | |||||||
							
								
								
									
										18
									
								
								.eslintrc
									
									
									
									
									
								
							
							
						
						| @ -1,4 +1,8 @@ | |||||||
| { | { | ||||||
|  |     "parser": "@typescript-eslint/parser", | ||||||
|  |     "parserOptions": { | ||||||
|  |       "project": "./tsconfig.json" | ||||||
|  |     }, | ||||||
|     "plugins": [ |     "plugins": [ | ||||||
|       "css-modules" |       "css-modules" | ||||||
|     ], |     ], | ||||||
| @ -11,6 +15,16 @@ | |||||||
|       "semi": [ |       "semi": [ | ||||||
|         "error", |         "error", | ||||||
|         "never" |         "never" | ||||||
|       ] |       ], | ||||||
|     } |       "react-hooks/exhaustive-deps": "off", | ||||||
|  |     }, | ||||||
|  |     "overrides": [ | ||||||
|  |       { | ||||||
|  |         "files": ["e2e/**/*.ts"], // Update the pattern based on your file structure | ||||||
|  |         "rules": { | ||||||
|  |           "@typescript-eslint/no-floating-promises": "warn", | ||||||
|  |           "testing-library/prefer-screen-queries": "off" | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     ] | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										3
									
								
								.github/workflows/cargo-build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -15,6 +15,9 @@ on: | |||||||
|       - '**/Cargo.lock' |       - '**/Cargo.lock' | ||||||
|       - '**/rust-toolchain.toml' |       - '**/rust-toolchain.toml' | ||||||
|       - .github/workflows/cargo-build.yml |       - .github/workflows/cargo-build.yml | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.head_ref || github.run_id }} | ||||||
|  |   cancel-in-progress: true | ||||||
| name: cargo build | name: cargo build | ||||||
| jobs: | jobs: | ||||||
|   cargobuild: |   cargobuild: | ||||||
|  | |||||||
							
								
								
									
										14
									
								
								.github/workflows/cargo-clippy.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -15,6 +15,9 @@ on: | |||||||
|       - '**/rust-toolchain.toml' |       - '**/rust-toolchain.toml' | ||||||
|       - '**.rs' |       - '**.rs' | ||||||
|       - .github/workflows/cargo-build.yml |       - .github/workflows/cargo-build.yml | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.head_ref || github.run_id }} | ||||||
|  |   cancel-in-progress: true | ||||||
| name: cargo clippy | name: cargo clippy | ||||||
| jobs: | jobs: | ||||||
|   cargoclippy: |   cargoclippy: | ||||||
| @ -40,17 +43,6 @@ jobs: | |||||||
|       - name: Rust Cache |       - name: Rust Cache | ||||||
|         uses: Swatinem/rust-cache@v2.6.1 |         uses: Swatinem/rust-cache@v2.6.1 | ||||||
|  |  | ||||||
|       - name: Install ffmpeg |  | ||||||
|         run: | |  | ||||||
|           sudo apt update |  | ||||||
|           sudo apt install \ |  | ||||||
|             ffmpeg \ |  | ||||||
|             libavformat-dev \ |  | ||||||
|             libavutil-dev \ |  | ||||||
|             libclang-dev \ |  | ||||||
|             libswscale-dev \ |  | ||||||
|             --no-install-recommends |  | ||||||
|  |  | ||||||
|       - name: Run clippy |       - name: Run clippy | ||||||
|         run: | |         run: | | ||||||
|           cd "${{ matrix.dir }}" |           cd "${{ matrix.dir }}" | ||||||
|  | |||||||
							
								
								
									
										3
									
								
								.github/workflows/cargo-criterion.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -17,6 +17,9 @@ on: | |||||||
|       - .github/workflows/cargo-criterion.yml |       - .github/workflows/cargo-criterion.yml | ||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
| permissions: read-all | permissions: read-all | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.head_ref || github.run_id }} | ||||||
|  |   cancel-in-progress: true | ||||||
| name: cargo criterion | name: cargo criterion | ||||||
| jobs: | jobs: | ||||||
|   cargocriterion: |   cargocriterion: | ||||||
|  | |||||||
							
								
								
									
										3
									
								
								.github/workflows/cargo-fmt.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -18,6 +18,9 @@ on: | |||||||
| permissions: | permissions: | ||||||
|   packages: read |   packages: read | ||||||
|   contents: read |   contents: read | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.head_ref || github.run_id }} | ||||||
|  |   cancel-in-progress: true | ||||||
| name: cargo fmt | name: cargo fmt | ||||||
| jobs: | jobs: | ||||||
|   cargofmt: |   cargofmt: | ||||||
|  | |||||||
							
								
								
									
										14
									
								
								.github/workflows/cargo-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -17,6 +17,9 @@ on: | |||||||
|       - .github/workflows/cargo-test.yml |       - .github/workflows/cargo-test.yml | ||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
| permissions: read-all | permissions: read-all | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.head_ref || github.run_id }} | ||||||
|  |   cancel-in-progress: true | ||||||
| name: cargo test | name: cargo test | ||||||
| jobs: | jobs: | ||||||
|   cargotest: |   cargotest: | ||||||
| @ -41,16 +44,6 @@ jobs: | |||||||
|       - uses: taiki-e/install-action@nextest |       - uses: taiki-e/install-action@nextest | ||||||
|       - name: Rust Cache |       - name: Rust Cache | ||||||
|         uses: Swatinem/rust-cache@v2.6.1 |         uses: Swatinem/rust-cache@v2.6.1 | ||||||
|       - name: Install ffmpeg |  | ||||||
|         run: | |  | ||||||
|           sudo apt update |  | ||||||
|           sudo apt install \ |  | ||||||
|             ffmpeg \ |  | ||||||
|             libavformat-dev \ |  | ||||||
|             libavutil-dev \ |  | ||||||
|             libclang-dev \ |  | ||||||
|             libswscale-dev \ |  | ||||||
|             --no-install-recommends |  | ||||||
|       - name: cargo test |       - name: cargo test | ||||||
|         shell: bash |         shell: bash | ||||||
|         run: |- |         run: |- | ||||||
| @ -58,4 +51,5 @@ jobs: | |||||||
|           cargo nextest run --workspace --no-fail-fast -P ci |           cargo nextest run --workspace --no-fail-fast -P ci | ||||||
|         env: |         env: | ||||||
|           KITTYCAD_API_TOKEN: ${{secrets.KITTYCAD_API_TOKEN}} |           KITTYCAD_API_TOKEN: ${{secrets.KITTYCAD_API_TOKEN}} | ||||||
|  |           RUST_MIN_STACK: 10485760000 | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										253
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -7,13 +7,24 @@ on: | |||||||
|       - main |       - main | ||||||
|   release: |   release: | ||||||
|     types: [published] |     types: [published] | ||||||
|  |   schedule: | ||||||
|  |     - cron: '0 4 * * *' | ||||||
|  |   # Daily at 04:00 AM UTC | ||||||
|  |   # Will checkout the last commit from the default branch (main as of 2023-10-04) | ||||||
|  |  | ||||||
|  | env: | ||||||
|  |   BUILD_RELEASE: ${{ github.event_name == 'release' || github.event_name == 'schedule' || github.event_name == 'pull_request' && contains(github.event.pull_request.title, 'Cut release v') }} | ||||||
|  |  | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.head_ref || github.run_id }} | ||||||
|  |   cancel-in-progress: true | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   check-format: |   check-format: | ||||||
|     runs-on: 'ubuntu-20.04' |     runs-on: 'ubuntu-latest' | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v4 | ||||||
|       - uses: actions/setup-node@v3 |       - uses: actions/setup-node@v4 | ||||||
|         with: |         with: | ||||||
|           node-version-file: '.nvmrc' |           node-version-file: '.nvmrc' | ||||||
|           cache: 'yarn' |           cache: 'yarn' | ||||||
| @ -21,11 +32,11 @@ jobs: | |||||||
|       - run: yarn fmt-check |       - run: yarn fmt-check | ||||||
|  |  | ||||||
|   check-types: |   check-types: | ||||||
|     runs-on: ubuntu-20.04 |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v4 | ||||||
|       - uses: actions/setup-node@v3 |       - uses: actions/setup-node@v4 | ||||||
|         with: |         with: | ||||||
|           node-version-file: '.nvmrc' |           node-version-file: '.nvmrc' | ||||||
|           cache: 'yarn' |           cache: 'yarn' | ||||||
| @ -35,16 +46,30 @@ jobs: | |||||||
|           workspaces: './src/wasm-lib' |           workspaces: './src/wasm-lib' | ||||||
|  |  | ||||||
|       - run: yarn build:wasm |       - run: yarn build:wasm | ||||||
|  |       - run: yarn xstate:typegen | ||||||
|       - run: yarn tsc |       - run: yarn tsc | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   check-typos:  | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - name: Checkout | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|  |       - name: Set up Python | ||||||
|  |         uses: actions/setup-python@v5 | ||||||
|  |       - name: Install codespell | ||||||
|  |         run: | | ||||||
|  |             python -m pip install codespell | ||||||
|  |       - name: Run codespell | ||||||
|  |         run: codespell --config .codespellrc # Edit this file to tweak the typo list and other configuration. | ||||||
|  |  | ||||||
|  |  | ||||||
|   build-test-web: |   build-test-web: | ||||||
|     runs-on: ubuntu-20.04 |     runs-on: ubuntu-latest | ||||||
|     outputs: |  | ||||||
|       version: ${{ steps.export_version.outputs.version }} |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|       - uses: actions/setup-node@v3 |       - uses: actions/setup-node@v4 | ||||||
|         with: |         with: | ||||||
|           node-version-file: '.nvmrc' |           node-version-file: '.nvmrc' | ||||||
|           cache: 'yarn' |           cache: 'yarn' | ||||||
| @ -61,38 +86,82 @@ jobs: | |||||||
|  |  | ||||||
|       - run: yarn test:nowatch |       - run: yarn test:nowatch | ||||||
|  |  | ||||||
|       - run: yarn test:cov |  | ||||||
|  |   prepare-json-files: | ||||||
|  |     runs-on: ubuntu-latest  # seperate job on Ubuntu for easy string manipulations (compared to Windows) | ||||||
|  |     outputs: | ||||||
|  |       version: ${{ steps.export_version.outputs.version }} | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|  |       - uses: actions/setup-node@v4 | ||||||
|  |         with: | ||||||
|  |           node-version-file: '.nvmrc' | ||||||
|  |           cache: 'yarn' | ||||||
|  |            | ||||||
|  |       - name: Set nightly version | ||||||
|  |         if: github.event_name == 'schedule' | ||||||
|  |         run: | | ||||||
|  |           VERSION=$(date +'%-y.%-m.%-d') yarn bump-jsons | ||||||
|  |           echo "$(jq --arg url 'https://dl.zoo.dev/releases/modeling-app/nightly/last_update.json' \ | ||||||
|  |             '.tauri.updater.endpoints[]=$url' src-tauri/tauri.release.conf.json --indent 2)" > src-tauri/tauri.release.conf.json | ||||||
|  |  | ||||||
|  |       - uses: actions/upload-artifact@v3 | ||||||
|  |         if: github.event_name == 'schedule' | ||||||
|  |         with: | ||||||
|  |           path: | | ||||||
|  |             package.json | ||||||
|  |             src-tauri/tauri.conf.json | ||||||
|  |             src-tauri/tauri.release.conf.json | ||||||
|  |  | ||||||
|       - id: export_version |       - id: export_version | ||||||
|         run: echo "version=`cat package.json | jq -r '.version'`" >> "$GITHUB_OUTPUT" |         run: echo "version=`cat package.json | jq -r '.version'`" >> "$GITHUB_OUTPUT" | ||||||
|  |  | ||||||
|   build-apps: |  | ||||||
|     needs: [check-format, build-test-web, check-types] |   build-test-apps: | ||||||
|  |     needs: [prepare-json-files] | ||||||
|     runs-on: ${{ matrix.os }} |     runs-on: ${{ matrix.os }} | ||||||
|     strategy: |     strategy: | ||||||
|  |       fail-fast: false | ||||||
|       matrix: |       matrix: | ||||||
|         os: [macos-latest, ubuntu-20.04, windows-latest] |         os: [macos-14, ubuntu-latest, windows-latest] | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|       - name: install ubuntu system dependencies |       - uses: actions/download-artifact@v3 | ||||||
|         if: matrix.os == 'ubuntu-20.04' |  | ||||||
|  |       - name: Copy updated .json files | ||||||
|  |         if: github.event_name == 'schedule' | ||||||
|         run: | |         run: | | ||||||
|           sudo apt-get update |           ls -l artifact | ||||||
|           sudo apt-get install -y libgtk-3-dev libwebkit2gtk-4.0-dev libayatana-appindicator3-dev librsvg2-dev |           cp artifact/package.json package.json | ||||||
|  |           cp artifact/src-tauri/tauri.conf.json src-tauri/tauri.conf.json | ||||||
|  |           cp artifact/src-tauri/tauri.release.conf.json src-tauri/tauri.release.conf.json  | ||||||
|  |  | ||||||
|  |       - name: Install ubuntu system dependencies | ||||||
|  |         if: matrix.os == 'ubuntu-latest' | ||||||
|  |         run: > | ||||||
|  |           sudo apt-get update && | ||||||
|  |           sudo apt-get install -y | ||||||
|  |           libgtk-3-dev | ||||||
|  |           libgtksourceview-3.0-dev | ||||||
|  |           webkit2gtk-4.0 | ||||||
|  |           libappindicator3-dev | ||||||
|  |           webkit2gtk-driver | ||||||
|  |           xvfb | ||||||
|  |  | ||||||
|       - name: Sync node version and setup cache |       - name: Sync node version and setup cache | ||||||
|         uses: actions/setup-node@v3 |         uses: actions/setup-node@v4 | ||||||
|         with: |         with: | ||||||
|           node-version-file: '.nvmrc' |           node-version-file: '.nvmrc' | ||||||
|           cache: 'yarn' # Set this to npm, yarn or pnpm. |           cache: 'yarn' # Set this to npm, yarn or pnpm. | ||||||
|  |  | ||||||
|       - run: yarn install |       - run: yarn install | ||||||
|  |  | ||||||
|       - name: Rust setup |       - name: Setup Rust | ||||||
|         uses: dtolnay/rust-toolchain@stable |         uses: dtolnay/rust-toolchain@stable | ||||||
|  |  | ||||||
|       - name: Rust cache |       - name: Setup Rust cache | ||||||
|         uses: swatinem/rust-cache@v2 |         uses: swatinem/rust-cache@v2 | ||||||
|         with: |         with: | ||||||
|           workspaces: './src-tauri -> target' |           workspaces: './src-tauri -> target' | ||||||
| @ -101,24 +170,30 @@ jobs: | |||||||
|         with: |         with: | ||||||
|           workspaces: './src/wasm-lib' |           workspaces: './src/wasm-lib' | ||||||
|  |  | ||||||
|       - name: wasm prep |       - name: Run build:wasm manually | ||||||
|         shell: bash |         shell: bash | ||||||
|  |         env: | ||||||
|  |           MODE: ${{ env.BUILD_RELEASE == 'true' && '--release' || '--debug' }} | ||||||
|         run: | |         run: | | ||||||
|           mkdir src/wasm-lib/pkg; cd src/wasm-lib |           mkdir src/wasm-lib/pkg; cd src/wasm-lib | ||||||
|           npx wasm-pack build --target web --out-dir pkg |           echo "building with ${{ env.MODE }}" | ||||||
|  |           npx wasm-pack build --target web --out-dir pkg ${{ env.MODE }} | ||||||
|           cd ../../ |           cd ../../ | ||||||
|           cp src/wasm-lib/pkg/wasm_lib_bg.wasm public |           cp src/wasm-lib/pkg/wasm_lib_bg.wasm public | ||||||
|  |  | ||||||
|  |       - name: Run vite build (build:both) | ||||||
|  |         run: yarn vite build --mode ${{ env.BUILD_RELEASE == 'true' && 'production' || 'development' }} | ||||||
|  |  | ||||||
|       - name: Fix format |       - name: Fix format | ||||||
|         run: yarn fmt |         run: yarn fmt | ||||||
|  |  | ||||||
|       - name: install apple silicon target mac |       - name: Install x86 target for Universal builds (MacOS only) | ||||||
|         if: matrix.os == 'macos-latest' |         if: matrix.os == 'macos-14' | ||||||
|         run: | |         run: | | ||||||
|           rustup target add aarch64-apple-darwin |           rustup target add x86_64-apple-darwin | ||||||
|  |  | ||||||
|       - name: Prepare Windows certificate and variables |       - name: Prepare certificate and variables (Windows only) | ||||||
|         if: matrix.os == 'windows-latest' |         if: ${{ matrix.os == 'windows-latest' && env.BUILD_RELEASE == 'true' }} | ||||||
|         run: | |         run: | | ||||||
|           echo "${{secrets.SM_CLIENT_CERT_FILE_B64 }}" | base64 --decode > /d/Certificate_pkcs12.p12 |           echo "${{secrets.SM_CLIENT_CERT_FILE_B64 }}" | base64 --decode > /d/Certificate_pkcs12.p12 | ||||||
|           cat /d/Certificate_pkcs12.p12 |           cat /d/Certificate_pkcs12.p12 | ||||||
| @ -132,8 +207,8 @@ jobs: | |||||||
|           echo "C:\Program Files\DigiCert\DigiCert One Signing Manager Tools" >> $GITHUB_PATH |           echo "C:\Program Files\DigiCert\DigiCert One Signing Manager Tools" >> $GITHUB_PATH | ||||||
|         shell: bash |         shell: bash | ||||||
|  |  | ||||||
|       - name: Setup Windows certicate with SSM KSP |       - name: Setup certicate with SSM KSP (Windows only) | ||||||
|         if: matrix.os == 'windows-latest' |         if: ${{ matrix.os == 'windows-latest' && env.BUILD_RELEASE == 'true' }} | ||||||
|         run: | |         run: | | ||||||
|           curl -X GET  https://one.digicert.com/signingmanager/api-ui/v1/releases/smtools-windows-x64.msi/download -H "x-api-key:%SM_API_KEY%" -o smtools-windows-x64.msi |           curl -X GET  https://one.digicert.com/signingmanager/api-ui/v1/releases/smtools-windows-x64.msi/download -H "x-api-key:%SM_API_KEY%" -o smtools-windows-x64.msi | ||||||
|           msiexec /i smtools-windows-x64.msi /quiet /qn |           msiexec /i smtools-windows-x64.msi /quiet /qn | ||||||
| @ -143,8 +218,17 @@ jobs: | |||||||
|           smksp_cert_sync.exe |           smksp_cert_sync.exe | ||||||
|         shell: cmd |         shell: cmd | ||||||
|  |  | ||||||
|       - name: Build and sign the app for the current platform |       - name: Build the app (debug) | ||||||
|         uses: tauri-apps/tauri-action@v0 |         uses: tauri-apps/tauri-action@v0 | ||||||
|  |         if: ${{ env.BUILD_RELEASE == 'false' }} | ||||||
|  |         with: | ||||||
|  |           includeRelease: false | ||||||
|  |           includeDebug: true | ||||||
|  |           args: ${{ matrix.os == 'macos-14' && '--target universal-apple-darwin' || '' }} | ||||||
|  |  | ||||||
|  |       - name: Build the app (release) and sign | ||||||
|  |         uses: tauri-apps/tauri-action@v0 | ||||||
|  |         if: ${{ env.BUILD_RELEASE == 'true' }} | ||||||
|         env: |         env: | ||||||
|           TAURI_PRIVATE_KEY: ${{ secrets.TAURI_PRIVATE_KEY }} |           TAURI_PRIVATE_KEY: ${{ secrets.TAURI_PRIVATE_KEY }} | ||||||
|           TAURI_KEY_PASSWORD: ${{ secrets.TAURI_KEY_PASSWORD }} |           TAURI_KEY_PASSWORD: ${{ secrets.TAURI_KEY_PASSWORD }} | ||||||
| @ -153,41 +237,59 @@ jobs: | |||||||
|           APPLE_SIGNING_IDENTITY: ${{ secrets.APPLE_SIGNING_IDENTITY }} |           APPLE_SIGNING_IDENTITY: ${{ secrets.APPLE_SIGNING_IDENTITY }} | ||||||
|           APPLE_ID: ${{ secrets.APPLE_ID }} |           APPLE_ID: ${{ secrets.APPLE_ID }} | ||||||
|           APPLE_PASSWORD: ${{ secrets.APPLE_PASSWORD }} |           APPLE_PASSWORD: ${{ secrets.APPLE_PASSWORD }} | ||||||
|  |           APPLE_TEAM_ID: ${{ secrets.APPLE_TEAM_ID }} | ||||||
|  |           TAURI_CONF_ARGS: "--config ${{ matrix.os == 'windows-latest' && 'src-tauri\\tauri.release.conf.json' || 'src-tauri/tauri.release.conf.json' }}" | ||||||
|         with: |         with: | ||||||
|           args: ${{ matrix.os == 'macos-latest' && '--target universal-apple-darwin' || '' }} |           args: "${{ matrix.os == 'macos-14' && '--target universal-apple-darwin' || '' }} ${{ env.TAURI_CONF_ARGS }}" | ||||||
|  |  | ||||||
|       - uses: actions/upload-artifact@v3 |       - uses: actions/upload-artifact@v3 | ||||||
|  |         if: matrix.os != 'ubuntu-latest' | ||||||
|  |         env: | ||||||
|  |           PREFIX: ${{ matrix.os == 'macos-14' && 'src-tauri/target/universal-apple-darwin' || 'src-tauri/target' }} | ||||||
|  |           MODE: ${{ env.BUILD_RELEASE == 'true' && 'release' || 'debug' }} | ||||||
|         with: |         with: | ||||||
|           path: ${{ matrix.os == 'macos-latest' && 'src-tauri/target/universal-apple-darwin/release/bundle/*/*' || 'src-tauri/target/release/bundle/*/*' }} |           path: "${{ env.PREFIX }}/${{ env.MODE }}/bundle/*/*" | ||||||
|  |  | ||||||
|  |       - name: Run e2e tests (linux only) | ||||||
|  |         if: matrix.os == 'ubuntu-latest' | ||||||
|  |         run: | | ||||||
|  |           cargo install tauri-driver@0.1.3 | ||||||
|  |           source .env.${{ env.BUILD_RELEASE == 'true' && 'production' || 'development' }} | ||||||
|  |           export VITE_KC_API_BASE_URL | ||||||
|  |           xvfb-run yarn test:e2e:tauri | ||||||
|  |         env: | ||||||
|  |           E2E_APPLICATION: "./src-tauri/target/${{ env.BUILD_RELEASE == 'true' && 'release' || 'debug' }}/zoo-modeling-app" | ||||||
|  |           KITTYCAD_API_TOKEN: ${{ env.BUILD_RELEASE == 'true' && secrets.KITTYCAD_API_TOKEN || secrets.KITTYCAD_API_TOKEN_DEV }} | ||||||
|  |  | ||||||
|  |  | ||||||
|   publish-apps-release: |   publish-apps-release: | ||||||
|     runs-on: ubuntu-20.04 |     runs-on: ubuntu-latest | ||||||
|     if: github.event_name == 'release' |     if: ${{ github.event_name == 'release' || github.event_name == 'schedule' }} | ||||||
|     needs: [build-test-web, build-apps] |     needs: [check-format, check-types, check-typos, build-test-web, prepare-json-files, build-test-apps] | ||||||
|     env: |     env: | ||||||
|       VERSION_NO_V: ${{ needs.build-test-web.outputs.version }} |       VERSION_NO_V: ${{ needs.prepare-json-files.outputs.version }} | ||||||
|       PUB_DATE: ${{ github.event.release.created_at }} |       VERSION: ${{ github.event_name == 'release' && format('v{0}', needs.prepare-json-files.outputs.version) || needs.prepare-json-files.outputs.version }} | ||||||
|       NOTES: ${{ github.event.release.body }} |       PUB_DATE: ${{ github.event_name == 'release' && github.event.release.created_at || github.event.repository.updated_at }} | ||||||
|  |       NOTES: ${{ github.event_name == 'release' && github.event.release.body || format('Nightly build, commit {0}', github.sha) }} | ||||||
|  |       BUCKET_DIR: ${{ github.event_name == 'release' && 'dl.kittycad.io/releases/modeling-app' || 'dl.kittycad.io/releases/modeling-app/nightly' }} | ||||||
|  |       WEBSITE_DIR: ${{ github.event_name == 'release' && 'dl.zoo.dev/releases/modeling-app' || 'dl.zoo.dev/releases/modeling-app/nightly' }} | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/download-artifact@v3 |       - uses: actions/download-artifact@v3 | ||||||
|  |  | ||||||
|       - name: Generate the update static endpoint |       - name: Generate the update static endpoint | ||||||
|         run: | |         run: | | ||||||
|           ls -l artifact/*/*itty* |           ls -l artifact/*/*oo* | ||||||
|           DARWIN_SIG=`cat artifact/macos/*.app.tar.gz.sig` |           DARWIN_SIG=`cat artifact/macos/*.app.tar.gz.sig` | ||||||
|           LINUX_SIG=`cat artifact/appimage/*.AppImage.tar.gz.sig` |  | ||||||
|           WINDOWS_SIG=`cat artifact/msi/*.msi.zip.sig` |           WINDOWS_SIG=`cat artifact/msi/*.msi.zip.sig` | ||||||
|           RELEASE_DIR=https://dl.kittycad.io/releases/modeling-app/v${VERSION_NO_V} |           RELEASE_DIR=https://${WEBSITE_DIR}/${VERSION} | ||||||
|           jq --null-input \ |           jq --null-input \ | ||||||
|             --arg version "v${VERSION_NO_V}" \ |             --arg version "${VERSION}" \ | ||||||
|             --arg pub_date "${PUB_DATE}" \ |             --arg pub_date "${PUB_DATE}" \ | ||||||
|             --arg notes "${NOTES}" \ |             --arg notes "${NOTES}" \ | ||||||
|             --arg darwin_sig "$DARWIN_SIG" \ |             --arg darwin_sig "$DARWIN_SIG" \ | ||||||
|             --arg darwin_url "$RELEASE_DIR/macos/KittyCAD%20Modeling.app.tar.gz" \ |             --arg darwin_url "$RELEASE_DIR/macos/Zoo%20Modeling%20App.app.tar.gz" \ | ||||||
|             --arg linux_sig "$LINUX_SIG" \ |  | ||||||
|             --arg linux_url "$RELEASE_DIR/appimage/kittycad-modeling_${VERSION_NO_V}_amd64.AppImage.tar.gz" \ |  | ||||||
|             --arg windows_sig "$WINDOWS_SIG" \ |             --arg windows_sig "$WINDOWS_SIG" \ | ||||||
|             --arg windows_url "$RELEASE_DIR/msi/KittyCAD%20Modeling_${VERSION_NO_V}_x64_en-US.msi.zip" \ |             --arg windows_url "$RELEASE_DIR/msi/Zoo%20Modeling%20App_${VERSION_NO_V}_x64_en-US.msi.zip" \ | ||||||
|             '{ |             '{ | ||||||
|               "version": $version, |               "version": $version, | ||||||
|               "pub_date": $pub_date, |               "pub_date": $pub_date, | ||||||
| @ -201,10 +303,6 @@ jobs: | |||||||
|                   "signature": $darwin_sig, |                   "signature": $darwin_sig, | ||||||
|                   "url": $darwin_url |                   "url": $darwin_url | ||||||
|                 }, |                 }, | ||||||
|                 "linux-x86_64": { |  | ||||||
|                   "signature": $linux_sig, |  | ||||||
|                   "url": $linux_url |  | ||||||
|                 }, |  | ||||||
|                 "windows-x86_64": { |                 "windows-x86_64": { | ||||||
|                   "signature": $windows_sig, |                   "signature": $windows_sig, | ||||||
|                   "url": $windows_url |                   "url": $windows_url | ||||||
| @ -215,14 +313,13 @@ jobs: | |||||||
|  |  | ||||||
|       - name: Generate the download static endpoint |       - name: Generate the download static endpoint | ||||||
|         run: | |         run: | | ||||||
|           RELEASE_DIR=https://dl.kittycad.io/releases/modeling-app/v${VERSION_NO_V} |           RELEASE_DIR=https://${WEBSITE_DIR}/${VERSION} | ||||||
|           jq --null-input \ |           jq --null-input \ | ||||||
|             --arg version "v${VERSION_NO_V}" \ |             --arg version "${VERSION}" \ | ||||||
|             --arg pub_date "${PUB_DATE}" \ |             --arg pub_date "${PUB_DATE}" \ | ||||||
|             --arg notes "${NOTES}" \ |             --arg notes "${NOTES}" \ | ||||||
|             --arg darwin_url "$RELEASE_DIR/dmg/KittyCAD%20Modeling_${VERSION_NO_V}_universal.dmg" \ |             --arg darwin_url "$RELEASE_DIR/dmg/Zoo%20Modeling%20App_${VERSION_NO_V}_universal.dmg" \ | ||||||
|             --arg linux_url "$RELEASE_DIR/appimage/kittycad-modeling_${VERSION_NO_V}_amd64.AppImage" \ |             --arg windows_url "$RELEASE_DIR/msi/Zoo%20Modeling%20App_${VERSION_NO_V}_x64_en-US.msi" \ | ||||||
|             --arg windows_url "$RELEASE_DIR/msi/KittyCAD%20Modeling_${VERSION_NO_V}_x64_en-US.msi" \ |  | ||||||
|             '{ |             '{ | ||||||
|               "version": $version, |               "version": $version, | ||||||
|               "pub_date": $pub_date, |               "pub_date": $pub_date, | ||||||
| @ -231,9 +328,6 @@ jobs: | |||||||
|                 "dmg-universal": { |                 "dmg-universal": { | ||||||
|                   "url": $darwin_url |                   "url": $darwin_url | ||||||
|                 }, |                 }, | ||||||
|                 "appimage-x86_64": { |  | ||||||
|                   "url": $linux_url |  | ||||||
|                 }, |  | ||||||
|                 "msi-x86_64": { |                 "msi-x86_64": { | ||||||
|                   "url": $windows_url |                   "url": $windows_url | ||||||
|                 } |                 } | ||||||
| @ -242,36 +336,61 @@ jobs: | |||||||
|             cat last_download.json |             cat last_download.json | ||||||
|  |  | ||||||
|       - name: Authenticate to Google Cloud |       - name: Authenticate to Google Cloud | ||||||
|         uses: 'google-github-actions/auth@v1.1.1' |         uses: 'google-github-actions/auth@v2.1.1' | ||||||
|         with: |         with: | ||||||
|           credentials_json: '${{ secrets.GOOGLE_CLOUD_DL_SA }}' |           credentials_json: '${{ secrets.GOOGLE_CLOUD_DL_SA }}' | ||||||
|  |  | ||||||
|       - name: Set up Google Cloud SDK |       - name: Set up Google Cloud SDK | ||||||
|         uses: google-github-actions/setup-gcloud@v1.1.1 |         uses: google-github-actions/setup-gcloud@v2.1.0 | ||||||
|         with: |         with: | ||||||
|           project_id: kittycadapi |           project_id: kittycadapi | ||||||
|  |  | ||||||
|       - name: Upload release files to public bucket |       - name: Upload release files to public bucket | ||||||
|         uses: google-github-actions/upload-cloud-storage@v1.0.3 |         uses: google-github-actions/upload-cloud-storage@v2.1.0 | ||||||
|         with: |         with: | ||||||
|           path: artifact |           path: artifact | ||||||
|           glob: '*/*itty*' |           glob: '*/Zoo*' | ||||||
|           parent: false |           parent: false | ||||||
|           destination: dl.kittycad.io/releases/modeling-app/v${{ env.VERSION_NO_V }} |           destination: ${{ env.BUCKET_DIR }}/${{ env.VERSION }} | ||||||
|  |  | ||||||
|       - name: Upload update endpoint to public bucket |       - name: Upload update endpoint to public bucket | ||||||
|         uses: google-github-actions/upload-cloud-storage@v1.0.3 |         uses: google-github-actions/upload-cloud-storage@v2.1.0 | ||||||
|         with: |         with: | ||||||
|           path: last_update.json |           path: last_update.json | ||||||
|           destination: dl.kittycad.io/releases/modeling-app |           destination: ${{ env.BUCKET_DIR }} | ||||||
|  |  | ||||||
|       - name: Upload download endpoint to public bucket |       - name: Upload download endpoint to public bucket | ||||||
|         uses: google-github-actions/upload-cloud-storage@v1.0.3 |         uses: google-github-actions/upload-cloud-storage@v2.1.0 | ||||||
|         with: |         with: | ||||||
|           path: last_download.json |           path: last_download.json | ||||||
|           destination: dl.kittycad.io/releases/modeling-app |           destination: ${{ env.BUCKET_DIR }} | ||||||
|  |  | ||||||
|       - name: Upload release files to Github |       - name: Upload release files to Github | ||||||
|  |         if: ${{ github.event_name == 'release' }} | ||||||
|         uses: softprops/action-gh-release@v1 |         uses: softprops/action-gh-release@v1 | ||||||
|         with: |         with: | ||||||
|           files: artifact/*/*itty* |           files: 'artifact/*/Zoo*' | ||||||
|  |            | ||||||
|  |   announce_release: | ||||||
|  |     needs: [publish-apps-release] | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - name: Check out code | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|  |            | ||||||
|  |       - name: Set up Python | ||||||
|  |         uses: actions/setup-python@v5 | ||||||
|  |         with: | ||||||
|  |           python-version: '3.x' | ||||||
|  |    | ||||||
|  |       - name: Install dependencies | ||||||
|  |         run: | | ||||||
|  |           python -m pip install --upgrade pip | ||||||
|  |           pip install requests | ||||||
|  |    | ||||||
|  |       - name: Announce Release | ||||||
|  |         env: | ||||||
|  |           DISCORD_WEBHOOK_URL: ${{ secrets.DISCORD_WEBHOOK_URL }} | ||||||
|  |           RELEASE_VERSION: ${{ github.event.release.tag_name }} | ||||||
|  |           RELEASE_BODY: ${{ github.event.release.body}} | ||||||
|  |         run: python public/announce_release.py | ||||||
							
								
								
									
										116
									
								
								.github/workflows/playwright.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,116 @@ | |||||||
|  | name: Playwright Tests | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: [ main ] | ||||||
|  |   pull_request: | ||||||
|  |     branches: [ main ] | ||||||
|  | jobs: | ||||||
|  |   playwright-ubuntu: | ||||||
|  |     timeout-minutes: 60 | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v4 | ||||||
|  |     - uses: actions/setup-node@v4 | ||||||
|  |       with: | ||||||
|  |         node-version-file: '.nvmrc' | ||||||
|  |         cache: 'yarn' | ||||||
|  |     - uses: KittyCAD/action-install-cli@v0.2.16 | ||||||
|  |     - name: Install dependencies | ||||||
|  |       run: yarn | ||||||
|  |     - name: Install Playwright Browsers | ||||||
|  |       run: yarn playwright install --with-deps | ||||||
|  |     - name: Setup Rust | ||||||
|  |       uses: dtolnay/rust-toolchain@stable | ||||||
|  |     - name: Cache wasm | ||||||
|  |       uses: Swatinem/rust-cache@v2 | ||||||
|  |       with: | ||||||
|  |         workspaces: './src/wasm-lib' | ||||||
|  |     - name: build wasm | ||||||
|  |       run: yarn build:wasm | ||||||
|  |     - name: build web | ||||||
|  |       run: yarn build:local | ||||||
|  |     - name: Run ubuntu/chrome snapshots | ||||||
|  |       run: yarn playwright test --project="Google Chrome" --update-snapshots e2e/playwright/snapshot-tests.spec.ts | ||||||
|  |       env: | ||||||
|  |         CI: true | ||||||
|  |         token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||||
|  |         snapshottoken: ${{ secrets.KITTYCAD_API_TOKEN }} | ||||||
|  |     - uses: actions/upload-artifact@v3 | ||||||
|  |       if: always() | ||||||
|  |       with: | ||||||
|  |         name: playwright-report | ||||||
|  |         path: playwright-report/ | ||||||
|  |         retention-days: 30 | ||||||
|  |     - name: check for changes | ||||||
|  |       id: git-check | ||||||
|  |       run: | | ||||||
|  |           git add . | ||||||
|  |           if git status | grep -q "Changes to be committed" | ||||||
|  |           then | ||||||
|  |             echo "::set-output name=modified::true" | ||||||
|  |           else | ||||||
|  |             echo "::set-output name=modified::false" | ||||||
|  |           fi | ||||||
|  |     - name: Commit changes, if any | ||||||
|  |       if: steps.git-check.outputs.modified == 'true' | ||||||
|  |       run: | | ||||||
|  |         git add . | ||||||
|  |         git config --local user.email "github-actions[bot]@users.noreply.github.com" | ||||||
|  |         git config --local user.name "github-actions[bot]" | ||||||
|  |         git remote set-url origin https://${{ github.actor }}:${{ secrets.GITHUB_TOKEN }}@github.com/${{ github.repository }}.git | ||||||
|  |         git fetch origin | ||||||
|  |         echo ${{ github.head_ref }} | ||||||
|  |         git checkout ${{ github.head_ref }} | ||||||
|  |         # TODO when safari works on ubuntu remove the os part of the commit message | ||||||
|  |         git commit -am "A snapshot a day keeps the bugs away! 📷🐛 (OS: ubuntu)" || true | ||||||
|  |         git push | ||||||
|  |         git push origin ${{ github.head_ref }} | ||||||
|  |     - name: Run ubuntu/chrome flow | ||||||
|  |       run: yarn playwright test --project="Google Chrome" e2e/playwright/flow-tests.spec.ts | ||||||
|  |       env: | ||||||
|  |         CI: true | ||||||
|  |         token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||||
|  |     - uses: actions/upload-artifact@v3 | ||||||
|  |       if: always() | ||||||
|  |       with: | ||||||
|  |         name: playwright-report | ||||||
|  |         path: playwright-report/ | ||||||
|  |         retention-days: 30 | ||||||
|  |  | ||||||
|  |   playwright-macos: | ||||||
|  |     timeout-minutes: 60 | ||||||
|  |     runs-on: macos-14 | ||||||
|  |     needs: playwright-ubuntu | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v4 | ||||||
|  |     - uses: actions/setup-node@v4 | ||||||
|  |       with: | ||||||
|  |         node-version-file: '.nvmrc' | ||||||
|  |         cache: 'yarn' | ||||||
|  |     - name: Install dependencies | ||||||
|  |       run: yarn | ||||||
|  |     - name: Install Playwright Browsers | ||||||
|  |       run: yarn playwright install --with-deps | ||||||
|  |     - name: Setup Rust | ||||||
|  |       uses: dtolnay/rust-toolchain@stable | ||||||
|  |     - name: Cache wasm | ||||||
|  |       uses: Swatinem/rust-cache@v2 | ||||||
|  |       with: | ||||||
|  |         workspaces: './src/wasm-lib' | ||||||
|  |     - name: build wasm | ||||||
|  |       run: yarn build:wasm | ||||||
|  |     - name: build web | ||||||
|  |       run: yarn build:local | ||||||
|  |     - name: Run macos/safari flow | ||||||
|  |       # safari doesn't work on Ubuntu because of the same reason tauri doesn't (webRTC issues) | ||||||
|  |       # TODO remove this and the matrix and run all tests on ubuntu when this is fixed | ||||||
|  |       run: yarn playwright test --project="webkit" e2e/playwright/flow-tests.spec.ts | ||||||
|  |       env: | ||||||
|  |         CI: true | ||||||
|  |         token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||||
|  |     - uses: actions/upload-artifact@v3 | ||||||
|  |       if: always() | ||||||
|  |       with: | ||||||
|  |         name: playwright-report | ||||||
|  |         path: playwright-report/ | ||||||
|  |         retention-days: 30 | ||||||
							
								
								
									
										24
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -22,7 +22,10 @@ npm-debug.log* | |||||||
| yarn-debug.log* | yarn-debug.log* | ||||||
| yarn-error.log* | yarn-error.log* | ||||||
|  |  | ||||||
|  | .idea | ||||||
|  | .vscode | ||||||
| src/wasm-lib/.idea | src/wasm-lib/.idea | ||||||
|  | src/wasm-lib/.vscode | ||||||
|  |  | ||||||
| # rust | # rust | ||||||
| src/wasm-lib/target | src/wasm-lib/target | ||||||
| @ -30,3 +33,24 @@ src/wasm-lib/bindings | |||||||
| src/wasm-lib/kcl/bindings | src/wasm-lib/kcl/bindings | ||||||
| public/wasm_lib_bg.wasm | public/wasm_lib_bg.wasm | ||||||
| src/wasm-lib/lcov.info | src/wasm-lib/lcov.info | ||||||
|  |  | ||||||
|  | e2e/playwright/playwright-secrets.env | ||||||
|  | e2e/playwright/temp1.png | ||||||
|  | e2e/playwright/temp2.png | ||||||
|  | # exports from snapshot-tests.spec.ts | ||||||
|  | e2e/playwright/export-snapshots/*.ply | ||||||
|  | e2e/playwright/export-snapshots/*.obj | ||||||
|  | e2e/playwright/export-snapshots/*.step | ||||||
|  | e2e/playwright/export-snapshots/*.stl | ||||||
|  | e2e/playwright/export-snapshots/*binary.gltf | ||||||
|  | e2e/playwright/export-snapshots/*embedded.gltf | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /test-results/ | ||||||
|  | /playwright-report/ | ||||||
|  | /blob-report/ | ||||||
|  | /playwright/.cache/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## generated files | ||||||
|  | src/**/*.typegen.ts | ||||||
|  | |||||||
| @ -7,3 +7,7 @@ coverage | |||||||
| target | target | ||||||
| src/wasm-lib/pkg | src/wasm-lib/pkg | ||||||
| src/wasm-lib/kcl/bindings | src/wasm-lib/kcl/bindings | ||||||
|  | e2e/playwright/export-snapshots | ||||||
|  |  | ||||||
|  | # XState generated files | ||||||
|  | src/machines/**.typegen.ts | ||||||
|  | |||||||
							
								
								
									
										21
									
								
								LICENSE
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,21 @@ | |||||||
|  | The MIT License (MIT) | ||||||
|  |  | ||||||
|  | Copyright (c) 2023 The Zoo Authors | ||||||
|  |  | ||||||
|  | Permission is hereby granted, free of charge, to any person obtaining a copy | ||||||
|  | of this software and associated documentation files (the "Software"), to deal | ||||||
|  | in the Software without restriction, including without limitation the rights | ||||||
|  | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||||
|  | copies of the Software, and to permit persons to whom the Software is | ||||||
|  | furnished to do so, subject to the following conditions: | ||||||
|  |  | ||||||
|  | The above copyright notice and this permission notice shall be included in all | ||||||
|  | copies or substantial portions of the Software. | ||||||
|  |  | ||||||
|  | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||||
|  | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||||
|  | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||||
|  | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||||
|  | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||||
|  | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||||
|  | SOFTWARE. | ||||||
							
								
								
									
										160
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @ -1,17 +1,17 @@ | |||||||
|  |  | ||||||
|  |  | ||||||
| ## KittyCAD Modeling App | ## Zoo Modeling App | ||||||
|  |  | ||||||
| live at [app.kittycad.io](https://app.kittycad.io/) | live at [app.zoo.dev](https://app.zoo.dev/) | ||||||
|  |  | ||||||
| A CAD application from the future, brought to you by the [KittyCAD team](https://kittycad.io). | A CAD application from the future, brought to you by the [Zoo team](https://zoo.dev). | ||||||
|  |  | ||||||
| The KittyCAD modeling app is our take on what a modern modelling experience can be. It is applying several lessons learned in the decades since most major CAD tools came into existence: | Modeling App is our take on what a modern modelling experience can be. It is applying several lessons learned in the decades since most major CAD tools came into existence: | ||||||
|  |  | ||||||
| - All artifacts—including parts and assemblies—should be represented as human-readable code. At the end of the day, your CAD project should be "plain text" | - All artifacts—including parts and assemblies—should be represented as human-readable code. At the end of the day, your CAD project should be "plain text" | ||||||
|   - This makes version control—which is a solved problem in software engineering—trivial for CAD |   - This makes version control—which is a solved problem in software engineering—trivial for CAD | ||||||
| - All GUI (or point-and-click) interactions should be actions performed on this code representation under the hood | - All GUI (or point-and-click) interactions should be actions performed on this code representation under the hood | ||||||
|   - This unlocks a hybrid approach to modeling. Whether you point-and-click as you always have or you write your own KCL code, you are performing the same action in KittyCAD Modeling App |   - This unlocks a hybrid approach to modeling. Whether you point-and-click as you always have or you write your own KCL code, you are performing the same action in Modeling App | ||||||
| - Everything graphics _has_ to be built for the GPU | - Everything graphics _has_ to be built for the GPU | ||||||
|   - Most CAD applications have had to retrofit support for GPUs, but our geometry engine is made for GPUs (primarily Nvidia's Vulkan), getting the order of magnitude rendering performance boost with it |   - Most CAD applications have had to retrofit support for GPUs, but our geometry engine is made for GPUs (primarily Nvidia's Vulkan), getting the order of magnitude rendering performance boost with it | ||||||
| - Make the resource-intensive pieces of an application auto-scaling | - Make the resource-intensive pieces of an application auto-scaling | ||||||
| @ -19,9 +19,9 @@ The KittyCAD modeling app is our take on what a modern modelling experience can | |||||||
|  |  | ||||||
| We are excited about what a small team of people could build in a short time with our API. We welcome you to try our API, build your own applications, or contribute to ours! | We are excited about what a small team of people could build in a short time with our API. We welcome you to try our API, build your own applications, or contribute to ours! | ||||||
|  |  | ||||||
| KittyCAD Modeling App is a _hybrid_ user interface for CAD modeling. You can point-and-click to design parts (and soon assemblies), but everything you make is really just [`kcl` code](https://github.com/KittyCAD/kcl-experiments) under the hood. All of your CAD models can be checked into source control such as GitHub and responsibly versioned, rolled back, and more. | Modeling App is a _hybrid_ user interface for CAD modeling. You can point-and-click to design parts (and soon assemblies), but everything you make is really just [`kcl` code](https://github.com/KittyCAD/kcl-experiments) under the hood. All of your CAD models can be checked into source control such as GitHub and responsibly versioned, rolled back, and more. | ||||||
|  |  | ||||||
| The 3D view in KittyCAD Modeling App is just a video stream from our hosted geometry engine. The app sends new modeling commands to the engine via WebSockets, which returns back video frames of the view within the engine. | The 3D view in Modeling App is just a video stream from our hosted geometry engine. The app sends new modeling commands to the engine via WebSockets, which returns back video frames of the view within the engine. | ||||||
|  |  | ||||||
| ## Tools | ## Tools | ||||||
|  |  | ||||||
| @ -29,6 +29,7 @@ The 3D view in KittyCAD Modeling App is just a video stream from our hosted geom | |||||||
|   - [React](https://react.dev/) |   - [React](https://react.dev/) | ||||||
|   - [Headless UI](https://headlessui.com/) |   - [Headless UI](https://headlessui.com/) | ||||||
|   - [TailwindCSS](https://tailwindcss.com/) |   - [TailwindCSS](https://tailwindcss.com/) | ||||||
|  |   - [XState](https://xstate.js.org/) | ||||||
| - Networking | - Networking | ||||||
|   - WebSockets (via [KittyCAD TS client](https://github.com/KittyCAD/kittycad.ts)) |   - WebSockets (via [KittyCAD TS client](https://github.com/KittyCAD/kittycad.ts)) | ||||||
| - Code Editor | - Code Editor | ||||||
| @ -47,7 +48,7 @@ We recommend downloading the latest application binary from [our Releases page]( | |||||||
|  |  | ||||||
| ## Running a development build | ## Running a development build | ||||||
|  |  | ||||||
| First, [install Rust via `rustup`](https://www.rust-lang.org/tools/install). This project uses a lot of Rust compiled to [WASM](https://webassembly.org/) within it. Then, run: | First, [install Rust via `rustup`](https://www.rust-lang.org/tools/install). This project uses a lot of Rust compiled to [WASM](https://webassembly.org/) within it. We always use the latest stable version of Rust, so you may need to run `rustup update stable`. Then, run: | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| yarn install | yarn install | ||||||
| @ -56,7 +57,7 @@ yarn install | |||||||
| followed by: | followed by: | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| yarn build:wasm | yarn build:wasm-dev | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| That will build the WASM binary and put in the `public` dir (though gitignored) | That will build the WASM binary and put in the `public` dir (though gitignored) | ||||||
| @ -88,15 +89,21 @@ yarn test | |||||||
|  |  | ||||||
| Which will run our suite of [Vitest unit](https://vitest.dev/) and [React Testing Library E2E](https://testing-library.com/docs/react-testing-library/intro/) tests, in interactive mode by default. | Which will run our suite of [Vitest unit](https://vitest.dev/) and [React Testing Library E2E](https://testing-library.com/docs/react-testing-library/intro/) tests, in interactive mode by default. | ||||||
|  |  | ||||||
|  | For running the rust (not tauri rust though) only, you can | ||||||
|  | ```bash | ||||||
|  | cd src/wasm-lib | ||||||
|  | cargo test | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ## Tauri | ## Tauri | ||||||
|  |  | ||||||
| To spin up up tauri dev, `yarn install` and `yarn build:wasm` need to have been done before hand then | To spin up up tauri dev, `yarn install` and `yarn build:wasm-dev` need to have been done before hand then | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| yarn tauri dev | yarn tauri dev | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| Will spin up the web app before opening up the tauri dev desktop app. Note that it's probably a good idea to close the browser tab that gets opened since at the time of writting they can conflict. | Will spin up the web app before opening up the tauri dev desktop app. Note that it's probably a good idea to close the browser tab that gets opened since at the time of writing they can conflict. | ||||||
|  |  | ||||||
| The dev instance automatically opens up the browser devtools which can be disabled by [commenting it out](https://github.com/KittyCAD/modeling-app/blob/main/src-tauri/src/main.rs#L92.) | The dev instance automatically opens up the browser devtools which can be disabled by [commenting it out](https://github.com/KittyCAD/modeling-app/blob/main/src-tauri/src/main.rs#L92.) | ||||||
|  |  | ||||||
| @ -123,13 +130,29 @@ Before you submit a contribution PR to this repo, please ensure that: | |||||||
|  |  | ||||||
| ## Release a new version | ## Release a new version | ||||||
|  |  | ||||||
| 1. Bump the versions in the .json files by creating a `Bump to v{x}.{y}.{z}` PR, committing the changes from | 1. Bump the versions in the .json files by creating a `Cut release v{x}.{y}.{z}` PR, committing the changes from | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| VERSION=x.y.z yarn run bump-jsons | VERSION=x.y.z yarn run bump-jsons | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| The PR may serve as a place to discuss the human-readable changelog and extra QA. | Alternatively you can try the experimental `make-release.sh` bash script that will create the branch with the updated json files for you. | ||||||
|  | run `./make-release.sh` for a patch update | ||||||
|  | run `./make-release.sh "minor"` for minor | ||||||
|  | run `./make-release.sh "major"` for major | ||||||
|  |  | ||||||
|  | The PR may serve as a place to discuss the human-readable changelog and extra QA. A quick way of getting PR's merged since the last bump is to [use this PR filter](https://github.com/KittyCAD/modeling-app/pulls?q=is%3Apr+sort%3Aupdated-desc+is%3Amerged+), open up the browser console and past in the following | ||||||
|  |  | ||||||
|  | ```typescript | ||||||
|  | console.log( | ||||||
|  |   '- ' + | ||||||
|  |     Array.from( | ||||||
|  |       document.querySelectorAll('[data-hovercard-type="pull_request"]') | ||||||
|  |     ).map((a) => `[${a.innerText}](${a.href})`).join(` | ||||||
|  | - `) | ||||||
|  | ) | ||||||
|  | ``` | ||||||
|  | grab the md list and delete any that are older than the last bump | ||||||
|  |  | ||||||
| 2. Merge the PR | 2. Merge the PR | ||||||
|  |  | ||||||
| @ -157,3 +180,112 @@ $ cargo +nightly fuzz run parser | |||||||
|  |  | ||||||
| For more information on fuzzing you can check out | For more information on fuzzing you can check out | ||||||
| [this guide](https://rust-fuzz.github.io/book/cargo-fuzz.html). | [this guide](https://rust-fuzz.github.io/book/cargo-fuzz.html). | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ### Playwright | ||||||
|  |  | ||||||
|  | First time running plawright locally, you'll need to add the secrets file | ||||||
|  | ```bash | ||||||
|  | touch ./e2e/playwright/playwright-secrets.env | ||||||
|  | printf 'token="your-token"\nsnapshottoken="your-snapshot-token"' > ./e2e/playwright/playwright-secrets.env | ||||||
|  | ``` | ||||||
|  | then replace "your-token" with a dev token from dev.zoo.dev/account/api-tokens | ||||||
|  |  | ||||||
|  | then: | ||||||
|  | run playwright | ||||||
|  | ``` | ||||||
|  | yarn playwright test | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | run a specific test suite | ||||||
|  | ``` | ||||||
|  | yarn playwright test src/e2e-tests/example.spec.ts | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | run a specific test change the test from `test('...` to `test.only('...` | ||||||
|  | (note if you commit this, the tests will instantly fail without running any of the tests) | ||||||
|  |  | ||||||
|  | run headed | ||||||
|  | ``` | ||||||
|  | yarn playwright test --headed | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | run with step through debugger | ||||||
|  | ``` | ||||||
|  | PWDEBUG=1 yarn playwright test | ||||||
|  | ``` | ||||||
|  | However, if you want a debugger I recommend using VSCode and the `playwright` extension, as the above command is a cruder debugger that steps into every function call which is annoying. | ||||||
|  | With the extension you can set a breakpoint after `waitForDefaultPlanesVisibilityChange` in order to skip app loading, then the vscode debugger's "step over" is much better for being able to stay at the right level of abstraction as you debug the code. | ||||||
|  |  | ||||||
|  | If you want to limit to a single browser use `--project="webkit"` or `firefox`, `Google Chrome` | ||||||
|  | Or comment out browsers in `playwright.config.ts`. | ||||||
|  |  | ||||||
|  | note chromium has encoder compat issues which is why were testing against the branded 'Google Chrome' | ||||||
|  |  | ||||||
|  | You may consider using the VSCode extension, it's useful for running individual threads, but some some reason the "record a test" is locked to chromium with we can't use. A work around is to us the CI `yarn playwright codegen -b wk --load-storage ./store localhost:3000` | ||||||
|  |  | ||||||
|  | <details> | ||||||
|  | <summary> | ||||||
|  |  | ||||||
|  | Where `./store` should look like this | ||||||
|  |  | ||||||
|  | </summary> | ||||||
|  |  | ||||||
|  | ```JSON | ||||||
|  | { | ||||||
|  |   "cookies": [], | ||||||
|  |   "origins": [ | ||||||
|  |     { | ||||||
|  |       "origin": "http://localhost:3000", | ||||||
|  |       "localStorage": [ | ||||||
|  |         { | ||||||
|  |           "name": "store", | ||||||
|  |           "value": "{\"state\":{\"openPanes\":[\"code\"]},\"version\":0}" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           "name": "persistCode", | ||||||
|  |           "value": "" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           "name": "TOKEN_PERSIST_KEY", | ||||||
|  |           "value": "your-token" | ||||||
|  |         } | ||||||
|  |       ] | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | </details> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | However because much of our tests involve clicking in the stream at specific locations, it's code-gen looks `await page.locator('video').click();` when really we need to use a pixel coord, so I think it's of limited use. | ||||||
|  |  | ||||||
|  | #### Some notes on CI | ||||||
|  |  | ||||||
|  | The tests are broken into snapshot tests and non-snapshot tests, and they run in that order, they automatically commit new snap shots, so if you see an image commit check it was an intended change. If we have non-determinism in the snapshots such that they are always committing new images, hopefully this annoyance makes us fix them asap, if you notice this happening let Kurt know. But for the odd occasion  `git reset --hard HEAD~ && git push -f` is your friend. | ||||||
|  |  | ||||||
|  | How to interpret failing playwright tests? | ||||||
|  | If your tests fail, click through to the action and see that the tests failed on a line that includes `await page.getByTestId('loading').waitFor({ state: 'detached' })`, this means the test fail because the stream never started. It's you choice if you want to re-run the test, or ignore the failure. | ||||||
|  |  | ||||||
|  | We run on ubuntu and macos, because safari doesn't work on linux because of the dreaded "no RTCPeerConnection variable" error. But linux runs first and then macos for the same reason that we limit the number of parallel tests to 1 because we limit stream connections per user, so tests would start failing we if let them run together. | ||||||
|  |  | ||||||
|  | If something fails on CI you can download the artifact, unzip it and then open `playwright-report/data/<UUID>.zip` with https://trace.playwright.dev/ to see what happened. | ||||||
|  |  | ||||||
|  | #### Getting started writing a playwright test in our app | ||||||
|  |  | ||||||
|  | Besides following the instructions above and using the playwright docs, our app is weird because of the whole stream thing, which means our testing is weird. Because we've just figured out this stuff and therefore docs might go stale quick here's a 15min vid/tutorial | ||||||
|  |  | ||||||
|  | https://github.com/KittyCAD/modeling-app/assets/29681384/6f5e8e85-1003-4fd9-be7f-f36ce833942d | ||||||
|  |  | ||||||
|  | <details> | ||||||
|  |  | ||||||
|  | <summary> | ||||||
|  | Ps for the debug panel, the following JSON is useful for snapping the camera | ||||||
|  | </summary> | ||||||
|  |  | ||||||
|  | ```JSON | ||||||
|  | {"type":"modeling_cmd_req","cmd_id":"054e5472-e5e9-4071-92d7-1ce3bac61956","cmd":{"type":"default_camera_look_at","center":{"x":15,"y":0,"z":0},"up":{"x":0,"y":0,"z":1},"vantage":{"x":30,"y":30,"z":30}}} | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | </details> | ||||||
|  | |||||||
							
								
								
									
										
											BIN
										
									
								
								app-icon.png
									
									
									
									
									
								
							
							
						
						| Before Width: | Height: | Size: 207 KiB After Width: | Height: | Size: 120 KiB | 
							
								
								
									
										16
									
								
								docs/kcl/KNOWN-ISSUES.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,16 @@ | |||||||
|  | # Known Issues | ||||||
|  |  | ||||||
|  | The following are bugs that are not in modeling-app or kcl itself. These bugs | ||||||
|  | once fixed in engine will just start working here with no language changes. | ||||||
|  |  | ||||||
|  | - **Sketch on Face**: If your sketch is outside the edges of the face (on which you | ||||||
|  |     are sketching) you will get multiple models returned instead of one single | ||||||
|  |     model for that sketch and its underlying 3D object. | ||||||
|  |  | ||||||
|  | - **Patterns**: If you try and pass a pattern to `hole` currently only the first | ||||||
|  |     item in the pattern is being subtracted. This is an engine bug that is being | ||||||
|  |     worked on. | ||||||
|  |    | ||||||
|  | - **Import**: Right now you can import a file, even if that file has brep data | ||||||
|  |     you cannot edit it. You also cannot move or transform the imported objects at | ||||||
|  |    all. In the future, after v1, the engine will account for this. | ||||||
							
								
								
									
										33783
									
								
								docs/kcl/std.json
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										7365
									
								
								docs/kcl/std.md
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/gltf-binary.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 193 KiB | 
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/gltf-embedded.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 193 KiB | 
							
								
								
									
										3056
									
								
								e2e/playwright/export-snapshots/gltf-standard-2.gltf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/gltf-standard.gltf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/obj-.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 259 KiB | 
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/ply-ascii.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 220 KiB | 
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/ply-binary_big_endian.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 220 KiB | 
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/ply-binary_little_endian.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 220 KiB | 
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/step-.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 193 KiB | 
							
								
								
									
										494
									
								
								e2e/playwright/export-snapshots/step-.step
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,494 @@ | |||||||
|  | ISO-10303-21; | ||||||
|  | HEADER; | ||||||
|  | FILE_DESCRIPTION((('zoo.dev export')), '2;1'); | ||||||
|  | FILE_NAME('dump.step', '1970-01-01T00:00:00.0+00:00', ('Author unknown'), ('Organization unknown'), 'zoo.dev beta', 'zoo.dev', 'Authorization unknown'); | ||||||
|  | FILE_SCHEMA(('AP203_CONFIGURATION_CONTROLLED_3D_DESIGN_OF_MECHANICAL_PARTS_AND_ASSEMBLIES_MIM_LF')); | ||||||
|  | ENDSEC; | ||||||
|  | DATA; | ||||||
|  | #1 = ( | ||||||
|  |   LENGTH_UNIT() | ||||||
|  |   NAMED_UNIT(*) | ||||||
|  |   SI_UNIT($, .METRE.) | ||||||
|  | ); | ||||||
|  | #2 = UNCERTAINTY_MEASURE_WITH_UNIT(0.00001, #1, 'DISTANCE_ACCURACY_VALUE', $); | ||||||
|  | #3 = ( | ||||||
|  |   GEOMETRIC_REPRESENTATION_CONTEXT(3) | ||||||
|  |   GLOBAL_UNCERTAINTY_ASSIGNED_CONTEXT((#2)) | ||||||
|  |   GLOBAL_UNIT_ASSIGNED_CONTEXT((#1)) | ||||||
|  |   REPRESENTATION_CONTEXT('', '3D') | ||||||
|  | ); | ||||||
|  | #4 = CARTESIAN_POINT('NONE', (0, 0, -0)); | ||||||
|  | #5 = VERTEX_POINT('NONE', #4); | ||||||
|  | #6 = CARTESIAN_POINT('NONE', (0, -0.64516, -0)); | ||||||
|  | #7 = VERTEX_POINT('NONE', #6); | ||||||
|  | #8 = CARTESIAN_POINT('NONE', (0, -0.64516, 2.58064)); | ||||||
|  | #9 = VERTEX_POINT('NONE', #8); | ||||||
|  | #10 = CARTESIAN_POINT('NONE', (0, 0, 2.58064)); | ||||||
|  | #11 = VERTEX_POINT('NONE', #10); | ||||||
|  | #12 = CARTESIAN_POINT('NONE', (1.996782122555674, -0.64516, -0)); | ||||||
|  | #13 = VERTEX_POINT('NONE', #12); | ||||||
|  | #14 = CARTESIAN_POINT('NONE', (1.996782122555674, -0.64516, 2.58064)); | ||||||
|  | #15 = VERTEX_POINT('NONE', #14); | ||||||
|  | #16 = CARTESIAN_POINT('NONE', (3.839550058615159, -1.9354799999999992, -0)); | ||||||
|  | #17 = VERTEX_POINT('NONE', #16); | ||||||
|  | #18 = CARTESIAN_POINT('NONE', (3.839550058615159, -1.9354799999999992, 2.58064)); | ||||||
|  | #19 = VERTEX_POINT('NONE', #18); | ||||||
|  | #20 = CARTESIAN_POINT('NONE', (6.12902, -1.93548, -0)); | ||||||
|  | #21 = VERTEX_POINT('NONE', #20); | ||||||
|  | #22 = CARTESIAN_POINT('NONE', (6.12902, -1.93548, 2.58064)); | ||||||
|  | #23 = VERTEX_POINT('NONE', #22); | ||||||
|  | #24 = CARTESIAN_POINT('NONE', (6.12902, -1.6129, -0)); | ||||||
|  | #25 = VERTEX_POINT('NONE', #24); | ||||||
|  | #26 = CARTESIAN_POINT('NONE', (6.12902, -1.6129, 2.58064)); | ||||||
|  | #27 = VERTEX_POINT('NONE', #26); | ||||||
|  | #28 = CARTESIAN_POINT('NONE', (3.9412591419317424, -1.6129, -0)); | ||||||
|  | #29 = VERTEX_POINT('NONE', #28); | ||||||
|  | #30 = CARTESIAN_POINT('NONE', (3.9412591419317424, -1.6129, 2.58064)); | ||||||
|  | #31 = VERTEX_POINT('NONE', #30); | ||||||
|  | #32 = CARTESIAN_POINT('NONE', (1.6377992218573856, 0, -0)); | ||||||
|  | #33 = VERTEX_POINT('NONE', #32); | ||||||
|  | #34 = CARTESIAN_POINT('NONE', (1.6377992218573856, 0, 2.58064)); | ||||||
|  | #35 = VERTEX_POINT('NONE', #34); | ||||||
|  | #36 = CARTESIAN_POINT('NONE', (3.7131243491113075, 0.9677400000000002, -0)); | ||||||
|  | #37 = VERTEX_POINT('NONE', #36); | ||||||
|  | #38 = CARTESIAN_POINT('NONE', (3.7131243491113075, 0.9677400000000002, 2.58064)); | ||||||
|  | #39 = VERTEX_POINT('NONE', #38); | ||||||
|  | #40 = CARTESIAN_POINT('NONE', (6.12902, 0.9677399999999998, -0)); | ||||||
|  | #41 = VERTEX_POINT('NONE', #40); | ||||||
|  | #42 = CARTESIAN_POINT('NONE', (6.12902, 0.9677399999999998, 2.58064)); | ||||||
|  | #43 = VERTEX_POINT('NONE', #42); | ||||||
|  | #44 = CARTESIAN_POINT('NONE', (6.12902, 1.29032, -0)); | ||||||
|  | #45 = VERTEX_POINT('NONE', #44); | ||||||
|  | #46 = CARTESIAN_POINT('NONE', (6.12902, 1.29032, 2.58064)); | ||||||
|  | #47 = VERTEX_POINT('NONE', #46); | ||||||
|  | #48 = CARTESIAN_POINT('NONE', (3.6416100848359463, 1.29032, -0)); | ||||||
|  | #49 = VERTEX_POINT('NONE', #48); | ||||||
|  | #50 = CARTESIAN_POINT('NONE', (3.6416100848359463, 1.29032, 2.58064)); | ||||||
|  | #51 = VERTEX_POINT('NONE', #50); | ||||||
|  | #52 = CARTESIAN_POINT('NONE', (2.2580599999999995, 0.64516, -0)); | ||||||
|  | #53 = VERTEX_POINT('NONE', #52); | ||||||
|  | #54 = CARTESIAN_POINT('NONE', (2.2580599999999995, 0.64516, 2.58064)); | ||||||
|  | #55 = VERTEX_POINT('NONE', #54); | ||||||
|  | #56 = CARTESIAN_POINT('NONE', (0, 0.64516, -0)); | ||||||
|  | #57 = VERTEX_POINT('NONE', #56); | ||||||
|  | #58 = CARTESIAN_POINT('NONE', (0, 0.64516, 2.58064)); | ||||||
|  | #59 = VERTEX_POINT('NONE', #58); | ||||||
|  | #60 = DIRECTION('NONE', (0, -1, 0)); | ||||||
|  | #61 = VECTOR('NONE', #60, 1); | ||||||
|  | #62 = CARTESIAN_POINT('NONE', (0, 0, -0)); | ||||||
|  | #63 = LINE('NONE', #62, #61); | ||||||
|  | #64 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #65 = VECTOR('NONE', #64, 1); | ||||||
|  | #66 = CARTESIAN_POINT('NONE', (0, -0.64516, -0)); | ||||||
|  | #67 = LINE('NONE', #66, #65); | ||||||
|  | #68 = DIRECTION('NONE', (0, -1, 0)); | ||||||
|  | #69 = VECTOR('NONE', #68, 1); | ||||||
|  | #70 = CARTESIAN_POINT('NONE', (0, 0, 2.58064)); | ||||||
|  | #71 = LINE('NONE', #70, #69); | ||||||
|  | #72 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #73 = VECTOR('NONE', #72, 1); | ||||||
|  | #74 = CARTESIAN_POINT('NONE', (0, 0, -0)); | ||||||
|  | #75 = LINE('NONE', #74, #73); | ||||||
|  | #76 = DIRECTION('NONE', (1, 0, 0)); | ||||||
|  | #77 = VECTOR('NONE', #76, 1); | ||||||
|  | #78 = CARTESIAN_POINT('NONE', (0, -0.64516, -0)); | ||||||
|  | #79 = LINE('NONE', #78, #77); | ||||||
|  | #80 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #81 = VECTOR('NONE', #80, 1); | ||||||
|  | #82 = CARTESIAN_POINT('NONE', (1.996782122555674, -0.64516, -0)); | ||||||
|  | #83 = LINE('NONE', #82, #81); | ||||||
|  | #84 = DIRECTION('NONE', (1, 0, 0)); | ||||||
|  | #85 = VECTOR('NONE', #84, 1); | ||||||
|  | #86 = CARTESIAN_POINT('NONE', (0, -0.64516, 2.58064)); | ||||||
|  | #87 = LINE('NONE', #86, #85); | ||||||
|  | #88 = DIRECTION('NONE', (0.819152044288992, -0.5735764363510459, 0)); | ||||||
|  | #89 = VECTOR('NONE', #88, 1); | ||||||
|  | #90 = CARTESIAN_POINT('NONE', (1.996782122555674, -0.64516, -0)); | ||||||
|  | #91 = LINE('NONE', #90, #89); | ||||||
|  | #92 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #93 = VECTOR('NONE', #92, 1); | ||||||
|  | #94 = CARTESIAN_POINT('NONE', (3.839550058615159, -1.9354799999999992, -0)); | ||||||
|  | #95 = LINE('NONE', #94, #93); | ||||||
|  | #96 = DIRECTION('NONE', (0.819152044288992, -0.5735764363510459, 0)); | ||||||
|  | #97 = VECTOR('NONE', #96, 1); | ||||||
|  | #98 = CARTESIAN_POINT('NONE', (1.996782122555674, -0.64516, 2.58064)); | ||||||
|  | #99 = LINE('NONE', #98, #97); | ||||||
|  | #100 = DIRECTION('NONE', (1, -0.00000000000000038794063361359933, 0)); | ||||||
|  | #101 = VECTOR('NONE', #100, 1); | ||||||
|  | #102 = CARTESIAN_POINT('NONE', (3.839550058615159, -1.9354799999999992, -0)); | ||||||
|  | #103 = LINE('NONE', #102, #101); | ||||||
|  | #104 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #105 = VECTOR('NONE', #104, 1); | ||||||
|  | #106 = CARTESIAN_POINT('NONE', (6.12902, -1.93548, -0)); | ||||||
|  | #107 = LINE('NONE', #106, #105); | ||||||
|  | #108 = DIRECTION('NONE', (1, -0.00000000000000038794063361359933, 0)); | ||||||
|  | #109 = VECTOR('NONE', #108, 1); | ||||||
|  | #110 = CARTESIAN_POINT('NONE', (3.839550058615159, -1.9354799999999992, 2.58064)); | ||||||
|  | #111 = LINE('NONE', #110, #109); | ||||||
|  | #112 = DIRECTION('NONE', (0, 1, 0)); | ||||||
|  | #113 = VECTOR('NONE', #112, 1); | ||||||
|  | #114 = CARTESIAN_POINT('NONE', (6.12902, -1.93548, -0)); | ||||||
|  | #115 = LINE('NONE', #114, #113); | ||||||
|  | #116 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #117 = VECTOR('NONE', #116, 1); | ||||||
|  | #118 = CARTESIAN_POINT('NONE', (6.12902, -1.6129, -0)); | ||||||
|  | #119 = LINE('NONE', #118, #117); | ||||||
|  | #120 = DIRECTION('NONE', (0, 1, 0)); | ||||||
|  | #121 = VECTOR('NONE', #120, 1); | ||||||
|  | #122 = CARTESIAN_POINT('NONE', (6.12902, -1.93548, 2.58064)); | ||||||
|  | #123 = LINE('NONE', #122, #121); | ||||||
|  | #124 = DIRECTION('NONE', (-1, 0, 0)); | ||||||
|  | #125 = VECTOR('NONE', #124, 1); | ||||||
|  | #126 = CARTESIAN_POINT('NONE', (6.12902, -1.6129, -0)); | ||||||
|  | #127 = LINE('NONE', #126, #125); | ||||||
|  | #128 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #129 = VECTOR('NONE', #128, 1); | ||||||
|  | #130 = CARTESIAN_POINT('NONE', (3.9412591419317424, -1.6129, -0)); | ||||||
|  | #131 = LINE('NONE', #130, #129); | ||||||
|  | #132 = DIRECTION('NONE', (-1, 0, 0)); | ||||||
|  | #133 = VECTOR('NONE', #132, 1); | ||||||
|  | #134 = CARTESIAN_POINT('NONE', (6.12902, -1.6129, 2.58064)); | ||||||
|  | #135 = LINE('NONE', #134, #133); | ||||||
|  | #136 = DIRECTION('NONE', (-0.8191520442889919, 0.573576436351046, 0)); | ||||||
|  | #137 = VECTOR('NONE', #136, 1); | ||||||
|  | #138 = CARTESIAN_POINT('NONE', (3.9412591419317424, -1.6129, -0)); | ||||||
|  | #139 = LINE('NONE', #138, #137); | ||||||
|  | #140 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #141 = VECTOR('NONE', #140, 1); | ||||||
|  | #142 = CARTESIAN_POINT('NONE', (1.6377992218573856, 0, -0)); | ||||||
|  | #143 = LINE('NONE', #142, #141); | ||||||
|  | #144 = DIRECTION('NONE', (-0.8191520442889919, 0.573576436351046, 0)); | ||||||
|  | #145 = VECTOR('NONE', #144, 1); | ||||||
|  | #146 = CARTESIAN_POINT('NONE', (3.9412591419317424, -1.6129, 2.58064)); | ||||||
|  | #147 = LINE('NONE', #146, #145); | ||||||
|  | #148 = DIRECTION('NONE', (0.90630778703665, 0.4226182617406992, 0)); | ||||||
|  | #149 = VECTOR('NONE', #148, 1); | ||||||
|  | #150 = CARTESIAN_POINT('NONE', (1.6377992218573856, 0, -0)); | ||||||
|  | #151 = LINE('NONE', #150, #149); | ||||||
|  | #152 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #153 = VECTOR('NONE', #152, 1); | ||||||
|  | #154 = CARTESIAN_POINT('NONE', (3.7131243491113075, 0.9677400000000002, -0)); | ||||||
|  | #155 = LINE('NONE', #154, #153); | ||||||
|  | #156 = DIRECTION('NONE', (0.90630778703665, 0.4226182617406992, 0)); | ||||||
|  | #157 = VECTOR('NONE', #156, 1); | ||||||
|  | #158 = CARTESIAN_POINT('NONE', (1.6377992218573856, 0, 2.58064)); | ||||||
|  | #159 = LINE('NONE', #158, #157); | ||||||
|  | #160 = DIRECTION('NONE', (1, -0.0000000000000001378647737807002, 0)); | ||||||
|  | #161 = VECTOR('NONE', #160, 1); | ||||||
|  | #162 = CARTESIAN_POINT('NONE', (3.7131243491113075, 0.9677400000000002, -0)); | ||||||
|  | #163 = LINE('NONE', #162, #161); | ||||||
|  | #164 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #165 = VECTOR('NONE', #164, 1); | ||||||
|  | #166 = CARTESIAN_POINT('NONE', (6.12902, 0.9677399999999998, -0)); | ||||||
|  | #167 = LINE('NONE', #166, #165); | ||||||
|  | #168 = DIRECTION('NONE', (1, -0.0000000000000001378647737807002, 0)); | ||||||
|  | #169 = VECTOR('NONE', #168, 1); | ||||||
|  | #170 = CARTESIAN_POINT('NONE', (3.7131243491113075, 0.9677400000000002, 2.58064)); | ||||||
|  | #171 = LINE('NONE', #170, #169); | ||||||
|  | #172 = DIRECTION('NONE', (0, 1, 0)); | ||||||
|  | #173 = VECTOR('NONE', #172, 1); | ||||||
|  | #174 = CARTESIAN_POINT('NONE', (6.12902, 0.9677399999999998, -0)); | ||||||
|  | #175 = LINE('NONE', #174, #173); | ||||||
|  | #176 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #177 = VECTOR('NONE', #176, 1); | ||||||
|  | #178 = CARTESIAN_POINT('NONE', (6.12902, 1.29032, -0)); | ||||||
|  | #179 = LINE('NONE', #178, #177); | ||||||
|  | #180 = DIRECTION('NONE', (0, 1, 0)); | ||||||
|  | #181 = VECTOR('NONE', #180, 1); | ||||||
|  | #182 = CARTESIAN_POINT('NONE', (6.12902, 0.9677399999999998, 2.58064)); | ||||||
|  | #183 = LINE('NONE', #182, #181); | ||||||
|  | #184 = DIRECTION('NONE', (-1, 0, 0)); | ||||||
|  | #185 = VECTOR('NONE', #184, 1); | ||||||
|  | #186 = CARTESIAN_POINT('NONE', (6.12902, 1.29032, -0)); | ||||||
|  | #187 = LINE('NONE', #186, #185); | ||||||
|  | #188 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #189 = VECTOR('NONE', #188, 1); | ||||||
|  | #190 = CARTESIAN_POINT('NONE', (3.6416100848359463, 1.29032, -0)); | ||||||
|  | #191 = LINE('NONE', #190, #189); | ||||||
|  | #192 = DIRECTION('NONE', (-1, 0, 0)); | ||||||
|  | #193 = VECTOR('NONE', #192, 1); | ||||||
|  | #194 = CARTESIAN_POINT('NONE', (6.12902, 1.29032, 2.58064)); | ||||||
|  | #195 = LINE('NONE', #194, #193); | ||||||
|  | #196 = DIRECTION('NONE', (-0.90630778703665, -0.4226182617406995, 0)); | ||||||
|  | #197 = VECTOR('NONE', #196, 1); | ||||||
|  | #198 = CARTESIAN_POINT('NONE', (3.6416100848359463, 1.29032, -0)); | ||||||
|  | #199 = LINE('NONE', #198, #197); | ||||||
|  | #200 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #201 = VECTOR('NONE', #200, 1); | ||||||
|  | #202 = CARTESIAN_POINT('NONE', (2.2580599999999995, 0.64516, -0)); | ||||||
|  | #203 = LINE('NONE', #202, #201); | ||||||
|  | #204 = DIRECTION('NONE', (-0.90630778703665, -0.4226182617406995, 0)); | ||||||
|  | #205 = VECTOR('NONE', #204, 1); | ||||||
|  | #206 = CARTESIAN_POINT('NONE', (3.6416100848359463, 1.29032, 2.58064)); | ||||||
|  | #207 = LINE('NONE', #206, #205); | ||||||
|  | #208 = DIRECTION('NONE', (-1, 0, 0)); | ||||||
|  | #209 = VECTOR('NONE', #208, 1); | ||||||
|  | #210 = CARTESIAN_POINT('NONE', (2.2580599999999995, 0.64516, -0)); | ||||||
|  | #211 = LINE('NONE', #210, #209); | ||||||
|  | #212 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #213 = VECTOR('NONE', #212, 1); | ||||||
|  | #214 = CARTESIAN_POINT('NONE', (0, 0.64516, -0)); | ||||||
|  | #215 = LINE('NONE', #214, #213); | ||||||
|  | #216 = DIRECTION('NONE', (-1, 0, 0)); | ||||||
|  | #217 = VECTOR('NONE', #216, 1); | ||||||
|  | #218 = CARTESIAN_POINT('NONE', (2.2580599999999995, 0.64516, 2.58064)); | ||||||
|  | #219 = LINE('NONE', #218, #217); | ||||||
|  | #220 = DIRECTION('NONE', (0, -1, 0)); | ||||||
|  | #221 = VECTOR('NONE', #220, 1); | ||||||
|  | #222 = CARTESIAN_POINT('NONE', (0, 0.64516, -0)); | ||||||
|  | #223 = LINE('NONE', #222, #221); | ||||||
|  | #224 = DIRECTION('NONE', (0, -1, 0)); | ||||||
|  | #225 = VECTOR('NONE', #224, 1); | ||||||
|  | #226 = CARTESIAN_POINT('NONE', (0, 0.64516, 2.58064)); | ||||||
|  | #227 = LINE('NONE', #226, #225); | ||||||
|  | #228 = EDGE_CURVE('NONE', #5, #7, #63, .T.); | ||||||
|  | #229 = EDGE_CURVE('NONE', #7, #9, #67, .T.); | ||||||
|  | #230 = EDGE_CURVE('NONE', #11, #9, #71, .T.); | ||||||
|  | #231 = EDGE_CURVE('NONE', #5, #11, #75, .T.); | ||||||
|  | #232 = EDGE_CURVE('NONE', #7, #13, #79, .T.); | ||||||
|  | #233 = EDGE_CURVE('NONE', #13, #15, #83, .T.); | ||||||
|  | #234 = EDGE_CURVE('NONE', #9, #15, #87, .T.); | ||||||
|  | #235 = EDGE_CURVE('NONE', #13, #17, #91, .T.); | ||||||
|  | #236 = EDGE_CURVE('NONE', #17, #19, #95, .T.); | ||||||
|  | #237 = EDGE_CURVE('NONE', #15, #19, #99, .T.); | ||||||
|  | #238 = EDGE_CURVE('NONE', #17, #21, #103, .T.); | ||||||
|  | #239 = EDGE_CURVE('NONE', #21, #23, #107, .T.); | ||||||
|  | #240 = EDGE_CURVE('NONE', #19, #23, #111, .T.); | ||||||
|  | #241 = EDGE_CURVE('NONE', #21, #25, #115, .T.); | ||||||
|  | #242 = EDGE_CURVE('NONE', #25, #27, #119, .T.); | ||||||
|  | #243 = EDGE_CURVE('NONE', #23, #27, #123, .T.); | ||||||
|  | #244 = EDGE_CURVE('NONE', #25, #29, #127, .T.); | ||||||
|  | #245 = EDGE_CURVE('NONE', #29, #31, #131, .T.); | ||||||
|  | #246 = EDGE_CURVE('NONE', #27, #31, #135, .T.); | ||||||
|  | #247 = EDGE_CURVE('NONE', #29, #33, #139, .T.); | ||||||
|  | #248 = EDGE_CURVE('NONE', #33, #35, #143, .T.); | ||||||
|  | #249 = EDGE_CURVE('NONE', #31, #35, #147, .T.); | ||||||
|  | #250 = EDGE_CURVE('NONE', #33, #37, #151, .T.); | ||||||
|  | #251 = EDGE_CURVE('NONE', #37, #39, #155, .T.); | ||||||
|  | #252 = EDGE_CURVE('NONE', #35, #39, #159, .T.); | ||||||
|  | #253 = EDGE_CURVE('NONE', #37, #41, #163, .T.); | ||||||
|  | #254 = EDGE_CURVE('NONE', #41, #43, #167, .T.); | ||||||
|  | #255 = EDGE_CURVE('NONE', #39, #43, #171, .T.); | ||||||
|  | #256 = EDGE_CURVE('NONE', #41, #45, #175, .T.); | ||||||
|  | #257 = EDGE_CURVE('NONE', #45, #47, #179, .T.); | ||||||
|  | #258 = EDGE_CURVE('NONE', #43, #47, #183, .T.); | ||||||
|  | #259 = EDGE_CURVE('NONE', #45, #49, #187, .T.); | ||||||
|  | #260 = EDGE_CURVE('NONE', #49, #51, #191, .T.); | ||||||
|  | #261 = EDGE_CURVE('NONE', #47, #51, #195, .T.); | ||||||
|  | #262 = EDGE_CURVE('NONE', #49, #53, #199, .T.); | ||||||
|  | #263 = EDGE_CURVE('NONE', #53, #55, #203, .T.); | ||||||
|  | #264 = EDGE_CURVE('NONE', #51, #55, #207, .T.); | ||||||
|  | #265 = EDGE_CURVE('NONE', #53, #57, #211, .T.); | ||||||
|  | #266 = EDGE_CURVE('NONE', #57, #59, #215, .T.); | ||||||
|  | #267 = EDGE_CURVE('NONE', #55, #59, #219, .T.); | ||||||
|  | #268 = EDGE_CURVE('NONE', #57, #5, #223, .T.); | ||||||
|  | #269 = EDGE_CURVE('NONE', #59, #11, #227, .T.); | ||||||
|  | #270 = ORIENTED_EDGE('NONE', *, *, #228, .T.); | ||||||
|  | #271 = ORIENTED_EDGE('NONE', *, *, #229, .T.); | ||||||
|  | #272 = ORIENTED_EDGE('NONE', *, *, #230, .F.); | ||||||
|  | #273 = ORIENTED_EDGE('NONE', *, *, #231, .F.); | ||||||
|  | #274 = EDGE_LOOP('NONE', (#270, #271, #272, #273)); | ||||||
|  | #275 = ORIENTED_EDGE('NONE', *, *, #232, .T.); | ||||||
|  | #276 = ORIENTED_EDGE('NONE', *, *, #233, .T.); | ||||||
|  | #277 = ORIENTED_EDGE('NONE', *, *, #234, .F.); | ||||||
|  | #278 = ORIENTED_EDGE('NONE', *, *, #229, .F.); | ||||||
|  | #279 = EDGE_LOOP('NONE', (#275, #276, #277, #278)); | ||||||
|  | #280 = ORIENTED_EDGE('NONE', *, *, #235, .T.); | ||||||
|  | #281 = ORIENTED_EDGE('NONE', *, *, #236, .T.); | ||||||
|  | #282 = ORIENTED_EDGE('NONE', *, *, #237, .F.); | ||||||
|  | #283 = ORIENTED_EDGE('NONE', *, *, #233, .F.); | ||||||
|  | #284 = EDGE_LOOP('NONE', (#280, #281, #282, #283)); | ||||||
|  | #285 = ORIENTED_EDGE('NONE', *, *, #238, .T.); | ||||||
|  | #286 = ORIENTED_EDGE('NONE', *, *, #239, .T.); | ||||||
|  | #287 = ORIENTED_EDGE('NONE', *, *, #240, .F.); | ||||||
|  | #288 = ORIENTED_EDGE('NONE', *, *, #236, .F.); | ||||||
|  | #289 = EDGE_LOOP('NONE', (#285, #286, #287, #288)); | ||||||
|  | #290 = ORIENTED_EDGE('NONE', *, *, #241, .T.); | ||||||
|  | #291 = ORIENTED_EDGE('NONE', *, *, #242, .T.); | ||||||
|  | #292 = ORIENTED_EDGE('NONE', *, *, #243, .F.); | ||||||
|  | #293 = ORIENTED_EDGE('NONE', *, *, #239, .F.); | ||||||
|  | #294 = EDGE_LOOP('NONE', (#290, #291, #292, #293)); | ||||||
|  | #295 = ORIENTED_EDGE('NONE', *, *, #244, .T.); | ||||||
|  | #296 = ORIENTED_EDGE('NONE', *, *, #245, .T.); | ||||||
|  | #297 = ORIENTED_EDGE('NONE', *, *, #246, .F.); | ||||||
|  | #298 = ORIENTED_EDGE('NONE', *, *, #242, .F.); | ||||||
|  | #299 = EDGE_LOOP('NONE', (#295, #296, #297, #298)); | ||||||
|  | #300 = ORIENTED_EDGE('NONE', *, *, #247, .T.); | ||||||
|  | #301 = ORIENTED_EDGE('NONE', *, *, #248, .T.); | ||||||
|  | #302 = ORIENTED_EDGE('NONE', *, *, #249, .F.); | ||||||
|  | #303 = ORIENTED_EDGE('NONE', *, *, #245, .F.); | ||||||
|  | #304 = EDGE_LOOP('NONE', (#300, #301, #302, #303)); | ||||||
|  | #305 = ORIENTED_EDGE('NONE', *, *, #250, .T.); | ||||||
|  | #306 = ORIENTED_EDGE('NONE', *, *, #251, .T.); | ||||||
|  | #307 = ORIENTED_EDGE('NONE', *, *, #252, .F.); | ||||||
|  | #308 = ORIENTED_EDGE('NONE', *, *, #248, .F.); | ||||||
|  | #309 = EDGE_LOOP('NONE', (#305, #306, #307, #308)); | ||||||
|  | #310 = ORIENTED_EDGE('NONE', *, *, #253, .T.); | ||||||
|  | #311 = ORIENTED_EDGE('NONE', *, *, #254, .T.); | ||||||
|  | #312 = ORIENTED_EDGE('NONE', *, *, #255, .F.); | ||||||
|  | #313 = ORIENTED_EDGE('NONE', *, *, #251, .F.); | ||||||
|  | #314 = EDGE_LOOP('NONE', (#310, #311, #312, #313)); | ||||||
|  | #315 = ORIENTED_EDGE('NONE', *, *, #256, .T.); | ||||||
|  | #316 = ORIENTED_EDGE('NONE', *, *, #257, .T.); | ||||||
|  | #317 = ORIENTED_EDGE('NONE', *, *, #258, .F.); | ||||||
|  | #318 = ORIENTED_EDGE('NONE', *, *, #254, .F.); | ||||||
|  | #319 = EDGE_LOOP('NONE', (#315, #316, #317, #318)); | ||||||
|  | #320 = ORIENTED_EDGE('NONE', *, *, #259, .T.); | ||||||
|  | #321 = ORIENTED_EDGE('NONE', *, *, #260, .T.); | ||||||
|  | #322 = ORIENTED_EDGE('NONE', *, *, #261, .F.); | ||||||
|  | #323 = ORIENTED_EDGE('NONE', *, *, #257, .F.); | ||||||
|  | #324 = EDGE_LOOP('NONE', (#320, #321, #322, #323)); | ||||||
|  | #325 = ORIENTED_EDGE('NONE', *, *, #262, .T.); | ||||||
|  | #326 = ORIENTED_EDGE('NONE', *, *, #263, .T.); | ||||||
|  | #327 = ORIENTED_EDGE('NONE', *, *, #264, .F.); | ||||||
|  | #328 = ORIENTED_EDGE('NONE', *, *, #260, .F.); | ||||||
|  | #329 = EDGE_LOOP('NONE', (#325, #326, #327, #328)); | ||||||
|  | #330 = ORIENTED_EDGE('NONE', *, *, #265, .T.); | ||||||
|  | #331 = ORIENTED_EDGE('NONE', *, *, #266, .T.); | ||||||
|  | #332 = ORIENTED_EDGE('NONE', *, *, #267, .F.); | ||||||
|  | #333 = ORIENTED_EDGE('NONE', *, *, #263, .F.); | ||||||
|  | #334 = EDGE_LOOP('NONE', (#330, #331, #332, #333)); | ||||||
|  | #335 = ORIENTED_EDGE('NONE', *, *, #268, .T.); | ||||||
|  | #336 = ORIENTED_EDGE('NONE', *, *, #231, .T.); | ||||||
|  | #337 = ORIENTED_EDGE('NONE', *, *, #269, .F.); | ||||||
|  | #338 = ORIENTED_EDGE('NONE', *, *, #266, .F.); | ||||||
|  | #339 = EDGE_LOOP('NONE', (#335, #336, #337, #338)); | ||||||
|  | #340 = ORIENTED_EDGE('NONE', *, *, #228, .T.); | ||||||
|  | #341 = ORIENTED_EDGE('NONE', *, *, #232, .T.); | ||||||
|  | #342 = ORIENTED_EDGE('NONE', *, *, #235, .T.); | ||||||
|  | #343 = ORIENTED_EDGE('NONE', *, *, #238, .T.); | ||||||
|  | #344 = ORIENTED_EDGE('NONE', *, *, #241, .T.); | ||||||
|  | #345 = ORIENTED_EDGE('NONE', *, *, #244, .T.); | ||||||
|  | #346 = ORIENTED_EDGE('NONE', *, *, #247, .T.); | ||||||
|  | #347 = ORIENTED_EDGE('NONE', *, *, #250, .T.); | ||||||
|  | #348 = ORIENTED_EDGE('NONE', *, *, #253, .T.); | ||||||
|  | #349 = ORIENTED_EDGE('NONE', *, *, #256, .T.); | ||||||
|  | #350 = ORIENTED_EDGE('NONE', *, *, #259, .T.); | ||||||
|  | #351 = ORIENTED_EDGE('NONE', *, *, #262, .T.); | ||||||
|  | #352 = ORIENTED_EDGE('NONE', *, *, #265, .T.); | ||||||
|  | #353 = ORIENTED_EDGE('NONE', *, *, #268, .T.); | ||||||
|  | #354 = EDGE_LOOP('NONE', (#340, #341, #342, #343, #344, #345, #346, #347, #348, #349, #350, #351, #352, #353)); | ||||||
|  | #355 = ORIENTED_EDGE('NONE', *, *, #230, .T.); | ||||||
|  | #356 = ORIENTED_EDGE('NONE', *, *, #234, .T.); | ||||||
|  | #357 = ORIENTED_EDGE('NONE', *, *, #237, .T.); | ||||||
|  | #358 = ORIENTED_EDGE('NONE', *, *, #240, .T.); | ||||||
|  | #359 = ORIENTED_EDGE('NONE', *, *, #243, .T.); | ||||||
|  | #360 = ORIENTED_EDGE('NONE', *, *, #246, .T.); | ||||||
|  | #361 = ORIENTED_EDGE('NONE', *, *, #249, .T.); | ||||||
|  | #362 = ORIENTED_EDGE('NONE', *, *, #252, .T.); | ||||||
|  | #363 = ORIENTED_EDGE('NONE', *, *, #255, .T.); | ||||||
|  | #364 = ORIENTED_EDGE('NONE', *, *, #258, .T.); | ||||||
|  | #365 = ORIENTED_EDGE('NONE', *, *, #261, .T.); | ||||||
|  | #366 = ORIENTED_EDGE('NONE', *, *, #264, .T.); | ||||||
|  | #367 = ORIENTED_EDGE('NONE', *, *, #267, .T.); | ||||||
|  | #368 = ORIENTED_EDGE('NONE', *, *, #269, .T.); | ||||||
|  | #369 = EDGE_LOOP('NONE', (#355, #356, #357, #358, #359, #360, #361, #362, #363, #364, #365, #366, #367, #368)); | ||||||
|  | #370 = CARTESIAN_POINT('NONE', (0, -0.3225799999999985, 1.2903199999999995)); | ||||||
|  | #371 = DIRECTION('NONE', (-1, -0, 0)); | ||||||
|  | #372 = AXIS2_PLACEMENT_3D('NONE', #370, #371, $); | ||||||
|  | #373 = PLANE('NONE', #372); | ||||||
|  | #374 = CARTESIAN_POINT('NONE', (0.9983910612778368, -0.6451599999999998, 1.2903199999999997)); | ||||||
|  | #375 = DIRECTION('NONE', (0, -1, 0)); | ||||||
|  | #376 = AXIS2_PLACEMENT_3D('NONE', #374, #375, $); | ||||||
|  | #377 = PLANE('NONE', #376); | ||||||
|  | #378 = CARTESIAN_POINT('NONE', (2.918166090585415, -1.2903199999999988, 1.2903199999999997)); | ||||||
|  | #379 = DIRECTION('NONE', (-0.5735764363510459, -0.8191520442889919, 0)); | ||||||
|  | #380 = AXIS2_PLACEMENT_3D('NONE', #378, #379, $); | ||||||
|  | #381 = PLANE('NONE', #380); | ||||||
|  | #382 = CARTESIAN_POINT('NONE', (4.984285029307579, -1.9354799999999992, 1.2903199999999997)); | ||||||
|  | #383 = DIRECTION('NONE', (0, -1, 0)); | ||||||
|  | #384 = AXIS2_PLACEMENT_3D('NONE', #382, #383, $); | ||||||
|  | #385 = PLANE('NONE', #384); | ||||||
|  | #386 = CARTESIAN_POINT('NONE', (6.129019999999999, -1.7741899999999997, 1.2903199999999997)); | ||||||
|  | #387 = DIRECTION('NONE', (1, -0, 0)); | ||||||
|  | #388 = AXIS2_PLACEMENT_3D('NONE', #386, #387, $); | ||||||
|  | #389 = PLANE('NONE', #388); | ||||||
|  | #390 = CARTESIAN_POINT('NONE', (5.035139570965871, -1.6128999999999998, 1.2903199999999997)); | ||||||
|  | #391 = DIRECTION('NONE', (0, 1, -0)); | ||||||
|  | #392 = AXIS2_PLACEMENT_3D('NONE', #390, #391, $); | ||||||
|  | #393 = PLANE('NONE', #392); | ||||||
|  | #394 = CARTESIAN_POINT('NONE', (2.7895291818945633, -0.8064499999999998, 1.2903199999999995)); | ||||||
|  | #395 = DIRECTION('NONE', (0.5735764363510459, 0.8191520442889918, -0)); | ||||||
|  | #396 = AXIS2_PLACEMENT_3D('NONE', #394, #395, $); | ||||||
|  | #397 = PLANE('NONE', #396); | ||||||
|  | #398 = CARTESIAN_POINT('NONE', (2.6754617854843468, 0.4838700000000003, 1.2903199999999997)); | ||||||
|  | #399 = DIRECTION('NONE', (0.4226182617406992, -0.90630778703665, 0)); | ||||||
|  | #400 = AXIS2_PLACEMENT_3D('NONE', #398, #399, $); | ||||||
|  | #401 = PLANE('NONE', #400); | ||||||
|  | #402 = CARTESIAN_POINT('NONE', (4.921072174555653, 0.9677399999999998, 1.2903199999999995)); | ||||||
|  | #403 = DIRECTION('NONE', (0, -1, 0)); | ||||||
|  | #404 = AXIS2_PLACEMENT_3D('NONE', #402, #403, $); | ||||||
|  | #405 = PLANE('NONE', #404); | ||||||
|  | #406 = CARTESIAN_POINT('NONE', (6.129019999999998, 1.1290299999999989, 1.2903199999999995)); | ||||||
|  | #407 = DIRECTION('NONE', (1, -0, 0)); | ||||||
|  | #408 = AXIS2_PLACEMENT_3D('NONE', #406, #407, $); | ||||||
|  | #409 = PLANE('NONE', #408); | ||||||
|  | #410 = CARTESIAN_POINT('NONE', (4.8853150424179725, 1.2903199999999997, 1.2903199999999997)); | ||||||
|  | #411 = DIRECTION('NONE', (0, 1, -0)); | ||||||
|  | #412 = AXIS2_PLACEMENT_3D('NONE', #410, #411, $); | ||||||
|  | #413 = PLANE('NONE', #412); | ||||||
|  | #414 = CARTESIAN_POINT('NONE', (2.9498350424179733, 0.9677399999999998, 1.2903199999999997)); | ||||||
|  | #415 = DIRECTION('NONE', (-0.42261826174069933, 0.9063077870366499, -0)); | ||||||
|  | #416 = AXIS2_PLACEMENT_3D('NONE', #414, #415, $); | ||||||
|  | #417 = PLANE('NONE', #416); | ||||||
|  | #418 = CARTESIAN_POINT('NONE', (1.1290299999999998, 0.6451599999999998, 1.29032)); | ||||||
|  | #419 = DIRECTION('NONE', (0, 1, -0)); | ||||||
|  | #420 = AXIS2_PLACEMENT_3D('NONE', #418, #419, $); | ||||||
|  | #421 = PLANE('NONE', #420); | ||||||
|  | #422 = CARTESIAN_POINT('NONE', (0, 0.32257999999999987, 1.2903199999999995)); | ||||||
|  | #423 = DIRECTION('NONE', (-1, -0, 0)); | ||||||
|  | #424 = AXIS2_PLACEMENT_3D('NONE', #422, #423, $); | ||||||
|  | #425 = PLANE('NONE', #424); | ||||||
|  | #426 = CARTESIAN_POINT('NONE', (0, 0, -0)); | ||||||
|  | #427 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #428 = AXIS2_PLACEMENT_3D('NONE', #426, #427, $); | ||||||
|  | #429 = PLANE('NONE', #428); | ||||||
|  | #430 = CARTESIAN_POINT('NONE', (0, 0, 2.58064)); | ||||||
|  | #431 = DIRECTION('NONE', (0, 0, 1)); | ||||||
|  | #432 = AXIS2_PLACEMENT_3D('NONE', #430, #431, $); | ||||||
|  | #433 = PLANE('NONE', #432); | ||||||
|  | #434 = FACE_OUTER_BOUND('NONE', #274, .T.); | ||||||
|  | #435 = ADVANCED_FACE('NONE', (#434), #373, .T.); | ||||||
|  | #436 = FACE_OUTER_BOUND('NONE', #279, .T.); | ||||||
|  | #437 = ADVANCED_FACE('NONE', (#436), #377, .T.); | ||||||
|  | #438 = FACE_OUTER_BOUND('NONE', #284, .T.); | ||||||
|  | #439 = ADVANCED_FACE('NONE', (#438), #381, .T.); | ||||||
|  | #440 = FACE_OUTER_BOUND('NONE', #289, .T.); | ||||||
|  | #441 = ADVANCED_FACE('NONE', (#440), #385, .T.); | ||||||
|  | #442 = FACE_OUTER_BOUND('NONE', #294, .T.); | ||||||
|  | #443 = ADVANCED_FACE('NONE', (#442), #389, .T.); | ||||||
|  | #444 = FACE_OUTER_BOUND('NONE', #299, .T.); | ||||||
|  | #445 = ADVANCED_FACE('NONE', (#444), #393, .T.); | ||||||
|  | #446 = FACE_OUTER_BOUND('NONE', #304, .T.); | ||||||
|  | #447 = ADVANCED_FACE('NONE', (#446), #397, .T.); | ||||||
|  | #448 = FACE_OUTER_BOUND('NONE', #309, .T.); | ||||||
|  | #449 = ADVANCED_FACE('NONE', (#448), #401, .T.); | ||||||
|  | #450 = FACE_OUTER_BOUND('NONE', #314, .T.); | ||||||
|  | #451 = ADVANCED_FACE('NONE', (#450), #405, .T.); | ||||||
|  | #452 = FACE_OUTER_BOUND('NONE', #319, .T.); | ||||||
|  | #453 = ADVANCED_FACE('NONE', (#452), #409, .T.); | ||||||
|  | #454 = FACE_OUTER_BOUND('NONE', #324, .T.); | ||||||
|  | #455 = ADVANCED_FACE('NONE', (#454), #413, .T.); | ||||||
|  | #456 = FACE_OUTER_BOUND('NONE', #329, .T.); | ||||||
|  | #457 = ADVANCED_FACE('NONE', (#456), #417, .T.); | ||||||
|  | #458 = FACE_OUTER_BOUND('NONE', #334, .T.); | ||||||
|  | #459 = ADVANCED_FACE('NONE', (#458), #421, .T.); | ||||||
|  | #460 = FACE_OUTER_BOUND('NONE', #339, .T.); | ||||||
|  | #461 = ADVANCED_FACE('NONE', (#460), #425, .T.); | ||||||
|  | #462 = FACE_OUTER_BOUND('NONE', #354, .F.); | ||||||
|  | #463 = ADVANCED_FACE('NONE', (#462), #429, .F.); | ||||||
|  | #464 = FACE_OUTER_BOUND('NONE', #369, .T.); | ||||||
|  | #465 = ADVANCED_FACE('NONE', (#464), #433, .T.); | ||||||
|  | #466 = CLOSED_SHELL('NONE', (#435, #437, #439, #441, #443, #445, #447, #449, #451, #453, #455, #457, #459, #461, #463, #465)); | ||||||
|  | #467 = ORIENTED_CLOSED_SHELL('NONE', *, #466, .T.); | ||||||
|  | #468 = MANIFOLD_SOLID_BREP('NONE', #467); | ||||||
|  | #469 = APPLICATION_CONTEXT('configuration controlled 3D design of mechanical parts and assemblies'); | ||||||
|  | #470 = PRODUCT_DEFINITION_CONTEXT('part definition', #469, 'design'); | ||||||
|  | #471 = PRODUCT('UNIDENTIFIED_PRODUCT', 'NONE', $, ()); | ||||||
|  | #472 = PRODUCT_DEFINITION_FORMATION('', $, #471); | ||||||
|  | #473 = PRODUCT_DEFINITION('design', $, #472, #470); | ||||||
|  | #474 = PRODUCT_DEFINITION_SHAPE('NONE', $, #473); | ||||||
|  | #475 = ADVANCED_BREP_SHAPE_REPRESENTATION('NONE', (#468), #3); | ||||||
|  | #476 = SHAPE_DEFINITION_REPRESENTATION(#474, #475); | ||||||
|  | ENDSEC; | ||||||
|  | END-ISO-10303-21; | ||||||
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/stl-ascii.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 221 KiB | 
							
								
								
									
										478
									
								
								e2e/playwright/export-snapshots/stl-ascii.stl
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,478 @@ | |||||||
|  | solid unnamed | ||||||
|  | facet normal -1 0 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 0 -101.600006 0 | ||||||
|  |         vertex 0 -0 0 | ||||||
|  |         vertex 0 -101.600006 -25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -1 0 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 0 -101.600006 -25.400002 | ||||||
|  |         vertex 0 -0 0 | ||||||
|  |         vertex 0 -0 -25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 0 -1 | ||||||
|  |     outer loop | ||||||
|  |         vertex 0 -101.600006 -25.400002 | ||||||
|  |         vertex 0 -0 -25.400002 | ||||||
|  |         vertex 78.613464 -101.600006 -25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 0 -1 | ||||||
|  |     outer loop | ||||||
|  |         vertex 78.613464 -101.600006 -25.400002 | ||||||
|  |         vertex 0 -0 -25.400002 | ||||||
|  |         vertex 78.613464 -0 -25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -0.5735764 0 -0.8191522 | ||||||
|  |     outer loop | ||||||
|  |         vertex 78.613464 -101.600006 -25.400002 | ||||||
|  |         vertex 78.613464 -0 -25.400002 | ||||||
|  |         vertex 151.16339 -101.600006 -76.2 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -0.5735764 0 -0.8191522 | ||||||
|  |     outer loop | ||||||
|  |         vertex 151.16339 -101.600006 -76.2 | ||||||
|  |         vertex 78.613464 -0 -25.400002 | ||||||
|  |         vertex 151.16339 -0 -76.2 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 0 -1 | ||||||
|  |     outer loop | ||||||
|  |         vertex 151.16339 -101.600006 -76.2 | ||||||
|  |         vertex 151.16339 -0 -76.2 | ||||||
|  |         vertex 241.3 -101.600006 -76.2 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 0 -1 | ||||||
|  |     outer loop | ||||||
|  |         vertex 241.3 -101.600006 -76.2 | ||||||
|  |         vertex 151.16339 -0 -76.2 | ||||||
|  |         vertex 241.3 -0 -76.2 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 1 0 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 241.3 -101.600006 -76.2 | ||||||
|  |         vertex 241.3 -0 -76.2 | ||||||
|  |         vertex 241.3 -101.600006 -63.5 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 1 -0 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 241.3 -101.600006 -63.5 | ||||||
|  |         vertex 241.3 -0 -76.2 | ||||||
|  |         vertex 241.3 -0 -63.5 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -0 1 | ||||||
|  |     outer loop | ||||||
|  |         vertex 241.3 -101.600006 -63.5 | ||||||
|  |         vertex 241.3 -0 -63.5 | ||||||
|  |         vertex 155.16768 -101.600006 -63.5 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 0 1 | ||||||
|  |     outer loop | ||||||
|  |         vertex 155.16768 -101.600006 -63.5 | ||||||
|  |         vertex 241.3 -0 -63.5 | ||||||
|  |         vertex 155.16768 -0 -63.5 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.5735765 0 0.81915194 | ||||||
|  |     outer loop | ||||||
|  |         vertex 87.15214 -101.600006 -15.875 | ||||||
|  |         vertex 109.82398 -101.600006 -31.75 | ||||||
|  |         vertex 109.82398 -0 -31.75 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.57357645 0 0.819152 | ||||||
|  |     outer loop | ||||||
|  |         vertex 109.82398 -101.600006 -31.75 | ||||||
|  |         vertex 155.16768 -101.600006 -63.5 | ||||||
|  |         vertex 155.16768 -0 -63.5 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.57357645 0 0.81915206 | ||||||
|  |     outer loop | ||||||
|  |         vertex 87.15214 -0 -15.875 | ||||||
|  |         vertex 64.480286 -101.600006 0 | ||||||
|  |         vertex 87.15214 -101.600006 -15.875 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.5735765 0 0.81915194 | ||||||
|  |     outer loop | ||||||
|  |         vertex 109.82398 -0 -31.75 | ||||||
|  |         vertex 87.15214 -0 -15.875 | ||||||
|  |         vertex 87.15214 -101.600006 -15.875 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.57357645 -0 0.819152 | ||||||
|  |     outer loop | ||||||
|  |         vertex 109.82398 -101.600006 -31.75 | ||||||
|  |         vertex 155.16768 -0 -63.5 | ||||||
|  |         vertex 109.82398 -0 -31.75 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.57357645 -0 0.81915206 | ||||||
|  |     outer loop | ||||||
|  |         vertex 64.480286 -101.600006 0 | ||||||
|  |         vertex 87.15214 -0 -15.875 | ||||||
|  |         vertex 64.480286 -0 0 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.4226182 0 -0.9063078 | ||||||
|  |     outer loop | ||||||
|  |         vertex 84.906715 -101.600006 9.525 | ||||||
|  |         vertex 64.480286 -101.600006 0 | ||||||
|  |         vertex 64.480286 -0 0 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.42261833 0 -0.90630776 | ||||||
|  |     outer loop | ||||||
|  |         vertex 105.33314 -101.600006 19.05 | ||||||
|  |         vertex 84.906715 -101.600006 9.525 | ||||||
|  |         vertex 84.906715 -0 9.525 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.4226182 0 -0.9063078 | ||||||
|  |     outer loop | ||||||
|  |         vertex 84.906715 -0 9.525 | ||||||
|  |         vertex 84.906715 -101.600006 9.525 | ||||||
|  |         vertex 64.480286 -0 0 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.4226183 0 -0.9063078 | ||||||
|  |     outer loop | ||||||
|  |         vertex 105.33314 -0 19.05 | ||||||
|  |         vertex 146.18599 -101.600006 38.1 | ||||||
|  |         vertex 105.33314 -101.600006 19.05 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.42261833 0 -0.90630776 | ||||||
|  |     outer loop | ||||||
|  |         vertex 105.33314 -101.600006 19.05 | ||||||
|  |         vertex 84.906715 -0 9.525 | ||||||
|  |         vertex 105.33314 -0 19.05 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0.4226183 0 -0.9063078 | ||||||
|  |     outer loop | ||||||
|  |         vertex 146.18599 -101.600006 38.1 | ||||||
|  |         vertex 105.33314 -0 19.05 | ||||||
|  |         vertex 146.18599 -0 38.1 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 0 -1 | ||||||
|  |     outer loop | ||||||
|  |         vertex 146.18599 -101.600006 38.1 | ||||||
|  |         vertex 146.18599 -0 38.1 | ||||||
|  |         vertex 241.3 -101.600006 38.1 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 0 -1 | ||||||
|  |     outer loop | ||||||
|  |         vertex 241.3 -101.600006 38.1 | ||||||
|  |         vertex 146.18599 -0 38.1 | ||||||
|  |         vertex 241.3 -0 38.1 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 1 0 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 241.3 -101.600006 38.1 | ||||||
|  |         vertex 241.3 -0 38.1 | ||||||
|  |         vertex 241.3 -101.600006 50.800003 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 1 -0 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 241.3 -101.600006 50.800003 | ||||||
|  |         vertex 241.3 -0 38.1 | ||||||
|  |         vertex 241.3 -0 50.800003 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -0 0.99999994 | ||||||
|  |     outer loop | ||||||
|  |         vertex 241.3 -101.600006 50.800003 | ||||||
|  |         vertex 241.3 -0 50.800003 | ||||||
|  |         vertex 143.37048 -101.600006 50.800003 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 0 0.99999994 | ||||||
|  |     outer loop | ||||||
|  |         vertex 143.37048 -101.600006 50.800003 | ||||||
|  |         vertex 241.3 -0 50.800003 | ||||||
|  |         vertex 143.37048 -0 50.800003 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -0.42261827 0 0.9063078 | ||||||
|  |     outer loop | ||||||
|  |         vertex 143.37048 -101.600006 50.800003 | ||||||
|  |         vertex 143.37048 -0 50.800003 | ||||||
|  |         vertex 88.9 -101.600006 25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -0.42261827 0 0.9063078 | ||||||
|  |     outer loop | ||||||
|  |         vertex 88.9 -101.600006 25.400002 | ||||||
|  |         vertex 143.37048 -0 50.800003 | ||||||
|  |         vertex 88.9 -0 25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -0 1 | ||||||
|  |     outer loop | ||||||
|  |         vertex 88.9 -101.600006 25.400002 | ||||||
|  |         vertex 88.9 -0 25.400002 | ||||||
|  |         vertex 0 -101.600006 25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 0 1 | ||||||
|  |     outer loop | ||||||
|  |         vertex 0 -101.600006 25.400002 | ||||||
|  |         vertex 88.9 -0 25.400002 | ||||||
|  |         vertex 0 -0 25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -1 0 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 0 -101.600006 25.400002 | ||||||
|  |         vertex 0 -0 25.400002 | ||||||
|  |         vertex 0 -101.600006 0 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -1 0 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 0 -101.600006 0 | ||||||
|  |         vertex 0 -0 25.400002 | ||||||
|  |         vertex 0 -0 0 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 -0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 84.906715 -0 9.525 | ||||||
|  |         vertex 64.480286 -0 0 | ||||||
|  |         vertex 88.9 -0 25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 105.33314 -0 19.05 | ||||||
|  |         vertex 84.906715 -0 9.525 | ||||||
|  |         vertex 88.9 -0 25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 87.15214 -0 -15.875 | ||||||
|  |         vertex 109.82398 -0 -31.75 | ||||||
|  |         vertex 78.613464 -0 -25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 105.33314 -0 19.05 | ||||||
|  |         vertex 143.37048 -0 50.800003 | ||||||
|  |         vertex 146.18599 -0 38.1 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 0 -0 25.400002 | ||||||
|  |         vertex 88.9 -0 25.400002 | ||||||
|  |         vertex 64.480286 -0 0 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 0 -0 25.400002 | ||||||
|  |         vertex 64.480286 -0 0 | ||||||
|  |         vertex 0 -0 0 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 143.37048 -0 50.800003 | ||||||
|  |         vertex 241.3 -0 50.800003 | ||||||
|  |         vertex 146.18599 -0 38.1 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 241.3 -0 50.800003 | ||||||
|  |         vertex 241.3 -0 38.1 | ||||||
|  |         vertex 146.18599 -0 38.1 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 -0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 105.33314 -0 19.05 | ||||||
|  |         vertex 88.9 -0 25.400002 | ||||||
|  |         vertex 143.37048 -0 50.800003 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 0.99999994 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 64.480286 -0 0 | ||||||
|  |         vertex 87.15214 -0 -15.875 | ||||||
|  |         vertex 78.613464 -0 -25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 109.82398 -0 -31.75 | ||||||
|  |         vertex 151.16339 -0 -76.2 | ||||||
|  |         vertex 78.613464 -0 -25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 155.16768 -0 -63.5 | ||||||
|  |         vertex 151.16339 -0 -76.2 | ||||||
|  |         vertex 109.82398 -0 -31.75 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 241.3 -0 -63.5 | ||||||
|  |         vertex 241.3 -0 -76.2 | ||||||
|  |         vertex 155.16768 -0 -63.5 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 155.16768 -0 -63.5 | ||||||
|  |         vertex 241.3 -0 -76.2 | ||||||
|  |         vertex 151.16339 -0 -76.2 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 64.480286 -0 0 | ||||||
|  |         vertex 78.613464 -0 -25.400002 | ||||||
|  |         vertex 0 -0 -25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 0 -0 -25.400002 | ||||||
|  |         vertex 0 -0 0 | ||||||
|  |         vertex 64.480286 -0 0 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -0 -1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 84.906715 -101.600006 9.525 | ||||||
|  |         vertex 88.9 -101.600006 25.400002 | ||||||
|  |         vertex 64.480286 -101.600006 0 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -0 -1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 105.33314 -101.600006 19.05 | ||||||
|  |         vertex 88.9 -101.600006 25.400002 | ||||||
|  |         vertex 84.906715 -101.600006 9.525 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -1 -0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 87.15214 -101.600006 -15.875 | ||||||
|  |         vertex 78.613464 -101.600006 -25.400002 | ||||||
|  |         vertex 109.82398 -101.600006 -31.75 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 105.33314 -101.600006 19.05 | ||||||
|  |         vertex 146.18599 -101.600006 38.1 | ||||||
|  |         vertex 143.37048 -101.600006 50.800003 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 0 -101.600006 25.400002 | ||||||
|  |         vertex 64.480286 -101.600006 0 | ||||||
|  |         vertex 88.9 -101.600006 25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 0 -101.600006 25.400002 | ||||||
|  |         vertex 0 -101.600006 0 | ||||||
|  |         vertex 64.480286 -101.600006 0 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 143.37048 -101.600006 50.800003 | ||||||
|  |         vertex 146.18599 -101.600006 38.1 | ||||||
|  |         vertex 241.3 -101.600006 50.800003 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -1 -0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 241.3 -101.600006 50.800003 | ||||||
|  |         vertex 146.18599 -101.600006 38.1 | ||||||
|  |         vertex 241.3 -101.600006 38.1 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 105.33314 -101.600006 19.05 | ||||||
|  |         vertex 143.37048 -101.600006 50.800003 | ||||||
|  |         vertex 88.9 -101.600006 25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -0.99999994 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 64.480286 -101.600006 0 | ||||||
|  |         vertex 78.613464 -101.600006 -25.400002 | ||||||
|  |         vertex 87.15214 -101.600006 -15.875 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -0 -1 -0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 109.82398 -101.600006 -31.75 | ||||||
|  |         vertex 78.613464 -101.600006 -25.400002 | ||||||
|  |         vertex 151.16339 -101.600006 -76.2 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -0 -1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 155.16768 -101.600006 -63.5 | ||||||
|  |         vertex 109.82398 -101.600006 -31.75 | ||||||
|  |         vertex 151.16339 -101.600006 -76.2 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal -0 -1 -0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 241.3 -101.600006 -63.5 | ||||||
|  |         vertex 155.16768 -101.600006 -63.5 | ||||||
|  |         vertex 241.3 -101.600006 -76.2 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -1 -0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 155.16768 -101.600006 -63.5 | ||||||
|  |         vertex 151.16339 -101.600006 -76.2 | ||||||
|  |         vertex 241.3 -101.600006 -76.2 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -1 -0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 64.480286 -101.600006 0 | ||||||
|  |         vertex 0 -101.600006 -25.400002 | ||||||
|  |         vertex 78.613464 -101.600006 -25.400002 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | facet normal 0 -1 0 | ||||||
|  |     outer loop | ||||||
|  |         vertex 0 -101.600006 -25.400002 | ||||||
|  |         vertex 64.480286 -101.600006 0 | ||||||
|  |         vertex 0 -101.600006 0 | ||||||
|  |     endloop | ||||||
|  | endfacet | ||||||
|  | endsolid unnamed | ||||||
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/stl-binary.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 221 KiB | 
							
								
								
									
										
											BIN
										
									
								
								e2e/playwright/export-snapshots/stl-binary.stl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										1016
									
								
								e2e/playwright/flow-tests.spec.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										21
									
								
								e2e/playwright/secrets.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,21 @@ | |||||||
|  | import { readFileSync } from 'fs' | ||||||
|  |  | ||||||
|  | const secrets: Record<string, string> = {} | ||||||
|  | try { | ||||||
|  |   const file = readFileSync('./e2e/playwright/playwright-secrets.env', 'utf8') | ||||||
|  |   file | ||||||
|  |     .split('\n') | ||||||
|  |     .filter((line) => line && line.length > 1) | ||||||
|  |     .forEach((line) => { | ||||||
|  |       const [key, value] = line.split('=') | ||||||
|  |       // prefer env vars over secrets file | ||||||
|  |       secrets[key] = process.env[key] || (value as any).replaceAll('"', '') | ||||||
|  |     }) | ||||||
|  | } catch (err) { | ||||||
|  |   // probably running in CI | ||||||
|  |   secrets.token = process.env.token || '' | ||||||
|  |   secrets.snapshottoken = process.env.snapshottoken || '' | ||||||
|  |   // add more env vars here to make them available in CI | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export { secrets } | ||||||
							
								
								
									
										479
									
								
								e2e/playwright/snapshot-tests.spec.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,479 @@ | |||||||
|  | import { test, expect } from '@playwright/test' | ||||||
|  | import { secrets } from './secrets' | ||||||
|  | import { getUtils } from './test-utils' | ||||||
|  | import { Models } from '@kittycad/lib' | ||||||
|  | import fsp from 'fs/promises' | ||||||
|  | import { spawn } from 'child_process' | ||||||
|  | import { APP_NAME } from 'lib/constants' | ||||||
|  |  | ||||||
|  | test.beforeEach(async ({ context, page }) => { | ||||||
|  |   await context.addInitScript(async (token) => { | ||||||
|  |     localStorage.setItem('TOKEN_PERSIST_KEY', token) | ||||||
|  |     localStorage.setItem('persistCode', ``) | ||||||
|  |     localStorage.setItem( | ||||||
|  |       'SETTINGS_PERSIST_KEY', | ||||||
|  |       JSON.stringify({ | ||||||
|  |         baseUnit: 'in', | ||||||
|  |         cameraControls: 'KittyCAD', | ||||||
|  |         defaultDirectory: '', | ||||||
|  |         defaultProjectName: 'project-$nnn', | ||||||
|  |         onboardingStatus: 'dismissed', | ||||||
|  |         showDebugPanel: true, | ||||||
|  |         textWrapping: 'On', | ||||||
|  |         theme: 'system', | ||||||
|  |         unitSystem: 'imperial', | ||||||
|  |       }) | ||||||
|  |     ) | ||||||
|  |   }, secrets.token) | ||||||
|  |   // reducedMotion kills animations, which speeds up tests and reduces flakiness | ||||||
|  |   await page.emulateMedia({ reducedMotion: 'reduce' }) | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | test.setTimeout(60000) | ||||||
|  |  | ||||||
|  | test('change camera, show planes', async ({ page, context }) => { | ||||||
|  |   const u = getUtils(page) | ||||||
|  |   await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |   await page.goto('/') | ||||||
|  |   await u.waitForAuthSkipAppStart() | ||||||
|  |   await u.openAndClearDebugPanel() | ||||||
|  |  | ||||||
|  |   const camPos: [number, number, number] = [0, 85, 85] | ||||||
|  |   await u.updateCamPosition(camPos) | ||||||
|  |  | ||||||
|  |   // rotate | ||||||
|  |   await u.closeDebugPanel() | ||||||
|  |   await page.mouse.move(700, 200) | ||||||
|  |   await page.mouse.down({ button: 'right' }) | ||||||
|  |   await page.mouse.move(600, 300) | ||||||
|  |   await page.mouse.up({ button: 'right' }) | ||||||
|  |  | ||||||
|  |   await u.openDebugPanel() | ||||||
|  |   await page.waitForTimeout(500) | ||||||
|  |   await u.clearCommandLogs() | ||||||
|  |  | ||||||
|  |   await page.getByRole('button', { name: 'Start Sketch' }).click() | ||||||
|  |  | ||||||
|  |   await u.closeDebugPanel() | ||||||
|  |  | ||||||
|  |   await expect(page).toHaveScreenshot({ | ||||||
|  |     maxDiffPixels: 100, | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await u.openAndClearDebugPanel() | ||||||
|  |   await page.getByRole('button', { name: 'Exit Sketch' }).click() | ||||||
|  |  | ||||||
|  |   await u.updateCamPosition(camPos) | ||||||
|  |  | ||||||
|  |   await u.clearCommandLogs() | ||||||
|  |   await u.closeDebugPanel() | ||||||
|  |   // pan | ||||||
|  |   await page.keyboard.down('Shift') | ||||||
|  |   await page.mouse.move(600, 200) | ||||||
|  |   await page.mouse.down({ button: 'right' }) | ||||||
|  |   await page.mouse.move(700, 200) | ||||||
|  |   await page.mouse.up({ button: 'right' }) | ||||||
|  |   await page.keyboard.up('Shift') | ||||||
|  |  | ||||||
|  |   await u.openDebugPanel() | ||||||
|  |   await page.waitForTimeout(300) | ||||||
|  |   await u.clearCommandLogs() | ||||||
|  |  | ||||||
|  |   await page.getByRole('button', { name: 'Start Sketch' }).click() | ||||||
|  |   await u.closeDebugPanel() | ||||||
|  |  | ||||||
|  |   await expect(page).toHaveScreenshot({ | ||||||
|  |     maxDiffPixels: 100, | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await u.openAndClearDebugPanel() | ||||||
|  |   await page.getByRole('button', { name: 'Exit Sketch' }).click() | ||||||
|  |  | ||||||
|  |   await u.updateCamPosition(camPos) | ||||||
|  |  | ||||||
|  |   await u.clearCommandLogs() | ||||||
|  |   await u.closeDebugPanel() | ||||||
|  |  | ||||||
|  |   // zoom | ||||||
|  |   await page.keyboard.down('Control') | ||||||
|  |   await page.mouse.move(700, 400) | ||||||
|  |   await page.mouse.down({ button: 'right' }) | ||||||
|  |   await page.mouse.move(700, 300) | ||||||
|  |   await page.mouse.up({ button: 'right' }) | ||||||
|  |   await page.keyboard.up('Control') | ||||||
|  |  | ||||||
|  |   await u.openDebugPanel() | ||||||
|  |   await page.waitForTimeout(300) | ||||||
|  |   await u.clearCommandLogs() | ||||||
|  |  | ||||||
|  |   await page.getByRole('button', { name: 'Start Sketch' }).click() | ||||||
|  |   await u.closeDebugPanel() | ||||||
|  |  | ||||||
|  |   await expect(page).toHaveScreenshot({ | ||||||
|  |     maxDiffPixels: 100, | ||||||
|  |   }) | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | test('exports of each format should work', async ({ page, context }) => { | ||||||
|  |   // FYI this test doesn't work with only engine running locally | ||||||
|  |   // And you will need to have the KittyCAD CLI installed | ||||||
|  |   const u = getUtils(page) | ||||||
|  |   await context.addInitScript(async () => { | ||||||
|  |     ;(window as any).playwrightSkipFilePicker = true | ||||||
|  |     localStorage.setItem( | ||||||
|  |       'persistCode', | ||||||
|  |       `const topAng = 25 | ||||||
|  | const bottomAng = 35 | ||||||
|  | const baseLen = 3.5 | ||||||
|  | const baseHeight = 1 | ||||||
|  | const totalHeightHalf = 2 | ||||||
|  | const armThick = 0.5 | ||||||
|  | const totalLen = 9.5 | ||||||
|  | const part001 = startSketchOn('-XZ') | ||||||
|  |   |> startProfileAt([0, 0], %) | ||||||
|  |   |> yLine(baseHeight, %) | ||||||
|  |   |> xLine(baseLen, %) | ||||||
|  |   |> angledLineToY({ | ||||||
|  |         angle: topAng, | ||||||
|  |         to: totalHeightHalf, | ||||||
|  |         tag: 'seg04' | ||||||
|  |       }, %) | ||||||
|  |   |> xLineTo({ to: totalLen, tag: 'seg03' }, %) | ||||||
|  |   |> yLine({ length: -armThick, tag: 'seg01' }, %) | ||||||
|  |   |> angledLineThatIntersects({ | ||||||
|  |         angle: HALF_TURN, | ||||||
|  |         offset: -armThick, | ||||||
|  |         intersectTag: 'seg04' | ||||||
|  |       }, %) | ||||||
|  |   |> angledLineToY([segAng('seg04', %) + 180, ZERO], %) | ||||||
|  |   |> angledLineToY({ | ||||||
|  |         angle: -bottomAng, | ||||||
|  |         to: -totalHeightHalf - armThick, | ||||||
|  |         tag: 'seg02' | ||||||
|  |       }, %) | ||||||
|  |   |> xLineTo(segEndX('seg03', %) + 0, %) | ||||||
|  |   |> yLine(-segLen('seg01', %), %) | ||||||
|  |   |> angledLineThatIntersects({ | ||||||
|  |         angle: HALF_TURN, | ||||||
|  |         offset: -armThick, | ||||||
|  |         intersectTag: 'seg02' | ||||||
|  |       }, %) | ||||||
|  |   |> angledLineToY([segAng('seg02', %) + 180, -baseHeight], %) | ||||||
|  |   |> xLineTo(ZERO, %) | ||||||
|  |   |> close(%)  | ||||||
|  |   |> extrude(4, %)` | ||||||
|  |     ) | ||||||
|  |   }) | ||||||
|  |   await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |   await page.goto('/') | ||||||
|  |   await u.waitForAuthSkipAppStart() | ||||||
|  |   await u.openDebugPanel() | ||||||
|  |   await u.expectCmdLog('[data-message-type="execution-done"]') | ||||||
|  |   await u.waitForCmdReceive('extrude') | ||||||
|  |   await page.waitForTimeout(1000) | ||||||
|  |   await u.clearAndCloseDebugPanel() | ||||||
|  |  | ||||||
|  |   await page.getByRole('button', { name: APP_NAME }).click() | ||||||
|  |  | ||||||
|  |   interface Paths { | ||||||
|  |     modelPath: string | ||||||
|  |     imagePath: string | ||||||
|  |     outputType: string | ||||||
|  |   } | ||||||
|  |   const doExport = async ( | ||||||
|  |     output: Models['OutputFormat_type'] | ||||||
|  |   ): Promise<Paths> => { | ||||||
|  |     await page.getByRole('button', { name: 'Export Model' }).click() | ||||||
|  |  | ||||||
|  |     const exportSelect = page.getByTestId('export-type') | ||||||
|  |     await exportSelect.selectOption({ label: output.type }) | ||||||
|  |  | ||||||
|  |     if ('storage' in output) { | ||||||
|  |       const storageSelect = page.getByTestId('export-storage') | ||||||
|  |       await storageSelect.selectOption({ label: output.storage }) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const downloadPromise = page.waitForEvent('download') | ||||||
|  |     await page.getByRole('button', { name: 'Export', exact: true }).click() | ||||||
|  |     const download = await downloadPromise | ||||||
|  |     const downloadLocationer = (extra = '', isImage = false) => | ||||||
|  |       `./e2e/playwright/export-snapshots/${output.type}-${ | ||||||
|  |         'storage' in output ? output.storage : '' | ||||||
|  |       }${extra}.${isImage ? 'png' : output.type}` | ||||||
|  |     const downloadLocation = downloadLocationer() | ||||||
|  |     const downloadLocation2 = downloadLocationer('-2') | ||||||
|  |  | ||||||
|  |     if (output.type === 'gltf' && output.storage === 'standard') { | ||||||
|  |       // wait for second download | ||||||
|  |       const download2 = await page.waitForEvent('download') | ||||||
|  |       await download.saveAs(downloadLocation) | ||||||
|  |       await download2.saveAs(downloadLocation2) | ||||||
|  |  | ||||||
|  |       // rewrite uri to reference our file name | ||||||
|  |       const fileContents = await fsp.readFile(downloadLocation, 'utf-8') | ||||||
|  |       const isJson = fileContents.includes('buffers') | ||||||
|  |       let contents = fileContents | ||||||
|  |       let reWriteLocation = downloadLocation | ||||||
|  |       let uri = downloadLocation2.split('/').pop() | ||||||
|  |       if (!isJson) { | ||||||
|  |         contents = await fsp.readFile(downloadLocation2, 'utf-8') | ||||||
|  |         reWriteLocation = downloadLocation2 | ||||||
|  |         uri = downloadLocation.split('/').pop() | ||||||
|  |       } | ||||||
|  |       contents = contents.replace(/"uri": ".*"/g, `"uri": "${uri}"`) | ||||||
|  |       await fsp.writeFile(reWriteLocation, contents) | ||||||
|  |     } else { | ||||||
|  |       await download.saveAs(downloadLocation) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (output.type === 'step') { | ||||||
|  |       // stable timestamps for step files | ||||||
|  |       const fileContents = await fsp.readFile(downloadLocation, 'utf-8') | ||||||
|  |       const newFileContents = fileContents.replace( | ||||||
|  |         /[0-9]{4}-[0-9]{2}-[0-9]{2}T[0-9]{2}:[0-9]{2}:[0-9]{2}\.[0-9]+[0-9]+[0-9]\+[0-9]{2}:[0-9]{2}/g, | ||||||
|  |         '1970-01-01T00:00:00.0+00:00' | ||||||
|  |       ) | ||||||
|  |       await fsp.writeFile(downloadLocation, newFileContents) | ||||||
|  |     } | ||||||
|  |     return { | ||||||
|  |       modelPath: downloadLocation, | ||||||
|  |       imagePath: downloadLocationer('', true), | ||||||
|  |       outputType: output.type, | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   const axisDirectionPair: Models['AxisDirectionPair_type'] = { | ||||||
|  |     axis: 'z', | ||||||
|  |     direction: 'positive', | ||||||
|  |   } | ||||||
|  |   const sysType: Models['System_type'] = { | ||||||
|  |     forward: axisDirectionPair, | ||||||
|  |     up: axisDirectionPair, | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const exportLocations: Paths[] = [] | ||||||
|  |  | ||||||
|  |   // NOTE it was easiest to leverage existing types and have doExport take Models['OutputFormat_type'] as in input | ||||||
|  |   // just note that only `type` and `storage` are used for selecting the drop downs is the app | ||||||
|  |   // the rest are only there to make typescript happy | ||||||
|  |   exportLocations.push( | ||||||
|  |     await doExport({ | ||||||
|  |       type: 'step', | ||||||
|  |       coords: sysType, | ||||||
|  |     }) | ||||||
|  |   ) | ||||||
|  |   exportLocations.push( | ||||||
|  |     await doExport({ | ||||||
|  |       type: 'ply', | ||||||
|  |       coords: sysType, | ||||||
|  |       selection: { type: 'default_scene' }, | ||||||
|  |       storage: 'ascii', | ||||||
|  |       units: 'in', | ||||||
|  |     }) | ||||||
|  |   ) | ||||||
|  |   exportLocations.push( | ||||||
|  |     await doExport({ | ||||||
|  |       type: 'ply', | ||||||
|  |       storage: 'binary_little_endian', | ||||||
|  |       coords: sysType, | ||||||
|  |       selection: { type: 'default_scene' }, | ||||||
|  |       units: 'in', | ||||||
|  |     }) | ||||||
|  |   ) | ||||||
|  |   exportLocations.push( | ||||||
|  |     await doExport({ | ||||||
|  |       type: 'ply', | ||||||
|  |       storage: 'binary_big_endian', | ||||||
|  |       coords: sysType, | ||||||
|  |       selection: { type: 'default_scene' }, | ||||||
|  |       units: 'in', | ||||||
|  |     }) | ||||||
|  |   ) | ||||||
|  |   exportLocations.push( | ||||||
|  |     await doExport({ | ||||||
|  |       type: 'stl', | ||||||
|  |       storage: 'ascii', | ||||||
|  |       coords: sysType, | ||||||
|  |       units: 'in', | ||||||
|  |       selection: { type: 'default_scene' }, | ||||||
|  |     }) | ||||||
|  |   ) | ||||||
|  |   exportLocations.push( | ||||||
|  |     await doExport({ | ||||||
|  |       type: 'stl', | ||||||
|  |       storage: 'binary', | ||||||
|  |       coords: sysType, | ||||||
|  |       units: 'in', | ||||||
|  |       selection: { type: 'default_scene' }, | ||||||
|  |     }) | ||||||
|  |   ) | ||||||
|  |   exportLocations.push( | ||||||
|  |     await doExport({ | ||||||
|  |       // obj seems to be a little flaky, times out tests sometimes | ||||||
|  |       type: 'obj', | ||||||
|  |       coords: sysType, | ||||||
|  |       units: 'in', | ||||||
|  |     }) | ||||||
|  |   ) | ||||||
|  |   exportLocations.push( | ||||||
|  |     await doExport({ | ||||||
|  |       type: 'gltf', | ||||||
|  |       storage: 'embedded', | ||||||
|  |       presentation: 'pretty', | ||||||
|  |     }) | ||||||
|  |   ) | ||||||
|  |   exportLocations.push( | ||||||
|  |     await doExport({ | ||||||
|  |       type: 'gltf', | ||||||
|  |       storage: 'binary', | ||||||
|  |       presentation: 'pretty', | ||||||
|  |     }) | ||||||
|  |   ) | ||||||
|  |  | ||||||
|  |   // TODO: gltfs don't seem to work with snap shots. push onto exportLocations once it's figured out | ||||||
|  |   await doExport({ | ||||||
|  |     type: 'gltf', | ||||||
|  |     storage: 'standard', | ||||||
|  |     presentation: 'pretty', | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   // close page to disconnect websocket since we can only have one open atm | ||||||
|  |   await page.close() | ||||||
|  |  | ||||||
|  |   // snapshot exports, good compromise to capture that exports are healthy without getting bogged down in "did the formatting change" changes | ||||||
|  |   // context: https://github.com/KittyCAD/modeling-app/issues/1222 | ||||||
|  |   for (const { modelPath, imagePath, outputType } of exportLocations) { | ||||||
|  |     const cliCommand = `export KITTYCAD_TOKEN=${secrets.snapshottoken} && kittycad file snapshot --output-format=png --src-format=${outputType} ${modelPath} ${imagePath}` | ||||||
|  |     const child = spawn(cliCommand, { shell: true }) | ||||||
|  |     await new Promise((resolve, reject) => { | ||||||
|  |       child.on('error', (code: any, msg: any) => { | ||||||
|  |         console.log('error', code, msg) | ||||||
|  |         reject() | ||||||
|  |       }) | ||||||
|  |       child.on('exit', (code, msg) => { | ||||||
|  |         console.log('exit', code, msg) | ||||||
|  |         if (code !== 0) { | ||||||
|  |           reject(`exit code ${code} for model ${modelPath}`) | ||||||
|  |         } else { | ||||||
|  |           resolve(true) | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  |       child.stderr.on('data', (data) => console.log(`stderr: ${data}`)) | ||||||
|  |       child.stdout.on('data', (data) => console.log(`stdout: ${data}`)) | ||||||
|  |     }) | ||||||
|  |   } | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | test('extrude on each default plane should be stable', async ({ | ||||||
|  |   page, | ||||||
|  |   context, | ||||||
|  | }) => { | ||||||
|  |   const u = getUtils(page) | ||||||
|  |   const makeCode = (plane = 'XY') => `const part001 = startSketchOn('${plane}') | ||||||
|  |   |> startProfileAt([0.70, 0.44], %) | ||||||
|  |   |> line([0.66, -0.02], %) | ||||||
|  |   |> line([0.28, 0.50], %) | ||||||
|  |   |> line([-0.56, 0.44], %) | ||||||
|  |   |> line([-0.54, -0.38], %) | ||||||
|  |   |> close(%) | ||||||
|  |   |> extrude(1.00, %) | ||||||
|  | ` | ||||||
|  |   await context.addInitScript(async (code) => { | ||||||
|  |     localStorage.setItem('persistCode', code) | ||||||
|  |   }, makeCode('XY')) | ||||||
|  |   await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |   await page.goto('/') | ||||||
|  |   await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|  |   // wait for execution done | ||||||
|  |   await u.openDebugPanel() | ||||||
|  |   await u.expectCmdLog('[data-message-type="execution-done"]') | ||||||
|  |   await u.clearAndCloseDebugPanel() | ||||||
|  |  | ||||||
|  |   await page.getByText('Code').click() | ||||||
|  |   await expect(page).toHaveScreenshot({ | ||||||
|  |     maxDiffPixels: 100, | ||||||
|  |   }) | ||||||
|  |   await page.getByText('Code').click() | ||||||
|  |  | ||||||
|  |   const runSnapshotsForOtherPlanes = async (plane = 'XY') => { | ||||||
|  |     // clear code | ||||||
|  |     await u.removeCurrentCode() | ||||||
|  |     // add makeCode('XZ') | ||||||
|  |     await page.locator('.cm-content').fill(makeCode(plane)) | ||||||
|  |     // wait for execution done | ||||||
|  |     await u.openDebugPanel() | ||||||
|  |     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||||
|  |     await u.clearAndCloseDebugPanel() | ||||||
|  |  | ||||||
|  |     await page.getByText('Code').click() | ||||||
|  |     await expect(page).toHaveScreenshot({ | ||||||
|  |       maxDiffPixels: 100, | ||||||
|  |     }) | ||||||
|  |     await page.getByText('Code').click() | ||||||
|  |   } | ||||||
|  |   await runSnapshotsForOtherPlanes('-XY') | ||||||
|  |  | ||||||
|  |   await runSnapshotsForOtherPlanes('XZ') | ||||||
|  |   await runSnapshotsForOtherPlanes('-XZ') | ||||||
|  |  | ||||||
|  |   await runSnapshotsForOtherPlanes('YZ') | ||||||
|  |   await runSnapshotsForOtherPlanes('-YZ') | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | test('Draft segments should look right', async ({ page }) => { | ||||||
|  |   const u = getUtils(page) | ||||||
|  |   await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |   const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|  |   await page.goto('/') | ||||||
|  |   await u.waitForAuthSkipAppStart() | ||||||
|  |   await u.openDebugPanel() | ||||||
|  |  | ||||||
|  |   await expect(page.getByRole('button', { name: 'Start Sketch' })).toBeVisible() | ||||||
|  |  | ||||||
|  |   // click on "Start Sketch" button | ||||||
|  |   await u.clearCommandLogs() | ||||||
|  |   await u.doAndWaitForImageDiff( | ||||||
|  |     () => page.getByRole('button', { name: 'Start Sketch' }).click(), | ||||||
|  |     200 | ||||||
|  |   ) | ||||||
|  |  | ||||||
|  |   // select a plane | ||||||
|  |   await page.mouse.click(700, 200) | ||||||
|  |  | ||||||
|  |   await expect(page.locator('.cm-content')).toHaveText( | ||||||
|  |     `const part001 = startSketchOn('-XZ')` | ||||||
|  |   ) | ||||||
|  |  | ||||||
|  |   await page.waitForTimeout(300) // TODO detect animation ending, or disable animation | ||||||
|  |  | ||||||
|  |   const startXPx = 600 | ||||||
|  |   await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) | ||||||
|  |   const startAt = '[23.74, -32.03]' | ||||||
|  |   await expect(page.locator('.cm-content')) | ||||||
|  |     .toHaveText(`const part001 = startSketchOn('-XZ') | ||||||
|  |   |> startProfileAt(${startAt}, %)`) | ||||||
|  |   await page.waitForTimeout(100) | ||||||
|  |  | ||||||
|  |   await u.closeDebugPanel() | ||||||
|  |   await page.mouse.move(startXPx + PUR * 20, 500 - PUR * 10) | ||||||
|  |   await expect(page).toHaveScreenshot({ | ||||||
|  |     maxDiffPixels: 100, | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) | ||||||
|  |   await page.waitForTimeout(100) | ||||||
|  |  | ||||||
|  |   const num = 23.97 | ||||||
|  |   await expect(page.locator('.cm-content')) | ||||||
|  |     .toHaveText(`const part001 = startSketchOn('-XZ') | ||||||
|  |   |> startProfileAt(${startAt}, %) | ||||||
|  |   |> line([${num}, 0], %)`) | ||||||
|  |  | ||||||
|  |   await page.getByRole('button', { name: 'Tangential Arc' }).click() | ||||||
|  |  | ||||||
|  |   await page.mouse.move(startXPx + PUR * 30, 500 - PUR * 20, { steps: 10 }) | ||||||
|  |  | ||||||
|  |   await expect(page).toHaveScreenshot({ | ||||||
|  |     maxDiffPixels: 100, | ||||||
|  |   }) | ||||||
|  | }) | ||||||
| After Width: | Height: | Size: 41 KiB | 
| After Width: | Height: | Size: 44 KiB | 
| After Width: | Height: | Size: 95 KiB | 
| After Width: | Height: | Size: 79 KiB | 
| After Width: | Height: | Size: 111 KiB | 
| After Width: | Height: | Size: 53 KiB | 
| After Width: | Height: | Size: 52 KiB | 
| After Width: | Height: | Size: 53 KiB | 
| After Width: | Height: | Size: 56 KiB | 
| After Width: | Height: | Size: 50 KiB | 
| After Width: | Height: | Size: 52 KiB | 
							
								
								
									
										167
									
								
								e2e/playwright/test-utils.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,167 @@ | |||||||
|  | import { expect, Page } from '@playwright/test' | ||||||
|  | import { EngineCommand } from '../../src/lang/std/engineConnection' | ||||||
|  | import fsp from 'fs/promises' | ||||||
|  | import pixelMatch from 'pixelmatch' | ||||||
|  | import { PNG } from 'pngjs' | ||||||
|  |  | ||||||
|  | async function waitForPageLoad(page: Page) { | ||||||
|  |   // wait for 'Loading stream...' spinner | ||||||
|  |   await page.getByTestId('loading-stream').waitFor() | ||||||
|  |   // wait for all spinners to be gone | ||||||
|  |   await page.getByTestId('loading').waitFor({ state: 'detached' }) | ||||||
|  |  | ||||||
|  |   await page.getByTestId('start-sketch').waitFor() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function removeCurrentCode(page: Page) { | ||||||
|  |   const hotkey = process.platform === 'darwin' ? 'Meta' : 'Control' | ||||||
|  |   await page.click('.cm-content') | ||||||
|  |   await page.keyboard.down(hotkey) | ||||||
|  |   await page.keyboard.press('a') | ||||||
|  |   await page.keyboard.up(hotkey) | ||||||
|  |   await page.keyboard.press('Backspace') | ||||||
|  |   await expect(page.locator('.cm-content')).toHaveText('') | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function sendCustomCmd(page: Page, cmd: EngineCommand) { | ||||||
|  |   await page.fill('[data-testid="custom-cmd-input"]', JSON.stringify(cmd)) | ||||||
|  |   await page.click('[data-testid="custom-cmd-send-button"]') | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function clearCommandLogs(page: Page) { | ||||||
|  |   await page.click('[data-testid="clear-commands"]') | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function expectCmdLog(page: Page, locatorStr: string) { | ||||||
|  |   await expect(page.locator(locatorStr)).toBeVisible() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function waitForDefaultPlanesToBeVisible(page: Page) { | ||||||
|  |   await page.waitForFunction( | ||||||
|  |     () => | ||||||
|  |       document.querySelectorAll('[data-receive-command-type="object_visible"]') | ||||||
|  |         .length >= 3 | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function openDebugPanel(page: Page) { | ||||||
|  |   const isOpen = | ||||||
|  |     (await page | ||||||
|  |       .locator('[data-testid="debug-panel"]') | ||||||
|  |       ?.getAttribute('open')) === '' | ||||||
|  |  | ||||||
|  |   if (!isOpen) { | ||||||
|  |     await page.getByText('Debug').click() | ||||||
|  |     await page.getByTestId('debug-panel').and(page.locator('[open]')).waitFor() | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function closeDebugPanel(page: Page) { | ||||||
|  |   const isOpen = | ||||||
|  |     (await page.getByTestId('debug-panel')?.getAttribute('open')) === '' | ||||||
|  |   if (isOpen) { | ||||||
|  |     await page.getByText('Debug').click() | ||||||
|  |     await page | ||||||
|  |       .getByTestId('debug-panel') | ||||||
|  |       .and(page.locator(':not([open])')) | ||||||
|  |       .waitFor() | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function waitForCmdReceive(page: Page, commandType: string) { | ||||||
|  |   return page | ||||||
|  |     .locator(`[data-receive-command-type="${commandType}"]`) | ||||||
|  |     .first() | ||||||
|  |     .waitFor() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export function getUtils(page: Page) { | ||||||
|  |   return { | ||||||
|  |     waitForAuthSkipAppStart: () => waitForPageLoad(page), | ||||||
|  |     removeCurrentCode: () => removeCurrentCode(page), | ||||||
|  |     sendCustomCmd: (cmd: EngineCommand) => sendCustomCmd(page, cmd), | ||||||
|  |     updateCamPosition: async (xyz: [number, number, number]) => { | ||||||
|  |       const fillInput = async () => { | ||||||
|  |         await page.fill('[data-testid="cam-x-position"]', String(xyz[0])) | ||||||
|  |         await page.fill('[data-testid="cam-y-position"]', String(xyz[1])) | ||||||
|  |         await page.fill('[data-testid="cam-z-position"]', String(xyz[2])) | ||||||
|  |       } | ||||||
|  |       await fillInput() | ||||||
|  |       await page.waitForTimeout(100) | ||||||
|  |       await fillInput() | ||||||
|  |       await page.waitForTimeout(100) | ||||||
|  |       await fillInput() | ||||||
|  |       await page.waitForTimeout(100) | ||||||
|  |     }, | ||||||
|  |     clearCommandLogs: () => clearCommandLogs(page), | ||||||
|  |     expectCmdLog: (locatorStr: string) => expectCmdLog(page, locatorStr), | ||||||
|  |     openDebugPanel: () => openDebugPanel(page), | ||||||
|  |     closeDebugPanel: () => closeDebugPanel(page), | ||||||
|  |     openAndClearDebugPanel: async () => { | ||||||
|  |       await openDebugPanel(page) | ||||||
|  |       return clearCommandLogs(page) | ||||||
|  |     }, | ||||||
|  |     clearAndCloseDebugPanel: async () => { | ||||||
|  |       await clearCommandLogs(page) | ||||||
|  |       return closeDebugPanel(page) | ||||||
|  |     }, | ||||||
|  |     waitForCmdReceive: (commandType: string) => | ||||||
|  |       waitForCmdReceive(page, commandType), | ||||||
|  |     doAndWaitForCmd: async ( | ||||||
|  |       fn: () => Promise<void>, | ||||||
|  |       commandType: string, | ||||||
|  |       endWithDebugPanelOpen = true | ||||||
|  |     ) => { | ||||||
|  |       await openDebugPanel(page) | ||||||
|  |       await clearCommandLogs(page) | ||||||
|  |       await closeDebugPanel(page) | ||||||
|  |       await fn() | ||||||
|  |       await openDebugPanel(page) | ||||||
|  |       await waitForCmdReceive(page, commandType) | ||||||
|  |       if (!endWithDebugPanelOpen) { | ||||||
|  |         await closeDebugPanel(page) | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     doAndWaitForImageDiff: (fn: () => Promise<any>, diffCount = 200) => | ||||||
|  |       new Promise(async (resolve) => { | ||||||
|  |         await page.screenshot({ | ||||||
|  |           path: './e2e/playwright/temp1.png', | ||||||
|  |           fullPage: true, | ||||||
|  |         }) | ||||||
|  |         await fn() | ||||||
|  |         const isImageDiff = async () => { | ||||||
|  |           await page.screenshot({ | ||||||
|  |             path: './e2e/playwright/temp2.png', | ||||||
|  |             fullPage: true, | ||||||
|  |           }) | ||||||
|  |           const screenshot1 = PNG.sync.read( | ||||||
|  |             await fsp.readFile('./e2e/playwright/temp1.png') | ||||||
|  |           ) | ||||||
|  |           const screenshot2 = PNG.sync.read( | ||||||
|  |             await fsp.readFile('./e2e/playwright/temp2.png') | ||||||
|  |           ) | ||||||
|  |           const actualDiffCount = pixelMatch( | ||||||
|  |             screenshot1.data, | ||||||
|  |             screenshot2.data, | ||||||
|  |             null, | ||||||
|  |             screenshot1.width, | ||||||
|  |             screenshot2.height | ||||||
|  |           ) | ||||||
|  |           return actualDiffCount > diffCount | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // run isImageDiff every 50ms until it returns true or 5 seconds have passed (100 times) | ||||||
|  |         let count = 0 | ||||||
|  |         const interval = setInterval(async () => { | ||||||
|  |           count++ | ||||||
|  |           if (await isImageDiff()) { | ||||||
|  |             clearInterval(interval) | ||||||
|  |             resolve(true) | ||||||
|  |           } else if (count > 100) { | ||||||
|  |             clearInterval(interval) | ||||||
|  |             resolve(false) | ||||||
|  |           } | ||||||
|  |         }, 50) | ||||||
|  |       }), | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										105
									
								
								e2e/tauri/specs/auth.e2e.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,105 @@ | |||||||
|  | import { browser, $, expect } from '@wdio/globals' | ||||||
|  | import fs from 'fs/promises' | ||||||
|  |  | ||||||
|  | const defaultDir = `${process.env.HOME}/Documents/zoo-modeling-app-projects` | ||||||
|  | const userCodeDir = '/tmp/kittycad_user_code' | ||||||
|  |  | ||||||
|  | async function click(element: WebdriverIO.Element): Promise<void> { | ||||||
|  |   // Workaround for .click(), see https://github.com/tauri-apps/tauri/issues/6541 | ||||||
|  |   await element.waitForClickable() | ||||||
|  |   await browser.execute('arguments[0].click();', element) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | describe('ZMA (Tauri, Linux)', () => { | ||||||
|  |   it('opens the auth page and signs in', async () => { | ||||||
|  |     // Clean up filesystem from previous tests | ||||||
|  |     await new Promise((resolve) => setTimeout(resolve, 100)) | ||||||
|  |     await fs.rm(defaultDir, { force: true, recursive: true }) | ||||||
|  |     await fs.rm(userCodeDir, { force: true }) | ||||||
|  |  | ||||||
|  |     const signInButton = await $('[data-testid="sign-in-button"]') | ||||||
|  |     expect(await signInButton.getText()).toEqual('Sign in') | ||||||
|  |  | ||||||
|  |     await click(signInButton) | ||||||
|  |     await new Promise((resolve) => setTimeout(resolve, 2000)) | ||||||
|  |  | ||||||
|  |     // Get from main.rs | ||||||
|  |     const userCode = await ( | ||||||
|  |       await fs.readFile('/tmp/kittycad_user_code') | ||||||
|  |     ).toString() | ||||||
|  |     console.log(`Found user code ${userCode}`) | ||||||
|  |  | ||||||
|  |     // Device flow: verify | ||||||
|  |     const token = process.env.KITTYCAD_API_TOKEN | ||||||
|  |     const headers = { | ||||||
|  |       Authorization: `Bearer ${token}`, | ||||||
|  |       Accept: 'application/json', | ||||||
|  |       'Content-Type': 'application/json', | ||||||
|  |     } | ||||||
|  |     const apiBaseUrl = process.env.VITE_KC_API_BASE_URL | ||||||
|  |     const verifyUrl = `${apiBaseUrl}/oauth2/device/verify?user_code=${userCode}` | ||||||
|  |     console.log(`GET ${verifyUrl}`) | ||||||
|  |     const vr = await fetch(verifyUrl, { headers }) | ||||||
|  |     console.log(vr.status) | ||||||
|  |  | ||||||
|  |     // Device flow: confirm | ||||||
|  |     const confirmUrl = `${apiBaseUrl}/oauth2/device/confirm` | ||||||
|  |     const data = JSON.stringify({ user_code: userCode }) | ||||||
|  |     console.log(`POST ${confirmUrl} ${data}`) | ||||||
|  |     const cr = await fetch(confirmUrl, { | ||||||
|  |       headers, | ||||||
|  |       method: 'POST', | ||||||
|  |       body: data, | ||||||
|  |     }) | ||||||
|  |     console.log(cr.status) | ||||||
|  |  | ||||||
|  |     // Now should be signed in | ||||||
|  |     const newFileButton = await $('[data-testid="home-new-file"]') | ||||||
|  |     expect(await newFileButton.getText()).toEqual('New file') | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   it('opens the settings page, checks filesystem settings, and closes the settings page', async () => { | ||||||
|  |     const menuButton = await $('[data-testid="user-sidebar-toggle"]') | ||||||
|  |     await click(menuButton) | ||||||
|  |  | ||||||
|  |     const settingsButton = await $('[data-testid="settings-button"]') | ||||||
|  |     await click(settingsButton) | ||||||
|  |  | ||||||
|  |     const defaultDirInput = await $('[data-testid="default-directory-input"]') | ||||||
|  |     expect(await defaultDirInput.getValue()).toEqual(defaultDir) | ||||||
|  |  | ||||||
|  |     const nameInput = await $('[data-testid="name-input"]') | ||||||
|  |     expect(await nameInput.getValue()).toEqual('project-$nnn') | ||||||
|  |  | ||||||
|  |     const closeButton = await $('[data-testid="close-button"]') | ||||||
|  |     await click(closeButton) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   it('checks that no file exists, creates a new file', async () => { | ||||||
|  |     const homeSection = await $('[data-testid="home-section"]') | ||||||
|  |     expect(await homeSection.getText()).toContain('No Projects found') | ||||||
|  |  | ||||||
|  |     const newFileButton = await $('[data-testid="home-new-file"]') | ||||||
|  |     await click(newFileButton) | ||||||
|  |     await new Promise((resolve) => setTimeout(resolve, 1000)) | ||||||
|  |  | ||||||
|  |     expect(await homeSection.getText()).toContain('project-000') | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   it('opens the new file and expects a loading stream', async () => { | ||||||
|  |     const projectLink = await $('[data-testid="project-link"]') | ||||||
|  |     await click(projectLink) | ||||||
|  |     const loadingText = await $('[data-testid="loading-stream"]') | ||||||
|  |     expect(await loadingText.getText()).toContain('Loading stream...') | ||||||
|  |     await browser.execute('window.location.href = "tauri://localhost/home"') | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   it('signs out', async () => { | ||||||
|  |     const menuButton = await $('[data-testid="user-sidebar-toggle"]') | ||||||
|  |     await click(menuButton) | ||||||
|  |     const signoutButton = await $('[data-testid="user-sidebar-sign-out"]') | ||||||
|  |     await click(signoutButton) | ||||||
|  |     const newSignInButton = await $('[data-testid="sign-in-button"]') | ||||||
|  |     expect(await newSignInButton.getText()).toEqual('Sign in') | ||||||
|  |   }) | ||||||
|  | }) | ||||||
							
								
								
									
										11
									
								
								index.html
									
									
									
									
									
								
							
							
						
						| @ -7,12 +7,17 @@ | |||||||
|     <meta name="theme-color" content="#000000" /> |     <meta name="theme-color" content="#000000" /> | ||||||
|     <meta |     <meta | ||||||
|       name="description" |       name="description" | ||||||
|       content="An open-source CAD modeling tool from the future by KittyCAD." |       content="An open-source CAD modeling tool from the future by Zoo." | ||||||
|     /> |     /> | ||||||
|     <link rel="apple-touch-icon" href="/logo192.png" /> |     <link rel="apple-touch-icon" href="/logo192.png" /> | ||||||
|     <link rel="manifest" href="/manifest.json" /> |     <link rel="manifest" href="/manifest.json" /> | ||||||
|     <script defer data-domain="app.kittycad.io" src="https://plausible.corp.kittycad.io/js/script.js"></script> |     <link rel="stylesheet" href="https://use.typekit.net/zzv8rvm.css" /> | ||||||
|     <title>KittyCAD Modeling App</title> |     <script | ||||||
|  |       defer | ||||||
|  |       data-domain="app.zoo.dev" | ||||||
|  |       src="https://plausible.corp.zoo.dev/js/script.js" | ||||||
|  |     ></script> | ||||||
|  |     <title>Zoo Modeling App</title> | ||||||
|   </head> |   </head> | ||||||
|   <body class="body-bg"> |   <body class="body-bg"> | ||||||
|     <noscript>You need to enable JavaScript to run this app.</noscript> |     <noscript>You need to enable JavaScript to run this app.</noscript> | ||||||
|  | |||||||
							
								
								
									
										69
									
								
								make-release.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @ -0,0 +1,69 @@ | |||||||
|  | #!/bin/bash | ||||||
|  |  | ||||||
|  | if ! git diff-index --quiet HEAD --; then | ||||||
|  |   echo "Please stash uncommitted changes before running release script" | ||||||
|  |   exit 1 | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | git checkout main | ||||||
|  | git pull | ||||||
|  | git fetch --all | ||||||
|  |  | ||||||
|  | # Get the latest semver tag from git | ||||||
|  | latest_tag=$(jq -r '.version' package.json) | ||||||
|  | latest_tag="v$latest_tag" | ||||||
|  |  | ||||||
|  | # Print the latest semver tag | ||||||
|  | echo "Latest semver tag: $latest_tag" | ||||||
|  |  | ||||||
|  | # Function to bump version numbers | ||||||
|  | bump_version() { | ||||||
|  |   local version=$1 | ||||||
|  |   local bump_type=$2 | ||||||
|  |   local major=$(echo $version | cut -d '.' -f 1 | sed 's/v//') | ||||||
|  |   local minor=$(echo $version | cut -d '.' -f 2) | ||||||
|  |   local patch=$(echo $version | cut -d '.' -f 3) | ||||||
|  |  | ||||||
|  |   case "$bump_type" in | ||||||
|  |     major) | ||||||
|  |       major=$((major + 1)) | ||||||
|  |       minor=0 | ||||||
|  |       patch=0 | ||||||
|  |       ;; | ||||||
|  |     minor) | ||||||
|  |       minor=$((minor + 1)) | ||||||
|  |       patch=0 | ||||||
|  |       ;; | ||||||
|  |     *) | ||||||
|  |       patch=$((patch + 1)) | ||||||
|  |       ;; | ||||||
|  |   esac | ||||||
|  |  | ||||||
|  |   echo "v${major}.${minor}.${patch}" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | # Determine the type of bump based on the argument | ||||||
|  | bump_type=${1:-patch} | ||||||
|  |  | ||||||
|  | # Bump the version | ||||||
|  | new_version=$(bump_version $latest_tag $bump_type) | ||||||
|  |  | ||||||
|  | # Print the new semver tag | ||||||
|  | echo "New semver tag: $new_version" | ||||||
|  | new_version_number=${new_version:1} | ||||||
|  | echo "New version number without 'v': $new_version_number" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | git checkout -b "cut-release-$new_version" | ||||||
|  |  | ||||||
|  | echo "$(jq --arg v "$new_version_number" '.version=$v' package.json --indent 2)" > package.json | ||||||
|  | echo "$(jq --arg v "$new_version_number" '.package.version=$v' src-tauri/tauri.conf.json --indent 2)" > src-tauri/tauri.conf.json | ||||||
|  |  | ||||||
|  | git add package.json src-tauri/tauri.conf.json | ||||||
|  | git commit -m "Cut release $new_version" | ||||||
|  |  | ||||||
|  | echo "" | ||||||
|  | echo "Versions has been bumped in relevant json files, a branch has been created and committed to." | ||||||
|  | echo "" | ||||||
|  | echo "What's left for you to do is, push the branch and make the release PR." | ||||||
|  | echo "" | ||||||
							
								
								
									
										104
									
								
								package.json
									
									
									
									
									
								
							
							
						
						| @ -1,37 +1,41 @@ | |||||||
| { | { | ||||||
|   "name": "untitled-app", |   "name": "untitled-app", | ||||||
|   "version": "0.8.2", |   "version": "0.15.2", | ||||||
|   "private": true, |   "private": true, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@codemirror/autocomplete": "^6.9.0", |     "@codemirror/autocomplete": "^6.10.2", | ||||||
|     "@fortawesome/fontawesome-svg-core": "^6.4.2", |     "@fortawesome/fontawesome-svg-core": "^6.4.2", | ||||||
|     "@fortawesome/free-brands-svg-icons": "^6.4.2", |     "@fortawesome/free-brands-svg-icons": "^6.4.2", | ||||||
|     "@fortawesome/free-solid-svg-icons": "^6.4.2", |     "@fortawesome/free-solid-svg-icons": "^6.4.2", | ||||||
|     "@fortawesome/react-fontawesome": "^0.2.0", |     "@fortawesome/react-fontawesome": "^0.2.0", | ||||||
|     "@headlessui/react": "^1.7.13", |     "@headlessui/react": "^1.7.17", | ||||||
|     "@headlessui/tailwindcss": "^0.2.0", |     "@headlessui/tailwindcss": "^0.2.0", | ||||||
|     "@kittycad/lib": "^0.0.37", |     "@kittycad/lib": "^0.0.53", | ||||||
|     "@lezer/javascript": "^1.4.7", |     "@lezer/javascript": "^1.4.9", | ||||||
|     "@open-rpc/client-js": "^1.8.1", |     "@open-rpc/client-js": "^1.8.1", | ||||||
|     "@react-hook/resize-observer": "^1.2.6", |     "@react-hook/resize-observer": "^1.2.6", | ||||||
|     "@replit/codemirror-interact": "^6.3.0", |     "@replit/codemirror-interact": "^6.3.0", | ||||||
|     "@sentry/react": "^7.65.0", |     "@sentry/react": "^7.77.0", | ||||||
|     "@tauri-apps/api": "^1.3.0", |     "@tauri-apps/api": "^1.5.1", | ||||||
|     "@testing-library/jest-dom": "^5.14.1", |     "@testing-library/jest-dom": "^5.14.1", | ||||||
|     "@testing-library/react": "^13.0.0", |     "@testing-library/react": "^14.0.0", | ||||||
|     "@testing-library/user-event": "^13.2.1", |     "@testing-library/user-event": "^14.5.1", | ||||||
|     "@ts-stack/markdown": "^1.5.0", |     "@ts-stack/markdown": "^1.5.0", | ||||||
|  |     "@tweenjs/tween.js": "^23.1.1", | ||||||
|     "@types/node": "^16.7.13", |     "@types/node": "^16.7.13", | ||||||
|     "@types/react": "^18.0.0", |     "@types/react": "^18.2.41", | ||||||
|     "@types/react-dom": "^18.0.0", |     "@types/react-dom": "^18.0.0", | ||||||
|     "@uiw/react-codemirror": "^4.21.13", |     "@uiw/react-codemirror": "^4.21.20", | ||||||
|  |     "@xstate/inspect": "^0.8.0", | ||||||
|     "@xstate/react": "^3.2.2", |     "@xstate/react": "^3.2.2", | ||||||
|     "crypto-js": "^4.1.1", |     "crypto-js": "^4.2.0", | ||||||
|  |     "debounce-promise": "^3.1.2", | ||||||
|     "formik": "^2.4.3", |     "formik": "^2.4.3", | ||||||
|     "fuse.js": "^6.6.2", |     "fuse.js": "^7.0.0", | ||||||
|     "http-server": "^14.1.1", |     "http-server": "^14.1.1", | ||||||
|     "json-rpc-2.0": "^1.6.0", |     "json-rpc-2.0": "^1.6.0", | ||||||
|     "re-resizable": "^6.9.9", |     "node-fetch": "^3.3.2", | ||||||
|  |     "re-resizable": "^6.9.11", | ||||||
|     "react": "^18.2.0", |     "react": "^18.2.0", | ||||||
|     "react-dom": "^18.2.0", |     "react-dom": "^18.2.0", | ||||||
|     "react-hot-toast": "^2.4.1", |     "react-hot-toast": "^2.4.1", | ||||||
| @ -41,23 +45,26 @@ | |||||||
|     "react-modal-promise": "^1.0.2", |     "react-modal-promise": "^1.0.2", | ||||||
|     "react-router-dom": "^6.14.2", |     "react-router-dom": "^6.14.2", | ||||||
|     "sketch-helpers": "^0.0.4", |     "sketch-helpers": "^0.0.4", | ||||||
|     "swr": "^2.0.4", |     "swr": "^2.2.2", | ||||||
|     "tauri-plugin-fs-extra-api": "https://github.com/tauri-apps/tauri-plugin-fs-extra#v1", |     "tauri-plugin-fs-extra-api": "https://github.com/tauri-apps/tauri-plugin-fs-extra#v1", | ||||||
|  |     "three": "^0.160.0", | ||||||
|     "toml": "^3.0.0", |     "toml": "^3.0.0", | ||||||
|     "ts-node": "^10.9.1", |     "ts-node": "^10.9.1", | ||||||
|     "typescript": "^4.4.2", |     "typescript": "^5.2.2", | ||||||
|     "uuid": "^9.0.0", |     "uuid": "^9.0.1", | ||||||
|     "vitest": "^0.34.1", |     "vitest": "^1.3.1", | ||||||
|     "vscode-jsonrpc": "^8.1.0", |     "vscode-jsonrpc": "^8.1.0", | ||||||
|     "vscode-languageserver-protocol": "^3.17.3", |     "vscode-languageserver-protocol": "^3.17.5", | ||||||
|     "wasm-pack": "^0.12.1", |     "wasm-pack": "^0.12.1", | ||||||
|     "web-vitals": "^2.1.0", |     "web-vitals": "^3.5.0", | ||||||
|     "ws": "^8.13.0", |     "ws": "^8.13.0", | ||||||
|     "xstate": "^4.38.2", |     "xstate": "^4.38.2", | ||||||
|     "zustand": "^4.1.4" |     "zustand": "^4.4.5" | ||||||
|   }, |   }, | ||||||
|   "scripts": { |   "scripts": { | ||||||
|     "start": "vite", |     "start": "vite", | ||||||
|  |     "start:prod": "vite preview --port=3000", | ||||||
|  |     "serve": "vite serve --port=3000", | ||||||
|     "build": "curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y && source \"$HOME/.cargo/env\" && curl https://rustwasm.github.io/wasm-pack/installer/init.sh -sSf | sh -s -- -y && yarn build:wasm && vite build", |     "build": "curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y && source \"$HOME/.cargo/env\" && curl https://rustwasm.github.io/wasm-pack/installer/init.sh -sSf | sh -s -- -y && yarn build:wasm && vite build", | ||||||
|     "build:local": "vite build", |     "build:local": "vite build", | ||||||
|     "build:both": "vite build", |     "build:both": "vite build", | ||||||
| @ -66,17 +73,20 @@ | |||||||
|     "test": "vitest --mode development", |     "test": "vitest --mode development", | ||||||
|     "test:nowatch": "vitest run --mode development", |     "test:nowatch": "vitest run --mode development", | ||||||
|     "test:rust": "(cd src/wasm-lib && cargo test --all && cargo clippy --all --tests --benches)", |     "test:rust": "(cd src/wasm-lib && cargo test --all && cargo clippy --all --tests --benches)", | ||||||
|     "test:cov": "vitest run --coverage --mode development", |     "test:e2e:tauri": "E2E_TAURI_ENABLED=true TS_NODE_COMPILER_OPTIONS='{\"module\": \"commonjs\"}' wdio run wdio.conf.ts", | ||||||
|     "simpleserver:ci": "yarn pretest && http-server ./public --cors -p 3000 &", |     "simpleserver:ci": "yarn pretest && http-server ./public --cors -p 3000 &", | ||||||
|     "simpleserver": "yarn pretest && http-server ./public --cors -p 3000", |     "simpleserver": "yarn pretest && http-server ./public --cors -p 3000", | ||||||
|     "fmt": "prettier --write ./src", |     "fmt": "prettier --write ./src && prettier --write ./e2e", | ||||||
|     "fmt-check": "prettier --check ./src", |     "fmt-check": "prettier --check ./src && prettier --check ./e2e", | ||||||
|  |     "build:wasm-dev": "(cd src/wasm-lib && wasm-pack build --dev --target web --out-dir pkg && cargo test -p kcl-lib export_bindings) && cp src/wasm-lib/pkg/wasm_lib_bg.wasm public && yarn fmt", | ||||||
|     "build:wasm": "(cd src/wasm-lib && wasm-pack build --target web --out-dir pkg && cargo test -p kcl-lib export_bindings) && cp src/wasm-lib/pkg/wasm_lib_bg.wasm public && yarn fmt", |     "build:wasm": "(cd src/wasm-lib && wasm-pack build --target web --out-dir pkg && cargo test -p kcl-lib export_bindings) && cp src/wasm-lib/pkg/wasm_lib_bg.wasm public && yarn fmt", | ||||||
|     "build:wasm-clean": "yarn wasm-prep && yarn build:wasm", |     "build:wasm-clean": "yarn wasm-prep && yarn build:wasm", | ||||||
|     "remove-importmeta": "sed -i 's/import.meta.url/window.location.origin/g' \"./src/wasm-lib/pkg/wasm_lib.js\"; sed -i '' 's/import.meta.url/window.location.origin/g' \"./src/wasm-lib/pkg/wasm_lib.js\" || echo \"sed for both mac and linux\"", |     "remove-importmeta": "sed -i 's/import.meta.url/window.location.origin/g' \"./src/wasm-lib/pkg/wasm_lib.js\"; sed -i '' 's/import.meta.url/window.location.origin/g' \"./src/wasm-lib/pkg/wasm_lib.js\" || echo \"sed for both mac and linux\"", | ||||||
|     "wasm-prep": "rm -rf src/wasm-lib/pkg && mkdir src/wasm-lib/pkg && rm -rf src/wasm-lib/kcl/bindings", |     "wasm-prep": "rm -rf src/wasm-lib/pkg && mkdir src/wasm-lib/pkg && rm -rf src/wasm-lib/kcl/bindings", | ||||||
|     "lint": "eslint --fix src", |     "lint": "eslint --fix src", | ||||||
|     "bump-jsons": "echo \"$(jq --arg v \"$VERSION\" '.version=$v' package.json --indent 2)\" > package.json && echo \"$(jq --arg v \"$VERSION\" '.package.version=$v' src-tauri/tauri.conf.json --indent 2)\" > src-tauri/tauri.conf.json" |     "bump-jsons": "echo \"$(jq --arg v \"$VERSION\" '.version=$v' package.json --indent 2)\" > package.json && echo \"$(jq --arg v \"$VERSION\" '.package.version=$v' src-tauri/tauri.conf.json --indent 2)\" > src-tauri/tauri.conf.json", | ||||||
|  |     "postinstall": "patch-package && yarn xstate:typegen", | ||||||
|  |     "xstate:typegen": "yarn xstate typegen \"src/**/*.ts?(x)\"" | ||||||
|   }, |   }, | ||||||
|   "prettier": { |   "prettier": { | ||||||
|     "trailingComma": "es5", |     "trailingComma": "es5", | ||||||
| @ -98,30 +108,46 @@ | |||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@babel/plugin-proposal-private-property-in-object": "^7.21.11", |     "@babel/plugin-proposal-private-property-in-object": "^7.21.11", | ||||||
|     "@babel/preset-env": "^7.22.9", |     "@babel/preset-env": "^7.23.3", | ||||||
|     "@tauri-apps/cli": "^1.3.1", |     "@playwright/test": "^1.39.0", | ||||||
|  |     "@tauri-apps/cli": "^1.5.6", | ||||||
|     "@types/crypto-js": "^4.1.1", |     "@types/crypto-js": "^4.1.1", | ||||||
|     "@types/debounce": "^1.2.1", |     "@types/debounce-promise": "^3.1.8", | ||||||
|     "@types/isomorphic-fetch": "^0.0.36", |     "@types/pixelmatch": "^5.2.6", | ||||||
|     "@types/react-modal": "^3.16.0", |     "@types/pngjs": "^6.0.4", | ||||||
|     "@types/uuid": "^9.0.1", |     "@types/react-modal": "^3.16.3", | ||||||
|  |     "@types/three": "^0.160.0", | ||||||
|  |     "@types/uuid": "^9.0.4", | ||||||
|  |     "@types/wait-on": "^5.3.4", | ||||||
|     "@types/wicg-file-system-access": "^2020.9.6", |     "@types/wicg-file-system-access": "^2020.9.6", | ||||||
|     "@types/ws": "^8.5.5", |     "@types/ws": "^8.5.5", | ||||||
|     "@vitejs/plugin-react": "^4.0.3", |     "@vitejs/plugin-react": "^4.2.1", | ||||||
|     "@vitest/coverage-istanbul": "^0.34.1", |     "@wdio/cli": "^8.24.3", | ||||||
|  |     "@wdio/globals": "^8.24.3", | ||||||
|  |     "@wdio/local-runner": "^8.24.3", | ||||||
|  |     "@wdio/mocha-framework": "^8.24.3", | ||||||
|  |     "@wdio/spec-reporter": "^8.24.2", | ||||||
|  |     "@xstate/cli": "^0.5.17", | ||||||
|     "autoprefixer": "^10.4.13", |     "autoprefixer": "^10.4.13", | ||||||
|     "eslint": "^8.44.0", |     "eslint": "^8.53.0", | ||||||
|     "eslint-config-react-app": "^7.0.1", |     "eslint-config-react-app": "^7.0.1", | ||||||
|     "eslint-plugin-css-modules": "^2.11.0", |     "eslint-plugin-css-modules": "^2.12.0", | ||||||
|     "happy-dom": "^10.8.0", |     "happy-dom": "^10.8.0", | ||||||
|     "husky": "^8.0.3", |     "husky": "^8.0.3", | ||||||
|     "postcss": "^8.4.19", |     "patch-package": "^8.0.0", | ||||||
|  |     "pixelmatch": "^5.3.0", | ||||||
|  |     "pngjs": "^7.0.0", | ||||||
|  |     "postcss": "^8.4.31", | ||||||
|  |     "postinstall-postinstall": "^2.1.0", | ||||||
|     "prettier": "^2.8.0", |     "prettier": "^2.8.0", | ||||||
|     "setimmediate": "^1.0.5", |     "setimmediate": "^1.0.5", | ||||||
|     "tailwindcss": "^3.2.4", |     "tailwindcss": "^3.3.6", | ||||||
|     "vite": "^4.4.3", |     "vite": "^5.1.3", | ||||||
|     "vite-plugin-eslint": "^1.8.1", |     "vite-plugin-eslint": "^1.8.1", | ||||||
|     "vite-tsconfig-paths": "^4.2.0", |     "vite-plugin-package-version": "^1.1.0", | ||||||
|  |     "vite-tsconfig-paths": "^4.3.1", | ||||||
|  |     "vitest-webgl-canvas-mock": "^1.1.0", | ||||||
|  |     "wait-on": "^7.2.0", | ||||||
|     "yarn": "^1.22.19" |     "yarn": "^1.22.19" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										138
									
								
								patches/three+0.160.0.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,138 @@ | |||||||
|  | diff --git a/node_modules/three/examples/jsm/controls/OrbitControls.js b/node_modules/three/examples/jsm/controls/OrbitControls.js | ||||||
|  | index f29e7fe..0ef636b 100644 | ||||||
|  | --- a/node_modules/three/examples/jsm/controls/OrbitControls.js | ||||||
|  | +++ b/node_modules/three/examples/jsm/controls/OrbitControls.js | ||||||
|  | @@ -113,6 +113,25 @@ class OrbitControls extends EventDispatcher { | ||||||
|  |  		// public methods | ||||||
|  |  		// | ||||||
|  |   | ||||||
|  | +		this.interactionGuards = { | ||||||
|  | +			pan: { | ||||||
|  | +				description: 'Right click + Shift + drag or middle click + drag', | ||||||
|  | +				callback: (e) => e.button === 2 && !e.ctrlKey, | ||||||
|  | +			}, | ||||||
|  | +			zoom: { | ||||||
|  | +				description: 'Scroll wheel or Right click + Ctrl + drag', | ||||||
|  | +				dragCallback: (e) => e.button === 2 && e.ctrlKey, | ||||||
|  | +				scrollCallback: () => true, | ||||||
|  | +			}, | ||||||
|  | +			rotate: { | ||||||
|  | +				description: 'Right click + drag', | ||||||
|  | +				callback: (e) => e.button === 0, | ||||||
|  | +			}, | ||||||
|  | +		} | ||||||
|  | +		this.setMouseGuards = (interactionGuards) => { | ||||||
|  | +			this.interactionGuards = interactionGuards | ||||||
|  | +		} | ||||||
|  | + | ||||||
|  |  		this.getPolarAngle = function () { | ||||||
|  |   | ||||||
|  |  			return spherical.phi; | ||||||
|  | @@ -1057,92 +1076,21 @@ class OrbitControls extends EventDispatcher { | ||||||
|  |   | ||||||
|  |  		function onMouseDown( event ) { | ||||||
|  |         | ||||||
|  | -			let mouseAction; | ||||||
|  | - | ||||||
|  | -			switch ( event.button ) { | ||||||
|  | - | ||||||
|  | -				case 0: | ||||||
|  | - | ||||||
|  | -					mouseAction = scope.mouseButtons.LEFT; | ||||||
|  | -					break; | ||||||
|  | - | ||||||
|  | -				case 1: | ||||||
|  | - | ||||||
|  | -					mouseAction = scope.mouseButtons.MIDDLE; | ||||||
|  | -					break; | ||||||
|  | - | ||||||
|  | -				case 2: | ||||||
|  | - | ||||||
|  | -					mouseAction = scope.mouseButtons.RIGHT; | ||||||
|  | -					break; | ||||||
|  | - | ||||||
|  | -				default: | ||||||
|  | - | ||||||
|  | -					mouseAction = - 1; | ||||||
|  | - | ||||||
|  | -			} | ||||||
|  | - | ||||||
|  | -			switch ( mouseAction ) { | ||||||
|  | - | ||||||
|  | -				case MOUSE.DOLLY: | ||||||
|  | - | ||||||
|  | -					if ( scope.enableZoom === false ) return; | ||||||
|  | - | ||||||
|  | -					handleMouseDownDolly( event ); | ||||||
|  | - | ||||||
|  | -					state = STATE.DOLLY; | ||||||
|  | - | ||||||
|  | -					break; | ||||||
|  | - | ||||||
|  | -				case MOUSE.ROTATE: | ||||||
|  | - | ||||||
|  | -					if ( event.ctrlKey || event.metaKey || event.shiftKey ) { | ||||||
|  | - | ||||||
|  | -						if ( scope.enablePan === false ) return; | ||||||
|  | - | ||||||
|  | -						handleMouseDownPan( event ); | ||||||
|  | - | ||||||
|  | -						state = STATE.PAN; | ||||||
|  | - | ||||||
|  | -					} else { | ||||||
|  | - | ||||||
|  | -						if ( scope.enableRotate === false ) return; | ||||||
|  | - | ||||||
|  | -						handleMouseDownRotate( event ); | ||||||
|  | - | ||||||
|  | -						state = STATE.ROTATE; | ||||||
|  | - | ||||||
|  | -					} | ||||||
|  | - | ||||||
|  | -					break; | ||||||
|  | - | ||||||
|  | -				case MOUSE.PAN: | ||||||
|  | - | ||||||
|  | -					if ( event.ctrlKey || event.metaKey || event.shiftKey ) { | ||||||
|  | - | ||||||
|  | -						if ( scope.enableRotate === false ) return; | ||||||
|  | - | ||||||
|  | -						handleMouseDownRotate( event ); | ||||||
|  | - | ||||||
|  | -						state = STATE.ROTATE; | ||||||
|  | - | ||||||
|  | -					} else { | ||||||
|  | - | ||||||
|  | -						if ( scope.enablePan === false ) return; | ||||||
|  | - | ||||||
|  | -						handleMouseDownPan( event ); | ||||||
|  | - | ||||||
|  | -						state = STATE.PAN; | ||||||
|  | - | ||||||
|  | -					} | ||||||
|  | - | ||||||
|  | -					break; | ||||||
|  | - | ||||||
|  | -				default: | ||||||
|  | - | ||||||
|  | -					state = STATE.NONE; | ||||||
|  | - | ||||||
|  | -			} | ||||||
|  | +      if (scope.interactionGuards.pan.callback(event)) { | ||||||
|  | +        if (scope.enablePan === false) return | ||||||
|  | +        handleMouseDownPan(event) | ||||||
|  | +        state = STATE.PAN | ||||||
|  | +      } else if (scope.interactionGuards.rotate.callback(event)) { | ||||||
|  | +        if (scope.enableRotate === false) return | ||||||
|  | +        handleMouseDownRotate(event) | ||||||
|  | +        state = STATE.ROTATE | ||||||
|  | +      } else if (scope.interactionGuards.zoom.dragCallback(event)) { | ||||||
|  | +        if (scope.enableZoom === false) return | ||||||
|  | +        handleMouseDownDolly(event) | ||||||
|  | +        state = STATE.DOLLY | ||||||
|  | +      } else { | ||||||
|  | +        return | ||||||
|  | +      } | ||||||
|  |   | ||||||
|  |  			if ( state !== STATE.NONE ) { | ||||||
|  |   | ||||||
							
								
								
									
										81
									
								
								playwright.config.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,81 @@ | |||||||
|  | import { defineConfig, devices } from '@playwright/test' | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Read environment variables from file. | ||||||
|  |  * https://github.com/motdotla/dotenv | ||||||
|  |  */ | ||||||
|  | // require('dotenv').config(); | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * See https://playwright.dev/docs/test-configuration. | ||||||
|  |  */ | ||||||
|  | export default defineConfig({ | ||||||
|  |   testDir: './e2e/playwright', | ||||||
|  |   /* Run tests in files in parallel */ | ||||||
|  |   fullyParallel: true, | ||||||
|  |   /* Fail the build on CI if you accidentally left test.only in the source code. */ | ||||||
|  |   forbidOnly: !!process.env.CI, | ||||||
|  |   /* Retry on CI only */ | ||||||
|  |   retries: process.env.CI ? 3 : 0, | ||||||
|  |   /* Opt out of parallel tests on CI. */ | ||||||
|  |   workers: process.env.CI ? 1 : 1, | ||||||
|  |   /* Reporter to use. See https://playwright.dev/docs/test-reporters */ | ||||||
|  |   reporter: 'html', | ||||||
|  |   /* Shared settings for all the projects below. See https://playwright.dev/docs/api/class-testoptions. */ | ||||||
|  |   use: { | ||||||
|  |     /* Base URL to use in actions like `await page.goto('/')`. */ | ||||||
|  |     baseURL: 'http://localhost:3000', | ||||||
|  |  | ||||||
|  |     /* Collect trace when retrying the failed test. See https://playwright.dev/docs/trace-viewer */ | ||||||
|  |     trace: 'on-first-retry', | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   /* Configure projects for major browsers */ | ||||||
|  |   projects: [ | ||||||
|  |     { | ||||||
|  |       name: 'Google Chrome', | ||||||
|  |       use: { ...devices['Desktop Chrome'], channel: 'chrome' }, // or 'chrome-beta' | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       name: 'webkit', | ||||||
|  |       use: { ...devices['Desktop Safari'] }, | ||||||
|  |     }, | ||||||
|  |     // { | ||||||
|  |     //   name: 'firefox', | ||||||
|  |     //   use: { ...devices['Desktop Firefox'] }, | ||||||
|  |     // }, | ||||||
|  |     // { | ||||||
|  |     //   name: 'chromium', // compat issue with encoding atm, so we're using the branded 'Google Chrome' instead | ||||||
|  |     //   use: { ...devices['Desktop Chrome'] }, | ||||||
|  |     // }, | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     /* Test against mobile viewports. */ | ||||||
|  |     // { | ||||||
|  |     //   name: 'Mobile Chrome', | ||||||
|  |     //   use: { ...devices['Pixel 5'] }, | ||||||
|  |     // }, | ||||||
|  |     // { | ||||||
|  |     //   name: 'Mobile Safari', | ||||||
|  |     //   use: { ...devices['iPhone 12'] }, | ||||||
|  |     // }, | ||||||
|  |  | ||||||
|  |     /* Test against branded browsers. */ | ||||||
|  |     // { | ||||||
|  |     //   name: 'Microsoft Edge', | ||||||
|  |     //   use: { ...devices['Desktop Edge'], channel: 'msedge' }, | ||||||
|  |     // }, | ||||||
|  |     // { | ||||||
|  |     //   name: 'Google Chrome', | ||||||
|  |     //   use: { ...devices['Desktop Chrome'], channel: 'chrome' }, | ||||||
|  |     // }, | ||||||
|  |   ], | ||||||
|  |  | ||||||
|  |   /* Run your local dev server before starting the tests */ | ||||||
|  |   webServer: { | ||||||
|  |     command: 'yarn serve', | ||||||
|  |     // url: 'http://127.0.0.1:3000', | ||||||
|  |     reuseExistingServer: !process.env.CI, | ||||||
|  |   }, | ||||||
|  | }) | ||||||
							
								
								
									
										26
									
								
								public/announce_release.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,26 @@ | |||||||
|  | import requests | ||||||
|  | import os | ||||||
|  |  | ||||||
|  | webhook_url = os.getenv('DISCORD_WEBHOOK_URL') | ||||||
|  | release_version = os.getenv('RELEASE_VERSION') | ||||||
|  | release_body = os.getenv('RELEASE_BODY') | ||||||
|  |  | ||||||
|  | # message to send to Discord | ||||||
|  | data = { | ||||||
|  |     "content":  | ||||||
|  |         f''' | ||||||
|  |         **{release_version}** is now available! Check out the latest features and improvements here: https://zoo.dev/modeling-app/download | ||||||
|  |         {release_body} | ||||||
|  |         ''', | ||||||
|  |     "username": "Modeling App Release Updates", | ||||||
|  |     "avatar_url": "https://raw.githubusercontent.com/KittyCAD/modeling-app/main/public/discord-avatar.png" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | # POST request to the Discord webhook | ||||||
|  | response = requests.post(webhook_url, json=data) | ||||||
|  |  | ||||||
|  | # Check for success | ||||||
|  | if response.status_code == 204: | ||||||
|  |     print("Successfully sent the message to Discord.") | ||||||
|  | else: | ||||||
|  |     print("Failed to send the message to Discord.") | ||||||
							
								
								
									
										
											BIN
										
									
								
								public/discord-avatar.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 11 KiB | 
| @ -4,9 +4,9 @@ | |||||||
|  |  | ||||||
| First off, thank you so much for your interest in being a part of the closed Alpha program! We are thrilled to have others use our product and see what you build with it (and truthfully, how you break it too). | First off, thank you so much for your interest in being a part of the closed Alpha program! We are thrilled to have others use our product and see what you build with it (and truthfully, how you break it too). | ||||||
|  |  | ||||||
| ### KittyCAD Modeling App (KCMA) | ### Zoo Modeling App (ZMA) | ||||||
|  |  | ||||||
| What we are introducing to you is our KittyCAD Modeling App (KCMA). KCMA is a CAD application that expresses a hybrid style of traditional CAD interface along with a code-CAD interface. KCMA is a great way for us to test our own APIs as well as inspire others to develop their own applications. | What we are introducing to you is our Zoo Modeling App (ZMA). ZMA is a CAD application that expresses a hybrid style of traditional CAD interface along with a code-CAD interface. ZMA is a great way for us to test our own APIs as well as inspire others to develop their own applications. | ||||||
|  |  | ||||||
| ### Why Code? | ### Why Code? | ||||||
|  |  | ||||||
| @ -18,11 +18,11 @@ Plenty of you have professional CAD experience, and may not understand why codin | |||||||
| - Reproducibility | - Reproducibility | ||||||
| - Easier integration with other tools | - Easier integration with other tools | ||||||
|  |  | ||||||
| ### Before You Use KCMA | ### Before You Use ZMA | ||||||
|  |  | ||||||
| Before you dive straight into the app, we wanted to lay some expectations out for you.  | Before you dive straight into the app, we wanted to lay some expectations out for you.  | ||||||
|  |  | ||||||
| - KCMA is in early development. Kurt pitched the idea back in January, and the team has been working hard on it since then. KCMA has really basic CAD features for now, but we have plenty of features on our roadmap. Most of the features that you may be currently used to in your CAD workflow today will be available down the road. | - ZMA is in early development. Kurt pitched the idea back in January, and the team has been working hard on it since then. ZMA has really basic CAD features for now, but we have plenty of features on our roadmap. Most of the features that you may be currently used to in your CAD workflow today will be available down the road. | ||||||
| - For a list of all scripting functions, please reference our [documentation](https://github.com/KittyCAD/modeling-app/blob/main/docs/kcl/std.md). For a basic rundown of our types, please reference [this document](https://github.com/KittyCAD/modeling-app/blob/main/docs/kcl/types.md). | - For a list of all scripting functions, please reference our [documentation](https://github.com/KittyCAD/modeling-app/blob/main/docs/kcl/std.md). For a basic rundown of our types, please reference [this document](https://github.com/KittyCAD/modeling-app/blob/main/docs/kcl/types.md). | ||||||
| - With that being said, we have created an external new features list in [GH Discussions](https://github.com/KittyCAD/modeling-app/discussions). For our current priority list, please click [here](https://github.com/KittyCAD/modeling-app/blob/main/public/roadmap.md). Please upvote any features in the GH Discussions page that you would like to see implemented first. We will prioritize the highest upvoted items or items that are foundational for other features on the list. You can also add your own, but we will review it to make sure it’s not a duplicate or it’s feasible for the current state of the app. | - With that being said, we have created an external new features list in [GH Discussions](https://github.com/KittyCAD/modeling-app/discussions). For our current priority list, please click [here](https://github.com/KittyCAD/modeling-app/blob/main/public/roadmap.md). Please upvote any features in the GH Discussions page that you would like to see implemented first. We will prioritize the highest upvoted items or items that are foundational for other features on the list. You can also add your own, but we will review it to make sure it’s not a duplicate or it’s feasible for the current state of the app. | ||||||
| - Please report any and all bugs/issues you find. Even the smallest bugs are important! You can report them in a GH Issue [here](https://github.com/KittyCAD/modeling-app/issues/new). You are more than welcome to link your GH Issue in the **bugs** section of our Discord, but if you want to discuss the bug further, please keep that in the GH Issue thread. Please include the severity of the bug in your GH Issue ticket (High, Medium, or Low). If you are having trouble deciding what severity the bug is, use this guideline: | - Please report any and all bugs/issues you find. Even the smallest bugs are important! You can report them in a GH Issue [here](https://github.com/KittyCAD/modeling-app/issues/new). You are more than welcome to link your GH Issue in the **bugs** section of our Discord, but if you want to discuss the bug further, please keep that in the GH Issue thread. Please include the severity of the bug in your GH Issue ticket (High, Medium, or Low). If you are having trouble deciding what severity the bug is, use this guideline: | ||||||
|  | |||||||
							
								
								
									
										3
									
								
								public/kcl-icon.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,3 @@ | |||||||
|  | <svg width="40" height="40" viewBox="0 0 40 40" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  | <path fill-rule="evenodd" clip-rule="evenodd" d="M40 0H0V40H40V0ZM7.34715 27.2143V15.6577L2.976 15.987V36.7949H7.34715V32.0645L8.00582 31.5256C8.24533 31.326 8.47487 31.1264 8.69442 30.9268L12.1075 36.7949H17.0475C16.1893 35.3978 15.311 33.9906 14.4128 32.5735C13.5346 31.1563 12.6664 29.7392 11.8081 28.3221L15.8499 24.9389C15.4308 24.4399 15.0017 23.931 14.5625 23.412L13.3051 21.8552L7.34715 27.2143ZM22.2581 26.6754C22.8769 25.9169 23.6753 25.5377 24.6533 25.5377C25.272 25.5377 25.8309 25.6175 26.3299 25.7772C26.8289 25.9169 27.4177 26.1465 28.0963 26.4658L29.3238 23.3521C28.5853 22.7933 27.7371 22.4041 26.779 22.1845C25.8409 21.9649 25.0625 21.8552 24.4437 21.8552C22.0885 21.8552 20.2223 22.5537 18.845 23.9509C17.4878 25.3281 16.8092 27.1944 16.8092 29.5496C16.8092 31.9048 17.4878 33.7611 18.845 35.1183C20.2223 36.4756 22.0885 37.1542 24.4437 37.1542C25.0625 37.1542 25.8509 37.0444 26.8089 36.8249C27.767 36.6053 28.6053 36.2161 29.3238 35.6572L28.0963 32.5435C27.4177 32.8629 26.8289 33.0924 26.3299 33.2321C25.8309 33.3718 25.272 33.4417 24.6533 33.4417C23.6753 33.4417 22.8769 33.0924 22.2581 32.3938C21.6594 31.6753 21.36 30.7272 21.36 29.5496C21.36 28.372 21.6594 27.4139 22.2581 26.6754ZM36.2796 36.7949V15.6577L31.9085 15.987V36.7949H36.2796Z" fill="#D0FF00"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 1.4 KiB | 
							
								
								
									
										
											BIN
										
									
								
								public/kcma-logomark-outlined.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 11 KiB | 
| @ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "short_name": "KCMA", |   "short_name": "ZMA", | ||||||
|   "name": "KittyCAD Modeling App", |   "name": "Zoo Modeling App", | ||||||
|   "icons": [ |   "icons": [ | ||||||
|     { |     { | ||||||
|       "src": "favicon.ico", |       "src": "favicon.ico", | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| ## KittyCAD Modeling App Roadmap | ## Zoo Modeling App Roadmap | ||||||
|  |  | ||||||
| This document ties into our [GH Discussions Feature List](https://github.com/KittyCAD/modeling-app/discussions). Please upvote any features that you want to see next, or add ones that are not listed and we will review.  | This document ties into our [GH Discussions Feature List](https://github.com/KittyCAD/modeling-app/discussions). Please upvote any features that you want to see next, or add ones that are not listed and we will review.  | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										13
									
								
								public/zma-logomark-dark.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 13 KiB | 
							
								
								
									
										
											BIN
										
									
								
								public/zma-logomark-outlined.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 11 KiB | 
							
								
								
									
										13
									
								
								public/zma-logomark.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 13 KiB | 
							
								
								
									
										7
									
								
								public/zoo-logo.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,7 @@ | |||||||
|  | <svg width="438" height="145" viewBox="0 0 438 145" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  | <path d="M88.2136 25.3021V3.12744H0.595269V34.3994H79.827L0.609484 120.312H0.595269V120.326L0.581055 120.34L0.595269 120.355V141.364H20.8936L41.3341 119.189V141.364H128.952V110.092H49.7349L128.952 24.1649V3.12744L108.64 3.15587L88.2136 25.3021Z" fill="white"/> | ||||||
|  | <path d="M167.36 72.4372C167.36 49.7366 185.824 31.2719 208.525 31.2719C216.514 31.2719 223.976 33.5605 230.288 37.5121L251.78 14.3709C239.698 5.34466 224.73 0 208.525 0C168.582 0 136.088 32.4944 136.088 72.4372C136.088 90.5465 142.769 107.135 153.828 119.857L175.32 96.7156C170.316 89.9069 167.36 81.5061 167.36 72.4372Z" fill="white"/> | ||||||
|  | <path d="M241.745 48.1442C246.734 54.9671 249.691 63.3679 249.691 72.4368C249.691 95.1232 231.226 113.588 208.525 113.588C200.537 113.588 193.088 111.299 186.777 107.348L165.271 130.503C177.353 139.515 192.321 144.86 208.525 144.86C248.468 144.86 280.963 112.365 280.963 72.4368C280.963 54.3133 274.282 37.7249 263.223 25.0029L241.745 48.1442Z" fill="white"/> | ||||||
|  | <path d="M419.312 25.0029L397.834 48.1442C402.823 54.9671 405.779 63.3679 405.779 72.4368C405.779 95.1232 387.315 113.588 364.614 113.588C356.626 113.588 349.177 111.299 342.866 107.348L321.359 130.503C333.442 139.515 348.41 144.86 364.614 144.86C404.557 144.86 437.051 112.365 437.051 72.4368C437.051 54.3133 430.371 37.7249 419.312 25.0029Z" fill="white"/> | ||||||
|  | <path d="M323.449 72.4372C323.449 49.7366 341.913 31.2719 364.614 31.2719C372.603 31.2719 380.065 33.5605 386.376 37.5121L407.869 14.3709C395.786 5.34466 380.819 0 364.614 0C324.671 0 292.177 32.4944 292.177 72.4372C292.177 90.5465 298.858 107.135 309.916 119.857L331.409 96.7156C326.405 89.9069 323.449 81.5061 323.449 72.4372Z" fill="white"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 1.7 KiB | 
							
								
								
									
										490
									
								
								src-tauri/Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						| @ -4,7 +4,7 @@ version = "0.1.0" | |||||||
| description = "A Tauri App" | description = "A Tauri App" | ||||||
| authors = ["you"] | authors = ["you"] | ||||||
| license = "" | license = "" | ||||||
| repository = "" | repository = "https://github.com/KittyCAD/modeling-app" | ||||||
| default-run = "app" | default-run = "app" | ||||||
| edition = "2021" | edition = "2021" | ||||||
| rust-version = "1.60" | rust-version = "1.60" | ||||||
| @ -12,18 +12,18 @@ rust-version = "1.60" | |||||||
| # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html | # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html | ||||||
|  |  | ||||||
| [build-dependencies] | [build-dependencies] | ||||||
| tauri-build = { version = "1.4.0", features = [] } | tauri-build = { version = "1.5.1", features = [] } | ||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| anyhow = "1" | anyhow = "1" | ||||||
| kittycad = "0.2.25" | kittycad = "0.2.53" | ||||||
| oauth2 = "4.4.2" | oauth2 = "4.4.2" | ||||||
| serde = { version = "1.0", features = ["derive"] } | serde = { version = "1.0", features = ["derive"] } | ||||||
| serde_json = "1.0" | serde_json = "1.0" | ||||||
| tauri = { version = "1.4.1", features = [ "os-all", "dialog-all", "fs-all", "http-request", "path-all", "shell-open", "shell-open-api", "updater", "devtools"] } | tauri = { version = "1.5.4", features = [ "os-all", "dialog-all", "fs-all", "http-request", "path-all", "shell-open", "shell-open-api", "devtools"] } | ||||||
| tauri-plugin-fs-extra = { git = "https://github.com/tauri-apps/plugins-workspace", branch = "v1" } | tauri-plugin-fs-extra = { git = "https://github.com/tauri-apps/plugins-workspace", branch = "v1" } | ||||||
| tokio = { version = "1.32.0", features = ["time"] } | tokio = { version = "1.36.0", features = ["time"] } | ||||||
| toml = "0.8.0" | toml = "0.8.2" | ||||||
|  |  | ||||||
| [features] | [features] | ||||||
| # this feature is used for production builds or when `devPath` points to the filesystem and the built-in dev server is disabled. | # this feature is used for production builds or when `devPath` points to the filesystem and the built-in dev server is disabled. | ||||||
|  | |||||||
| Before Width: | Height: | Size: 20 KiB After Width: | Height: | Size: 10 KiB | 
| Before Width: | Height: | Size: 48 KiB After Width: | Height: | Size: 25 KiB | 
| Before Width: | Height: | Size: 2.7 KiB After Width: | Height: | Size: 1.8 KiB | 
| Before Width: | Height: | Size: 16 KiB After Width: | Height: | Size: 8.4 KiB | 
| Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 12 KiB | 
| Before Width: | Height: | Size: 25 KiB After Width: | Height: | Size: 13 KiB | 
| Before Width: | Height: | Size: 55 KiB After Width: | Height: | Size: 29 KiB | 
| Before Width: | Height: | Size: 2.5 KiB After Width: | Height: | Size: 1.6 KiB | 
| Before Width: | Height: | Size: 62 KiB After Width: | Height: | Size: 33 KiB | 
| Before Width: | Height: | Size: 4.5 KiB After Width: | Height: | Size: 2.7 KiB | 
| Before Width: | Height: | Size: 9.2 KiB After Width: | Height: | Size: 5.0 KiB | 
| Before Width: | Height: | Size: 13 KiB After Width: | Height: | Size: 6.6 KiB | 
| Before Width: | Height: | Size: 5.4 KiB After Width: | Height: | Size: 3.1 KiB | 
| Before Width: | Height: | Size: 68 KiB After Width: | Height: | Size: 39 KiB | 
| Before Width: | Height: | Size: 122 KiB After Width: | Height: | Size: 69 KiB | 
| @ -1,10 +1,13 @@ | |||||||
| // Prevents additional console window on Windows in release, DO NOT REMOVE!! | // Prevents additional console window on Windows in release, DO NOT REMOVE!! | ||||||
| #![cfg_attr(not(debug_assertions), windows_subsystem = "windows")] | #![cfg_attr(not(debug_assertions), windows_subsystem = "windows")] | ||||||
|  |  | ||||||
|  | use std::env; | ||||||
|  | use std::fs; | ||||||
| use std::io::Read; | use std::io::Read; | ||||||
|  |  | ||||||
| use anyhow::Result; | use anyhow::Result; | ||||||
| use oauth2::TokenResponse; | use oauth2::TokenResponse; | ||||||
|  | use std::process::Command; | ||||||
| use tauri::{InvokeError, Manager}; | use tauri::{InvokeError, Manager}; | ||||||
| const DEFAULT_HOST: &str = "https://api.kittycad.io"; | const DEFAULT_HOST: &str = "https://api.kittycad.io"; | ||||||
|  |  | ||||||
| @ -68,10 +71,23 @@ async fn login(app: tauri::AppHandle, host: &str) -> Result<String, InvokeError> | |||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     // Open the system browser with the auth_uri. |     // Open the system browser with the auth_uri. | ||||||
|     // We do this in the browser and not a seperate window because we want 1password and |     // We do this in the browser and not a separate window because we want 1password and | ||||||
|     // other crap to work well. |     // other crap to work well. | ||||||
|     tauri::api::shell::open(&app.shell_scope(), auth_uri.secret(), None) |     // TODO: find a better way to share this value with tauri e2e tests | ||||||
|         .map_err(|e| InvokeError::from_anyhow(e.into()))?; |     // Here we're using an env var to enable the /tmp file (windows not supported for now) | ||||||
|  |     // and bypass the shell::open call as it fails on GitHub Actions. | ||||||
|  |     let e2e_tauri_enabled = env::var("E2E_TAURI_ENABLED").is_ok(); | ||||||
|  |     if e2e_tauri_enabled { | ||||||
|  |         println!( | ||||||
|  |             "E2E_TAURI_ENABLED is set, won't open {} externally", | ||||||
|  |             auth_uri.secret() | ||||||
|  |         ); | ||||||
|  |         fs::write("/tmp/kittycad_user_code", details.user_code().secret()) | ||||||
|  |             .expect("Unable to write /tmp/kittycad_user_code file"); | ||||||
|  |     } else { | ||||||
|  |         tauri::api::shell::open(&app.shell_scope(), auth_uri.secret(), None) | ||||||
|  |             .map_err(|e| InvokeError::from_anyhow(e.into()))?; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     // Wait for the user to login. |     // Wait for the user to login. | ||||||
|     let token = auth_client |     let token = auth_client | ||||||
| @ -129,10 +145,10 @@ async fn get_user( | |||||||
|  |  | ||||||
| fn main() { | fn main() { | ||||||
|     tauri::Builder::default() |     tauri::Builder::default() | ||||||
|         .setup(|app| { |         .setup(|_app| { | ||||||
|             #[cfg(debug_assertions)] // only include this code on debug builds |             #[cfg(debug_assertions)] // only include this code on debug builds | ||||||
|             { |             { | ||||||
|                 let window = app.get_window("main").unwrap(); |                 let window = _app.get_window("main").unwrap(); | ||||||
|                 // comment out the below if you don't devtools to open everytime. |                 // comment out the below if you don't devtools to open everytime. | ||||||
|                 // it's useful because otherwise devtools shuts everytime rust code changes. |                 // it's useful because otherwise devtools shuts everytime rust code changes. | ||||||
|                 window.open_devtools(); |                 window.open_devtools(); | ||||||
|  | |||||||
| @ -1,14 +1,13 @@ | |||||||
| { | { | ||||||
|   "$schema": "../node_modules/@tauri-apps/cli/schema.json", |   "$schema": "../node_modules/@tauri-apps/cli/schema.json", | ||||||
|   "build": { |   "build": { | ||||||
|     "beforeBuildCommand": "yarn build:both", |  | ||||||
|     "beforeDevCommand": "yarn start", |     "beforeDevCommand": "yarn start", | ||||||
|     "devPath": "http://localhost:3000", |     "devPath": "http://localhost:3000", | ||||||
|     "distDir": "../build" |     "distDir": "../build" | ||||||
|   }, |   }, | ||||||
|   "package": { |   "package": { | ||||||
|     "productName": "kittycad-modeling", |     "productName": "zoo-modeling-app", | ||||||
|     "version": "0.8.2" |     "version": "0.15.2" | ||||||
|   }, |   }, | ||||||
|   "tauri": { |   "tauri": { | ||||||
|     "allowlist": { |     "allowlist": { | ||||||
| @ -72,30 +71,20 @@ | |||||||
|       }, |       }, | ||||||
|       "resources": [], |       "resources": [], | ||||||
|       "shortDescription": "", |       "shortDescription": "", | ||||||
|       "targets": "all", |       "targets": "all" | ||||||
|       "windows": { |  | ||||||
|         "certificateThumbprint": "F4C9A52FF7BC26EE5E054946F6B11DEEA94C748D", |  | ||||||
|         "digestAlgorithm": "sha256", |  | ||||||
|         "timestampUrl": "http://timestamp.digicert.com" |  | ||||||
|       } |  | ||||||
|     }, |     }, | ||||||
|     "security": { |     "security": { | ||||||
|       "csp": null |       "csp": null | ||||||
|     }, |     }, | ||||||
|     "updater": { |     "updater": { | ||||||
|       "active": true, |       "active": false | ||||||
|       "endpoints": [ |  | ||||||
|         "https://dl.kittycad.io/releases/modeling-app/last_update.json" |  | ||||||
|       ], |  | ||||||
|       "dialog": true, |  | ||||||
|       "pubkey": "dW50cnVzdGVkIGNvbW1lbnQ6IG1pbmlzaWduIHB1YmxpYyBrZXk6IEUzNzA4MjBEQjFBRTY4NzYKUldSMmFLNnhEWUp3NCtsT21Jd05wQktOaGVkOVp6MUFma0hNTDRDSnI2RkJJTEZOWG1ncFhqcU8K" |  | ||||||
|     }, |     }, | ||||||
|     "windows": [ |     "windows": [ | ||||||
|       { |       { | ||||||
|         "fullscreen": false, |         "fullscreen": false, | ||||||
|         "height": 1200, |         "height": 1200, | ||||||
|         "resizable": true, |         "resizable": true, | ||||||
|         "title": "KittyCAD Modeling", |         "title": "Zoo Modeling App", | ||||||
|         "width": 1800 |         "width": 1800 | ||||||
|       } |       } | ||||||
|     ] |     ] | ||||||
|  | |||||||
| @ -1,7 +1,6 @@ | |||||||
|  |  | ||||||
| { | { | ||||||
|   "$schema": "../node_modules/@tauri-apps/cli/schema.json", |   "$schema": "../node_modules/@tauri-apps/cli/schema.json", | ||||||
|   "package": { |   "package": { | ||||||
|     "productName": "KittyCAD Modeling" |     "productName": "Zoo Modeling App" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										21
									
								
								src-tauri/tauri.release.conf.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,21 @@ | |||||||
|  | { | ||||||
|  |   "$schema": "../node_modules/@tauri-apps/cli/schema.json", | ||||||
|  |   "tauri": { | ||||||
|  |     "updater": { | ||||||
|  |       "active": true, | ||||||
|  |       "endpoints": [ | ||||||
|  |         "https://dl.zoo.dev/releases/modeling-app/last_update.json" | ||||||
|  |       ], | ||||||
|  |       "dialog": true, | ||||||
|  |       "pubkey": "dW50cnVzdGVkIGNvbW1lbnQ6IG1pbmlzaWduIHB1YmxpYyBrZXk6IEUzNzA4MjBEQjFBRTY4NzYKUldSMmFLNnhEWUp3NCtsT21Jd05wQktOaGVkOVp6MUFma0hNTDRDSnI2RkJJTEZOWG1ncFhqcU8K" | ||||||
|  |     }, | ||||||
|  |     "bundle": { | ||||||
|  |       "identifier": "io.kittycad.modeling-app", | ||||||
|  |       "windows": { | ||||||
|  |         "certificateThumbprint": "F4C9A52FF7BC26EE5E054946F6B11DEEA94C748D", | ||||||
|  |         "digestAlgorithm": "sha256", | ||||||
|  |         "timestampUrl": "http://timestamp.digicert.com" | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
| @ -1,7 +1,6 @@ | |||||||
|  |  | ||||||
| { | { | ||||||
|   "$schema": "../node_modules/@tauri-apps/cli/schema.json", |   "$schema": "../node_modules/@tauri-apps/cli/schema.json", | ||||||
|   "package": { |   "package": { | ||||||
|     "productName": "KittyCAD Modeling" |     "productName": "Zoo Modeling App" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,52 +0,0 @@ | |||||||
| import { render, screen } from '@testing-library/react' |  | ||||||
| import { App } from './App' |  | ||||||
| import { describe, test, vi } from 'vitest' |  | ||||||
| import { BrowserRouter } from 'react-router-dom' |  | ||||||
| import { GlobalStateProvider } from './components/GlobalStateProvider' |  | ||||||
| import CommandBarProvider from 'components/CommandBar' |  | ||||||
|  |  | ||||||
| let listener: ((rect: any) => void) | undefined = undefined |  | ||||||
| ;(global as any).ResizeObserver = class ResizeObserver { |  | ||||||
|   constructor(ls: ((rect: any) => void) | undefined) { |  | ||||||
|     listener = ls |  | ||||||
|   } |  | ||||||
|   observe() {} |  | ||||||
|   unobserve() {} |  | ||||||
|   disconnect() {} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| describe('App tests', () => { |  | ||||||
|   test('Renders the modeling app screen, including "Variables" pane.', () => { |  | ||||||
|     vi.mock('react-router-dom', async () => { |  | ||||||
|       const actual = (await vi.importActual('react-router-dom')) as Record< |  | ||||||
|         string, |  | ||||||
|         any |  | ||||||
|       > |  | ||||||
|       return { |  | ||||||
|         ...actual, |  | ||||||
|         useParams: () => ({ id: 'new' }), |  | ||||||
|         useLoaderData: () => ({ code: null }), |  | ||||||
|       } |  | ||||||
|     }) |  | ||||||
|     render( |  | ||||||
|       <TestWrap> |  | ||||||
|         <App /> |  | ||||||
|       </TestWrap> |  | ||||||
|     ) |  | ||||||
|     const linkElement = screen.getByText(/Variables/i) |  | ||||||
|     expect(linkElement).toBeInTheDocument() |  | ||||||
|  |  | ||||||
|     vi.restoreAllMocks() |  | ||||||
|   }) |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| function TestWrap({ children }: { children: React.ReactNode }) { |  | ||||||
|   // wrap in router and xState context |  | ||||||
|   return ( |  | ||||||
|     <BrowserRouter> |  | ||||||
|       <CommandBarProvider> |  | ||||||
|         <GlobalStateProvider>{children}</GlobalStateProvider> |  | ||||||
|       </CommandBarProvider> |  | ||||||
|     </BrowserRouter> |  | ||||||
|   ) |  | ||||||
| } |  | ||||||
							
								
								
									
										197
									
								
								src/App.tsx
									
									
									
									
									
								
							
							
						
						| @ -1,4 +1,4 @@ | |||||||
| import { useRef, useEffect, useCallback, MouseEventHandler } from 'react' | import { useCallback, MouseEventHandler } from 'react' | ||||||
| import { DebugPanel } from './components/DebugPanel' | import { DebugPanel } from './components/DebugPanel' | ||||||
| import { v4 as uuidv4 } from 'uuid' | import { v4 as uuidv4 } from 'uuid' | ||||||
| import { PaneType, useStore } from './useStore' | import { PaneType, useStore } from './useStore' | ||||||
| @ -19,56 +19,43 @@ import { | |||||||
| } from '@fortawesome/free-solid-svg-icons' | } from '@fortawesome/free-solid-svg-icons' | ||||||
| import { useHotkeys } from 'react-hotkeys-hook' | import { useHotkeys } from 'react-hotkeys-hook' | ||||||
| import { getNormalisedCoordinates } from './lib/utils' | import { getNormalisedCoordinates } from './lib/utils' | ||||||
| import { isTauri } from './lib/isTauri' | import { useLoaderData, useNavigate } from 'react-router-dom' | ||||||
| import { useLoaderData } from 'react-router-dom' | import { type IndexLoaderData } from 'lib/types' | ||||||
| import { IndexLoaderData } from './Router' | import { paths } from 'lib/paths' | ||||||
| import { useGlobalStateContext } from 'hooks/useGlobalStateContext' | import { useGlobalStateContext } from 'hooks/useGlobalStateContext' | ||||||
| import { onboardingPaths } from 'routes/Onboarding' | import { onboardingPaths } from 'routes/Onboarding/paths' | ||||||
| import { cameraMouseDragGuards } from 'lib/cameraControls' |  | ||||||
| import { CameraDragInteractionType_type } from '@kittycad/lib/dist/types/src/models' |  | ||||||
| import { CodeMenu } from 'components/CodeMenu' | import { CodeMenu } from 'components/CodeMenu' | ||||||
| import { TextEditor } from 'components/TextEditor' | import { TextEditor } from 'components/TextEditor' | ||||||
| import { Themes, getSystemTheme } from 'lib/theme' | import { Themes, getSystemTheme } from 'lib/theme' | ||||||
| import { useSetupEngineManager } from 'hooks/useSetupEngineManager' |  | ||||||
| import { useEngineConnectionSubscriptions } from 'hooks/useEngineConnectionSubscriptions' | import { useEngineConnectionSubscriptions } from 'hooks/useEngineConnectionSubscriptions' | ||||||
|  | import { engineCommandManager } from './lang/std/engineConnection' | ||||||
|  | import { useModelingContext } from 'hooks/useModelingContext' | ||||||
|  | import { useAbsoluteFilePath } from 'hooks/useAbsoluteFilePath' | ||||||
|  | import { isTauri } from 'lib/isTauri' | ||||||
|  |  | ||||||
| export function App() { | export function App() { | ||||||
|   const { code: loadedCode, project } = useLoaderData() as IndexLoaderData |   const { project, file } = useLoaderData() as IndexLoaderData | ||||||
|  |   const navigate = useNavigate() | ||||||
|  |   const filePath = useAbsoluteFilePath() | ||||||
|  |  | ||||||
|   const streamRef = useRef<HTMLDivElement>(null) |  | ||||||
|   useHotKeyListener() |   useHotKeyListener() | ||||||
|   const { |   const { | ||||||
|     setCode, |  | ||||||
|     engineCommandManager, |  | ||||||
|     buttonDownInStream, |     buttonDownInStream, | ||||||
|     openPanes, |     openPanes, | ||||||
|     setOpenPanes, |     setOpenPanes, | ||||||
|     didDragInStream, |     didDragInStream, | ||||||
|     streamDimensions, |     streamDimensions, | ||||||
|     guiMode, |  | ||||||
|     setGuiMode, |  | ||||||
|     executeAst, |  | ||||||
|   } = useStore((s) => ({ |   } = useStore((s) => ({ | ||||||
|     guiMode: s.guiMode, |  | ||||||
|     setGuiMode: s.setGuiMode, |  | ||||||
|     setCode: s.setCode, |  | ||||||
|     engineCommandManager: s.engineCommandManager, |  | ||||||
|     buttonDownInStream: s.buttonDownInStream, |     buttonDownInStream: s.buttonDownInStream, | ||||||
|     openPanes: s.openPanes, |     openPanes: s.openPanes, | ||||||
|     setOpenPanes: s.setOpenPanes, |     setOpenPanes: s.setOpenPanes, | ||||||
|     didDragInStream: s.didDragInStream, |     didDragInStream: s.didDragInStream, | ||||||
|     streamDimensions: s.streamDimensions, |     streamDimensions: s.streamDimensions, | ||||||
|     executeAst: s.executeAst, |  | ||||||
|   })) |   })) | ||||||
|  |  | ||||||
|   const { |   const { settings } = useGlobalStateContext() | ||||||
|     auth: { |   const { showDebugPanel, onboardingStatus, theme } = settings?.context || {} | ||||||
|       context: { token }, |   const { state, send } = useModelingContext() | ||||||
|     }, |  | ||||||
|     settings: { |  | ||||||
|       context: { showDebugPanel, onboardingStatus, cameraControls, theme }, |  | ||||||
|     }, |  | ||||||
|   } = useGlobalStateContext() |  | ||||||
|  |  | ||||||
|   const editorTheme = theme === Themes.System ? getSystemTheme() : theme |   const editorTheme = theme === Themes.System ? getSystemTheme() : theme | ||||||
|  |  | ||||||
| @ -85,50 +72,17 @@ export function App() { | |||||||
|   useHotkeys('shift + l', () => togglePane('logs')) |   useHotkeys('shift + l', () => togglePane('logs')) | ||||||
|   useHotkeys('shift + e', () => togglePane('kclErrors')) |   useHotkeys('shift + e', () => togglePane('kclErrors')) | ||||||
|   useHotkeys('shift + d', () => togglePane('debug')) |   useHotkeys('shift + d', () => togglePane('debug')) | ||||||
|   useHotkeys('esc', () => { |   useHotkeys('esc', () => send('Cancel')) | ||||||
|     if (guiMode.mode === 'sketch') { |   useHotkeys('backspace', (e) => { | ||||||
|       if (guiMode.sketchMode === 'selectFace') return |     e.preventDefault() | ||||||
|       if (guiMode.sketchMode === 'sketchEdit') { |  | ||||||
|         // TODO: share this with Toolbar's "Exit sketch" button |  | ||||||
|         // exiting sketch should be done consistently across all exits |  | ||||||
|         engineCommandManager?.sendSceneCommand({ |  | ||||||
|           type: 'modeling_cmd_req', |  | ||||||
|           cmd_id: uuidv4(), |  | ||||||
|           cmd: { type: 'edit_mode_exit' }, |  | ||||||
|         }) |  | ||||||
|         engineCommandManager?.sendSceneCommand({ |  | ||||||
|           type: 'modeling_cmd_req', |  | ||||||
|           cmd_id: uuidv4(), |  | ||||||
|           cmd: { type: 'default_camera_disable_sketch_mode' }, |  | ||||||
|         }) |  | ||||||
|         setGuiMode({ mode: 'default' }) |  | ||||||
|         // this is necessary to get the UI back into a consistent |  | ||||||
|         // state right now, hopefully won't need to rerender |  | ||||||
|         // when exiting sketch mode in the future |  | ||||||
|         executeAst() |  | ||||||
|       } else { |  | ||||||
|         engineCommandManager?.sendSceneCommand({ |  | ||||||
|           type: 'modeling_cmd_req', |  | ||||||
|           cmd_id: uuidv4(), |  | ||||||
|           cmd: { |  | ||||||
|             type: 'set_tool', |  | ||||||
|             tool: 'select', |  | ||||||
|           }, |  | ||||||
|         }) |  | ||||||
|         setGuiMode({ |  | ||||||
|           mode: 'sketch', |  | ||||||
|           sketchMode: 'sketchEdit', |  | ||||||
|           rotation: guiMode.rotation, |  | ||||||
|           position: guiMode.position, |  | ||||||
|           pathToNode: guiMode.pathToNode, |  | ||||||
|           pathId: guiMode.pathId, |  | ||||||
|           // todo: ...guiMod is adding isTooltip: true, will probably just fix with xstate migtaion |  | ||||||
|         }) |  | ||||||
|       } |  | ||||||
|     } else { |  | ||||||
|       setGuiMode({ mode: 'default' }) |  | ||||||
|     } |  | ||||||
|   }) |   }) | ||||||
|  |   useHotkeys( | ||||||
|  |     isTauri() ? 'mod + ,' : 'shift + mod + ,', | ||||||
|  |     () => navigate(filePath + paths.SETTINGS), | ||||||
|  |     { | ||||||
|  |       splitKey: '|', | ||||||
|  |     } | ||||||
|  |   ) | ||||||
|  |  | ||||||
|   const paneOpacity = [onboardingPaths.CAMERA, onboardingPaths.STREAMING].some( |   const paneOpacity = [onboardingPaths.CAMERA, onboardingPaths.STREAMING].some( | ||||||
|     (p) => p === onboardingStatus |     (p) => p === onboardingStatus | ||||||
| @ -138,28 +92,15 @@ export function App() { | |||||||
|     ? 'opacity-40' |     ? 'opacity-40' | ||||||
|     : '' |     : '' | ||||||
|  |  | ||||||
|   // Use file code loaded from disk |  | ||||||
|   // on mount, and overwrite any locally-stored code |  | ||||||
|   useEffect(() => { |  | ||||||
|     if (isTauri() && loadedCode !== null) { |  | ||||||
|       setCode(loadedCode) |  | ||||||
|     } |  | ||||||
|     return () => { |  | ||||||
|       // Clear code on unmount if in desktop app |  | ||||||
|       if (isTauri()) { |  | ||||||
|         setCode('') |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   }, [loadedCode, setCode]) |  | ||||||
|  |  | ||||||
|   useSetupEngineManager(streamRef, token) |  | ||||||
|   useEngineConnectionSubscriptions() |   useEngineConnectionSubscriptions() | ||||||
|  |  | ||||||
|   const debounceSocketSend = throttle<EngineCommand>((message) => { |   const debounceSocketSend = throttle<EngineCommand>((message) => { | ||||||
|     engineCommandManager?.sendSceneCommand(message) |     engineCommandManager.sendSceneCommand(message) | ||||||
|   }, 16) |   }, 1000 / 15) | ||||||
|   const handleMouseMove: MouseEventHandler<HTMLDivElement> = (e) => { |   const handleMouseMove: MouseEventHandler<HTMLDivElement> = (e) => { | ||||||
|     e.nativeEvent.preventDefault() |     if (state.matches('Sketch')) { | ||||||
|  |       return | ||||||
|  |     } | ||||||
|  |  | ||||||
|     const { x, y } = getNormalisedCoordinates({ |     const { x, y } = getNormalisedCoordinates({ | ||||||
|       clientX: e.clientX, |       clientX: e.clientX, | ||||||
| @ -170,62 +111,11 @@ export function App() { | |||||||
|  |  | ||||||
|     const newCmdId = uuidv4() |     const newCmdId = uuidv4() | ||||||
|     if (buttonDownInStream === undefined) { |     if (buttonDownInStream === undefined) { | ||||||
|       if ( |  | ||||||
|         guiMode.mode === 'sketch' && |  | ||||||
|         guiMode.sketchMode === ('sketch_line' as any) |  | ||||||
|       ) { |  | ||||||
|         debounceSocketSend({ |  | ||||||
|           type: 'modeling_cmd_req', |  | ||||||
|           cmd_id: newCmdId, |  | ||||||
|           cmd: { |  | ||||||
|             type: 'mouse_move', |  | ||||||
|             window: { x, y }, |  | ||||||
|           }, |  | ||||||
|         }) |  | ||||||
|       } else { |  | ||||||
|         debounceSocketSend({ |  | ||||||
|           type: 'modeling_cmd_req', |  | ||||||
|           cmd: { |  | ||||||
|             type: 'highlight_set_entity', |  | ||||||
|             selected_at_window: { x, y }, |  | ||||||
|           }, |  | ||||||
|           cmd_id: newCmdId, |  | ||||||
|         }) |  | ||||||
|       } |  | ||||||
|     } else { |  | ||||||
|       if (guiMode.mode === 'sketch' && guiMode.sketchMode === ('move' as any)) { |  | ||||||
|         debounceSocketSend({ |  | ||||||
|           type: 'modeling_cmd_req', |  | ||||||
|           cmd_id: newCmdId, |  | ||||||
|           cmd: { |  | ||||||
|             type: 'handle_mouse_drag_move', |  | ||||||
|             window: { x, y }, |  | ||||||
|           }, |  | ||||||
|         }) |  | ||||||
|         return |  | ||||||
|       } |  | ||||||
|       const interactionGuards = cameraMouseDragGuards[cameraControls] |  | ||||||
|       let interaction: CameraDragInteractionType_type |  | ||||||
|  |  | ||||||
|       const eWithButton = { ...e, button: buttonDownInStream } |  | ||||||
|  |  | ||||||
|       if (interactionGuards.pan.callback(eWithButton)) { |  | ||||||
|         interaction = 'pan' |  | ||||||
|       } else if (interactionGuards.rotate.callback(eWithButton)) { |  | ||||||
|         interaction = 'rotate' |  | ||||||
|       } else if (interactionGuards.zoom.dragCallback(eWithButton)) { |  | ||||||
|         interaction = 'zoom' |  | ||||||
|       } else { |  | ||||||
|         console.log('none') |  | ||||||
|         return |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       debounceSocketSend({ |       debounceSocketSend({ | ||||||
|         type: 'modeling_cmd_req', |         type: 'modeling_cmd_req', | ||||||
|         cmd: { |         cmd: { | ||||||
|           type: 'camera_drag_move', |           type: 'highlight_set_entity', | ||||||
|           interaction, |           selected_at_window: { x, y }, | ||||||
|           window: { x, y }, |  | ||||||
|         }, |         }, | ||||||
|         cmd_id: newCmdId, |         cmd_id: newCmdId, | ||||||
|       }) |       }) | ||||||
| @ -234,9 +124,8 @@ export function App() { | |||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <div |     <div | ||||||
|       className="h-screen overflow-hidden relative flex flex-col cursor-pointer select-none" |       className="relative h-full flex flex-col" | ||||||
|       onMouseMove={handleMouseMove} |       onMouseMove={handleMouseMove} | ||||||
|       ref={streamRef} |  | ||||||
|     > |     > | ||||||
|       <AppHeader |       <AppHeader | ||||||
|         className={ |         className={ | ||||||
| @ -244,17 +133,14 @@ export function App() { | |||||||
|           paneOpacity + |           paneOpacity + | ||||||
|           (buttonDownInStream ? ' pointer-events-none' : '') |           (buttonDownInStream ? ' pointer-events-none' : '') | ||||||
|         } |         } | ||||||
|         project={project} |         project={{ project, file }} | ||||||
|         enableMenu={true} |         enableMenu={true} | ||||||
|       /> |       /> | ||||||
|       <ModalContainer /> |       <ModalContainer /> | ||||||
|       <Resizable |       <Resizable | ||||||
|         className={ |         className={ | ||||||
|           'h-full flex flex-col flex-1 z-10 my-5 ml-5 pr-1 transition-opacity transition-duration-75 ' + |           'pointer-events-none h-full flex flex-col flex-1 z-10 my-5 ml-5 pr-1 transition-opacity transition-duration-75 ' + | ||||||
|           (buttonDownInStream || onboardingStatus === 'camera' |           +paneOpacity | ||||||
|             ? ' pointer-events-none ' |  | ||||||
|             : ' ') + |  | ||||||
|           paneOpacity |  | ||||||
|         } |         } | ||||||
|         defaultSize={{ |         defaultSize={{ | ||||||
|           width: '550px', |           width: '550px', | ||||||
| @ -266,10 +152,16 @@ export function App() { | |||||||
|         maxHeight={'auto'} |         maxHeight={'auto'} | ||||||
|         handleClasses={{ |         handleClasses={{ | ||||||
|           right: |           right: | ||||||
|             'hover:bg-liquid-30/40 dark:hover:bg-liquid-10/40 bg-transparent transition-colors duration-100 transition-ease-out delay-100', |             'hover:bg-chalkboard-10/50 bg-transparent transition-colors duration-75 transition-ease-out delay-100 ' + | ||||||
|  |             (buttonDownInStream || onboardingStatus === 'camera' | ||||||
|  |               ? 'pointer-events-none ' | ||||||
|  |               : 'pointer-events-auto'), | ||||||
|         }} |         }} | ||||||
|       > |       > | ||||||
|         <div id="code-pane" className="h-full flex flex-col justify-between"> |         <div | ||||||
|  |           id="code-pane" | ||||||
|  |           className="h-full flex flex-col justify-between pointer-events-none" | ||||||
|  |         > | ||||||
|           <CollapsiblePanel |           <CollapsiblePanel | ||||||
|             title="Code" |             title="Code" | ||||||
|             icon={faCode} |             icon={faCode} | ||||||
| @ -313,6 +205,7 @@ export function App() { | |||||||
|           open={openPanes.includes('debug')} |           open={openPanes.includes('debug')} | ||||||
|         /> |         /> | ||||||
|       )} |       )} | ||||||
|  |       {/* <CamToggle /> */} | ||||||
|     </div> |     </div> | ||||||
|   ) |   ) | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										10
									
								
								src/Auth.tsx
									
									
									
									
									
								
							
							
						
						| @ -3,13 +3,13 @@ import { useGlobalStateContext } from 'hooks/useGlobalStateContext' | |||||||
|  |  | ||||||
| // Wrapper around protected routes, used in src/Router.tsx | // Wrapper around protected routes, used in src/Router.tsx | ||||||
| export const Auth = ({ children }: React.PropsWithChildren) => { | export const Auth = ({ children }: React.PropsWithChildren) => { | ||||||
|   const { |   const { auth } = useGlobalStateContext() | ||||||
|     auth: { state }, |   const isLoggingIn = auth?.state.matches('checkIfLoggedIn') | ||||||
|   } = useGlobalStateContext() |  | ||||||
|   const isLoggingIn = state.matches('checkIfLoggedIn') |  | ||||||
|  |  | ||||||
|   return isLoggingIn ? ( |   return isLoggingIn ? ( | ||||||
|     <Loading>Loading KittyCAD Modeling App...</Loading> |     <Loading> | ||||||
|  |       <span data-testid="initial-load">Loading Modeling App...</span> | ||||||
|  |     </Loading> | ||||||
|   ) : ( |   ) : ( | ||||||
|     <>{children}</> |     <>{children}</> | ||||||
|   ) |   ) | ||||||
|  | |||||||
							
								
								
									
										125
									
								
								src/Router.tsx
									
									
									
									
									
								
							
							
						
						| @ -14,10 +14,7 @@ import { | |||||||
| import { useEffect } from 'react' | import { useEffect } from 'react' | ||||||
| import { ErrorPage } from './components/ErrorPage' | import { ErrorPage } from './components/ErrorPage' | ||||||
| import { Settings } from './routes/Settings' | import { Settings } from './routes/Settings' | ||||||
| import Onboarding, { | import Onboarding, { onboardingRoutes } from './routes/Onboarding' | ||||||
|   onboardingRoutes, |  | ||||||
|   onboardingPaths, |  | ||||||
| } from './routes/Onboarding' |  | ||||||
| import SignIn from './routes/SignIn' | import SignIn from './routes/SignIn' | ||||||
| import { Auth } from './Auth' | import { Auth } from './Auth' | ||||||
| import { isTauri } from './lib/isTauri' | import { isTauri } from './lib/isTauri' | ||||||
| @ -29,17 +26,25 @@ import { | |||||||
|   isProjectDirectory, |   isProjectDirectory, | ||||||
|   PROJECT_ENTRYPOINT, |   PROJECT_ENTRYPOINT, | ||||||
| } from './lib/tauriFS' | } from './lib/tauriFS' | ||||||
| import { metadata, type Metadata } from 'tauri-plugin-fs-extra-api' | import { metadata } from 'tauri-plugin-fs-extra-api' | ||||||
| import DownloadAppBanner from './components/DownloadAppBanner' | import DownloadAppBanner from './components/DownloadAppBanner' | ||||||
|  | import { WasmErrBanner } from './components/WasmErrBanner' | ||||||
| import { GlobalStateProvider } from './components/GlobalStateProvider' | import { GlobalStateProvider } from './components/GlobalStateProvider' | ||||||
| import { | import { | ||||||
|   SETTINGS_PERSIST_KEY, |   SETTINGS_PERSIST_KEY, | ||||||
|   settingsMachine, |   settingsMachine, | ||||||
| } from './machines/settingsMachine' | } from './machines/settingsMachine' | ||||||
| import { ContextFrom } from 'xstate' | import { ContextFrom } from 'xstate' | ||||||
| import CommandBarProvider from 'components/CommandBar' | import CommandBarProvider from 'components/CommandBar/CommandBar' | ||||||
| import { TEST, VITE_KC_SENTRY_DSN } from './env' | import { TEST, VITE_KC_SENTRY_DSN } from './env' | ||||||
| import * as Sentry from '@sentry/react' | import * as Sentry from '@sentry/react' | ||||||
|  | import ModelingMachineProvider from 'components/ModelingMachineProvider' | ||||||
|  | import { KclContextProvider, kclManager } from 'lang/KclSingleton' | ||||||
|  | import FileMachineProvider from 'components/FileMachineProvider' | ||||||
|  | import { sep } from '@tauri-apps/api/path' | ||||||
|  | import { paths } from 'lib/paths' | ||||||
|  | import { IndexLoaderData, HomeLoaderData } from 'lib/types' | ||||||
|  | import { fileSystemManager } from 'lang/std/fileSystemManager' | ||||||
|  |  | ||||||
| if (VITE_KC_SENTRY_DSN && !TEST) { | if (VITE_KC_SENTRY_DSN && !TEST) { | ||||||
|   Sentry.init({ |   Sentry.init({ | ||||||
| @ -73,38 +78,7 @@ if (VITE_KC_SENTRY_DSN && !TEST) { | |||||||
|   }) |   }) | ||||||
| } | } | ||||||
|  |  | ||||||
| const prependRoutes = | export const BROWSER_FILE_NAME = 'new' | ||||||
|   (routesObject: Record<string, string>) => (prepend: string) => { |  | ||||||
|     return Object.fromEntries( |  | ||||||
|       Object.entries(routesObject).map(([constName, path]) => [ |  | ||||||
|         constName, |  | ||||||
|         prepend + path, |  | ||||||
|       ]) |  | ||||||
|     ) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
| export const paths = { |  | ||||||
|   INDEX: '/', |  | ||||||
|   HOME: '/home', |  | ||||||
|   FILE: '/file', |  | ||||||
|   SETTINGS: '/settings', |  | ||||||
|   SIGN_IN: '/signin', |  | ||||||
|   ONBOARDING: prependRoutes(onboardingPaths)( |  | ||||||
|     '/onboarding' |  | ||||||
|   ) as typeof onboardingPaths, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export type IndexLoaderData = { |  | ||||||
|   code: string | null |  | ||||||
|   project?: ProjectWithEntryPointMetadata |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export type ProjectWithEntryPointMetadata = FileEntry & { |  | ||||||
|   entrypoint_metadata: Metadata |  | ||||||
| } |  | ||||||
| export type HomeLoaderData = { |  | ||||||
|   projects: ProjectWithEntryPointMetadata[] |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type CreateBrowserRouterArg = Parameters<typeof createBrowserRouter>[0] | type CreateBrowserRouterArg = Parameters<typeof createBrowserRouter>[0] | ||||||
|  |  | ||||||
| @ -129,17 +103,26 @@ const router = createBrowserRouter( | |||||||
|     { |     { | ||||||
|       path: paths.INDEX, |       path: paths.INDEX, | ||||||
|       loader: () => |       loader: () => | ||||||
|         isTauri() ? redirect(paths.HOME) : redirect(paths.FILE + '/new'), |         isTauri() | ||||||
|  |           ? redirect(paths.HOME) | ||||||
|  |           : redirect(paths.FILE + '/' + BROWSER_FILE_NAME), | ||||||
|       errorElement: <ErrorPage />, |       errorElement: <ErrorPage />, | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       path: paths.FILE + '/:id', |       path: paths.FILE + '/:id', | ||||||
|       element: ( |       element: ( | ||||||
|         <Auth> |         <KclContextProvider> | ||||||
|           <Outlet /> |           <Auth> | ||||||
|           <App /> |             <FileMachineProvider> | ||||||
|           {!isTauri() && import.meta.env.PROD && <DownloadAppBanner />} |               <ModelingMachineProvider> | ||||||
|         </Auth> |                 <Outlet /> | ||||||
|  |                 <App /> | ||||||
|  |               </ModelingMachineProvider> | ||||||
|  |               <WasmErrBanner /> | ||||||
|  |             </FileMachineProvider> | ||||||
|  |             {!isTauri() && import.meta.env.PROD && <DownloadAppBanner />} | ||||||
|  |           </Auth> | ||||||
|  |         </KclContextProvider> | ||||||
|       ), |       ), | ||||||
|       id: paths.FILE, |       id: paths.FILE, | ||||||
|       loader: async ({ |       loader: async ({ | ||||||
| @ -167,21 +150,46 @@ const router = createBrowserRouter( | |||||||
|           ) |           ) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (params.id && params.id !== 'new') { |         const defaultDir = persistedSettings.defaultDirectory || '' | ||||||
|  |  | ||||||
|  |         if (params.id && params.id !== BROWSER_FILE_NAME) { | ||||||
|  |           const decodedId = decodeURIComponent(params.id) | ||||||
|  |           const projectAndFile = decodedId.replace(defaultDir + sep, '') | ||||||
|  |           const firstSlashIndex = projectAndFile.indexOf(sep) | ||||||
|  |           const projectName = projectAndFile.slice(0, firstSlashIndex) | ||||||
|  |           const projectPath = defaultDir + sep + projectName | ||||||
|  |           const currentFileName = projectAndFile.slice(firstSlashIndex + 1) | ||||||
|  |  | ||||||
|  |           if (firstSlashIndex === -1 || !currentFileName) | ||||||
|  |             return redirect( | ||||||
|  |               `${paths.FILE}/${encodeURIComponent( | ||||||
|  |                 `${params.id}${sep}${PROJECT_ENTRYPOINT}` | ||||||
|  |               )}` | ||||||
|  |             ) | ||||||
|  |  | ||||||
|           // Note that PROJECT_ENTRYPOINT is hardcoded until we support multiple files |           // Note that PROJECT_ENTRYPOINT is hardcoded until we support multiple files | ||||||
|           const code = await readTextFile(params.id + '/' + PROJECT_ENTRYPOINT) |           const code = await readTextFile(decodedId) | ||||||
|           const entrypoint_metadata = await metadata( |           const entrypointMetadata = await metadata( | ||||||
|             params.id + '/' + PROJECT_ENTRYPOINT |             projectPath + sep + PROJECT_ENTRYPOINT | ||||||
|           ) |           ) | ||||||
|           const children = await readDir(params.id) |           const children = await readDir(projectPath, { recursive: true }) | ||||||
|  |           kclManager.setCodeAndExecute(code, false) | ||||||
|  |  | ||||||
|  |           // Set the file system manager to the project path | ||||||
|  |           // So that WASM gets an updated path for operations | ||||||
|  |           fileSystemManager.dir = projectPath | ||||||
|  |  | ||||||
|           return { |           return { | ||||||
|             code, |             code, | ||||||
|             project: { |             project: { | ||||||
|               name: params.id.slice(params.id.lastIndexOf('/') + 1), |               name: projectName, | ||||||
|               path: params.id, |               path: projectPath, | ||||||
|               children, |               children, | ||||||
|               entrypoint_metadata, |               entrypointMetadata, | ||||||
|  |             }, | ||||||
|  |             file: { | ||||||
|  |               name: currentFileName, | ||||||
|  |               path: params.id, | ||||||
|             }, |             }, | ||||||
|           } |           } | ||||||
|         } |         } | ||||||
| @ -210,9 +218,9 @@ const router = createBrowserRouter( | |||||||
|           <Home /> |           <Home /> | ||||||
|         </Auth> |         </Auth> | ||||||
|       ), |       ), | ||||||
|       loader: async () => { |       loader: async (): Promise<HomeLoaderData | Response> => { | ||||||
|         if (!isTauri()) { |         if (!isTauri()) { | ||||||
|           return redirect(paths.FILE + '/new') |           return redirect(paths.FILE + '/' + BROWSER_FILE_NAME) | ||||||
|         } |         } | ||||||
|         const fetchedStorage = localStorage?.getItem(SETTINGS_PERSIST_KEY) |         const fetchedStorage = localStorage?.getItem(SETTINGS_PERSIST_KEY) | ||||||
|         const persistedSettings = JSON.parse(fetchedStorage || '{}') as Partial< |         const persistedSettings = JSON.parse(fetchedStorage || '{}') as Partial< | ||||||
| @ -221,6 +229,7 @@ const router = createBrowserRouter( | |||||||
|         const projectDir = await initializeProjectDirectory( |         const projectDir = await initializeProjectDirectory( | ||||||
|           persistedSettings.defaultDirectory || '' |           persistedSettings.defaultDirectory || '' | ||||||
|         ) |         ) | ||||||
|  |         let newDefaultDirectory: string | undefined = undefined | ||||||
|         if (projectDir !== persistedSettings.defaultDirectory) { |         if (projectDir !== persistedSettings.defaultDirectory) { | ||||||
|           localStorage.setItem( |           localStorage.setItem( | ||||||
|             SETTINGS_PERSIST_KEY, |             SETTINGS_PERSIST_KEY, | ||||||
| @ -229,14 +238,15 @@ const router = createBrowserRouter( | |||||||
|               defaultDirectory: projectDir, |               defaultDirectory: projectDir, | ||||||
|             }) |             }) | ||||||
|           ) |           ) | ||||||
|  |           newDefaultDirectory = projectDir | ||||||
|         } |         } | ||||||
|         const projectsNoMeta = (await readDir(projectDir)).filter( |         const projectsNoMeta = (await readDir(projectDir)).filter( | ||||||
|           isProjectDirectory |           isProjectDirectory | ||||||
|         ) |         ) | ||||||
|         const projects = await Promise.all( |         const projects = await Promise.all( | ||||||
|           projectsNoMeta.map(async (p) => ({ |           projectsNoMeta.map(async (p: FileEntry) => ({ | ||||||
|             entrypoint_metadata: await metadata( |             entrypointMetadata: await metadata( | ||||||
|               p.path + '/' + PROJECT_ENTRYPOINT |               p.path + sep + PROJECT_ENTRYPOINT | ||||||
|             ), |             ), | ||||||
|             ...p, |             ...p, | ||||||
|           })) |           })) | ||||||
| @ -244,6 +254,7 @@ const router = createBrowserRouter( | |||||||
|  |  | ||||||
|         return { |         return { | ||||||
|           projects, |           projects, | ||||||
|  |           newDefaultDirectory, | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       children: [ |       children: [ | ||||||
|  | |||||||
| @ -1,106 +0,0 @@ | |||||||
| .toolbarWrapper { |  | ||||||
|   @apply relative; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .toolbar { |  | ||||||
|   @apply flex gap-4 items-center rounded-full; |  | ||||||
|   @apply border border-cool-20/30 bg-cool-10/50; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| :global(.dark) .toolbar { |  | ||||||
|   @apply border-cool-100/50 bg-cool-120/50; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| :global(.sketch) .toolbar { |  | ||||||
|   @apply border-fern-20/20 bg-fern-10/20; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| :global(.dark .sketch) .toolbar { |  | ||||||
|   @apply border-fern-120/50 bg-fern-100/30; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .toolbarCap { |  | ||||||
|   @apply text-sm font-bold; |  | ||||||
|   @apply bg-cool-20/50 text-cool-100; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| :global(.dark) .toolbarCap { |  | ||||||
|   @apply bg-cool-90/50 text-cool-30; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| :global(.sketch) .toolbarCap { |  | ||||||
|   @apply bg-fern-20/50 text-fern-100; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| :global(.dark .sketch) .toolbarCap { |  | ||||||
|   @apply bg-fern-90/50 text-fern-30; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .label { |  | ||||||
|   @apply self-stretch flex items-center px-4 py-1; |  | ||||||
|   @apply rounded-l-full; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .popoverToggle { |  | ||||||
|   @apply self-stretch m-0 flex items-center px-4 py-1; |  | ||||||
|   @apply rounded-r-full border-none; |  | ||||||
|   @apply hover:bg-cool-20; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .toolbarButtons::-webkit-scrollbar { |  | ||||||
|   @apply h-0.5; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .toolbarButtons { |  | ||||||
|   @apply flex items-center overflow-x-auto; |  | ||||||
|   scrollbar-width: thin; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .toolbarButtons button { |  | ||||||
|   @apply text-chalkboard-90 bg-chalkboard-10/50 border-chalkboard-50 whitespace-nowrap; |  | ||||||
|   display: inline-flex; |  | ||||||
|   align-items: center; |  | ||||||
|   justify-content: center; |  | ||||||
|   @apply gap-1.5 p-0.5 pr-1; |  | ||||||
|   @apply rounded-sm; |  | ||||||
| } |  | ||||||
| :global(.dark) .toolbarButtons button { |  | ||||||
|   @apply text-chalkboard-30 bg-chalkboard-90/50 border-chalkboard-50; |  | ||||||
| } |  | ||||||
| .toolbarButtons button:hover { |  | ||||||
|   @apply text-cool-90 bg-cool-10; |  | ||||||
| } |  | ||||||
| :global(.sketch) .toolbarButtons button:hover { |  | ||||||
|   @apply text-fern-90 bg-fern-10; |  | ||||||
| } |  | ||||||
| .toolbarButtons button:disabled { |  | ||||||
|   @apply text-chalkboard-70 bg-chalkboard-30; |  | ||||||
| } |  | ||||||
| .toolbarButtons button:disabled:hover { |  | ||||||
|   @apply !bg-inherit !text-inherit cursor-not-allowed; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| :global(.dark) .toolbarButtons button { |  | ||||||
|   @apply text-chalkboard-20 border-chalkboard-50; |  | ||||||
| } |  | ||||||
| :global(.dark) .toolbarButtons button:hover { |  | ||||||
|   @apply text-cool-10 border-chalkboard-50 bg-cool-90; |  | ||||||
| } |  | ||||||
| :global(.dark .sketch) .toolbarButtons button:hover { |  | ||||||
|   @apply text-fern-10 border-chalkboard-50 bg-fern-90; |  | ||||||
| } |  | ||||||
| :global(.dark) .toolbarButtons button:disabled { |  | ||||||
|   @apply text-chalkboard-40 bg-chalkboard-80; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| :global(.dark) .popoverToggle { |  | ||||||
|   @apply hover:bg-cool-90; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| :global(.sketch) .popoverToggle { |  | ||||||
|   @apply hover:bg-fern-20; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| :global(.dark .sketch) .popoverToggle { |  | ||||||
|   @apply hover:bg-fern-90; |  | ||||||
| } |  | ||||||
							
								
								
									
										525
									
								
								src/Toolbar.tsx
									
									
									
									
									
								
							
							
						
						| @ -1,335 +1,230 @@ | |||||||
| import { useStore, toolTips, ToolTip } from './useStore' | import { WheelEvent, useRef, useMemo } from 'react' | ||||||
| import { extrudeSketch, sketchOnExtrudedFace } from './lang/modifyAst' | import { isCursorInSketchCommandRange } from 'lang/util' | ||||||
| import { getNodePathFromSourceRange } from './lang/queryAst' | import { engineCommandManager } from './lang/std/engineConnection' | ||||||
| import { HorzVert } from './components/Toolbar/HorzVert' | import { useModelingContext } from 'hooks/useModelingContext' | ||||||
| import { RemoveConstrainingValues } from './components/Toolbar/RemoveConstrainingValues' | import { useCommandsContext } from 'hooks/useCommandsContext' | ||||||
| import { EqualLength } from './components/Toolbar/EqualLength' | import { ActionButton } from 'components/ActionButton' | ||||||
| import { EqualAngle } from './components/Toolbar/EqualAngle' | import usePlatform from 'hooks/usePlatform' | ||||||
| import { Intersect } from './components/Toolbar/Intersect' | import { isSingleCursorInPipe } from 'lang/queryAst' | ||||||
| import { SetHorzVertDistance } from './components/Toolbar/SetHorzVertDistance' | import { kclManager } from 'lang/KclSingleton' | ||||||
| import { SetAngleLength } from './components/Toolbar/setAngleLength' |  | ||||||
| import { SetAbsDistance } from './components/Toolbar/SetAbsDistance' |  | ||||||
| import { SetAngleBetween } from './components/Toolbar/SetAngleBetween' |  | ||||||
| import { Fragment, useEffect } from 'react' |  | ||||||
| import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' |  | ||||||
| import { faSearch, faX } from '@fortawesome/free-solid-svg-icons' |  | ||||||
| import { Popover, Transition } from '@headlessui/react' |  | ||||||
| import styles from './Toolbar.module.css' |  | ||||||
| import { v4 as uuidv4 } from 'uuid' |  | ||||||
| import { useAppMode } from 'hooks/useAppMode' |  | ||||||
| import { ActionIcon } from 'components/ActionIcon' |  | ||||||
|  |  | ||||||
| export const sketchButtonClassnames = { |  | ||||||
|   background: |  | ||||||
|     'bg-chalkboard-100 group-hover:bg-chalkboard-90 hover:bg-chalkboard-90 dark:bg-fern-20 dark:group-hover:bg-fern-10 dark:hover:bg-fern-10 group-disabled:bg-chalkboard-50 dark:group-disabled:bg-chalkboard-60 group-hover:group-disabled:bg-chalkboard-50 dark:group-hover:group-disabled:bg-chalkboard-50', |  | ||||||
|   icon: 'text-fern-20 h-auto group-hover:text-fern-10 hover:text-fern-10 dark:text-chalkboard-100 dark:group-hover:text-chalkboard-100 dark:hover:text-chalkboard-100 group-disabled:bg-chalkboard-60 hover:group-disabled:text-inherit', |  | ||||||
| } |  | ||||||
|  |  | ||||||
| const sketchFnLabels: Record<ToolTip | 'sketch_line' | 'move', string> = { |  | ||||||
|   sketch_line: 'Line', |  | ||||||
|   line: 'Line', |  | ||||||
|   move: 'Move', |  | ||||||
|   angledLine: 'Angled Line', |  | ||||||
|   angledLineThatIntersects: 'Angled Line That Intersects', |  | ||||||
|   angledLineOfXLength: 'Angled Line Of X Length', |  | ||||||
|   angledLineOfYLength: 'Angled Line Of Y Length', |  | ||||||
|   angledLineToX: 'Angled Line To X', |  | ||||||
|   angledLineToY: 'Angled Line To Y', |  | ||||||
|   lineTo: 'Line to Point', |  | ||||||
|   xLine: 'Horizontal Line', |  | ||||||
|   yLine: 'Vertical Line', |  | ||||||
|   xLineTo: 'Horizontal Line to Point', |  | ||||||
|   yLineTo: 'Vertical Line to Point', |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export const Toolbar = () => { | export const Toolbar = () => { | ||||||
|   const { |   const platform = usePlatform() | ||||||
|     setGuiMode, |   const { commandBarSend } = useCommandsContext() | ||||||
|     guiMode, |   const { state, send, context } = useModelingContext() | ||||||
|     selectionRanges, |   const toolbarButtonsRef = useRef<HTMLUListElement>(null) | ||||||
|     ast, |   const bgClassName = | ||||||
|     updateAst, |     'group-enabled:group-hover:bg-energy-10 group-pressed:bg-energy-10 dark:group-enabled:group-hover:bg-chalkboard-80 dark:group-pressed:bg-chalkboard-80' | ||||||
|     programMemory, |   const pathId = useMemo(() => { | ||||||
|     engineCommandManager, |     if (!isSingleCursorInPipe(context.selectionRanges, kclManager.ast)) { | ||||||
|     executeAst, |       return false | ||||||
|   } = useStore((s) => ({ |     } | ||||||
|     guiMode: s.guiMode, |     return isCursorInSketchCommandRange( | ||||||
|     setGuiMode: s.setGuiMode, |       engineCommandManager.artifactMap, | ||||||
|     selectionRanges: s.selectionRanges, |       context.selectionRanges | ||||||
|     ast: s.ast, |     ) | ||||||
|     updateAst: s.updateAst, |   }, [engineCommandManager.artifactMap, context.selectionRanges]) | ||||||
|     programMemory: s.programMemory, |  | ||||||
|     engineCommandManager: s.engineCommandManager, |  | ||||||
|     executeAst: s.executeAst, |  | ||||||
|   })) |  | ||||||
|   useAppMode() |  | ||||||
|  |  | ||||||
|   useEffect(() => { |   function handleToolbarButtonsWheelEvent(ev: WheelEvent<HTMLSpanElement>) { | ||||||
|     console.log('guiMode', guiMode) |     const span = toolbarButtonsRef.current | ||||||
|   }, [guiMode]) |     if (!span) { | ||||||
|  |       return | ||||||
|  |     } | ||||||
|  |  | ||||||
|   function ToolbarButtons({ className }: React.HTMLAttributes<HTMLElement>) { |     span.scrollLeft = span.scrollLeft += ev.deltaY | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function ToolbarButtons({ | ||||||
|  |     className = '', | ||||||
|  |     ...props | ||||||
|  |   }: React.HTMLAttributes<HTMLElement>) { | ||||||
|     return ( |     return ( | ||||||
|       <span className={styles.toolbarButtons + ' ' + className}> |       <ul | ||||||
|         {guiMode.mode === 'default' && ( |         {...props} | ||||||
|           <button |         ref={toolbarButtonsRef} | ||||||
|             onClick={() => { |         onWheel={handleToolbarButtonsWheelEvent} | ||||||
|               setGuiMode({ |         className={ | ||||||
|                 mode: 'sketch', |           'm-0 py-1 rounded-l-sm flex gap-2 items-center overflow-x-auto ' + | ||||||
|                 sketchMode: 'selectFace', |           className | ||||||
|               }) |         } | ||||||
|             }} |         style={{ scrollbarWidth: 'thin' }} | ||||||
|             className="group" |       > | ||||||
|           > |         {state.nextEvents.includes('Enter sketch') && ( | ||||||
|             <ActionIcon icon="sketch" className="!p-0.5" size="md" /> |           <li className="contents"> | ||||||
|             Start Sketch |             <ActionButton | ||||||
|           </button> |               Element="button" | ||||||
|  |               onClick={() => | ||||||
|  |                 send({ type: 'Enter sketch', data: { forceNewSketch: true } }) | ||||||
|  |               } | ||||||
|  |               icon={{ | ||||||
|  |                 icon: 'sketch', | ||||||
|  |                 bgClassName, | ||||||
|  |               }} | ||||||
|  |             > | ||||||
|  |               <span data-testid="start-sketch">Start Sketch</span> | ||||||
|  |             </ActionButton> | ||||||
|  |           </li> | ||||||
|         )} |         )} | ||||||
|         {guiMode.mode === 'canEditExtrude' && ( |         {state.nextEvents.includes('Enter sketch') && pathId && ( | ||||||
|           <button |           <li className="contents"> | ||||||
|             onClick={() => { |             <ActionButton | ||||||
|               if (!ast) return |               Element="button" | ||||||
|               const pathToNode = getNodePathFromSourceRange( |               onClick={() => send({ type: 'Enter sketch' })} | ||||||
|                 ast, |               icon={{ | ||||||
|                 selectionRanges.codeBasedSelections[0].range |                 icon: 'sketch', | ||||||
|               ) |                 bgClassName, | ||||||
|               const { modifiedAst } = sketchOnExtrudedFace( |               }} | ||||||
|                 ast, |             > | ||||||
|                 pathToNode, |               Edit Sketch | ||||||
|                 programMemory |             </ActionButton> | ||||||
|               ) |           </li> | ||||||
|               updateAst(modifiedAst, true) |  | ||||||
|             }} |  | ||||||
|             className="group" |  | ||||||
|           > |  | ||||||
|             <ActionIcon icon="sketch" className="!p-0.5" size="md" /> |  | ||||||
|             Sketch on Face |  | ||||||
|           </button> |  | ||||||
|         )} |         )} | ||||||
|         {guiMode.mode === 'canEditSketch' && ( |         {state.nextEvents.includes('Cancel') && !state.matches('idle') && ( | ||||||
|           <button |           <li className="contents"> | ||||||
|             onClick={() => { |             <ActionButton | ||||||
|               const pathToNode = getNodePathFromSourceRange( |               Element="button" | ||||||
|                 ast, |               onClick={() => send({ type: 'Cancel' })} | ||||||
|                 selectionRanges.codeBasedSelections[0].range |               icon={{ | ||||||
|               ) |                 icon: 'arrowLeft', | ||||||
|               setGuiMode({ |                 bgClassName, | ||||||
|                 mode: 'sketch', |               }} | ||||||
|                 sketchMode: 'enterSketchEdit', |             > | ||||||
|                 pathToNode: pathToNode, |               Exit Sketch | ||||||
|                 rotation: [0, 0, 0, 1], |             </ActionButton> | ||||||
|                 position: [0, 0, 0], |           </li> | ||||||
|                 pathId: guiMode.pathId, |  | ||||||
|               }) |  | ||||||
|             }} |  | ||||||
|             className="group" |  | ||||||
|           > |  | ||||||
|             <ActionIcon icon="sketch" className="!p-0.5" size="md" /> |  | ||||||
|             Edit Sketch |  | ||||||
|           </button> |  | ||||||
|         )} |         )} | ||||||
|         {guiMode.mode === 'canEditSketch' && ( |         {state.matches('Sketch') && !state.matches('idle') && ( | ||||||
|           <> |           <> | ||||||
|             <button |             <li className="contents" key="line-button"> | ||||||
|               onClick={() => { |               <ActionButton | ||||||
|                 if (!ast) return |                 Element="button" | ||||||
|                 const pathToNode = getNodePathFromSourceRange( |                 onClick={() => | ||||||
|                   ast, |                   state?.matches('Sketch.Line tool') | ||||||
|                   selectionRanges.codeBasedSelections[0].range |                     ? send('CancelSketch') | ||||||
|                 ) |                     : send('Equip Line tool') | ||||||
|                 const { modifiedAst, pathToExtrudeArg } = extrudeSketch( |                 } | ||||||
|                   ast, |                 aria-pressed={state?.matches('Sketch.Line tool')} | ||||||
|                   pathToNode |                 className="pressed:bg-energy-10/20 dark:pressed:bg-energy-80" | ||||||
|                 ) |                 icon={{ | ||||||
|                 updateAst(modifiedAst, true, { focusPath: pathToExtrudeArg }) |                   icon: 'line', | ||||||
|               }} |                   bgClassName, | ||||||
|               className="group" |  | ||||||
|             > |  | ||||||
|               <ActionIcon icon="extrude" className="!p-0.5" size="md" /> |  | ||||||
|               Extrude |  | ||||||
|             </button> |  | ||||||
|             <button |  | ||||||
|               onClick={() => { |  | ||||||
|                 if (!ast) return |  | ||||||
|                 const pathToNode = getNodePathFromSourceRange( |  | ||||||
|                   ast, |  | ||||||
|                   selectionRanges.codeBasedSelections[0].range |  | ||||||
|                 ) |  | ||||||
|                 const { modifiedAst, pathToExtrudeArg } = extrudeSketch( |  | ||||||
|                   ast, |  | ||||||
|                   pathToNode, |  | ||||||
|                   false |  | ||||||
|                 ) |  | ||||||
|                 updateAst(modifiedAst, true, { focusPath: pathToExtrudeArg }) |  | ||||||
|               }} |  | ||||||
|               className="group" |  | ||||||
|             > |  | ||||||
|               <ActionIcon icon="extrude" className="!p-0.5" size="md" /> |  | ||||||
|               Extrude as new |  | ||||||
|             </button> |  | ||||||
|           </> |  | ||||||
|         )} |  | ||||||
|  |  | ||||||
|         {guiMode.mode === 'sketch' && ( |  | ||||||
|           <button |  | ||||||
|             onClick={() => { |  | ||||||
|               engineCommandManager?.sendSceneCommand({ |  | ||||||
|                 type: 'modeling_cmd_req', |  | ||||||
|                 cmd_id: uuidv4(), |  | ||||||
|                 cmd: { type: 'edit_mode_exit' }, |  | ||||||
|               }) |  | ||||||
|               engineCommandManager?.sendSceneCommand({ |  | ||||||
|                 type: 'modeling_cmd_req', |  | ||||||
|                 cmd_id: uuidv4(), |  | ||||||
|                 cmd: { type: 'default_camera_disable_sketch_mode' }, |  | ||||||
|               }) |  | ||||||
|  |  | ||||||
|               setGuiMode({ mode: 'default' }) |  | ||||||
|               executeAst() |  | ||||||
|             }} |  | ||||||
|             className="group" |  | ||||||
|           > |  | ||||||
|             <ActionIcon |  | ||||||
|               icon="exit" |  | ||||||
|               className="!p-0.5" |  | ||||||
|               bgClassName={sketchButtonClassnames.background} |  | ||||||
|               iconClassName={sketchButtonClassnames.icon} |  | ||||||
|               size="md" |  | ||||||
|             /> |  | ||||||
|             Exit sketch |  | ||||||
|           </button> |  | ||||||
|         )} |  | ||||||
|         {toolTips |  | ||||||
|           .filter( |  | ||||||
|             // (sketchFnName) => !['angledLineThatIntersects'].includes(sketchFnName) |  | ||||||
|             (sketchFnName) => ['sketch_line', 'move'].includes(sketchFnName) |  | ||||||
|           ) |  | ||||||
|           .map((sketchFnName) => { |  | ||||||
|             if ( |  | ||||||
|               guiMode.mode !== 'sketch' || |  | ||||||
|               !('isTooltip' in guiMode || guiMode.sketchMode === 'sketchEdit') |  | ||||||
|             ) |  | ||||||
|               return null |  | ||||||
|             return ( |  | ||||||
|               <button |  | ||||||
|                 key={sketchFnName} |  | ||||||
|                 onClick={() => { |  | ||||||
|                   engineCommandManager?.sendSceneCommand({ |  | ||||||
|                     type: 'modeling_cmd_req', |  | ||||||
|                     cmd_id: uuidv4(), |  | ||||||
|                     cmd: { |  | ||||||
|                       type: 'set_tool', |  | ||||||
|                       tool: |  | ||||||
|                         guiMode.sketchMode === sketchFnName |  | ||||||
|                           ? 'select' |  | ||||||
|                           : (sketchFnName as any), |  | ||||||
|                     }, |  | ||||||
|                   }) |  | ||||||
|                   setGuiMode({ |  | ||||||
|                     ...guiMode, |  | ||||||
|                     ...(guiMode.sketchMode === sketchFnName |  | ||||||
|                       ? { |  | ||||||
|                           sketchMode: 'sketchEdit', |  | ||||||
|                           // todo: ...guiMod is adding isTooltip: true, will probably just fix with xstate migtaion |  | ||||||
|                         } |  | ||||||
|                       : { |  | ||||||
|                           sketchMode: sketchFnName, |  | ||||||
|                           waitingFirstClick: true, |  | ||||||
|                           isTooltip: true, |  | ||||||
|                           pathId: guiMode.pathId, |  | ||||||
|                         }), |  | ||||||
|                   }) |  | ||||||
|                 }} |                 }} | ||||||
|                 className={ |               > | ||||||
|                   'group ' + |                 Line | ||||||
|                   (guiMode.sketchMode === sketchFnName |               </ActionButton> | ||||||
|                     ? '!text-fern-70 !bg-fern-10 !dark:text-fern-20 !border-fern-50' |             </li> | ||||||
|                     : '') |             <li className="contents" key="tangential-arc-button"> | ||||||
|  |               <ActionButton | ||||||
|  |                 Element="button" | ||||||
|  |                 onClick={() => | ||||||
|  |                   state.matches('Sketch.Tangential arc to') | ||||||
|  |                     ? send('CancelSketch') | ||||||
|  |                     : send('Equip tangential arc to') | ||||||
|  |                 } | ||||||
|  |                 aria-pressed={state.matches('Sketch.Tangential arc to')} | ||||||
|  |                 className="pressed:bg-energy-10/20 dark:pressed:bg-energy-80" | ||||||
|  |                 icon={{ | ||||||
|  |                   icon: 'arc', | ||||||
|  |                   bgClassName, | ||||||
|  |                 }} | ||||||
|  |                 disabled={ | ||||||
|  |                   !state.can('Equip tangential arc to') && | ||||||
|  |                   !state.matches('Sketch.Tangential arc to') | ||||||
|                 } |                 } | ||||||
|               > |               > | ||||||
|                 <ActionIcon |                 Tangential Arc | ||||||
|                   icon={sketchFnName.includes('line') ? 'line' : 'move'} |               </ActionButton> | ||||||
|                   className="!p-0.5" |             </li> | ||||||
|                   bgClassName={sketchButtonClassnames.background} |           </> | ||||||
|                   iconClassName={sketchButtonClassnames.icon} |         )} | ||||||
|                   size="md" |         {state.matches('Sketch.SketchIdle') && | ||||||
|                 /> |           state.nextEvents | ||||||
|                 {sketchFnLabels[sketchFnName]} |             .filter( | ||||||
|               </button> |               (eventName) => | ||||||
|  |                 eventName.includes('Make segment') || | ||||||
|  |                 eventName.includes('Constrain') | ||||||
|             ) |             ) | ||||||
|           })} |             .sort((a, b) => { | ||||||
|         <HorzVert horOrVert="horizontal" /> |               const aisEnabled = state.nextEvents | ||||||
|         <HorzVert horOrVert="vertical" /> |                 .filter((event) => state.can(event as any)) | ||||||
|         <EqualLength /> |                 .includes(a) | ||||||
|         <EqualAngle /> |               const bIsEnabled = state.nextEvents | ||||||
|         <SetHorzVertDistance buttonType="alignEndsVertically" /> |                 .filter((event) => state.can(event as any)) | ||||||
|         <SetHorzVertDistance buttonType="setHorzDistance" /> |                 .includes(b) | ||||||
|         <SetAbsDistance buttonType="snapToYAxis" /> |               if (aisEnabled && !bIsEnabled) { | ||||||
|         <SetAbsDistance buttonType="xAbs" /> |                 return -1 | ||||||
|         <SetHorzVertDistance buttonType="alignEndsHorizontally" /> |               } | ||||||
|         <SetAbsDistance buttonType="snapToXAxis" /> |               if (!aisEnabled && bIsEnabled) { | ||||||
|         <SetHorzVertDistance buttonType="setVertDistance" /> |                 return 1 | ||||||
|         <SetAbsDistance buttonType="yAbs" /> |               } | ||||||
|         <SetAngleLength angleOrLength="setAngle" /> |               return 0 | ||||||
|         <SetAngleLength angleOrLength="setLength" /> |             }) | ||||||
|         <Intersect /> |             .map((eventName) => ( | ||||||
|         <RemoveConstrainingValues /> |               <li className="contents" key={eventName}> | ||||||
|         <SetAngleBetween /> |                 <ActionButton | ||||||
|       </span> |                   Element="button" | ||||||
|  |                   className="text-sm" | ||||||
|  |                   key={eventName} | ||||||
|  |                   onClick={() => send(eventName)} | ||||||
|  |                   disabled={ | ||||||
|  |                     !state.nextEvents | ||||||
|  |                       .filter((event) => state.can(event as any)) | ||||||
|  |                       .includes(eventName) | ||||||
|  |                   } | ||||||
|  |                   title={eventName} | ||||||
|  |                   icon={{ | ||||||
|  |                     icon: 'line', | ||||||
|  |                     bgClassName, | ||||||
|  |                   }} | ||||||
|  |                 > | ||||||
|  |                   {eventName | ||||||
|  |                     .replace('Make segment ', '') | ||||||
|  |                     .replace('Constrain ', '')} | ||||||
|  |                 </ActionButton> | ||||||
|  |               </li> | ||||||
|  |             ))} | ||||||
|  |         {state.matches('idle') && ( | ||||||
|  |           <li className="contents"> | ||||||
|  |             <ActionButton | ||||||
|  |               Element="button" | ||||||
|  |               className="text-sm" | ||||||
|  |               onClick={() => | ||||||
|  |                 commandBarSend({ | ||||||
|  |                   type: 'Find and select command', | ||||||
|  |                   data: { name: 'Extrude', ownerMachine: 'modeling' }, | ||||||
|  |                 }) | ||||||
|  |               } | ||||||
|  |               disabled={!state.can('Extrude')} | ||||||
|  |               title={ | ||||||
|  |                 state.can('Extrude') | ||||||
|  |                   ? 'extrude' | ||||||
|  |                   : 'sketches need to be closed, or not already extruded' | ||||||
|  |               } | ||||||
|  |               icon={{ | ||||||
|  |                 icon: 'extrude', | ||||||
|  |                 bgClassName, | ||||||
|  |               }} | ||||||
|  |             > | ||||||
|  |               Extrude | ||||||
|  |             </ActionButton> | ||||||
|  |           </li> | ||||||
|  |         )} | ||||||
|  |       </ul> | ||||||
|     ) |     ) | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <Popover className={styles.toolbarWrapper + ' ' + guiMode.mode}> |     <div className="max-w-full flex items-stretch rounded-l-sm rounded-r-full bg-chalkboard-10 dark:bg-chalkboard-100 relative"> | ||||||
|       <div className={styles.toolbar}> |       <menu className="flex-1 pl-1 pr-2 py-0 overflow-hidden rounded-l-sm whitespace-nowrap bg-chalkboard-10 dark:bg-chalkboard-100 border-solid border border-energy-10 dark:border-chalkboard-90 border-r-0"> | ||||||
|         <span className={styles.toolbarCap + ' ' + styles.label}> |         <ToolbarButtons /> | ||||||
|           {guiMode.mode === 'sketch' ? '2D' : '3D'} |       </menu> | ||||||
|         </span> |       <ActionButton | ||||||
|         <menu className="flex-1 gap-2 py-0.5 overflow-hidden whitespace-nowrap"> |         Element="button" | ||||||
|           <ToolbarButtons /> |         onClick={() => commandBarSend({ type: 'Open' })} | ||||||
|         </menu> |         className="rounded-r-full pr-4 self-stretch border-energy-10 hover:border-energy-10 dark:border-chalkboard-80 bg-energy-10/50 hover:bg-energy-10 dark:bg-chalkboard-80 dark:text-energy-10" | ||||||
|         <Popover.Button |  | ||||||
|           className={styles.toolbarCap + ' ' + styles.popoverToggle} |  | ||||||
|         > |  | ||||||
|           <FontAwesomeIcon icon={faSearch} /> |  | ||||||
|         </Popover.Button> |  | ||||||
|       </div> |  | ||||||
|       <Transition |  | ||||||
|         as={Fragment} |  | ||||||
|         enter="transition ease-out duration-200" |  | ||||||
|         enterFrom="opacity-0" |  | ||||||
|         enterTo="opacity-100" |  | ||||||
|         leave="transition ease-out duration-100" |  | ||||||
|         leaveFrom="opacity-100" |  | ||||||
|         leaveTo="opacity-0" |  | ||||||
|       > |       > | ||||||
|         <Popover.Overlay className="fixed inset-0 bg-chalkboard-110/20 dark:bg-chalkboard-110/50" /> |         {platform === 'darwin' ? '⌘K' : 'Ctrl+/'} | ||||||
|       </Transition> |       </ActionButton> | ||||||
|       <Transition |     </div> | ||||||
|         as={Fragment} |  | ||||||
|         enter="transition ease-out duration-100" |  | ||||||
|         enterFrom="opacity-0 translate-y-1 scale-95" |  | ||||||
|         enterTo="opacity-100 translate-y-0 scale-100" |  | ||||||
|         leave="transition ease-out duration-75" |  | ||||||
|         leaveFrom="opacity-100 translate-y-0" |  | ||||||
|         leaveTo="opacity-0 translate-y-2" |  | ||||||
|       > |  | ||||||
|         <Popover.Panel className="absolute top-0 w-screen max-w-xl left-1/2 -translate-x-1/2 flex flex-col gap-8 bg-chalkboard-10 dark:bg-chalkboard-100 p-5 rounded border border-chalkboard-20/30 dark:border-chalkboard-70/50"> |  | ||||||
|           <section className="flex justify-between items-center"> |  | ||||||
|             <p |  | ||||||
|               className={`${styles.toolbarCap} ${styles.label} !self-center rounded-r-full w-fit`} |  | ||||||
|             > |  | ||||||
|               You're in {guiMode.mode === 'sketch' ? '2D' : '3D'} |  | ||||||
|             </p> |  | ||||||
|             <Popover.Button className="p-2 flex items-center justify-center rounded-sm bg-chalkboard-20 text-chalkboard-110 dark:bg-chalkboard-70 dark:text-chalkboard-20 border-none hover:bg-chalkboard-30 dark:hover:bg-chalkboard-60"> |  | ||||||
|               <FontAwesomeIcon icon={faX} className="w-4 h-4" /> |  | ||||||
|             </Popover.Button> |  | ||||||
|           </section> |  | ||||||
|           <section> |  | ||||||
|             <ToolbarButtons className="flex-wrap" /> |  | ||||||
|           </section> |  | ||||||
|         </Popover.Panel> |  | ||||||
|       </Transition> |  | ||||||
|     </Popover> |  | ||||||
|   ) |   ) | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										252
									
								
								src/clientSideScene/ClientSideSceneComp.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,252 @@ | |||||||
|  | import { useRef, useEffect, useState } from 'react' | ||||||
|  | import { useModelingContext } from 'hooks/useModelingContext' | ||||||
|  |  | ||||||
|  | import { cameraMouseDragGuards } from 'lib/cameraControls' | ||||||
|  | import { useGlobalStateContext } from 'hooks/useGlobalStateContext' | ||||||
|  | import { useStore } from 'useStore' | ||||||
|  | import { | ||||||
|  |   DEBUG_SHOW_BOTH_SCENES, | ||||||
|  |   ReactCameraProperties, | ||||||
|  |   sceneInfra, | ||||||
|  | } from './sceneInfra' | ||||||
|  | import { throttle } from 'lib/utils' | ||||||
|  |  | ||||||
|  | function useShouldHideScene(): { hideClient: boolean; hideServer: boolean } { | ||||||
|  |   const [isCamMoving, setIsCamMoving] = useState(false) | ||||||
|  |   const [isTween, setIsTween] = useState(false) | ||||||
|  |  | ||||||
|  |   const { state } = useModelingContext() | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     sceneInfra.setIsCamMovingCallback((isMoving, isTween) => { | ||||||
|  |       setIsCamMoving(isMoving) | ||||||
|  |       setIsTween(isTween) | ||||||
|  |     }) | ||||||
|  |   }, []) | ||||||
|  |  | ||||||
|  |   if (DEBUG_SHOW_BOTH_SCENES || !isCamMoving) | ||||||
|  |     return { hideClient: false, hideServer: false } | ||||||
|  |   let hideServer = state.matches('Sketch') | ||||||
|  |   if (isTween) { | ||||||
|  |     hideServer = false | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return { hideClient: !hideServer, hideServer } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const ClientSideScene = ({ | ||||||
|  |   cameraControls, | ||||||
|  | }: { | ||||||
|  |   cameraControls: ReturnType< | ||||||
|  |     typeof useGlobalStateContext | ||||||
|  |   >['settings']['context']['cameraControls'] | ||||||
|  | }) => { | ||||||
|  |   const canvasRef = useRef<HTMLDivElement>(null) | ||||||
|  |   const { state, send } = useModelingContext() | ||||||
|  |   const { hideClient, hideServer } = useShouldHideScene() | ||||||
|  |   const { setHighlightRange } = useStore((s) => ({ | ||||||
|  |     setHighlightRange: s.setHighlightRange, | ||||||
|  |     highlightRange: s.highlightRange, | ||||||
|  |   })) | ||||||
|  |  | ||||||
|  |   // Listen for changes to the camera controls setting | ||||||
|  |   // and update the client-side scene's controls accordingly. | ||||||
|  |   useEffect(() => { | ||||||
|  |     sceneInfra.setInteractionGuards(cameraMouseDragGuards[cameraControls]) | ||||||
|  |   }, [cameraControls]) | ||||||
|  |   useEffect(() => { | ||||||
|  |     sceneInfra.updateOtherSelectionColors( | ||||||
|  |       state?.context?.selectionRanges?.otherSelections || [] | ||||||
|  |     ) | ||||||
|  |   }, [state?.context?.selectionRanges?.otherSelections]) | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     if (!canvasRef.current) return | ||||||
|  |     const canvas = canvasRef.current | ||||||
|  |     canvas.appendChild(sceneInfra.renderer.domElement) | ||||||
|  |     sceneInfra.animate() | ||||||
|  |     sceneInfra.setHighlightCallback(setHighlightRange) | ||||||
|  |     canvas.addEventListener('mousemove', sceneInfra.onMouseMove, false) | ||||||
|  |     canvas.addEventListener('mousedown', sceneInfra.onMouseDown, false) | ||||||
|  |     canvas.addEventListener('mouseup', sceneInfra.onMouseUp, false) | ||||||
|  |     sceneInfra.setSend(send) | ||||||
|  |     return () => { | ||||||
|  |       canvas?.removeEventListener('mousemove', sceneInfra.onMouseMove) | ||||||
|  |       canvas?.removeEventListener('mousedown', sceneInfra.onMouseDown) | ||||||
|  |       canvas?.removeEventListener('mouseup', sceneInfra.onMouseUp) | ||||||
|  |     } | ||||||
|  |   }, []) | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <div | ||||||
|  |       ref={canvasRef} | ||||||
|  |       className={`absolute inset-0 h-full w-full transition-all duration-300 ${ | ||||||
|  |         hideClient ? 'opacity-0' : 'opacity-100' | ||||||
|  |       } ${hideServer ? 'bg-black' : ''} ${ | ||||||
|  |         !hideClient && !hideServer && state.matches('Sketch') | ||||||
|  |           ? 'bg-black/80' | ||||||
|  |           : '' | ||||||
|  |       }`} | ||||||
|  |     ></div> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const throttled = throttle((a: ReactCameraProperties) => { | ||||||
|  |   if (a.type === 'perspective' && a.fov) { | ||||||
|  |     sceneInfra.dollyZoom(a.fov) | ||||||
|  |   } | ||||||
|  | }, 1000 / 15) | ||||||
|  |  | ||||||
|  | export const CamDebugSettings = () => { | ||||||
|  |   const [camSettings, setCamSettings] = useState<ReactCameraProperties>({ | ||||||
|  |     type: 'perspective', | ||||||
|  |     fov: 12, | ||||||
|  |     position: [0, 0, 0], | ||||||
|  |     quaternion: [0, 0, 0, 1], | ||||||
|  |   }) | ||||||
|  |   const [fov, setFov] = useState(12) | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     sceneInfra.setReactCameraPropertiesCallback(setCamSettings) | ||||||
|  |   }, [sceneInfra]) | ||||||
|  |   useEffect(() => { | ||||||
|  |     if (camSettings.type === 'perspective' && camSettings.fov) { | ||||||
|  |       setFov(camSettings.fov) | ||||||
|  |     } | ||||||
|  |   }, [(camSettings as any)?.fov]) | ||||||
|  |  | ||||||
|  |   return ( | ||||||
|  |     <div> | ||||||
|  |       <h3>cam settings</h3> | ||||||
|  |       perspective cam | ||||||
|  |       <input | ||||||
|  |         type="checkbox" | ||||||
|  |         checked={camSettings.type === 'perspective'} | ||||||
|  |         onChange={(e) => { | ||||||
|  |           if (camSettings.type === 'perspective') { | ||||||
|  |             sceneInfra.useOrthographicCamera() | ||||||
|  |           } else { | ||||||
|  |             sceneInfra.usePerspectiveCamera() | ||||||
|  |           } | ||||||
|  |         }} | ||||||
|  |       /> | ||||||
|  |       {camSettings.type === 'perspective' && ( | ||||||
|  |         <input | ||||||
|  |           type="range" | ||||||
|  |           min="4" | ||||||
|  |           max="90" | ||||||
|  |           step={0.5} | ||||||
|  |           value={fov} | ||||||
|  |           onChange={(e) => { | ||||||
|  |             setFov(parseFloat(e.target.value)) | ||||||
|  |  | ||||||
|  |             throttled({ | ||||||
|  |               ...camSettings, | ||||||
|  |               fov: parseFloat(e.target.value), | ||||||
|  |             }) | ||||||
|  |           }} | ||||||
|  |           className="w-full cursor-pointer pointer-events-auto" | ||||||
|  |         /> | ||||||
|  |       )} | ||||||
|  |       {camSettings.type === 'perspective' && ( | ||||||
|  |         <div> | ||||||
|  |           <span>fov</span> | ||||||
|  |           <input | ||||||
|  |             type="number" | ||||||
|  |             value={camSettings.fov} | ||||||
|  |             className="text-black w-16" | ||||||
|  |             onChange={(e) => { | ||||||
|  |               sceneInfra.setCam({ | ||||||
|  |                 ...camSettings, | ||||||
|  |                 fov: parseFloat(e.target.value), | ||||||
|  |               }) | ||||||
|  |             }} | ||||||
|  |           /> | ||||||
|  |         </div> | ||||||
|  |       )} | ||||||
|  |       {camSettings.type === 'orthographic' && ( | ||||||
|  |         <> | ||||||
|  |           <div> | ||||||
|  |             <span>fov</span> | ||||||
|  |             <input | ||||||
|  |               type="number" | ||||||
|  |               value={camSettings.zoom} | ||||||
|  |               className="text-black w-16" | ||||||
|  |               onChange={(e) => { | ||||||
|  |                 sceneInfra.setCam({ | ||||||
|  |                   ...camSettings, | ||||||
|  |                   zoom: parseFloat(e.target.value), | ||||||
|  |                 }) | ||||||
|  |               }} | ||||||
|  |             /> | ||||||
|  |           </div> | ||||||
|  |         </> | ||||||
|  |       )} | ||||||
|  |       <div> | ||||||
|  |         Position | ||||||
|  |         <ul className="flex"> | ||||||
|  |           <li> | ||||||
|  |             <span className="pl-2 pr-1">x:</span> | ||||||
|  |             <input | ||||||
|  |               type="number" | ||||||
|  |               step={5} | ||||||
|  |               data-testid="cam-x-position" | ||||||
|  |               value={camSettings.position[0]} | ||||||
|  |               className="text-black w-16" | ||||||
|  |               onChange={(e) => { | ||||||
|  |                 sceneInfra.setCam({ | ||||||
|  |                   ...camSettings, | ||||||
|  |                   position: [ | ||||||
|  |                     parseFloat(e.target.value), | ||||||
|  |                     camSettings.position[1], | ||||||
|  |                     camSettings.position[2], | ||||||
|  |                   ], | ||||||
|  |                 }) | ||||||
|  |               }} | ||||||
|  |             /> | ||||||
|  |           </li> | ||||||
|  |           <li> | ||||||
|  |             <span className="pl-2 pr-1">y:</span> | ||||||
|  |             <input | ||||||
|  |               type="number" | ||||||
|  |               step={5} | ||||||
|  |               data-testid="cam-y-position" | ||||||
|  |               value={camSettings.position[1]} | ||||||
|  |               className="text-black w-16" | ||||||
|  |               onChange={(e) => { | ||||||
|  |                 sceneInfra.setCam({ | ||||||
|  |                   ...camSettings, | ||||||
|  |                   position: [ | ||||||
|  |                     camSettings.position[0], | ||||||
|  |                     parseFloat(e.target.value), | ||||||
|  |                     camSettings.position[2], | ||||||
|  |                   ], | ||||||
|  |                 }) | ||||||
|  |               }} | ||||||
|  |             /> | ||||||
|  |           </li> | ||||||
|  |           <li> | ||||||
|  |             <span className="pl-2 pr-1">z:</span> | ||||||
|  |             <input | ||||||
|  |               type="number" | ||||||
|  |               step={5} | ||||||
|  |               data-testid="cam-z-position" | ||||||
|  |               value={camSettings.position[2]} | ||||||
|  |               className="text-black w-16" | ||||||
|  |               onChange={(e) => { | ||||||
|  |                 sceneInfra.setCam({ | ||||||
|  |                   ...camSettings, | ||||||
|  |                   position: [ | ||||||
|  |                     camSettings.position[0], | ||||||
|  |                     camSettings.position[1], | ||||||
|  |                     parseFloat(e.target.value), | ||||||
|  |                   ], | ||||||
|  |                 }) | ||||||
|  |               }} | ||||||
|  |             /> | ||||||
|  |           </li> | ||||||
|  |         </ul> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   ) | ||||||
|  | } | ||||||
							
								
								
									
										33
									
								
								src/clientSideScene/helpers.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,33 @@ | |||||||
|  | import { | ||||||
|  |   GridHelper, | ||||||
|  |   LineBasicMaterial, | ||||||
|  |   OrthographicCamera, | ||||||
|  |   PerspectiveCamera, | ||||||
|  |   Group, | ||||||
|  |   Mesh, | ||||||
|  | } from 'three' | ||||||
|  |  | ||||||
|  | export function createGridHelper({ | ||||||
|  |   size, | ||||||
|  |   divisions, | ||||||
|  | }: { | ||||||
|  |   size: number | ||||||
|  |   divisions: number | ||||||
|  | }) { | ||||||
|  |   const gridHelperMaterial = new LineBasicMaterial({ | ||||||
|  |     color: 0xaaaaaa, | ||||||
|  |     transparent: true, | ||||||
|  |     opacity: 0.5, | ||||||
|  |     depthTest: false, | ||||||
|  |   }) | ||||||
|  |   const gridHelper = new GridHelper(size, divisions, 0x0000ff, 0xffffff) | ||||||
|  |   gridHelper.material = gridHelperMaterial | ||||||
|  |   gridHelper.rotation.x = Math.PI / 2 | ||||||
|  |   return gridHelper | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const orthoScale = (cam: OrthographicCamera | PerspectiveCamera) => | ||||||
|  |   0.55 / cam.zoom | ||||||
|  |  | ||||||
|  | export const perspScale = (cam: PerspectiveCamera, group: Group | Mesh) => | ||||||
|  |   (group.position.distanceTo(cam.position) * cam.fov) / 4000 | ||||||
							
								
								
									
										1032
									
								
								src/clientSideScene/sceneEntities.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										28
									
								
								src/clientSideScene/sceneInfra.test.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,28 @@ | |||||||
|  | import { Quaternion } from 'three' | ||||||
|  | import { isQuaternionVertical } from './sceneInfra' | ||||||
|  |  | ||||||
|  | describe('isQuaternionVertical', () => { | ||||||
|  |   it('should identify vertical quaternions', () => { | ||||||
|  |     const verticalQuaternions = [ | ||||||
|  |       new Quaternion(1, 0, 0, 0).normalize(), // bottom | ||||||
|  |       new Quaternion(-0.7, 0.7, 0, 0).normalize(), // bottom 2 | ||||||
|  |       new Quaternion(0, 1, 0, 0).normalize(), // bottom 3 | ||||||
|  |       new Quaternion(0, 0, 0, 1).normalize(), // look from top | ||||||
|  |     ] | ||||||
|  |     verticalQuaternions.forEach((quaternion) => { | ||||||
|  |       expect(isQuaternionVertical(quaternion)).toBe(true) | ||||||
|  |     }) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   it('should identify non-vertical quaternions', () => { | ||||||
|  |     const nonVerticalQuaternions = [ | ||||||
|  |       new Quaternion(0.7, 0, 0, 0.7).normalize(), // front | ||||||
|  |       new Quaternion(0, 0.7, 0.7, 0).normalize(), // back | ||||||
|  |       new Quaternion(-0.5, 0.5, 0.5, -0.5).normalize(), // left side | ||||||
|  |       new Quaternion(0.5, 0.5, 0.5, 0.5).normalize(), // right side | ||||||
|  |     ] | ||||||
|  |     nonVerticalQuaternions.forEach((quaternion) => { | ||||||
|  |       expect(isQuaternionVertical(quaternion)).toBe(false) | ||||||
|  |     }) | ||||||
|  |   }) | ||||||
|  | }) | ||||||
