Compare commits
	
		
			25 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| f57b016b46 | |||
| 09907dbbf7 | |||
| e65f0a1670 | |||
| 941db9436f | |||
| 0874507025 | |||
| 2578ea4dc1 | |||
| d80e3e4658 | |||
| 188501423a | |||
| 668da28d2c | |||
| 8a5b402388 | |||
| ebb3bd9fb3 | |||
| d2cbe71731 | |||
| 8197e3c369 | |||
| 4b38dc0a1d | |||
| 0ce629ea49 | |||
| a77d19fb04 | |||
| d9604d606a | |||
| 57eb9e76a4 | |||
| b7dfc71335 | |||
| 38e01dad24 | |||
| 8576d25c2c | |||
| e54f998d29 | |||
| 7eee4fab7e | |||
| 9013229c20 | |||
| da9a69b916 | 
| @@ -1,6 +1,6 @@ | ||||
| name: Release Creation | ||||
|  | ||||
| on:  | ||||
| on: | ||||
|   release: | ||||
|     types: [published] | ||||
|  | ||||
| @@ -8,45 +8,45 @@ jobs: | ||||
|   build: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - run: echo "💡 The ${{ gitea.repository }} repository will cloned to the runner." | ||||
|      | ||||
|     #- uses: actions/checkout@v3 | ||||
|     - uses: RouxAntoine/checkout@v3.5.4 | ||||
|       - run: echo "💡 The ${{ gitea.repository }} repository will cloned to the runner." | ||||
|  | ||||
|     # get part of the tag after the `v` | ||||
|     - name: Extract tag version number | ||||
|       id: get_version | ||||
|       uses: battila7/get-version-action@v2 | ||||
|       #- uses: actions/checkout@v3 | ||||
|       - uses: RouxAntoine/checkout@v3.5.4 | ||||
|  | ||||
|     # Substitute the Manifest and Download URLs in the module.json | ||||
|     - name: Substitute Manifest and Download Links For Versioned Ones | ||||
|       id: sub_manifest_link_version | ||||
|       uses: microsoft/variable-substitution@v1 | ||||
|       with: | ||||
|         files: 'system.json' | ||||
|       env: | ||||
|         version: ${{steps.get_version.outputs.version-without-v}} | ||||
|         url: https://www.uberwald.me/gitea/${{gitea.repository}} | ||||
|         manifest: https://www.uberwald.me/gitea/uberwald/${{gitea.repository}}/releases/download/latest/system.json | ||||
|         download: https://www.uberwald.me/gitea/${{gitea.repository}}/releases/download/${{github.event.release.tag_name}}/fvtt-ftl-nomad-${{github.event.release.tag_name}}.zip | ||||
|      | ||||
|     # Create a zip file with all files required by the module to add to the release | ||||
|     - run: | | ||||
|          apt update -y | ||||
|          apt install -y zip | ||||
|       # get part of the tag after the `v` | ||||
|       - name: Extract tag version number | ||||
|         id: get_version | ||||
|         uses: battila7/get-version-action@v2 | ||||
|  | ||||
|     - run: zip -r ./fvtt-ftl-nomad-${{github.event.release.tag_name}}.zip system.json README.md LICENSE assets/ css/ lang/ module/ packs-system/ templates/ fvtt-ftl-nomad.mjs | ||||
|      | ||||
|     - name: setup go | ||||
|       uses: https://github.com/actions/setup-go@v4 | ||||
|       with: | ||||
|         go-version: '>=1.20.1' | ||||
|      | ||||
|     - name: Use Go Action   | ||||
|       id: use-go-action | ||||
|       uses: https://gitea.com/actions/release-action@main | ||||
|       with: | ||||
|         files: |- | ||||
|           ./fvtt-ftl-nomad-${{github.event.release.tag_name}}.zip | ||||
|           system.json | ||||
|         api_key: '${{secrets.ALLOW_PUSH_RELEASE}}' | ||||
|       # Substitute the Manifest and Download URLs in the module.json | ||||
|       - name: Substitute Manifest and Download Links For Versioned Ones | ||||
|         id: sub_manifest_link_version | ||||
|         uses: microsoft/variable-substitution@v1 | ||||
|         with: | ||||
|           files: "system.json" | ||||
|         env: | ||||
|           version: ${{steps.get_version.outputs.version-without-v}} | ||||
|           url: https://www.uberwald.me/gitea/${{gitea.repository}} | ||||
|           manifest: https://www.uberwald.me/gitea/${{gitea.repository}}/releases/download/latest/system.json | ||||
|           download: https://www.uberwald.me/gitea/${{gitea.repository}}/releases/download/${{github.event.release.tag_name}}/fvtt-ftl-nomad-${{github.event.release.tag_name}}.zip | ||||
|  | ||||
|       # Create a zip file with all files required by the module to add to the release | ||||
|       - run: | | ||||
|           apt update -y | ||||
|           apt install -y zip | ||||
|  | ||||
|       - run: zip -r ./fvtt-ftl-nomad-${{github.event.release.tag_name}}.zip system.json README.md LICENSE assets/ css/ lang/ module/ packs/ packs-system/ templates/ fvtt-ftl-nomad.mjs | ||||
|  | ||||
|       - name: setup go | ||||
|         uses: https://github.com/actions/setup-go@v4 | ||||
|         with: | ||||
|           go-version: ">=1.20.1" | ||||
|  | ||||
|       - name: Use Go Action | ||||
|         id: use-go-action | ||||
|         uses: https://gitea.com/actions/release-action@main | ||||
|         with: | ||||
|           files: |- | ||||
|             ./fvtt-ftl-nomad-${{github.event.release.tag_name}}.zip | ||||
|             system.json | ||||
|           api_key: "${{secrets.ALLOW_PUSH_RELEASE}}" | ||||
|   | ||||
							
								
								
									
										15
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -1,3 +1,14 @@ | ||||
| # fvtt-ftl-nomad | ||||
| Faster Than Light: Nomad (FTL: Nomad) for Foundry Virtual TableTop | ||||
|  | ||||
| Faster Than Light : Nomad system for FoundryVTT | ||||
| The official game system for playing Faster Than Light: Nomad on FoundryVTT. This framework covers all the fundemental aspects of Faster Than Light: Nomad and permits full FTL:N online gameplay. | ||||
|  | ||||
| This product's format, computer code, graphics, and presentation are copyrighted by Stellagama Publishing. | ||||
|  | ||||
| This system is used with permission granted as part of the partnership agreement between Foundry Gaming LLC and Stellagama Publishing. It uses the following copyright: | ||||
|  | ||||
| Faster Than Light: Nomad, © 2025 Stellagama Publishing. All rights reserved. | ||||
|  | ||||
| Stellagama Publishing's web page is available at https://stellagamapublishing.co/ and the company's owner, Omer Golan-Joel, may be contacted at golan2072@gmail.com. | ||||
|  | ||||
| Community | ||||
| You are invited to join Stellagama Publishing's Discord server: https://discord.gg/MHpMDBpcZ6 | ||||
|   | ||||
							
								
								
									
										1
									
								
								assets/icons/icon_combat.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| <svg style="height: 512px; width: 512px;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><defs><filter id="shadow-1" height="300%" width="300%" x="-100%" y="-100%"><feFlood flood-color="rgba(255, 255, 255, 1)" result="flood"></feFlood><feComposite in="flood" in2="SourceGraphic" operator="atop" result="composite"></feComposite><feGaussianBlur in="composite" stdDeviation="15" result="blur"></feGaussianBlur><feOffset dx="0" dy="0" result="offset"></feOffset><feComposite in="SourceGraphic" in2="offset" operator="over"></feComposite></filter></defs><g class="" style="" transform="translate(0,0)"><path d="M221.6 28.53c-5.2.1-10.6 2.18-15.8 6.35-7.7 6.03-14.4 16.44-17.8 29.58-3.3 13.14-2.5 26.2 1.1 36.24 3.6 9.9 9.4 16.4 16.6 18.9 6 2.1 12.6 1.3 19-2.6 2.3-13.6 9.9-24.28 21.3-31.6.1-.19.1-.38.2-.58 3.3-13.15 2.5-26.2-1.1-36.2-3.6-9.95-9.4-16.46-16.6-18.97-2.3-.78-4.6-1.16-6.9-1.12zm212.5 49.31c-3.2.72-11 3.44-18.7 7.15-8.7 4.14-17.4 8.86-22.4 11.5l-.9.45-.9.25c-38.1 10.61-57.5 17.91-88.3 23.21-2.2 6.1-6 12.5-11.3 19h-.1v.1c-6.2 7.4-12.4 14.7-18.7 21.9 45.8-8.5 78.8-16.7 132.7-38l4.1-1.6 3.9 2.3c5.5 3.4 13.4 5.8 19.6 6.3 3.1.3 5.8 0 7.4-.4 1.4-.4 1.4-.7 1.4-.6 5.4-11.9 7.3-24.4 5.5-33.8-1.7-8.89-5.7-14.64-13.3-17.76zM272.2 96.08c-8.9-.1-20.1 4.32-27.6 15.62-5.9 29.5-26.7 50.1-57.2 62.6l-6.8-16.6c12.7-5.2 22.6-11.2 30.1-18.6-3.1-.2-6.3-.9-9.4-2-10.4-3.7-18.6-11.6-24.1-21.4-14.6 2.8-22.6 6.3-32.8 14.6-4.1 36.5 6.8 67.5 27.6 96.6 41.7-25.4 73.7-60.5 105.7-98.9 8.2-10.2 10-17.6 9.5-21.9-.6-4.2-2.7-6.73-7.2-8.58-2.2-.91-4.9-1.42-7.8-1.44zM131.5 185.5c-1.7 23.2-.7 48.5-1 75v3.8l-2.7 2.6C75.51 317.5 33.9 400.5 20.36 495.7h40.62c9.12-41.2 17.47-82.3 34.69-115.7 18.23-35.3 47.83-62 96.23-67.5l4.8-.6 3.1 3.7c38.2 45.7 75.3 103.9 85.9 180.1h39.6c-6.3-88.4-46.8-156.5-97.1-229-9.8-14.1-10.2-29.8-6.5-44.9.9-3.8 2.1-7.7 3.4-11.7-15.5 13.6-32.3 26-51 36.7l-6.9 3.9-4.7-6.2c-14-18.2-24.8-37.8-31-59z" fill="#21d121" fill-opacity="1" stroke="#0f0101" stroke-opacity="1" stroke-width="8" filter="url(#shadow-1)"></path></g></svg> | ||||
| After Width: | Height: | Size: 2.0 KiB | 
							
								
								
									
										1
									
								
								assets/icons/icon_knowledge.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| <svg style="height: 512px; width: 512px;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><defs><filter id="shadow-1" height="300%" width="300%" x="-100%" y="-100%"><feFlood flood-color="rgba(255, 255, 255, 1)" result="flood"></feFlood><feComposite in="flood" in2="SourceGraphic" operator="atop" result="composite"></feComposite><feGaussianBlur in="composite" stdDeviation="15" result="blur"></feGaussianBlur><feOffset dx="0" dy="0" result="offset"></feOffset><feComposite in="SourceGraphic" in2="offset" operator="over"></feComposite></filter></defs><g class="" style="" transform="translate(0,0)"><path d="M104 37.25V215.1c3.6 9.9 10 14.1 20.9 17 11.4 2.9 27.1 3.1 43.9 3.1 16.8 0 34.6-.1 51.4 3.6 9.5 2.1 18.7 5.7 26.8 11.2V55.43c-7.2-9.9-15.9-13.69-27.3-15.09-12.2-1.49-27.3.55-42.9 2.83-15.7 2.28-31.9 4.81-47.7 2.88-8.7-1.07-17.3-3.87-25.1-8.8zm304 0c-7.8 4.93-16.4 7.73-25.1 8.8-15.8 1.93-32-.6-47.7-2.88-15.6-2.28-30.7-4.32-42.9-2.83-11.4 1.4-20.1 5.19-27.3 15.09V250c8.1-5.5 17.3-9.1 26.8-11.2 16.8-3.7 34.6-3.6 51.4-3.6 16.8 0 32.5-.2 43.9-3.1 10.9-2.9 17.3-7.1 20.9-17V37.25zM130.8 80.03h89.4v18h-89.4v-18zm161 0h89.4v18h-89.4v-18zm-161 44.47h89.4v18h-89.4v-18zm161 0h89.4v18h-89.4v-18zm-161 46.8h89.4v18h-89.4v-18zm161 0h89.4v18h-89.4v-18zM96 249c-17.3 0-29.19 7.3-37.77 18.9C49.66 279.4 45 295.7 45 312s4.66 32.6 13.23 44.1C66.81 367.7 78.7 375 96 375c22 0 35.7-7.1 44.4-14 8.6-6.9 11.5-13 11.5-13l2.5-5h203.2l2.5 5s2.9 6.1 11.5 13c8.7 6.9 22.4 14 44.4 14 17.3 0 29.2-7.3 37.8-18.9 8.5-11.5 13.2-27.8 13.2-44.1s-4.7-32.6-13.2-44.1c-8.6-11.6-20.5-18.9-37.8-18.9-22 0-35.7 7.1-44.4 14-8.6 6.9-11.5 13-11.5 13l-2.5 5H154.4l-2.5-5s-2.9-6.1-11.5-13c-8.7-6.9-22.4-14-44.4-14zm4.6 21.9c.9 0 1.9 0 2.9.1 13.5 1.2 28.2 8.9 44.1 24h216.8c15.9-15.1 30.6-22.8 44.1-24 14.2-1.2 26.6 5.8 33.1 16.2 13.1 20.9 7 53.9-20.6 72.3l-10-15c20.4-13.6 22.3-36.6 15.4-47.7-3.5-5.6-8.1-8.6-16.4-7.8-8.2.7-20.6 6.3-35.6 21.4l-2.7 2.6H140.3l-2.7-2.6c-15-15.1-27.4-20.7-35.6-21.4-8.26-.8-12.9 2.2-16.37 7.8-6.92 11.1-4.99 34.1 15.37 47.7l-9.99 15c-27.65-18.4-33.72-51.4-20.64-72.3 5.72-9.1 15.9-15.6 27.92-16.3h2.31zM169 361v126h30V361h-30zm48 0v126h30V361h-30zm48 0v126h30V361h-30zm48 0v126h30V361h-30z" fill="#21d121" fill-opacity="1" stroke="#0f0101" stroke-opacity="1" stroke-width="8" filter="url(#shadow-1)"></path></g></svg> | ||||
| After Width: | Height: | Size: 2.3 KiB | 
							
								
								
									
										1
									
								
								assets/icons/icon_physical.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| <svg style="height: 512px; width: 512px;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><defs><filter id="shadow-1" height="300%" width="300%" x="-100%" y="-100%"><feFlood flood-color="rgba(255, 255, 255, 1)" result="flood"></feFlood><feComposite in="flood" in2="SourceGraphic" operator="atop" result="composite"></feComposite><feGaussianBlur in="composite" stdDeviation="15" result="blur"></feGaussianBlur><feOffset dx="0" dy="0" result="offset"></feOffset><feComposite in="SourceGraphic" in2="offset" operator="over"></feComposite></filter></defs><g class="" style="" transform="translate(0,0)"><path d="M165.906 18.688C15.593 59.28-42.187 198.55 92.72 245.375h-1.095c.635.086 1.274.186 1.906.28 8.985 3.077 18.83 5.733 29.532 7.94C173.36 273.35 209.74 321.22 212.69 368c-33.514 23.096-59.47 62.844-59.47 62.844L179.5 469.53 138.28 493h81.97c-40.425-40.435-11.76-85.906 36.125-85.906 48.54 0 73.945 48.112 36.156 85.906h81.126l-40.375-23.47 26.283-38.686s-26.376-40.4-60.282-63.406c3.204-46.602 39.5-94.167 89.595-113.844 10.706-2.207 20.546-4.86 29.53-7.938.633-.095 1.273-.195 1.908-.28h-1.125c134.927-46.82 77.163-186.094-73.157-226.69-40.722 39.37 6.54 101.683 43.626 56.877 36.9 69.08 8.603 127.587-72.28 83.406-11.88 24.492-34.213 41.374-60.688 41.374-26.703 0-49.168-17.167-60.97-42-81.774 45.38-110.512-13.372-73.437-82.78 37.09 44.805 84.35-17.508 43.626-56.876zm90.79 35.92c-27.388 0-51.33 27.556-51.33 63.61 0 36.056 23.942 62.995 51.33 62.995 27.387 0 51.327-26.94 51.327-62.994 0-36.058-23.94-63.61-51.328-63.61z" fill="#21d121" fill-opacity="1" stroke="#0f0101" stroke-opacity="1" stroke-width="8" filter="url(#shadow-1)"></path></g></svg> | ||||
| After Width: | Height: | Size: 1.6 KiB | 
							
								
								
									
										1
									
								
								assets/icons/icon_robot.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| <svg style="height: 512px; width: 512px;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><defs><filter id="shadow-1" height="300%" width="300%" x="-100%" y="-100%"><feFlood flood-color="rgba(255, 255, 255, 1)" result="flood"></feFlood><feComposite in="flood" in2="SourceGraphic" operator="atop" result="composite"></feComposite><feGaussianBlur in="composite" stdDeviation="15" result="blur"></feGaussianBlur><feOffset dx="0" dy="0" result="offset"></feOffset><feComposite in="SourceGraphic" in2="offset" operator="over"></feComposite></filter></defs><g class="" style="" transform="translate(0,0)"><path d="M265.344 18.563v302.863h-18.69V18.59c-61.403 3.005-115.44 34.412-149.238 81.336l35.022 60.658a46.633 46.633 0 0 1 12.11-1.598c25.747 0 46.62 20.872 46.62 46.62 0 22.556-16.02 41.37-37.3 45.687v174.084c13.712 21.43 27.6 41.98 40.712 60.754V369.01h122.838v117.117c13.11-18.773 27-39.324 40.713-60.754V251.294c-21.282-4.317-37.302-23.13-37.302-45.688 0-25.747 20.873-46.62 46.62-46.62 4.19 0 8.25.562 12.113 1.6l35.13-60.847c-33.74-47-87.758-78.375-149.346-81.175zM426.03 117.47l-29.74 51.52c10.824 8.537 17.782 21.76 17.782 36.615 0 22.54-15.995 41.343-37.254 45.68V395.38c9.833-16.237 19.31-32.676 28.014-48.927 27.3-50.976 46.564-100.37 46.564-132.36 0-35.127-9.22-68.104-25.365-96.622zm-339.973.157c-16.196 28.486-25.46 61.408-25.46 96.465 0 31.992 19.267 81.385 46.567 132.36 8.704 16.253 18.182 32.694 28.016 48.933v-144.1c-21.26-4.336-37.254-23.14-37.254-45.68 0-14.856 6.957-28.08 17.783-36.617l-29.653-51.36zm127.213 270.07V488.98h85.46V387.696h-85.46z" fill="#7ed321" fill-opacity="1" filter="url(#shadow-1)" stroke="#ffffff" stroke-opacity="1" stroke-width="8"></path></g></svg> | ||||
| After Width: | Height: | Size: 1.7 KiB | 
							
								
								
									
										1
									
								
								assets/icons/icon_social.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| <svg style="height: 512px; width: 512px;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><defs><filter id="shadow-1" height="300%" width="300%" x="-100%" y="-100%"><feFlood flood-color="rgba(255, 255, 255, 1)" result="flood"></feFlood><feComposite in="flood" in2="SourceGraphic" operator="atop" result="composite"></feComposite><feGaussianBlur in="composite" stdDeviation="15" result="blur"></feGaussianBlur><feOffset dx="0" dy="0" result="offset"></feOffset><feComposite in="SourceGraphic" in2="offset" operator="over"></feComposite></filter></defs><g class="" style="" transform="translate(0,0)"><path d="M494 61.363l-82.58 77.934 78.994 132.96 3.586-4.458V61.362zM18 62.5v225.893c4.48.582 9.863.903 15.295.96 11.87.125 21.654-.65 27.15-1.144L113.1 154.974 18 62.5zm389.154 104.86l-7.04 4.556c-.15.097-5.362 3.336-6.893 4.29l-10.605 6.42.15.09c-4.914 3.057-6.28 3.917-11.857 7.38-2.83 1.757-2.9 1.798-5.584 3.465-20.29-10.907-42.306-19.29-67.998-25.882-32.312 9.762-66.542 23.888-100.722 37.142 14.19 17.087 29.96 22.651 45.845 22.85 18.42.23 37.25-7.78 50.218-16.754l7.4-5.12 7.426 10.73 115.453 83.33 45.112-29.987-60.906-102.51zM126.477 170.1L81.11 284.887 97.76 297.69l30.795-34.905 2.467-2.795 3.72-.232c1.5-.094 2.98-.138 4.44-.13 10.212.066 19.342 2.716 26.19 8.76 5.072 4.472 8.444 10.426 10.4 17.32l2.28-.142c11.995-.75 22.802 1.725 30.63 8.63 7.827 6.907 11.63 17.323 12.38 29.32l.07 1.08c6.44 1.216 12.205 3.752 16.893 7.888 7.828 6.906 11.63 17.32 12.38 29.317l.197 3.12c.642.202 1.275.424 1.9.658l2.033-2.853 5.47-7.678 2.813-3.95 7.33 5.223 59.428 42.336c6.464-1.594 10.317-4.075 12.46-7.086 2.147-3.012 3.233-7.47 2.624-14.107l-71.258-51.03-7.318-5.24 5.19-7.246 6.67-9.365 7.33 5.223 80.335 57.226c6.464-1.593 10.32-4.074 12.463-7.085 2.144-3.01 3.23-7.457 2.625-14.082l-92.398-65.55-7.34-5.21 10.414-14.68 7.343 5.208 92.414 65.565c6.47-1.594 10.327-4.075 12.473-7.088 2.148-3.015 3.233-7.476 2.62-14.125l-110.44-79.71c-14.655 8.688-33.402 15.648-53.557 15.396-23.587-.295-48.817-11.566-67.377-40.05a9 9 0 0 1 4.343-13.327c13.014-4.945 26.163-10.17 39.343-15.354l-92.056-6.834zm12.902 107.62l-47.564 53.91c.927 6.746 3.04 10.942 5.887 13.454 2.847 2.512 7.275 4.085 14.084 4.164l47.563-53.908c-.927-6.747-3.04-10.945-5.887-13.457-2.847-2.512-7.274-4.084-14.084-4.162zm43.308 25.81l-53.713 60.88c.926 6.747 3.04 10.945 5.886 13.457 2.85 2.51 7.275 4.083 14.085 4.16l53.713-60.878c-.926-6.748-3.04-10.944-5.887-13.457-2.846-2.512-7.273-4.085-14.083-4.164zm29.34 38.286l-47.56 53.91c.927 6.746 3.04 10.943 5.887 13.456 2.848 2.512 7.275 4.083 14.084 4.162L232 359.44c-.927-6.75-3.04-10.947-5.887-13.46-2.847-2.512-7.274-4.083-14.084-4.162zm24.702 39.137l-38.794 44.28c.925 6.76 3.038 10.962 5.888 13.476 2.845 2.51 7.267 4.082 14.067 4.163l38.796-44.28c-.926-6.758-3.04-10.96-5.89-13.476-2.844-2.51-7.266-4.08-14.066-4.162zm35.342 4.79c1.694 4.62 2.673 9.74 3.014 15.192l.232 3.704-8.277 9.448 26.724 19.037c6.464-1.594 10.316-4.075 12.46-7.086 2.145-3.01 3.233-7.464 2.628-14.093l-36.78-26.2z" fill="#21d121" fill-opacity="1" stroke="#0f0101" stroke-opacity="1" stroke-width="8" filter="url(#shadow-1)"></path></g></svg> | ||||
| After Width: | Height: | Size: 3.1 KiB | 
							
								
								
									
										1
									
								
								assets/icons/icon_starship.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| <svg style="height: 512px; width: 512px;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><defs><filter id="shadow-1" height="300%" width="300%" x="-100%" y="-100%"><feFlood flood-color="rgba(11, 1, 1, 1)" result="flood"></feFlood><feComposite in="flood" in2="SourceGraphic" operator="atop" result="composite"></feComposite><feGaussianBlur in="composite" stdDeviation="15" result="blur"></feGaussianBlur><feOffset dx="0" dy="0" result="offset"></feOffset><feComposite in="SourceGraphic" in2="offset" operator="over"></feComposite></filter></defs><g class="" style="" transform="translate(0,0)"><path d="m265 34 47.898 35.924 61.563 123.123-8.057 32.23-24.943-4.158 3.16-10.533 2.842-9.473L256 182.823l-91.463 18.29 6.002 20.006-24.943 4.156-8.057-32.228L199.1 69.926 247 34v56h-39l-16 32 64 38 64-38-16-32h-39zm188.313 169.258 30.3 10.101-13.478 29.205-30.016-5.001zm-394.626 0 13.194 34.304-30.016 5.002-13.478-29.205zM256 205.32l53.8 58.692L281.306 359h-50.61L202.2 264.012zm25.254.909 43.283 8.658-8.715 29.052zm-50.508.002-34.568 37.709-8.715-29.053zm105.5 32.267L482.5 262.873 429.799 368.28 329.98 259.385zm-160.492 0 6.266 20.887L82.2 368.279 29.5 262.873zm148.205 40.96 72.201 78.765-84.556-37.582zm-135.918 0 12.355 41.183-84.556 37.582zm118.348 58.564 28.646 12.732L312.973 439H265v-62h29.695zm-100.778 0L217.305 377H247v62h-47.973l-22.062-88.246zM387.6 374.115l18.105 8.047-9.984 21.635-16.387-8.193zm-263.2 0 8.266 21.489-16.387 8.193-9.984-21.635zM311 457v30h-30v-30zm-80 0v30h-30v-30z" fill="#7ed321" fill-opacity="1" filter="url(#shadow-1)" stroke="#ccc" stroke-opacity="1" stroke-width="8"></path></g></svg> | ||||
| After Width: | Height: | Size: 1.6 KiB | 
							
								
								
									
										1
									
								
								assets/icons/icon_stealth.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| <svg style="height: 512px; width: 512px;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><defs><filter id="shadow-1" height="300%" width="300%" x="-100%" y="-100%"><feFlood flood-color="rgba(255, 255, 255, 1)" result="flood"></feFlood><feComposite in="flood" in2="SourceGraphic" operator="atop" result="composite"></feComposite><feGaussianBlur in="composite" stdDeviation="15" result="blur"></feGaussianBlur><feOffset dx="0" dy="0" result="offset"></feOffset><feComposite in="SourceGraphic" in2="offset" operator="over"></feComposite></filter></defs><g class="" style="" transform="translate(0,0)"><path d="M254.07 19.707c-56.303 28.998-106.297 107.317-122.64 168.707 32.445 2.11 58.63 12.963 78.638 30.848l9.334-10.198c-13.336-13.056-30.596-23.9-52.994-34.707 12.68-31.542 32.01-79.29 56.598-82.07 9.62-1.088 19.92 4.722 31.13 21.068 35.08-58.334 68.394 18.705 87.727 61.002-21.94 11.897-39.132 22.82-52.63 36.024l8.68 9.76c19.68-17.732 45.72-29.358 78.55-31.673C358.24 127.335 311.515 50.14 254.07 19.707zM219.617 144.57c-8.894 0-16.103 3.952-16.103 8.826 0 4.875 7.21 8.827 16.103 8.827 8.894 0 16.106-3.95 16.106-8.827 0-4.874-7.212-8.826-16.106-8.826zm68.965 0c-8.894 0-16.105 3.952-16.105 8.826 0 4.875 7.21 8.827 16.105 8.827 8.894 0 16.106-3.95 16.106-8.827 0-4.874-7.212-8.826-16.106-8.826zm-118.894 70.88c-2.19 3.672-4.343 7.497-6.444 11.52-25.587 48.98-43.26 123.643-43.896 223.48 32.776 18.89 64.322 31.324 95.707 36.988-35.5-24.36-60.375-80.893-60.375-146.754 0-45.97 12.12-87.39 31.51-116.506-5.098-3.372-10.583-6.29-16.502-8.727zm168.933.35c-5.852 2.477-11.27 5.412-16.298 8.764 19.24 29.095 31.254 70.354 31.254 116.12 0 65.82-24.844 122.322-60.306 146.707 30.88-5.598 62.44-17.812 95.656-36.947-.638-99.57-18.31-174.163-43.9-223.177-2.088-4.002-4.228-7.81-6.405-11.467zm-97.665 23.61c7.026 22.543 9.128 45.086.98 67.63h-41.552v18.513c10.057-3.24 20.25-5.39 30.502-6.594.066 50.215 1.313 96.574 19.82 145.435l4.193 11.074 4.485-10.962c19.48-47.615 18.045-95.297 17.933-145.024 10.257 1.333 20.463 3.4 30.545 6.07v-18.515h-41.374c-6.888-22.544-5.932-45.087.803-67.63h-26.335z" fill="#21d121" fill-opacity="1" stroke="#0f0101" stroke-opacity="1" stroke-width="8" filter="url(#shadow-1)"></path></g></svg> | ||||
| After Width: | Height: | Size: 2.2 KiB | 
							
								
								
									
										1
									
								
								assets/icons/icon_technology.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| <svg style="height: 512px; width: 512px;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><defs><filter id="shadow-1" height="300%" width="300%" x="-100%" y="-100%"><feFlood flood-color="rgba(255, 255, 255, 1)" result="flood"></feFlood><feComposite in="flood" in2="SourceGraphic" operator="atop" result="composite"></feComposite><feGaussianBlur in="composite" stdDeviation="15" result="blur"></feGaussianBlur><feOffset dx="0" dy="0" result="offset"></feOffset><feComposite in="SourceGraphic" in2="offset" operator="over"></feComposite></filter></defs><g class="" style="" transform="translate(0,0)"><path d="M228.844 32.22v114.218h17.687V32.218h-17.686zm-108.25.624c-15.507 0-28.094 12.586-28.094 28.093C92.5 76.444 105.087 89 120.594 89c12.655 0 23.34-8.372 26.844-19.875h44.937v77.313h17.688v-95H147.03c-3.888-10.837-14.262-18.593-26.436-18.593zm193.25 0c-15.507 0-28.063 12.586-28.063 28.093 0 12.124 7.677 22.45 18.44 26.376v59.124h17.655V87.844c11.596-3.452 20.063-14.193 20.063-26.906 0-15.508-12.587-28.094-28.094-28.094zM266.124 92.5v53.938h17.657V92.5h-17.655zm188.532 4.03c-15.507 0-28.094 12.588-28.094 28.095 0 13.083 8.948 24.074 21.063 27.188v27.468h-92.938v17.657h110.624v-46.342c10.223-4.192 17.407-14.233 17.407-25.97 0-15.507-12.557-28.094-28.064-28.094zM30.187 123.657v17.688H96.75v55.594h62.814V179.28h-45.126v-55.624h-84.25zm147.032 40.47v159.718h159.81v-159.72H177.22zm17.56 15.655h17.657v78.595l32.407 32.406h75.28v17.658H237.5l-2.594-2.594-10.75-10.75c-1.033 7.385-7.36 13.062-15.03 13.062-8.392 0-15.19-6.796-15.19-15.187 0-7.682 5.696-13.98 13.095-15l-9.655-9.658-2.594-2.593V179.78zm54.94.157h17.686v55.313h52.53l.002 17.688H249.72v-73zM53.124 217.375V307.344c-11.49 3.512-19.844 14.198-19.844 26.844 0 15.505 12.557 28.093 28.064 28.093s28.093-12.587 28.093-28.092c0-12.195-7.79-22.564-18.656-26.438v-72.72h88.782v-17.655H53.124zm301.563 0v17.656h53.968v-17.655h-53.97zm99.968 21.97c-10.898 0-20.342 6.21-25 15.28h-74.97l.002 17.688H427c2.325 13.168 13.824 23.187 27.656 23.187 15.507 0 28.063-12.588 28.063-28.094 0-15.507-12.557-28.062-28.064-28.062zm-349.062 15.28v17.688h53.97v-17.688h-53.97zm17.156 36.47v84.217c-11.498 3.513-19.875 14.2-19.875 26.844 0 15.506 12.587 28.094 28.094 28.094 15.506 0 28.06-12.588 28.06-28.094 0-12.194-7.766-22.564-18.624-26.437v-66.94h19.156v-17.686H122.75zm231.938 0v17.686h45.156v95.283c-11.323 3.624-19.53 14.26-19.53 26.78-.002 15.506 12.585 28.063 28.092 28.063 15.507 0 28.063-12.557 28.063-28.062 0-12.32-7.935-22.778-18.97-26.563V291.095h-62.814zM192.375 341.53v54.033h17.688V341.53h-17.688zm36.47 0v86.564c-11.013 3.794-18.94 14.233-18.94 26.53 0 15.506 12.588 28.095 28.095 28.095s28.063-12.59 28.063-28.095c0-12.53-8.203-23.14-19.532-26.75V341.53h-17.686zm37.28 0v54.033h17.688l-.032-54.032h-17.655zm38.094 0v140.064h17.655V341.53H304.22z" fill="#21d121" fill-opacity="1" stroke="#0f0101" stroke-opacity="1" stroke-width="8" filter="url(#shadow-1)"></path></g></svg> | ||||
| After Width: | Height: | Size: 2.9 KiB | 
							
								
								
									
										1
									
								
								assets/icons/icon_vehicle.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| <svg style="height: 512px; width: 512px;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><defs><filter id="shadow-1" height="300%" width="300%" x="-100%" y="-100%"><feFlood flood-color="rgba(11, 1, 1, 1)" result="flood"></feFlood><feComposite in="flood" in2="SourceGraphic" operator="atop" result="composite"></feComposite><feGaussianBlur in="composite" stdDeviation="15" result="blur"></feGaussianBlur><feOffset dx="0" dy="0" result="offset"></feOffset><feComposite in="SourceGraphic" in2="offset" operator="over"></feComposite></filter></defs><g class="" style="" transform="translate(0,0)"><path d="M304 25c-9.496 0-17.002 7.504-17.002 17S294.504 59.002 304 59.002c9.496 0 17.002-7.506 17.002-17.002 0-9.496-7.506-17-17.002-17zm-9 50.809V201h-95.377l-13.314 79.885L256 294.822l69.691-13.937L313 204.738V75.808A34.63 34.63 0 0 1 304 77a34.63 34.63 0 0 1-9-1.191zM228.2 137l-38.557 46H277v-46zM55 179.402v139.405l-14.621 13.209 22.738 23.492 164.969-5.906-3.014-29.946-15.199-15.705-35.639-7.127-8.543-1.709 4.266-25.597-42.064-16.569L73 302.543v-123.14zm128.07 11.438-23.379 27.892 17.844 5.323zM329.123 192l5.342 32.055 17.844-5.323c-5.456-6.278-8.806-10.732-23.186-26.732zm54.984 60.95-42.064 16.568 4.266 25.597-44.182 8.836-15.2 15.705-3.013 29.946 164.969 5.906 22.738-23.492zm-265.744 28.687 48 48-12.726 12.726-48-48zm275.274 0 12.726 12.726-48 48-12.726-12.726zm-240.453 88.119-20.323.396-24.914 49.83L96 416l-32 32 48 16 32-32-18.805-6.268zm205.906.547 27.715 55.43L368 432l32 32 48-16-32-32-11.947 3.982-24.473-48.945z" fill="#7ed321" fill-opacity="1" stroke="#ccc" stroke-opacity="1" stroke-width="8" filter="url(#shadow-1)"></path></g></svg> | ||||
| After Width: | Height: | Size: 1.6 KiB | 
							
								
								
									
										1
									
								
								assets/icons/icon_vehicles.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| <svg style="height: 512px; width: 512px;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><defs><filter id="shadow-1" height="300%" width="300%" x="-100%" y="-100%"><feFlood flood-color="rgba(255, 255, 255, 1)" result="flood"></feFlood><feComposite in="flood" in2="SourceGraphic" operator="atop" result="composite"></feComposite><feGaussianBlur in="composite" stdDeviation="15" result="blur"></feGaussianBlur><feOffset dx="0" dy="0" result="offset"></feOffset><feComposite in="SourceGraphic" in2="offset" operator="over"></feComposite></filter></defs><g class="" style="" transform="translate(0,0)"><path d="M18.313 17.78v62.095L170.22 231.75 43 211.625s64.877 30.418 116.344 54.438l-47.375 23.25L279.687 355c-3.582-3.086-7.13-6.347-10.594-9.813-39.543-39.54-54.888-88.3-34.28-108.906 20.605-20.604 69.362-5.26 108.905 34.282 3.355 3.356 6.53 6.754 9.53 10.22l-64.938-167.813L265 162.187c-24.094-51.623-55.03-117.532-55.03-117.532l19.468 122.875L79.655 17.78H18.312zm425.312 243.345l-43.406 46.938-23.595 5.875-33.594-18.344-50.06 50.094 18.31 33.562-5.874 23.563-46.03 42.562 80.093 14.03 28.717 28.72 13.188-13.22-51.5-51.53L469 471.625l-48.656-140.25 52.28 52.28 13.22-13.217-27.97-27.97-14.25-81.343zM392.062 357.5c69.03 70.208 33.083 106.35-37.218 37.22l37.22-37.22z" fill="#21d121" fill-opacity="1" stroke="#0f0101" stroke-opacity="1" stroke-width="8" filter="url(#shadow-1)"></path></g></svg> | ||||
| After Width: | Height: | Size: 1.4 KiB | 
							
								
								
									
										114
									
								
								assets/json_data/armors.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,114 @@ | ||||
| [ | ||||
|   { | ||||
|     "name": "Leather", | ||||
|     "protection": 2, | ||||
|     "cost": 50, | ||||
|     "enc": 1, | ||||
|     "techAge": "Early Primitive", | ||||
|     "description": "Made from Boiled animal hide or other non-metallic material, this armor can be worn as regular clothing. When facing firearms or other weapons from the Early Mechanical Eras or later, this armor provides 1 point of Protection [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Chainmail", | ||||
|     "protection": 4, | ||||
|     "cost": 300, | ||||
|     "enc": 2, | ||||
|     "techAge": "Late Primitive", | ||||
|     "description": "Manufactured from rings of iron or primitive steel woven together. Typically, chainmail covers the upper body and hangs to the knees. When facing firearms or other weapons from the Early Mechanical Age or later, this armor provides 2 points of Protection [2]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Plate Mail", | ||||
|     "protection": 6, | ||||
|     "cost": 1000, | ||||
|     "enc": 3, | ||||
|     "techAge": "Late Primitive", | ||||
|     "description": "A full suit of metal plated armor with a thick woven undergarment to reduce irritation. Plate mail is normally only worn by the wealthy nobles. When facing firearms or other weapons from the Early Mechanical Eras or later, this armor provides 2 points of Protection [3]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Bulletproof Vest", | ||||
|     "protection": 4, | ||||
|     "cost": 150, | ||||
|     "enc": 1, | ||||
|     "techAge": "Early Atomic", | ||||
|     "description": "A lightweight ballistic armor for torso protection. Bulletproof vests are easy to wear under clothes and are quite concealable [3]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Ceramic Plates", | ||||
|     "protection": 8, | ||||
|     "cost": 300, | ||||
|     "enc": 2, | ||||
|     "techAge": "Early Atomic", | ||||
|     "description": "Modern heavy-duty ballistic-cloth armor. Commonly worn by soldiers, well-equipped security, and Atomic Age mercenaries. Ceramic Plate counts as two Items for encumbrance purposes and is not concealable [4]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Synthsilk Armor", | ||||
|     "protection": 7, | ||||
|     "cost": 6000, | ||||
|     "enc": 0, | ||||
|     "techAge": "Early Interstellar", | ||||
|     "description": "This advanced personal armor is bio-engineered from spider silk. Synthsilk is easily concealable and does not count as an item for Encumbrance [5]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Energy Shield", | ||||
|     "protection": 0, | ||||
|     "cost": 20000, | ||||
|     "enc": 1, | ||||
|     "techAge": "Late Galactic", | ||||
|     "description": "This high-energy emitter is worn on a belt clip. The energy shield is capable of stopping massive amounts of damage and may be worn with other armor. An energy shield completely negates the first hit it receives in combat but is then depleted. Roll 1d6 every following combat round; on a result of 4-6, the shield regenerates on the wearer’s turn, and may negate another hit afterwards [6]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Envirosuit", | ||||
|     "protection": 4, | ||||
|     "cost": 700, | ||||
|     "enc": 2, | ||||
|     "techAge": "Early Atomic", | ||||
|     "description": "A full body sealed suit with attached filters and ports to attach oxygen tanks as well. The Envirosuit fully protects against airborne toxins and pathogens. An Envirosuit will absorb the first 20 rads of radiation exposure [6]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Spacesuit", | ||||
|     "protection": 6, | ||||
|     "cost": 2000, | ||||
|     "enc": 2, | ||||
|     "techAge": "Early Space", | ||||
|     "description": "This is a standard space suit, much lighter and cheaper than a 20th Century Terran space suit. A space suit carries life support supplies for up to 6 hours. It protects against Vacuum, Tainted, and Exotic atmospheres, as well as airborne pathogens. Absorbs 60 rads [7]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Spacesuit, Advanced", | ||||
|     "protection": 6, | ||||
|     "cost": 4000, | ||||
|     "enc": 0, | ||||
|     "techAge": "Early Interstellar", | ||||
|     "description": "This represents the final evolution of the space suit. An advanced space suit does not count as an item for encumbrance and provides life support for 12 hours. It offers the same protection as a regular space suit [7]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Spacesuit, Hostile Environment", | ||||
|     "protection": 10, | ||||
|     "cost": 10000, | ||||
|     "enc": 3, | ||||
|     "techAge": "Late Space", | ||||
|     "description": "A heavy-duty and reinforced space suit designed for extreme conditions. It protects against vacuum and corrosive atmospheres, as well as airborne pathogens and toxins, and provides 2d6 hours of protection from highly corrosive atmospheres. The Hostile Environment space suit carries life support supplies for 6 hours and absorbs 150 rads’ worth of radiation [5]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Envirosuit, Advanced", | ||||
|     "protection": 6, | ||||
|     "cost": 5000, | ||||
|     "enc": 0, | ||||
|     "techAge": "Early Interstellar", | ||||
|     "description": "This is a much higher-tech version of the Enrivosuit, and works the same way, but is ultra-light. It can absorb 40 rads’ worth of radiation [8]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Encased Armor", | ||||
|     "protection": 10, | ||||
|     "cost": 10000, | ||||
|     "enc": 2, | ||||
|     "techAge": "Late Space", | ||||
|     "description": "A suit of heavy, full body rigid armor, often used by corporate SWAT and high-end mercenaries. The suit offers environmental protection as an Envirosuit (see below) and includes integrated comms and a Heads-Up Display [4]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Powered Armor", | ||||
|     "protection": 14, | ||||
|     "cost": 200000, | ||||
|     "enc": 0, | ||||
|     "techAge": "Late Interstellar", | ||||
|     "description": "This is servo-assisted, fully enclosed, combat armor. This powerful armor grants +3D to all Physical checks related to brute strength and doubles the wearer’s normal Encumbrance limits. Powered armor grants +1D to all melee damage done by its wearer. It also removes the Bulky aspect from any weapons the wearer is using. Powered Armor protects against vacuum and hostile environments, as a space suit, with 6 hours of life support, and absorbs 50 rads of radiation. Powered armor does not count as an item for encumbrance when worn. It counts as 3 encumbrance items when carried [8]." | ||||
|   } | ||||
| ] | ||||
							
								
								
									
										38
									
								
								assets/json_data/drugs.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,38 @@ | ||||
| [ | ||||
|   { | ||||
|     "name": "Antirad", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 1000, | ||||
|     "description": "Administering this drug before exposure to radiation will prevent the accumulation of rads for 1d6 hours. After being exposed to radiation, it removes 2d6 rads [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Fast Forward", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 200, | ||||
|     "description": "This emergency life support drug dramatically slows the subject’s metabolism; this drastically reduces life support costs and can keep a subject alive for 1d6 days without the use of any cryogenic medical technology [2]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Hyper", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 50, | ||||
|     "description": "A dangerous drug that allows a person to go 72 hours without sleep at full alertness and functionality. The subject ignores Fatigue for the full duration. Afterwards, the subject must sleep for 24 hours. Another dose of Hyperspace before the 24 hours is up forces a roll on the Wound Triage Table [2]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Ice", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 20, | ||||
|     "description": "A combat drug used by many professional mercenaries; Ice gives a +2D bonus to all morale checks. It is rumored that prolonged use of Ice will result in lack of empathy and sociopathic behavior. Ice manufacturers vehemently deny such calumnies [3]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Panacea", | ||||
|     "tech_age": "Late Space", | ||||
|     "cost": 50, | ||||
|     "description": "When using this drug, roll Knowledge 8+ to cure most diseases or poisons. This drug does not work on tailored viruses or biological weapons [3]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "SlowMo", | ||||
|     "tech_age": "Late Space", | ||||
|     "cost": 300, | ||||
|     "description": "This drug will heal one Wound in 10 minutes. This metabolic accelerator can only be used safely once per week. It gets its name because those who take it spend ten minutes watching the world go by very, very slowly. If it is used more than once per week, roll on the Wound Triage Table with a -2D penalty [3]." | ||||
|   } | ||||
| ] | ||||
							
								
								
									
										320
									
								
								assets/json_data/exploration_equipment.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,320 @@ | ||||
| [ | ||||
|   { | ||||
|     "name": "Artificial Gill", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 4000, | ||||
|     "description": "This device allows unlimited underwater activity on worlds with otherwise breathable atmospheres." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Auto Doc", | ||||
|     "tech_age": "Early Interstellar", | ||||
|     "cost": 10000, | ||||
|     "description": "A large device normally carried in a vehicle or space craft, the Auto Doc uses advanced AI and sensors to heal almost any disease, poison, or injury. The Auto Doc provides a +1D advantage on any Triage rolls or any roll to resist poisons or diseases. It can serve as a qualified nurse or doctor for the purposes of supervising healing from Wounds." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Autokitchen", | ||||
|     "tech_age": "Early Interstellar", | ||||
|     "cost": 10000, | ||||
|     "description": "An automated kitchen which can store and prepare meals. The standard kitchen can prepare 25 meals from internal supplies. The mass of the kitchen is 100 kg. Resupply costs depend on the quality of ingredients desired and can be determined using the Cost-of-Living table above." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Autokitchen, Emergency", | ||||
|     "tech_age": "Early Interstellar", | ||||
|     "cost": 5000, | ||||
|     "description": "A smaller and much more restrictive version of an autokitchen, this emergency kitchen provides no-frills survival food for 10 meals. This unit weighs 5kg, or two items of Encumbrance." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Backpack", | ||||
|     "tech_age": "Early Primitive", | ||||
|     "cost": 10, | ||||
|     "description": "A backpack does not count as an item for encumbrance when worn. The items in the backpack still count towards encumbrance. It takes one Action to remove an item from a backpack while in combat." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Binoculars", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "cost": 50, | ||||
|     "description": "A standard piece of survival and exploration equipment. At the Early Mechanical Age, they are ruggedized for extended wilderness use, and come equipped with a compass and scaled optics. Beyond the Late Atomic Age, hand-held binoculars become integrated into high-tech optical sensors like the Night Visor." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Bioscanner", | ||||
|     "tech_age": "Early Interstellar", | ||||
|     "cost": 5000, | ||||
|     "description": "This handheld device provides data on an organic body’s composition, chemistry, physiology, and life signs. A Knowledge roll is required to properly interpret the data." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Breather Mask", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 100, | ||||
|     "description": "A dual-purpose gas/filter mask apparatus that condenses thin atmospheres for comfortable breathing. It also filters out tainted atmospheres and provides protection against airborne toxins. Attachments for oxygen bottles make the Breather useful for operating in oxygen-poor or poisonous atmospheres." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Chameleon Suit", | ||||
|     "tech_age": "Late Interstellar", | ||||
|     "cost": 2500, | ||||
|     "description": "This full body suit can shift its color patterns to blend in with the surroundings. It does not make a wearer invisible, but at medium and long ranges the wearer will be almost impossible to see and gains +2D to Stealth tests. It includes infra-red cloaking. The same technology can be applied to an Encased Armor Suit for an additional cost of 10000 Credits." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Cold Weather Clothing", | ||||
|     "tech_age": "Early Primitive", | ||||
|     "cost": 50, | ||||
|     "description": "These protect against inclement weather. When wearing Cold Weather Clothing, characters can survive in Arctic environments without taking any damage. The Early Interstellar Age version costs 800 Credits and does not count as an item for encumbrance. Note that surviving in a harsher environment, like one that is far colder than a typical arctic climate, may require additional protection, like a space suit." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Compass", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "cost": 10, | ||||
|     "description": "This is a standard piece of equipment in all wilderness survival kits. By the Early Space Age, they incorporate gyroscopic and electronic features that make them usable on any world, given a few minutes of calibration." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Desert Suit", | ||||
|     "tech_age": "Later Space", | ||||
|     "cost": 1000, | ||||
|     "description": "An all-in-one suit that traps the bodies moisture and recycles it as drinking water. It includes a hood, breather mask, and goggles." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Disguise Kit", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "cost": 500, | ||||
|     "description": "This kit consists of highly personalized collections of makeup, clothing, wigs, and other accouterments that allow a skilled user to not be recognized, or even pass as someone else. At more advanced Tech Ages, Disguise Kits may include 3-D printed latex masks (Late Atomic Age), retina scanner spoofers (Early Space Age), or even holographic enhancements designed to spoof facial recognition algorithms (Early Interstellar Age). Increase the kit’s price by 100% for each of these listed features." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Emergency Beacon", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 1000, | ||||
|     "description": "A portable device that can be activated and will transmit an emergency signal up to a 1000 km. The device has an internal battery that will last for 48 hours. Most vehicles at Late Space Age and above include one automatically, but this portable version can be easily carried along with other survival supplies." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Exoskeleton", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 20000, | ||||
|     "description": "This marvel of robotics grants wearers +2D to athletics-related Physical checks. It also allows the wearer to jump up to 2m vertically or 5m horizontally. Soldiers wearing exoskeletons may march their normal cross-country speed while tiring at half the rate. The exoskeleton’s batteries are good for 48 hours of activity before requiring replacement. At higher tech levels, exoskeletons are often replaced by cybernetics." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Field Radio", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 100, | ||||
|     "description": "A radio transmitter and receiver with a 100 km range. At the Late Mechanical Age is counts as 2 items of Encumbrance, reduced to 1 item at Early Atomic, and 0 encumbrance at the Late Atomic Age." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Field Rations", | ||||
|     "tech_age": "Late Primitive", | ||||
|     "cost": 5, | ||||
|     "description": "These are foodstuffs specially prepared to stay edible for 6 months. Each ration counts as a single item for encumbrance. Early Space Age field rations are much lighter: 25 field rations count as one item, and they last for 6 decades." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Glidesuit", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 500, | ||||
|     "description": "A form-fitting suit that includes adjustable wings that allow the wearer to glide. Flapping provides limited flight. The Glidesuit reduces a person’s falling rate in half. Roll Physical to avoid injury when landing. Wearers use their Physical skill to make complex maneuvers." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Handcuffs", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "cost": 50, | ||||
|     "description": "An item commonly used by bounty hunters and law enforcement officers. Escaping properly applied handcuffs requires a Physical roll at -3D. Palming the key and keeping it hidden until the right time is a standard Stealth roll." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Hotsuit", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 200, | ||||
|     "description": "A reflective suit designed to protect the wearer from temperatures up to 100 degrees centigrade. The suit is clumsy and imposes a -1D disadvantage on all physical activity while wearing it. At Early Interstellar Age, the suit is form-fitting and is not clumsy to wear and does not count as an item of encumbrance." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Intelligent Rooms", | ||||
|     "tech_age": "Late Space", | ||||
|     "cost": "Double Cost", | ||||
|     "description": "Fully integrated rooms, houses, or apartments with voice activated commands controlling lighting, heating, and general appearance controlled by an Independent robotic brain. Smartwalls provide 3-D entertainment. At Early Interstellar Age, Low Simulated AI rooms are expected even at the subsistence quality of living and are included in the listed prices." | ||||
|   }, | ||||
|   { | ||||
|     "name": "LED Flashlight", | ||||
|     "tech_age": "Late Atomic", | ||||
|     "cost": 5, | ||||
|     "description": "A hand-held, rubberized, sturdy light source with a variable light aperture. The 2m cone of light has a range of 50m, but it can be adjusted to illuminate a 5m radius. Smaller, more expensive (20 Credits) versions can be mounted on weapons or clipped to armor. A Late Atomic Age LED Flashlight’s battery will last for 24 hours. By Early Interstellar Age, the battery will last for a decade of continual use." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Lockpicks", | ||||
|     "tech_age": "Late Primitive", | ||||
|     "cost": 50, | ||||
|     "description": "A collection of specialized tools for opening locks, in a handy carrying pouch. Attempting to pick a lock without lockpicks penalizes the attempt by -2D." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Map Box", | ||||
|     "tech_age": "Late Space", | ||||
|     "cost": 3000, | ||||
|     "description": "Designed for wilderness exploration, this self-contained nav-computer system provides scalable maps of a world’s surface based on known data and any additional data obtained by the exploration team. Most inhabited worlds have detailed map drives which can be inserted and provide up to street-level details of any explored area and high-level maps of the entire surface. Map boxes count as 2 items of Encumbrance. At Early Interstellar, it is 1 item." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Medkit", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "cost": 100, | ||||
|     "description": "This is standard issue in all survival kits and can be found in ship’s lockers across known space. Medkits contain all the necessary equipment and compounds to deal with minor trauma and easily treatable emergencies." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Medkit, Advanced", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 1000, | ||||
|     "description": "Containing advanced scanners and additional medications, the advanced version of the medical kit provides a +1D advantage when rolling on the Triage table." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Multiscanner", | ||||
|     "tech_age": "Late Space", | ||||
|     "cost": 2500, | ||||
|     "description": "This hand-held sensor suite is a combination of several functions: it can detect radiation, electromagnetic emissions, metals, and a limited set of chemicals and biological compounds. It can also perform standard blood tests. A Knowledge roll is required to make real use of the data, but basic analysis is possible with the use of on-board software." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Night Visor", | ||||
|     "tech_age": "Late Atomic", | ||||
|     "cost": 300, | ||||
|     "description": "This stylish set of eye wear combines low-light and infra-red vision goggles into one handy device. Safety features prevent blindness if the light level suddenly changes. Wearing a Night Visor eliminates the disadvantages of working in dim light and reduces the penalty for total darkness to -1D." | ||||
|   }, | ||||
|   { | ||||
|     "name": "OmniComm", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 50, | ||||
|     "description": "Today’s smartphones will eventually evolve into the OmniComm, which handles all your telecom needs in one device, with holographic augmented reality capability. The OmniComm can automatically integrate with whatever telecommunication networks are present on a given world. Batteries last for a few days in the Early Space Age. By Early Interstellar, batteries will last for months, if not years." | ||||
|   }, | ||||
|   { | ||||
|     "name": "OmniComp", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 250, | ||||
|     "description": "A hand-held, light-weight computer which evolved from both the laptop and the tablet computer. It functions much like an OmniComm but is intended for more advanced content production and computing tasks. Early Space Age batteries for these devices last for a few days; Early Interstellar Age batteries last for months between recharging." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Omnitool", | ||||
|     "tech_age": "Late Space", | ||||
|     "cost": 500, | ||||
|     "description": "An auto-adjusting, auto-fitting repair tool. The Omnitool will fit exactly to any kind of bolt, screw or fastening device, it doubles as wire-cutters, small hacksaw, flashlight, and a myriad of other tools. All this is packed into one item’s worth of encumbrance!" | ||||
|   }, | ||||
|   { | ||||
|     "name": "Oxygen Tank", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 100, | ||||
|     "description": "An oxygen tank that can be attached to a space suit or breather mask to provide 90 minutes of breathable air. Each tank is 2 items of encumbrance. At the Late Space Age, the tanks are only 1 item of encumbrance." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Personal Reentry Capsule", | ||||
|     "tech_age": "Late Space", | ||||
|     "cost": 15000, | ||||
|     "description": "Used as a last-ditch survival tool, the personal capsule allows for reentry into an atmosphere and a relatively safe landing on a world’s surface. The capsule includes a primitive autopilot and a small reentry thruster to make a soft landing. The occupant has no control over the capsule. The autopilot will attempt to land on flat, dry land if possible. All uses must make a Physical check at +1D to avoid 2d6 damage upon landing." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Personal HUD", | ||||
|     "tech_age": "Early Interstellar", | ||||
|     "cost": 1500, | ||||
|     "description": "A personal Heads-Up-Display (HUD) can be fitted to any pair of goggles or a helmet visor. At the Late Interstellar Age, it can be fitted as a contact lens. The HUD interfaces with an OmniComp displaying information and messages. The HUD can also be synched to any Smart Gun system (see the Weapons Accessories section)." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Portable Fusion Generator", | ||||
|     "tech_age": "Early Interstellar", | ||||
|     "cost": 20000, | ||||
|     "description": "A marvel of interstellar technology, and a crucial piece of equipment for colonies across the galaxy. Fusion generators are also very useful for mercenary outfits that rely on energy weapons. These generators are small (20kg, roughly 2 Encumbrance), rugged, and designed to last for decades." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Portable Generators", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 1000, | ||||
|     "description": "A ubiquitous device still used across the galaxy to generate electricity far away from central power grids. Most runoff refined hydrocarbons that can be sourced locally. This one is 15kg and consumes 4 liters of fuel per hour. Higher tech level versions are smaller and more fuel efficient." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Prefab Space Base", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 50000, | ||||
|     "description": "This is a pressurized, modular, temporary structure that is capable of housing 6 persons. It has a single airlock and requires 12 person-hours to assemble. Life support lasts for 24 hours without a portable generator to power atmoscrubbers." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Pressure Tent", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 5000, | ||||
|     "description": "This basic survival shelter lacks an airlock and must be depressurized to enter or leave; it can house up to 2 people and requires an external oxygen tank." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Radiation Detector", | ||||
|     "tech_age": "Early Atomic", | ||||
|     "cost": 150, | ||||
|     "description": "A matchbox sized device that displays radiation levels in a 10m radius. It also indicates the type of radiation if that is important. The detector can also serve as an alarm, chiming loudly when radiation levels are dangerous to humans. This feature can be shut off." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Responsive Clothes", | ||||
|     "tech_age": "Late Space", | ||||
|     "cost": 200, | ||||
|     "description": "Clothing that can adapt to a wearer, fitting them perfectly. The clothing is self-cleaning and can change color and pattern within limits. Not quite one-size-fits-all, but almost." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Rope", | ||||
|     "tech_age": "Early Primitive", | ||||
|     "cost": 5, | ||||
|     "description": "At the Early Primitive Age, rope can hold 50kg. By the Late Atomic Age, the rope is fully synthetic, and can support 500kg of mass before breaking. Roughly 100m of rope is the equivalent of one encumbrance item." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Rope, Advanced", | ||||
|     "tech_age": "Early Interstellar", | ||||
|     "cost": 10, | ||||
|     "description": "This synthetic rope is incredibly light and strong. 10m of this rope can support 1000 kg before breaking." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Sensor Gloves", | ||||
|     "tech_age": "Late Interstellar", | ||||
|     "cost": 4000, | ||||
|     "description": "Thin gloves that incorporate sensitive tactile, pressure, chemical, and bioscanner sensors. They can interface with an OmniComp and provide detailed information about weight, mass, composition, even recent users. They can sense residual heat and chemical signatures for up to an hour after the object was used. The gloves provide a +1D advantage on checks for information where they would apply." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Sentry Gun", | ||||
|     "tech_age": "Late Space", | ||||
|     "cost": 750, | ||||
|     "description": "This security system consists of a weapons tripod mounted with a processor and sensor array. It can be fitted with a support weapon. The operator sets the sentry’s gun’s parameters for when and who it opens fire on. It has a Combat skill of 1." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Sleeping Bag", | ||||
|     "tech_age": "Late Primitive", | ||||
|     "cost": 30, | ||||
|     "description": "A cloth or synthetic padded bag for one person. It provides insulation while sleeping. In the Early Atomic Age, cold weather bags are available. Regular sleeping bags are effective to zero degrees centigrade, while cold weather bags, costing 50 Credits, are insulated to negative 30 degrees centigrade." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Smart Clothes", | ||||
|     "tech_age": "Early Interstellar", | ||||
|     "cost": 500, | ||||
|     "description": "Responsive clothes that incorporate microelectronics providing the equivalent of an Omnicomp and allowing images and messages to be displayed on the clothing if desired." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Super Adhesive", | ||||
|     "tech_age": "Late Atomic", | ||||
|     "cost": 20, | ||||
|     "description": "A powerful adhesive that bonds two materials together and dries in one combat round (6 seconds). The adhesive bond is limited by the strength of the weaker of the two objects. 20 Credits buys a small tube that can cover 1 square meter of surface." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Survival Kit", | ||||
|     "tech_age": "Early Atomic", | ||||
|     "cost": 50, | ||||
|     "description": "A standard-issue item in most aircraft, explorer vehicles, and spacecraft across known space. The survival kit’s contents vary over Tech Age but consist mainly of the basic supplies to keep a small group of humans alive, fed, and healthy for a few weeks. A kit usually contains thermal tarps, LED flashlights, fire starters, water filter and decontamination pills, magnesium flares, and a radio/flash rescue beacon. They count as one item for Encumbrance." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Technical Toolkit", | ||||
|     "tech_age": "Early Atomic", | ||||
|     "cost": 300, | ||||
|     "description": "This toolkit is fully equipped to handle almost any technical job that doesn’t require a full workshop. As the Tech Ages increase, many of the tools become self-modifying smart omnitools that can reconfigure depending on what the tech needs. This greatly reduces the bulk of the toolkit: Space Age Tool Kits are 2 Encumbrance points, while all but the largest Interstellar kits are 1 Encumbrance point." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Thruster Pack", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 5000, | ||||
|     "description": "This wearable harness allows maneuvering in zero-G conditions at normal movement rates. Characters and objects moving by thruster pack are still subject to the laws of motion." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Trauma Tab", | ||||
|     "tech_age": "Late Space", | ||||
|     "cost": 500, | ||||
|     "description": "A single-use medical application that automatically allows a character who has suffered a Wound to continue to fight. However, the strain adds a -1D penalty to the Wound Triage table roll after the fight. At Late Interstellar age, this penalty is dropped." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Water Purification Tablets", | ||||
|     "tech_age": "Late Atomic", | ||||
|     "cost": 10, | ||||
|     "description": "A pack of 50 tablets that are standard issue in survival kits. Each tablet can sterilize up to 1L of water and make it fit for drinking." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Welder, Laser", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 3000, | ||||
|     "description": "This portable workshop tool uses a focused laser beam to perform cutting and welding operations. It can cut through a reinforced bulkhead in 1d6 starship combat rounds (6-36 minutes). Roll Technology to reduce the time to one starship combat round (6 minutes). As a weapon, the laser welder is inaccurate and clumsy, but extremely dangerous: it does 4D (or AV 3D) damage with a range of Close and has energy for 5 attacks. When used as a weapon, the laser welder gains the Fire Weapon Aspect." | ||||
|   } | ||||
| ] | ||||
							
								
								
									
										82
									
								
								assets/json_data/grenades.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,82 @@ | ||||
| [ | ||||
|   { | ||||
|     "name": "Anti-Armor Grenade", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 50, | ||||
|     "damage": "2d6+1", | ||||
|     "aspects": ["Blast", "AV 3d6"], | ||||
|     "description": "Anti-tank grenades use a focused blast to penetrate armor. This grenade can only be fired from a grenade launcher [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Dynamite", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "cost": 75, | ||||
|     "damage": "2d6", | ||||
|     "aspects": ["Blast"], | ||||
|     "description": "A low-tech explosive made of nitroglycerin, sorbents (such as powdered shells or clay) and stabilizers. Dynamite is typically used for demolition but can also be thrown with a lit fuse as a \"grenade\" [2]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "EMP Grenade", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 50, | ||||
|     "damage": "4d6", | ||||
|     "aspects": ["Blast", "Stun (Machines)", "AV 3d6"], | ||||
|     "description": "This Electro-Magnetic Pulse grenade emits a powerful magnetic pulse capable of disrupting electronics. It causes full damage to robots but is otherwise harmless to humans. Characters with implanted cybernetics are vulnerable to EMP but reduce the damage from an EMP grenade by -2d6. Non-cybered organic beings are immune to this stun damage. This grenade’s AV dice are effective against Late Atomic Age or later vehicles [3]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Flashbang Grenade", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 30, | ||||
|     "damage": "3d6+1", | ||||
|     "aspects": ["Blast", "Stun"], | ||||
|     "description": "A non-lethal stun grenade emitting a powerful flash and deafening noise upon detonation [3]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Fragmentation Grenade", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "cost": 30, | ||||
|     "damage": "3d6+3", | ||||
|     "aspects": ["Blast"], | ||||
|     "description": "The basic anti-personnel hand grenade that explodes into a hail of shrapnel [4]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Incendiary Grenade", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 30, | ||||
|     "damage": "3d6", | ||||
|     "aspects": ["Blast", "Fire"], | ||||
|     "description": "A military-grade fire-bomb grenade [4]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Molotov Cocktail", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "cost": 5, | ||||
|     "damage": "2d6+1", | ||||
|     "aspects": ["Blast", "Fire"], | ||||
|     "description": "These are home-made incendiary bombs, commonly made with a glass container filled with highly flammable liquid and lit with a rag [4]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Plasma Grenade", | ||||
|     "tech_age": "Late Interstellar", | ||||
|     "cost": 250, | ||||
|     "damage": "4d6", | ||||
|     "aspects": ["Blast", "Fire", "AV 4d6"], | ||||
|     "description": "A plasma explosive device in a grenade-sized package [4]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Plastique", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 200, | ||||
|     "damage": "2d6+2", | ||||
|     "aspects": ["Greater Blast"], | ||||
|     "description": "This generic, multi-purpose plastic explosive is a favorite of military units, terrorists, demolition teams, and adventurers across known space [4]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Smoke Grenade", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "cost": 15, | ||||
|     "damage": null, | ||||
|     "aspects": ["Greater Blast (Smoke)"], | ||||
|     "description": "This grenade dispenses a cloud of thick smoke upon impact. Apply -1D to all attacks passing within Near range of its detonation point. Smoke also penalizes attacks by laser weapons by an additional -1D. The smoke stays in the air for 1d6+1 combat rounds [5]." | ||||
|   } | ||||
| ] | ||||
							
								
								
									
										219
									
								
								assets/json_data/implants.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,219 @@ | ||||
| [ | ||||
|   { | ||||
|     "name": "Advanced Cyberarms", | ||||
|     "cost": 20000, | ||||
|     "loss": 2, | ||||
|     "tech_age": "Late Space", | ||||
|     "description": "The arms are replaced by obvious cybernetic appendages. This implant grants +2D for brute strength tasks involving the arms. Unarmed attacks do 3d6 damage. Both real arms have to be replaced to provide proper structural support." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Advanced Cyberlegs", | ||||
|     "cost": 50000, | ||||
|     "loss": 2, | ||||
|     "tech_age": "Late Space", | ||||
|     "description": "The subject’s legs are replaced by obvious cybernetic versions. The hips and spine are reinforced to deal with the additional strain. The subject gains the Runner Talent, with a 15+2d6m base rate. As an action, the character may throw Physical to jump 10m horizontally in any direction or 3m upwards. The subject also ignores fall damage for any falls less than 10m." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Artificial Gill", | ||||
|     "cost": 2500, | ||||
|     "loss": 1, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "This implant permits the subject to breathe underwater with no difficulty. Diving too deep will still be lethal. The artificial gill runs off the character’s own energy and does not need recharging." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Bastet Frame Upgrade", | ||||
|     "cost": 1000000, | ||||
|     "loss": 3, | ||||
|     "tech_age": "Late Space", | ||||
|     "description": "This series of implants represents a radical reconstruction of the entire organic body to grant superhuman agility and speed. The Bastet upgrade grants +1D to all Physical and Stealth skill rolls related to speed and agility. The Bastet Frame Upgrade is incompatible with the Hercules Frame Upgrade." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Chem Injector", | ||||
|     "cost": 10000, | ||||
|     "loss": 1, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "An assassin’s implant par excellence, this cybernetic is mounted in the user’s fingertips or canines. Different chemicals are available, but typically they amount to lethal or non-lethal poisons. Attacks against unsuspecting targets are automatically successful. In combat, or against a wary target, roll Combat normally to hit. Targets must roll Physical to avoid the effects of the poison." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Control Rig Implant", | ||||
|     "cost": 1000000, | ||||
|     "loss": 1, | ||||
|     "tech_age": "Late Space", | ||||
|     "description": "This is a highly specialized implant designed to give elite vehicle pilots even more of an edge. The control rig allows the operator to fully interface their minds with vehicles. The control rig gives the character +1D on Vehicles skill throws with suitably modified vehicles. Position rolls remain unaffected. The Control Rig requires that the operator to be in the vehicle, and “plugged in.” Modifying a vehicle to work with a control rig increases its cost by 10%. When the vehicle takes any damage, the plugged in pilot must roll Physical to avoid taking 1D damage from electrical feedback—this damage ignores any armor the pilot may be wearing." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Cosmetic Modification", | ||||
|     "cost": "varies", | ||||
|     "loss": 0.25, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "It is a simple matter to effect changes to a person’s appearance with cybernetics. Small modifications cost 500 Credits; a general makeover costs 1000 Credits; and exotic options—for example, changing your body to resemble that of your favorite animal—cost 5000 Credits and upwards." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Cybernetic Eyes", | ||||
|     "cost": 7500, | ||||
|     "loss": 0.5, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "Both eyes are replaced with artificial versions. Cybereyes are considered to have internal Night Visor upgrades (low light and infrared optics). They also come with x4 image intensification, and anti-flare protection which provides a +1D bonus against flash-bang grenades and other similar effects." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Cyber-Blades", | ||||
|     "cost": 5000, | ||||
|     "loss": 0.5, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "The character’s arm carries a blade that can be extended or retracted at will. The blade does 2d6+1 damage. Internal blades are made with ceramic composites to avoid detection and cannot be detected by any means short of exploratory surgery." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Data safe", | ||||
|     "cost": 5000, | ||||
|     "loss": 0.5, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "This secured electronic storage device secures encrypted files in a small data chip implanted in the character’s skull. The data safe cannot be detected by a cursory sensor scan, but a medical scan will reveal its presence. The safe’s capacity is very large and can store (not run!) even advanced AI software. Data must be uploaded and downloaded from the data safe by means of a Neural Interface. The safe can be set so that failed attempts to break the encryption cause it to shut down and be safely dissolved by the subject’s body." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Filtration System", | ||||
|     "cost": 3000, | ||||
|     "loss": 0.5, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "This lung filtration implant protects from airborne and pathogens. It requires a few minutes of maintenance after every month of use, but otherwise draws its power from the subject’s own electrical impulses. Certain extremely high-tech artificially designed chemical agents might degrade the filtration system very quickly." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Gekko Implants", | ||||
|     "cost": 25000, | ||||
|     "loss": 0.5, | ||||
|     "tech_age": "Late Interstellar", | ||||
|     "description": "Miniature gravitic generators are implanted in the character’s hands and feet. This allows them to climb sheer surfaces and run along walls or ceilings. Earlier versions (same price, Late Space Age) deploy biomimickry micro-hairs that provide the same performance but require bare hands and feet to use." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Glow Tattoos", | ||||
|     "cost": 100, | ||||
|     "loss": 0, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "These advanced tattoos consist of smart bioluminescent skin grafts implanted in complex patterns across the subject’s skin. Patterns and designs are up to the subject. The subject may activate or deactivate the tattoos at will." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Head bomb", | ||||
|     "cost": 3000, | ||||
|     "loss": 0.25, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "A tiny remote-controlled explosive device implanted in the brain. They are designed so that the bomb’s detonation will kill the subject without breaking the skull: no collateral damage! These devices are rarely implanted voluntarily." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Hercules Frame Replacement", | ||||
|     "cost": 1000000, | ||||
|     "loss": 3, | ||||
|     "tech_age": "Late Space", | ||||
|     "description": "A radical reconstruction and replacement of the entire organic body grants superhuman endurance and might. The Hercules Frame grants the character +1D to all Physical rolls related to brute strength and doubles their Encumbrance capacity. All melee damage rolls gain +1d6 as well. The Hercules Frame also provides +10 Stamina. The Hercules frame upgrade is incompatible with the Bastet Frame Upgrade." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Hibernation Suite", | ||||
|     "cost": 100000, | ||||
|     "loss": 1, | ||||
|     "tech_age": "Early Interstellar", | ||||
|     "description": "This emergency cybernetic implant puts the character into a medical coma upon command. The user appears dead and has no need of any air, food, or water while in suspended animation. The Hybernation Suite also pauses any healing and prevents the spread of poisons or disease. The length of hybernation is determined by the user before it begins, and the maximum duration is four weeks. The Hybernation Suite cannot be used for a week after the user is woken from their coma." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Hypercosmetic Surgery", | ||||
|     "cost": 1000000, | ||||
|     "loss": 2, | ||||
|     "tech_age": "Late Space", | ||||
|     "description": "Perfect body sculpting and implanted synthetic pheromone glands radically reconstructs the subject’s body to be perfectly attractive to other members of their species. Grants +1D to all Social skill rolls in situations where physical attractiveness matters." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Neural Booster", | ||||
|     "cost": 1000000, | ||||
|     "loss": 3, | ||||
|     "tech_age": "Late Space", | ||||
|     "description": "This invasive augmentation to the character’s nervous system and adrenal glands grants the character super speed and reaction times for very short periods of time. Once per combat, the character may go twice, for a total of four actions per combat round rather than the usual two. As well, they may use this ability as an interrupt action, going before anyone else in the round." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Neural Interface, Military Grade", | ||||
|     "cost": 10000, | ||||
|     "loss": 0.5, | ||||
|     "tech_age": "Late Space", | ||||
|     "description": "A basic way to jack into a computer, allowing rapid upload of code directly from a hacker’s mind and both combat actions per round to be used as hacking actions." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Neural Interface, Military Grade (Ultimate Jack)", | ||||
|     "cost": 1000000, | ||||
|     "loss": 0.5, | ||||
|     "tech_age": "Late Space", | ||||
|     "description": "The ultimate interface between the human brain and a machine. In addition to all the benefits of the regular Neural Interface, this implant grants +1D to all throws made to hack computers and tamper with electronics." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Omnicomm Implant", | ||||
|     "cost": 5000, | ||||
|     "loss": 0.25, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "This implant functions as an Omnicomm but with a neural interface, allowing communication without audible speech. This implant allows the subject to send and receive images and videos as well. Nearly every operator in the starfaring future has such an implant." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Omnicomp Implant", | ||||
|     "cost": 10000, | ||||
|     "loss": 0.25, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "This implant provides full Omnicomp capabilities to any operator, but with a neural interface. Dedicated hackers tend to eschew this implant, as it is difficult to constantly upgrade with custom modifications." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Oxygen Supply", | ||||
|     "cost": 4000, | ||||
|     "loss": 0.25, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "This implant provides 3 hours of oxygen. This means that the subject can avoid breathing tainted air without any other implants. It is also an excellent emergency oxygen backup supply for spacers operating in dire circumstances." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Prehensile Tail", | ||||
|     "cost": 15000, | ||||
|     "loss": 1, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "This implant provides the subject with an additional grasping and manipulating appendage. The tail is not easily concealed but can be made to look like anything the character desires. The character still only has two actions per combat round. The tail is strong enough to support the weight of the owner." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Prosthetic Limb/Organ", | ||||
|     "cost": 5000, | ||||
|     "loss": 0.25, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "This entry covers a variety of medical cybernetics intended to replace missing or damaged hands, eyes, fingers, arms, legs, feet, or internal organs. These prosthetics are cosmetically indistinguishable from the real thing, but close inspection will reveal the truth." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Smart Gun, Internal", | ||||
|     "cost": 15000, | ||||
|     "loss": 1, | ||||
|     "tech_age": "Late Space", | ||||
|     "description": "Some gun-bunnies need to project their aiming reticule all the time. This cybernetic implant grants the standard Smart Gun advantages to all weapons the user fires, as long as the weapon itself has the Smart Gun sensors. The character must have Cybernetic Eyes or an Omnicomp implanted to benefit from the Internal Smart Gun. Modifying a weapon to work with an Internal Smart Gun costs 100 Credits." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Storage Pouch", | ||||
|     "cost": 2000, | ||||
|     "loss": 0.5, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "This is a discreet abdominal pouch accessible via a well-concealed sphincter. Stows items up to 0.5 kg in weight and 0.5 liters in volume: one Encumbrance point’s worth of items." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Subdermal armor", | ||||
|     "cost": 30000, | ||||
|     "loss": 0.5, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "This implant consists of subtle bone reinforcement and flexible mesh implants that provide 4 points of Protection. Subdermal armor is detectable upon visual inspection, but only if the subject removes their clothing. Subdermal armor is added to the Protection of any worn armor." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Vacuum Adaptation", | ||||
|     "cost": 1000000, | ||||
|     "loss": 2, | ||||
|     "tech_age": "Late Interstellar", | ||||
|     "description": "This implant turns the user into a natural spacer. Internal oxygen reserves, micro-seals, and metal weave skin permit the character an hour of operation in vacuum without a spacesuit. An integral grav unit allows short-range maneuvering in low and microgravity. The grav unit is too weak to maneuver in any gravity beyond 0.05g. This augment also provides 4 points of integral armor Protection and reduces radiation by 10 rads. The Vacuum Adaptation implant is visible upon medical inspection or sensor scan but is otherwise quite discreet. The Vacuum Adaptation requires 1d6+1 hours to recharge after use." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Weapon, Internal", | ||||
|     "cost": 5000, | ||||
|     "loss": 1, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "The character carries a concealed weapon in an internal arm mounting. This can be any pistol-sized weapon, according to the tech age of the society installing the cybernetic: stealth pistol, gyrojet pistol, gauss pistol, laser pistol, blaster pistol, or stunner. The weapon’s magazine is half normal capacity, but it can be reloaded normally via a retractable magazine holder. The cybernetic’s price is 5000 Credits, plus twice the weapon’s cost. Internal weapons can be detected with medical scans or detailed physical inspections. Internal weapons are made to be automatically compatible with a Smart Gun system (internal or external)." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Wound Triage Booster", | ||||
|     "cost": 15000, | ||||
|     "loss": 1, | ||||
|     "tech_age": "Early Space", | ||||
|     "description": "An implanted autodispenser of emergency combat drugs that stabilize wounds and minimize pain. Characters with this implant gain a +1D to Wound Triage rolls after combat. As well, they do not die until they suffer 4 Wounds in combat." | ||||
|   } | ||||
| ] | ||||
							
								
								
									
										142
									
								
								assets/json_data/psionics.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,142 @@ | ||||
| [ | ||||
|   { | ||||
|     "name": "Augmented Speed", | ||||
|     "description": "The character may move to Far distance in half the number of Move Actions normally required. This power lasts for 3 combat rounds [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Augmented Strength", | ||||
|     "description": "Gain +1D to Physical throws, as well as all melee and unarmed damage rolls for the rest of the battle [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Blink", | ||||
|     "description": "The character may teleport once to any point within Far distance, as long as they have line of sight to it, or they have been there previously [2]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Choke", | ||||
|     "description": "Telekinetically grab one humanoid creature by its throat (or analogous body part) and strangle them for 1D Damage every round, for a maximum of 5 rounds. The victim may throw Physical each round to break out of the chokehold [2]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Clairvoyance", | ||||
|     "description": "This power allows characters to psionically watch and listen at a spot they choose within Far distance. This power ignores solid objects, and the character does not have to be familiar with the location in question. Whether or not a psion can use Clairvoyance to familiarize themselves sufficiently for the Blink power (above) is up to the Referee [2]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Cryokinesis", | ||||
|     "description": "Characters with this power may freeze up to 5 cubic meters of material for 10 minutes. Cryokinesis can also be used offensively. Throw Combat to attack a target within Near range, causing 2D+2 damage, with AV 2D. This affects targets internally; survival clothes do not protect from this power [3]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Electrokinesis", | ||||
|     "description": "This character is able to manipulate electrical energy in their vicinity. They may use this power to recharge batteries, short-circuit electrical devices, or generate spectacular lightshows. They may also attack a target within Near range with lightning! This causes 3D damage, with AV 2D. The target may throw Physical to take half damage. Robots suffer -2D on that roll [3]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Emotion", | ||||
|     "description": "This power allows the psion to evoke a powerful emotion in a target at Near range; fear, joy, or friendliness are typical emotions. It is obvious to everyone, affected or not, that something has happened. This power can be used to force a Morale check among enemy (or friendly) forces. This effect lasts until the end of the battle, or for 1d6x10 minutes [4]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Empathy", | ||||
|     "description": "Psions may read a target’s emotions and subtly affect them. This includes changing a Near target’s NPC Reaction Roll result by one category up or down, player’s choice [4]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "False Perception", | ||||
|     "description": "The psion causes a single target within Near range to hallucinate. This effect lasts as long as the psion concentrates to maintain it. The details of the False Perception are left to the psion but can range from a normal illusion or distraction to something that closely resembles a drug-fueled fugue state. The target is permitted a Knowledge throw to shake off the effect after one round [5]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Hibernation", | ||||
|     "description": "The character may enter into a trance-like state and meditate for up to 7 days. While meditating, the psychic does not require food, water, or air. They remain aware of their surroundings and may end the hibernation at will [5]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Inspiration", | ||||
|     "description": "When the psion uses this power, they gain +1D to their next roll, regardless of what the roll is for. This power may only be used once per day [6]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Invisibility", | ||||
|     "description": "The psion is able to conceal themselves from nearby onlookers. While invisible, organic targets within Near range will not see the psion. This power works on electronic sensors and video cameras within Near range. The psion must concentrate for this power to be effective. Attacking any target immediately terminates this effect [6]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Kinetic Blast", | ||||
|     "description": "The psion generates a powerful if clumsy blast of telekinetic force! This invisible telekinetic bolt of force does 4D damage with AV 3D. The target may throw Physical (or Vehicles if in a vehicle) to completely avoid the attack [7]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Kinetic Shield", | ||||
|     "description": "This power grants the psion 8 points of armor Protection for the duration of the battle. The kinetic shield is not cumulative with worn armor [7]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Levitation", | ||||
|     "description": "The Psion may fly at a speed of one Move Action per round, for 5 rounds, in any direction desired [7]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Life Detection", | ||||
|     "description": "This power allows the psion to instinctively detect all living beings within a Far radius. This power cannot be used to pinpoint the exact whereabouts of a target but will give a general sense of the location of a hidden interlocutor or sniper [8]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Memory Block", | ||||
|     "description": "To use this power, the psion must be within Near range of a target. The target is allowed a Knowledge throw to avoid the effect. If the target fails the throw, the psion may block out up to 15 minutes of the target’s memories. The targeted memories must be targeted as a block for each use of this power. The memories can only be unblocked by another psion using Memory Block on the target [8, 9]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Microkinesis", | ||||
|     "description": "The psion gains telekinetic control of small or microscopic objects. Gain +2D on any one Technology throw involving components or systems containing small or microscopic objects [9]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Mind Blast", | ||||
|     "description": "This powerful telepathic blast renders a Near target unconscious for 10 minutes. The target is allowed a Knowledge throw to reduce this to being stunned (no actions) for one combat round [9]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Mind Control", | ||||
|     "description": "The psion gains control of a Near target, who becomes the psion’s ally for 15 minutes. The target may make a successful Knowledge throw to reduce the duration to 1D rounds. The target of Mind Control will only perform tasks a typical ally would do and will not perform any action that would harm themselves [10]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Mind Lock", | ||||
|     "description": "A Near target is paralyzed for 2d6 combat rounds and may not move. They may still speak. The target is allowed a Physical throw to resist this power [10]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Mind Shield", | ||||
|     "description": "The psion may erect an invisible mental shield that protects themselves and all allies within Near range any mind-affecting power or psionic ability. The Mind Shield will last for ten minutes [10]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Precognition", | ||||
|     "description": "This power allows the player to ask the Referee one question about what may happen to their character in the next 24 hours. The Referee must respond truthfully with either “yes” or “no.” Precognition may only be used once per day [11]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Psychic Healing", | ||||
|     "description": "This handy power allows the psion to heal 1 Wound on an injured target [11]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Pyrokinesis", | ||||
|     "description": "The psion may ignite a flammable object within a Near range. When used to directly attack a target, throw Combat to hit the target. A successful attack does 3D damage and has the Fire weapon Aspect. See the Environmental Hazard chapter for details on fire damage [11]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Regeneration", | ||||
|     "description": "This power allows a psychic to automatically treat a single Wound Triage Table result as though it were a result of 11: the wound will heal after 10 minutes of rest. Regeneration cannot be used on other targets but may be used after the results of the Triage Table roll are known. The psychic must be able to rest for ten minutes, and Regeneration only works on one wound at a time [12]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Second Wind", | ||||
|     "description": "This power allows the psion to ignore the effects of Fatigue for 1 hour [12]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Suggestion", | ||||
|     "description": "Psions may use this power to plant a simple hypnotic suggestion in one target. This effect will last until the target has performed their suggested task, or ten minutes have passed. Victims will not obey suggestions to harm themselves directly or indirectly [12, 13]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Telekinesis", | ||||
|     "description": "The psion may move objects up to 100kg by the power of their brain. Telekinetics may throw a heavy object at a target at Near range; such an attack requires a Combat roll and causes 4D damage, with AV 3D [13]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Telepathy", | ||||
|     "description": "This power allows the psion to read a Near target’s surface thoughts for one minute [13]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Thousand Faces", | ||||
|     "description": "The psion may change their appearance and biometric data to those of any target of the same or similar species for 10 minutes. To mimic the specific biometric data of another person, the psion must have a sample of the target’s DNA [13]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Truth", | ||||
|     "description": "This telepathic ability forces one Near target to answer questions truthfully. They must be able to speak or communicate to the psion, and the questions can be about anything: the target will fully answer to the best of their ability [14]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Vampire", | ||||
|     "description": "The psion must make a normal unarmed attack at Close range. The Vampire power inflicts a Wound on the target, and instantly heals a Wound the psion has suffered [14]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Zombie", | ||||
|     "description": "This gruesome power allows the psion to control one body of a creature killed in the past 24 hours. The target creature must be within Near range, and the power lasts for 5 combat rounds. The corpse may act, speak, and attack as the psion desires. The creature’s abilities might be limited depending on the amount and kinds of damage it suffered when it died [14]." | ||||
|   } | ||||
| ] | ||||
							
								
								
									
										268
									
								
								assets/json_data/starships.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,268 @@ | ||||
| [ | ||||
|   { | ||||
|     "name": "Orbital Transit Vehicle", | ||||
|     "agility": "+1", | ||||
|     "hullType": "Pod", | ||||
|     "endurance": "1 week Travel x2", | ||||
|     "armor": "2/4", | ||||
|     "crew": "1 (3)", | ||||
|     "cargo": "8", | ||||
|     "guns": "0", | ||||
|     "travelMultiplier": "x2", | ||||
|     "cost": 66000, | ||||
|     "monthlyCost": 4970, | ||||
|     "description": "The OTV (Orbital Transfer Vehicle) is a larger and slower version of a standard pod. It trades acceleration and travel velocity for a larger life module that holds a pilot and three passengers. It also has more Endurance, and more Cargo. OTV Pods are often used for moving cargo and passengers from planets to vessels in orbit. They are often carried by colony ships. Often, OTVs double as landers, and are able to convert into a shelter for its four occupants who rely on the OTV’s extended Endurance while waiting for resupply." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Defense Fighter", | ||||
|     "agility": "+4", | ||||
|     "hullType": "Micro", | ||||
|     "endurance": "1 Day Travel x2", | ||||
|     "armor": "14/17", | ||||
|     "crew": "1", | ||||
|     "cargo": "0.5", | ||||
|     "guns": "5d6", | ||||
|     "travelMultiplier": "x2", | ||||
|     "cost": 1120000, | ||||
|     "monthlyCost": 5300, | ||||
|     "description": "The Defense Fighter is a common sight on frontier worlds. Tough little vessels, these fighters have excellent acceleration, and the cutting edge FTL drive provides additional inertial dampening for truly breakneck maneuvers. The ship is otherwise incapable of FTL travel. The ship has minimal cargo space but carries decent guns making it a serious threat to small and medium-sized craft. Any colony world and many settled planets will have these ships on hand to deter pirate, and force down questionable ships for inspection." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Shuttle", | ||||
|     "agility": "+3", | ||||
|     "hullType": "Small", | ||||
|     "endurance": "1 Week", | ||||
|     "armor": "7/9", | ||||
|     "crew": "3 (7)", | ||||
|     "cargo": "8", | ||||
|     "guns": "2d6", | ||||
|     "travelMultiplier": "x1", | ||||
|     "cost": 2500000, | ||||
|     "monthlyCost": 22500, | ||||
|     "description": "A lightly armored and armed transfer vehicle. This shuttle is designed to move people and cargo to and from space. Shuttles take up a bit of cargo space, they are typically only found on larger vessels or based at a space station or spaceport. It lacks the armor or weapons for rougher frontier areas." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Tramp Freighter", | ||||
|     "agility": "0", | ||||
|     "hullType": "Scout", | ||||
|     "endurance": "1 Month", | ||||
|     "armor": "14/17", | ||||
|     "crew": "4 (6)", | ||||
|     "cargo": "64", | ||||
|     "guns": "4d6", | ||||
|     "travelMultiplier": "x1", | ||||
|     "cost": 6800000, | ||||
|     "monthlyCost": 39000, | ||||
|     "description": "The tramp freighter sacrifices Agility, Armor, and Guns for Cargo. Typically, a vessel like this can make a tidy profit of 6000-7000 credits a month, more if the crew/owners forego part of their salary to buy off the ship’s debt. This version retains some weapons and armor to deal with unsavory types and is suited to Frontier regions. Running cargo in risky systems can pay off: freight and passenger rates are liable to be two or three time the normal rate for emergency shipments via these dangerous systems." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Merchant", | ||||
|     "agility": "0", | ||||
|     "hullType": "Picket", | ||||
|     "endurance": "3 Months", | ||||
|     "armor": "18/21", | ||||
|     "crew": "10 (15)", | ||||
|     "cargo": "125", | ||||
|     "guns": "4d6", | ||||
|     "travelMultiplier": "x1", | ||||
|     "cost": 23800000, | ||||
|     "monthlyCost": 98000, | ||||
|     "description": "Merchant pickets are bought outright by mid-sized independent companies. They carry vital supplies to new colonies and outposts, something no large corporation wants to deal with. They are often stripped down former military ships. The largest hull capable of landing, they are very handy for frontier worlds. Many captains crew these ships at half strength to carry five more passengers and make an extra 40000 credits a month. This amounts to working the crew to the point of burn-out. Crew turnover on these vessels is exceedingly high." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Explorer", | ||||
|     "agility": "0", | ||||
|     "hullType": "Picket", | ||||
|     "endurance": "6 Months", | ||||
|     "armor": "18/21", | ||||
|     "crew": "10 (15)", | ||||
|     "cargo": "64", | ||||
|     "guns": "4d6", | ||||
|     "travelMultiplier": "x1", | ||||
|     "cost": 27700000, | ||||
|     "monthlyCost": 138500, | ||||
|     "description": "The Explorer is an under-gunned picket hull that possesses increased Endurance but reduces the armament even more. Propellant is sacrificed as well to allow a suite of advanced sensors. The 15 passengers are explorers and scientists. The shuttle can ferry field teams around a planet being investigated. Some missions forgo this for more cargo and simply land the ship." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Gunship", | ||||
|     "agility": "+1", | ||||
|     "hullType": "Picket", | ||||
|     "endurance": "3 Months", | ||||
|     "armor": "18/21", | ||||
|     "crew": "10 (15)", | ||||
|     "cargo": "32", | ||||
|     "guns": "6d6", | ||||
|     "travelMultiplier": "x1", | ||||
|     "cost": 28000000, | ||||
|     "monthlyCost": 98000, | ||||
|     "description": "The gunship is a vessel much-loved by pirates. Small enough to land on a planet for a raid of opportunity, yet with large enough guns to terrorize a small convoy or space station. Its light armor does mean that it must flee when larger escorts arrive. The 15 passengers comprise the boarding/looting party. The gunship is an all-too-common sight on the wild frontier." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Force Transport", | ||||
|     "agility": "0", | ||||
|     "hullType": "Destroyer", | ||||
|     "endurance": "6 Months", | ||||
|     "armor": "21/24", | ||||
|     "crew": "25 (75)", | ||||
|     "cargo": "177", | ||||
|     "guns": "6d6", | ||||
|     "travelMultiplier": "x1", | ||||
|     "cost": 102570000, | ||||
|     "monthlyCost": 331500, | ||||
|     "description": "The Force Transport is designed to deliver ground troops to serve as garrisons and pacify low-intensity conflicts: civil disorder, unrest, and similar situations. Its guns are supplemented by seven torpedoes to deal with space-borne threats. Torpedo reloads are stored as cargo. Since this hull is too large to land on a planet, three shuttles serve to deliver troops to all but the hottest landing zones." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Assault", | ||||
|     "agility": "0", | ||||
|     "hullType": "Destroyer", | ||||
|     "endurance": "3 Months", | ||||
|     "armor": "25/28", | ||||
|     "crew": "25", | ||||
|     "cargo": "7", | ||||
|     "guns": "8d6", | ||||
|     "travelMultiplier": "x1", | ||||
|     "cost": 145700000, | ||||
|     "monthlyCost": 264000, | ||||
|     "description": "The Assault ship is a powerful ship that is designed to do one thing: go into hot spots, fight the enemy, and win! Its powerful guns make it dangerous to smaller craft, and its compliment of torpedoes make it a threat to larger ships. This ship is common in all space forces anticipating sharp engagements. Force Transports are often paired with Assault ships: the Assault ships clear a path to a planet, and the Transports offload their troops to establish a beachhead." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Colonizer", | ||||
|     "agility": "-2", | ||||
|     "hullType": "Cruiser", | ||||
|     "endurance": "2 Years Travel x2", | ||||
|     "armor": "25/28", | ||||
|     "crew": "100 (1100)", | ||||
|     "cargo": "528", | ||||
|     "guns": "6d6", | ||||
|     "travelMultiplier": "x2", | ||||
|     "cost": 381400000, | ||||
|     "monthlyCost": 1532000, | ||||
|     "description": "Colonizer cruisers are the vanguard of colonization and immigration efforts. They carry the OTV pods and colonists to a hopefully promising planet and carries all the necessary supplies for the initial colony to be established. The colonists are transported in tight confines and spend their days training and preparing for their new lives. Some Colonizers carry cryo-tubes to transport even more colonists at a fraction of the life support cost. By the time the Colonizer arrives over the new planet, the crew will have programmed the on-board OTV pods with descent parameters. Occasionally a colonist will receive a week of training to fly the OTV pod. The pods descend, each with four passengers, and eight tons of supplies. The pods are designed to be dismantled and used to create permanent shelters. The colonizer remains in orbit for four to five months to assist the fledgling colony by using its shuttles to move additional colonists, supplies, and heavier equipment down to the planet. The ship has enough firepower to dissuade all but the most aggressive raiders. This is an older and slower colonizer. One thousand colonist are transported awake, while 800 more are in cryo-tubes!" | ||||
|   }, | ||||
|   { | ||||
|     "name": "Miner", | ||||
|     "agility": "-1", | ||||
|     "hullType": "Scout", | ||||
|     "endurance": "1 Month Travel x1.5", | ||||
|     "armor": "18/21", | ||||
|     "crew": "4", | ||||
|     "cargo": "32", | ||||
|     "guns": "2d6", | ||||
|     "travelMultiplier": "x1.5", | ||||
|     "cost": 7600000, | ||||
|     "monthlyCost": 32000, | ||||
|     "description": "The Mining Scout sacrifices Guns, travel speed and acceleration to mount mining gear and an enlarged cargo hold. It retains the Scout’s armor for protection against mining debris. The Scout is seldom troubled by pirates, who prefer looting a hold of luxury items to raw minerals. In areas where mining operations are harassed, these ships are guarded by interceptors. These ships regularly operate with military operations to provide fuel." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Escort Tender", | ||||
|     "agility": "0", | ||||
|     "hullType": "Picket", | ||||
|     "endurance": "1 Month Travel x0.75", | ||||
|     "armor": "18/21", | ||||
|     "crew": "19 (6)", | ||||
|     "cargo": "16", | ||||
|     "guns": "4d6", | ||||
|     "travelMultiplier": "x0.75", | ||||
|     "cost": 30520000, | ||||
|     "monthlyCost": 218800, | ||||
|     "description": "Escort tenders are the unglamorous workhorses of many navies, as they transport Defense Fighters, the agile combat craft with low Travel Multipliers, to a fight. Tenders are built around large propellant tanks, and very little else. The tenders will deploy their fighters and hang back while the fighters deploy to deal with threats. The six fighters’ pilots are considered passengers in this entry. Attacks that incapacitate crew if a tender while it is operating fighters will soon leave it under-crewed." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Express Picket", | ||||
|     "agility": "+2", | ||||
|     "hullType": "Picket", | ||||
|     "endurance": "1 Month Travel x1", | ||||
|     "armor": "21/24", | ||||
|     "crew": "4 Auto.", | ||||
|     "cargo": "32 (10 Cryo)", | ||||
|     "guns": "6d6", | ||||
|     "travelMultiplier": "x1", | ||||
|     "cost": 30800000, | ||||
|     "monthlyCost": 69000, | ||||
|     "description": "Express pickets are high speed craft that are designed to move vital information, supplies, and individuals over vast distances with very little external support. To reduce the logistical burden and simplify matters, many VIPs are placed in cryo-tubes for the entirety of the voyage. Express Pickets often possess redundant FTL Drives that permit two FTL voyages in rapid succession before requiring recharge. Another unique feature of this vessel is the use of an advanced Artificial Intelligence to reduce crew size to the bare minimum. Meanwhile, with such valuable components and the presence of sensitive information and persons, Express Pickets are armored and armed well enough to deter pirates." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Racer Rocket", | ||||
|     "agility": "+1", | ||||
|     "hullType": "Destroyer", | ||||
|     "endurance": "6 months Travel x0.5", | ||||
|     "armor": "7/9", | ||||
|     "crew": "25", | ||||
|     "cargo": "16", | ||||
|     "guns": "1d6", | ||||
|     "travelMultiplier": "x0.5", | ||||
|     "cost": 50000000, | ||||
|     "monthlyCost": 450000, | ||||
|     "description": "Competitive rocket racing is a popular and profitable sport, with many clubs and racing circuits across many sectors. The Racing Rocket is a standard high-speed, long-distance racer, though other types of rockets also exist for shorter races. This particular rocket is outfitted for dangerous and cutting-edge antimatter fuel. The Agility increases to +3 and the Travel Multiplier increases to x0.33 (faster!) when antimatter fuel is available. Racing rockets are also used as an emergency express courier to quickly send help or supplies to remote outposts in a given system." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Pleasure Craft", | ||||
|     "agility": "+1", | ||||
|     "hullType": "Picket", | ||||
|     "endurance": "3 months", | ||||
|     "armor": "18/21", | ||||
|     "crew": "10(20/70)", | ||||
|     "cargo": "16", | ||||
|     "guns": "4d6", | ||||
|     "travelMultiplier": "x1", | ||||
|     "cost": 33732000, | ||||
|     "monthlyCost": 167940, | ||||
|     "description": "Pleasure craft are popular among the elite and leaders of worlds. The ship is designed to carry many passengers. The ten crew are supported by another twenty stewards who tend to the seventy or so passengers. Of course, many of the passengers are themselves guards, agents, and other members of a wealthy VIP’s household or entourage. It is rare for these vessels to be attacked, but if an attack does occur, the ship uses its Agility to stay out of the field of fire, and launches its torpedoes before fleeing." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Refinery", | ||||
|     "agility": "-1", | ||||
|     "hullType": "Battleship", | ||||
|     "endurance": "6 Months Travel x2", | ||||
|     "armor": "32/35", | ||||
|     "crew": "300", | ||||
|     "cargo": "4000", | ||||
|     "guns": "6d6", | ||||
|     "travelMultiplier": "x2", | ||||
|     "cost": 4400000000, | ||||
|     "monthlyCost": 2364000, | ||||
|     "description": "A Refinery Ship costs is a large and costly vessel. Those that aren’t constructed outright by corporations or governments to serve as the lynchpins of resource exploitation projects are usually hobbled together by several clans of belters pooling together their hard-won resources. A Refinery ship will have huge cargo holds and often have an entire fleet of smaller mining ships operating out of its hangars. Refinery ships are heavily reinforced designs and are constructed to withstand the nearby explosion of an asteroid. They have enough defensive systems to carefully guard their resources. Many belter communities also hire mercenary ships to further bolster their refinery’s defenses. A refinery ship has never been successfully boarded or looted." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Prison Ship", | ||||
|     "agility": "+1", | ||||
|     "hullType": "Picket", | ||||
|     "endurance": "6 Months", | ||||
|     "armor": "18/21", | ||||
|     "crew": "10(15)", | ||||
|     "cargo": "22", | ||||
|     "guns": "5d6", | ||||
|     "travelMultiplier": "x1", | ||||
|     "cost": 28020000, | ||||
|     "monthlyCost": 116000, | ||||
|     "description": "This prison ship serves as a mobile base and prison for affiliated bounty hunters. Five of the passengers are guards, the rest prisoners. Many crews will store extra prisoners on ice in cryo-tubes in the hold. The ship sacrifices some armor and firepower for the larger life support system, and it cannot stand up to similar military ships. However, it remains a credible threat to most pirates who may want to mount a rescue. The guards on such a vessel are often replaced by robots, the likes of which are not likely to be bribed or coerced." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Torpedo Boat", | ||||
|     "agility": "+2", | ||||
|     "hullType": "Scout", | ||||
|     "endurance": "1 Week", | ||||
|     "armor": "18/21", | ||||
|     "crew": "4", | ||||
|     "cargo": "0", | ||||
|     "guns": "6d6", | ||||
|     "travelMultiplier": "x1", | ||||
|     "cost": 8000000, | ||||
|     "monthlyCost": 39000, | ||||
|     "description": "Torpedo Boats serve as a defensive vessel for mid-sized colonies that are occasionally the targets of larger pirate raids. The Torpedo Boat uses its high Agility to maintain an advantageous position, while launching torpedo after torpedo at pirate pickets. Torpedo Boats are often used with Defense Fighters or Gunships. They are also carried by larger ships like Cruisers." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Drop Ship", | ||||
|     "agility": "+3", | ||||
|     "hullType": "Scout", | ||||
|     "endurance": "1 Day", | ||||
|     "armor": "18/21", | ||||
|     "crew": "4(21)", | ||||
|     "cargo": "2", | ||||
|     "guns": "6d6", | ||||
|     "travelMultiplier": "x1", | ||||
|     "cost": 9600000, | ||||
|     "monthlyCost": 37175, | ||||
|     "description": "The Drop Ship is a large, militarized shuttle bristling with weapons. Its primary task is to deliver a platoon of 20 soldiers to a hostile planetary surface in very little time. Its engines give it excellent acceleration, hopefully enough to evade fire—any trooper will tell you that a Drop Ship ride is never a restful experience." | ||||
|   } | ||||
| ] | ||||
							
								
								
									
										270
									
								
								assets/json_data/talents.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,270 @@ | ||||
| [ | ||||
|   { | ||||
|     "name": "Armor Training", | ||||
|     "description": "This talent allows the character to use armor more effectively. The encumbrance values of all armors are reduced by 1 to a minimum of 0." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Artillery", | ||||
|     "description": "This character received advanced training in using heavy artillery. Usually, this means acting as a forward observer, but the character is also an expert in artillery gunnery, fuse setting, and demolishing fortifications with artillery. Gain +1D when firing artillery, or acting as a forward observer, or when conducting counter-battery fire." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Assault", | ||||
|     "description": "Assault specialists receive training in the use of heavy weapons. Gain +1D when using assault weapons, hunting tanks or other large vehicles, or when attacking prepared enemy fortifications." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Berserker", | ||||
|     "description": "When injured in combat but still conscious, the character may enter a Rage. This grants them +1D to hit and +1D to damage in melee combat, but also grants the character's enemies +1D to hit them. This lasts until the end of that combat encounter." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Blind Fighter", | ||||
|     "description": "The character can ignore the Penalty Die for fighting in darkness." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Bodyguard", | ||||
|     "description": "This talent allows the character to quickly move into harm’s way and take a hit meant for a comrade. This works as an interrupting action, and uses the bodyguard’s next Action in combat, if they survive." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Evasive Action", | ||||
|     "description": "This talent reflects innate luck, unconscious skill, or lightning fast reflexes. Once per battle, the character may avoid an otherwise successful attack and take no damage. They may elect to use this ability after damage is rolled. This talent can apply in vehicle combat as well as in personal combat." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Hardy", | ||||
|     "description": "The damage roll of the first (and only the first) successful attack on the character in a given battle incurs a -2D penalty." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Heavy Hitter", | ||||
|     "description": "Once per battle, you may opt to inflict maximum damage with an attack. You may use this ability after you have rolled damage." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Hunter", | ||||
|     "description": "The character is practiced at setting ambushes. Gain +1D on Surprise throws when attacking from ambush." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Leader", | ||||
|     "description": "Whether a natural, or an experienced, trained leader, this character knows how to motivate people into doing things they might not normally do. They gain +1D when attempting to rally troops and convince others to carry out their orders. As an action in Combat, they may roll Social 8+. If the roll is successful, the Leader generates Hero Points equal to the result of the roll, minus 7 (a result of 8 provides 1 Hero Point). These Hero Points can be distributed as a free action to comrades during the battle, but disappear when the fight is over." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Marksman", | ||||
|     "description": "The character is particularly adept at hitting where it counts most with ranged weapons. After spending one Hero Point, add +2D to damage rolls with ranged weapons." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Martial Artist", | ||||
|     "description": "Your character is a highly trained martial artist. Their feet and fists are lethal weapons. Add +1d6 to the damage rolls of all unarmed attacks." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Maverick", | ||||
|     "description": "A brash, risk-taking combatant, the Maverick has raw talent and little time for rules and personal safety. They may spend a Hero Point to perform an additional action in a combat round. They may use this ability as an interrupting action, and act out of initiative sequence. But beware: whenever they use this ability, enemies may target the character with +1D to hit in the next round." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Quick Draw", | ||||
|     "description": "If the character has a weapon at the ready—in a quick-draw holster or sheath, or just at hand—then they may make a rapid attack at the start of the first round of combat, before the initiative order. They incur one Penalty Die on the attack." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Relentless", | ||||
|     "description": "This talent allows the character to temporarily ignore the effects of one hit in combat. They may use this talent after they have learned what the damage might be. After the combat, roll on the Triage table." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Sneak Attack", | ||||
|     "description": "This character is well practiced at sneak attacks. When attacking a completely unaware target in melee, gain +1D to hit, and +1D to the attack's damage." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Tactics", | ||||
|     "description": "At the start of a fight, roll Knowledge 8+. The character’s training and drills provide a number of Advantage Dice equal to the result of the roll, minus 7 (with 8 providing +1D). These Advantage Dice can be spent by the character and their comrades at any time during the combat." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Two-Fisted", | ||||
|     "description": "The character may wield a one-handed weapon in each hand. They may attack with each of the two weapon once per round. Their second action in the round cannot be used to make another two attacks. The attack from the off-hand weapon is made with a -1D penalty. If the character is also Ambidextrous (see below), then the Penalty Die is removed." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Weak Point", | ||||
|     "description": "With training and experience, this character can spot an opening in an opponent’s armor or defensive posture. Throw Knowledge 12+ to grant you and your allies +1D to hit against a single enemy during the next round." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Whirlwind", | ||||
|     "description": "On the character’s turn, they may spend a Hero Point to make another melee attack that round." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Ambidextrous", | ||||
|     "description": "This character has an innate or learned ability to use their off-hand equally as well as their dominant hand. This is useful in case of injury, among other things. However, without the Two-Fisted Talent they can’t use a second weapon to attack with their off-hand." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Awareness", | ||||
|     "description": "Gain +1D on throws to avoid Surprise in combat. This does not refer to setting up ambushes, where the Hunter Talent applies." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Beast Friendship", | ||||
|     "description": "Gain +1D when identifying flora or fauna, as well as in Reaction throws with (non-magical) animals." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Big Mouth", | ||||
|     "description": "This character has a witty retort, insult, or inspiring word at the ready for all occasions. The character gains +1D whenever they are involved in an argument, debate, or they just want to aggravate or antagonize someone." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Charming", | ||||
|     "description": "This character’s natural habitat is a room full of people. Gain +1D when carousing, networking, or making a good impression in social settings." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Deduction", | ||||
|     "description": "Spend a Hero Point to ask the GM three yes-or-no questions about any person, place, or object you are investigating or observing." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Determined", | ||||
|     "description": "Gain one extra Hero Point at the start of each game session." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Eidetic Memory", | ||||
|     "description": "This character remembers everything their senses receive in raw excruciating detail. They gain +1D whenever they try to remember details of past events." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Empath", | ||||
|     "description": "This talent represents an innate ability to relate to other sentient beings. The character gains +1D on Reaction throws when interacting with humans and sentient aliens." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Endurance", | ||||
|     "description": "Your character is used to physical hardship. Ignore the first Penalty Die from fatigue, pain, Wounds (including the Physical roll to keep fighting), or other physical difficulties." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Pet", | ||||
|     "description": "The character has a special animal companion that is extremely loyal, surprisingly intelligent, but not particularly obedient. Distribute 4 points among Combat, Physical, Social, and Stealth skills, determine Stamina and Lifeblood normally, and select two innate Talents for it. The Referee has final say on what Talents are appropriate." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Immunity", | ||||
|     "description": "The character has a very strong immune system. They gain +1D on throws to resist the effects of poisons and disease." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Infuriating", | ||||
|     "description": "The character is a notorious troll. The character may taunt a target as an action: roll Social 8+. If successful, one target creature gains a Penalty Die on their actions for the next round." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Ingenious", | ||||
|     "description": "The character is talented at improvising with technology and their environmental surroundings in delightfully different ways. Gain +1D when the character uses tools or found objects in an interesting or creative manner." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Insightful", | ||||
|     "description": "Due to training or intuition, the character gains +1D when attempting to determine if someone is lying to them." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Inspiring", | ||||
|     "description": "As an Action in combat, they may offer a word of inspiration to a comrade, and grant +1D to whatever action the comrade is attempting." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Intimidating", | ||||
|     "description": "Gain +1D to rolls made to intimidate and terrify opponents." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Mighty", | ||||
|     "description": "The character is a physically powerful specimen. Gain +1D to all Physical skill rolls when attempting feats of strength and athleticism." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Runner", | ||||
|     "description": "The character gains +1D when engaging in foot races, sprints over obstacles, or other feats of athletic speed. This Talent does not apply to changing range bands in combat, unless the Referee allows it in specific situations. In cases where exact distances are required, the Runner talent adds 1d6 to the 10m of distance covered with each Move action. The d6 roll is an abstraction to take into account rougher terrain and unforeseen obstacles. If both Move actions are used to run, then the total distance is 20 + 2d6m that round." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Status", | ||||
|     "description": "This character has some sort of innate social status or rank that provides privileges. This may include special treatment, or higher level security access. The player and Referee should determine what the source of this status is." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Ace Pilot", | ||||
|     "description": "When piloting ships or other vehicles, this character gains +1D to Position rolls." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Acrobat", | ||||
|     "description": "The character is a highly trained gymnast and acrobat, and is comfortable operating at great heights. Gain +1D on Physical throws for tumbling, athletics, jumps, and landings." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Animal Handling", | ||||
|     "description": "Through training or natural affinity, the character is good with animals, and knows how to care for them. This also extends to riding animals. Gain +1D when dealing with animals, or when making throws to ride animals. This is cumulative with Beast Friendship." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Broker", | ||||
|     "description": "You have an expert eye for merchandise and deal making. Gain +1D when rolling to negotiate trade deals and haggle on prices." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Commando", | ||||
|     "description": "Commando training covers stealth and sabotage. The character gains +1D when infiltrating enemy positions and when setting explosives to destroy military targets like radar dishes and ammunition depots." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Con Artist", | ||||
|     "description": "The character is an accomplished liar, and has developed false identities. Gain +1D when acting as an imposter, or when running a confidence scheme." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Contortionism", | ||||
|     "description": "This character is trained in moving their body in very tight confines. The character may throw Physical 10+ to escape from bonds or manacles, or to slip through bars." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Demolitions Expert", | ||||
|     "description": "This talent grants the character +1D when setting explosives, carrying out controlled demolition operations of buildings, bridges, and other large structures, and disarming explosives." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Detective", | ||||
|     "description": "The character has a trained eye for details and is very familiar with investigative and forensic protocols. Gain +1D when looking for clues or conducting interrogations." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Drone Operator", | ||||
|     "description": "Whether a hard-working asteroid miner, a canny salvager, or a combat photographer, this character is a gifted drone operator. They gain +1D whenever they need to perform difficult tasks remotely with drones or other remotely controlled robots." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Gambler", | ||||
|     "description": "You know how to play games of chance very well. You also know how to cheat, and how to spot a cheater. Gain +1D for Knowledge throws to win a game. Gain +1D when using the Social skill to cheat, or to spot a cheater." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Ghost", | ||||
|     "description": "Whenever the character is attempting to sneak under cover of darkness, smoke, or other obscuring cover, they gain +1D." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Hacker", | ||||
|     "description": "Hackers are not just expert programmers. They are gifted improvisers who understand how to build workarounds on the fly, and make computers do their bidding. Gain +1D when attempting to hack a computer system, break encryption, jam sensors, or counter sensor jamming." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Intelligence Analyst", | ||||
|     "description": "This character has received specialized training in intelligence gathering and analysis. Whenever they are attempting to make sense of large amounts of data and determining what is actually going on, they gain +1D." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Logistics Expert", | ||||
|     "description": "This character has gained expertise in making plans and getting resources to where they have to be, on time. Gain +1D when drawing up plans, negotiating with suppliers, and ensuring that things run smoothly." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Medic", | ||||
|     "description": "This character has received enough life-saving medical training to act as a combat medic. They grant characters under their treatment with +1D on rolls on the Triage Table." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Professional", | ||||
|     "description": "This character had a well-developed career before becoming a Nomad. They have a non-combat professional background of some kind that might come in useful: administration, law, accounting, or the like. They gain +1D when practicing their profession." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Rigger", | ||||
|     "description": "You are an expert at quick battlefield repairs. Gain +1D when attempting to make a spot repair on a vehicle. The repair fail after the battle is over; more extensive repairs will have to be made later on." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Scholar", | ||||
|     "description": "Gain +1D when dealing in a specific field of knowledge, such as biology, physics, history, or mathematics. This includes research, experimentation, and drawing conclusions. This Talent can be taken multiple times, each applying to a different field of study." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Spacer", | ||||
|     "description": "Some individuals are born in outer space, while others become so acclimated that they could have been born there anyways. This character gains +1D when attempting physical skills in zero-gravity, or whenever they are operating in the vacuum of space." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Streetwise", | ||||
|     "description": "The character is wise to the criminal element wherever they find themselves. They gain +1D when attempting to fence stolen goods, find illegal work, or make connections with criminals." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Street Thief", | ||||
|     "description": "This character is a practiced petty criminal. Gain +1D whenever attempting to pick pockets, open locks, hotwire vehicles, or when fencing stolen goods." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Survivalist", | ||||
|     "description": "Gain +1D when making skill rolls involving wilderness survival, hunting, and overland navigation." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Tech Specialist", | ||||
|     "description": "Gain +1D when dealing with a specific technological field of knowledge: electronics, computers, vehicle mechanics, power systems, or weapons systems. This includes repairing devices and designing them." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Tinker", | ||||
|     "description": "This character spends all their free time tinkering with everyone’s gear. Once per game session, each player character gains +1D when using their personal equipment." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Untraceable", | ||||
|     "description": "You are especially well-practiced at passing through an area and covering your tracks. Anyone attempting to track you is penalized by -1D to the roll." | ||||
|   } | ||||
| ] | ||||
							
								
								
									
										114
									
								
								assets/json_data/vehicle_weapons.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,114 @@ | ||||
|  | ||||
|  | ||||
| [ | ||||
|   { | ||||
|     "name": "Autocannon, Light", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 10000, | ||||
|     "range": "HeavyWeapon", | ||||
|     "damage": "5d6", | ||||
|     "mag": 100, | ||||
|     "ammo_cost": 1000, | ||||
|     "aspects": ["Auto", "AV 3d6+1"], | ||||
|     "description": "This weapon can fire bursts of shells at high rates of fire. It is common to find this mounted on aircraft as a dogfighting weapon, as well as on light armored vehicles as a support weapon for infantry [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Autocannon, Heavy", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 20000, | ||||
|     "range": "HeavyWeapon", | ||||
|     "damage": "5d6", | ||||
|     "mag": 50, | ||||
|     "ammo_cost": 2000, | ||||
|     "aspects": ["Auto", "AV 3d6+2"], | ||||
|     "description": "A larger version of the light autocannon, often found on light armored vehicles, naval vessels, and ground attack aircraft [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Cannon, Light", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "cost": 15000, | ||||
|     "range": "HeavyWeapon", | ||||
|     "damage": "5d6", | ||||
|     "mag": 40, | ||||
|     "ammo_cost": 2000, | ||||
|     "aspects": ["AV 4d6+1"], | ||||
|     "description": "A multipurpose support weapon usually mounted on a vehicle or gun carriage [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Cannon, Heavy", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "cost": 30000, | ||||
|     "range": "HeavyWeapon", | ||||
|     "damage": "6d6", | ||||
|     "mag": 20, | ||||
|     "ammo_cost": 3000, | ||||
|     "aspects": ["AV 5d6"], | ||||
|     "description": "This cannon is a main battle tank’s primary weapon. It is often used as field artillery as well [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Heavy Flamer", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 1000, | ||||
|     "range": "Rifle", | ||||
|     "damage": "3d6", | ||||
|     "mag": 5, | ||||
|     "ammo_cost": 25, | ||||
|     "aspects": ["AV 2d6", "Fire"], | ||||
|     "description": "A brutal close-in anti-infantry weapon used to flush out fortifications and defend against infantry attack in built-up areas. Vehicles that carry flamethrowers are usually the first to be targeted by enemy infantry [1, 2]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Grenade Launcher", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 400, | ||||
|     "range": "Rifle", | ||||
|     "damage": "By gren.", | ||||
|     "mag": 50, | ||||
|     "ammo_cost": null, | ||||
|     "aspects": [], | ||||
|     "description": "A light support weapon mounted on many vehicles. This version has a 50 round belt magazine. Many vehicles mount smoke grenade launchers as defensive weapons [1, 2]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Guided Anti-Tank Missile", | ||||
|     "tech_age": "Early Atomic", | ||||
|     "cost": 4000, | ||||
|     "range": "HeavyWeapon", | ||||
|     "damage": "5d6", | ||||
|     "mag": 4, | ||||
|     "ammo_cost": 200, | ||||
|     "aspects": ["Blast", "AV 5d6"], | ||||
|     "description": "This vehicle mounted version of the portable anti-tank missile launcher has a magazine of four missiles. A system like this is often mounted on a light vehicle to give it significant anti-armor punch [1, 3]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Laser Cannon", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 10000, | ||||
|     "range": "HeavyWeapon", | ||||
|     "damage": "5d6", | ||||
|     "mag": 100, | ||||
|     "ammo_cost": 100, | ||||
|     "aspects": ["AV 4d6+2", "Laser"], | ||||
|     "description": "The Early Space Age’s replacement for all but the heaviest cannons, this vehicle weapon is highly efficient, and can be recharged from any power grid. This makes it the darling of logistics officers in any military where it is found [1, 3]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Laser, Support", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 5000, | ||||
|     "range": "HeavyWeapon", | ||||
|     "damage": "5d6", | ||||
|     "mag": 100, | ||||
|     "ammo_cost": 100, | ||||
|     "aspects": ["AV 3d6+1", "Laser"], | ||||
|     "description": "This vehicle mounted version of the portable support laser lacks auto fire capabilities but has a nearly inexhaustible ammunition supply [3, 4]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Rocket Launcher", | ||||
|     "tech_age": "Early Atomic", | ||||
|     "cost": 2000, | ||||
|     "range": "HeavyWeapon", | ||||
|     "damage": "4d6", | ||||
|     "mag": 1, | ||||
|     "ammo_cost": 150, | ||||
|     "aspects": ["Greater Blast", "AV 3d6"], | ||||
|     "description": "A portable rocket launcher utilizing unguided munitions. Rocket launchers are unsafe to fire in enclosed spaces. Rocket payloads are considered high explosive and are not particularly effective against armored vehicles. Armor piercing rockets cost 500 Credits per rocket: trade the Greater Blast for AV 4d6, Blast. Rocket launchers can also be mounted on vehicles, where they usually have 12 or even 24 shots before requiring reloading [4, 5]." | ||||
|   } | ||||
| ] | ||||
							
								
								
									
										254
									
								
								assets/json_data/vehicles.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,254 @@ | ||||
| [ | ||||
|   { | ||||
|     "name": "Ground Car", | ||||
|     "agility": 0, | ||||
|     "armor": "6/8", | ||||
|     "cargo": "250kg", | ||||
|     "crew": "1/4", | ||||
|     "force": 0, | ||||
|     "range": "400km", | ||||
|     "speed": "160km/h", | ||||
|     "techAge": "Early Mechanical", | ||||
|     "tonnage": "2 Tons", | ||||
|     "cost": 10000, | ||||
|     "description": "The ground car is popular with civilians on many worlds due to its rugged and simple construction. They are ubiquitous with models for sale or rental everywhere. Typically has no weapon." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Pickup Truck", | ||||
|     "agility": 0, | ||||
|     "armor": "6/8", | ||||
|     "cargo": "1 ton", | ||||
|     "crew": "1/6", | ||||
|     "force": 1, | ||||
|     "range": "400km", | ||||
|     "speed": "120km/h", | ||||
|     "techAge": "Early Mechanical", | ||||
|     "tonnage": "4 Tons", | ||||
|     "cost": 15000, | ||||
|     "description": "A bigger version of the ground car with a flat-bed cargo bay instead of passenger seating. Any passengers would sit in the open bay. This is a ubiquitous vehicle and has a notable reputation for being able to go anywhere, and in any condition. Many mercenary teams desperately get their hands on pickup trucks and fit any manner of weapons onto their flatbed. Some manufacturers are happy to see a new market for their vehicles. Others are less enthusiastic. Typically has no weapon." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Armored Personnel Carrier", | ||||
|     "agility": -1, | ||||
|     "armor": "12/16", | ||||
|     "cargo": "2 tons", | ||||
|     "crew": "2/8", | ||||
|     "force": 2, | ||||
|     "range": "400km", | ||||
|     "speed": "60km/h", | ||||
|     "techAge": "Late Mechanical", | ||||
|     "tonnage": "5 Tons", | ||||
|     "cost": 177250, | ||||
|     "description": "Introduced in the Mechanical Age, Armored Personnel Carriers are essential to modern warfare, allowing infantry to keep pace with armored units. Weapons include a Heavy Machinegun (Damage 5d6, Auto, AV 3d6). Some mount a Light Autocannon (Damage 5d6, Auto, AV 4d6)." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Light Tank", | ||||
|     "agility": -1, | ||||
|     "armor": "14/20", | ||||
|     "cargo": "1 ton", | ||||
|     "crew": "2/8", | ||||
|     "force": 2, | ||||
|     "range": "500km", | ||||
|     "speed": "60km/h", | ||||
|     "techAge": "Late Mechanical", | ||||
|     "tonnage": "10 Tons", | ||||
|     "cost": 177250, | ||||
|     "description": "The light tank is popular in various low-tech worlds for internal security and counter insurgency. In wartime they are used to reconnoiter for heavier armored vehicles. Later models include IR sensors and fire suppression systems for the crew. Weapons include a Heavy Machinegun (Dam 5d6, Auto, AV 3d6), Light Cannon (Dam 4d6, AV 4d6)." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Sports Car", | ||||
|     "agility": 0, | ||||
|     "armor": "5/7", | ||||
|     "cargo": "200kg", | ||||
|     "crew": "1/1", | ||||
|     "force": 1, | ||||
|     "range": "360km", | ||||
|     "speed": "240km/h", | ||||
|     "techAge": "Late Mechanical", | ||||
|     "tonnage": "2 Tons", | ||||
|     "cost": 14825, | ||||
|     "description": "The sports car is popular with the wealthy on many worlds due to its speed and comfort. Typically has no weapon." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Explorer", | ||||
|     "agility": -2, | ||||
|     "armor": "7/10", | ||||
|     "cargo": "1500kg", | ||||
|     "crew": "2/2", | ||||
|     "force": 2, | ||||
|     "range": "Unlimited", | ||||
|     "speed": "60km/h", | ||||
|     "techAge": "Early Interstellar", | ||||
|     "tonnage": "10 Tons", | ||||
|     "cost": 159300, | ||||
|     "description": "Explorers are popular ground vehicles in Early Interstellar Age cultures. It is powered by a Fusion pack which provides effectively unlimited range. The Explorer carries a driver, a sensor technician, and two passengers in cramped seats. The sensor operator has dual controls for driving. Two bunks allow sleeping in rotation to recover from fatigue. Life Support lasts indefinitely. The ample cargo bay, rest accommodations, and amenities make the Explorer very popular with survey crews. Sensors: IR, Radar, Ladar, Dynamometer. Has no weapon." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Hovercraft", | ||||
|     "agility": -1, | ||||
|     "armor": "6/8", | ||||
|     "cargo": "1750kg", | ||||
|     "crew": "1/5", | ||||
|     "force": 1, | ||||
|     "range": "400km", | ||||
|     "speed": "160km/h", | ||||
|     "techAge": "Early Atomic", | ||||
|     "tonnage": "10 Tons", | ||||
|     "cost": 40800, | ||||
|     "description": "Hovercraft can travel over virtually any terrain or water. This civilian hovercraft has an unarmored chassis, and transports five passengers in cramped seats alongside the driver. This vehicle also has a waterproof body, allowing it to float. Has no weapon." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Cabin Cruiser", | ||||
|     "agility": -2, | ||||
|     "armor": "6/8", | ||||
|     "cargo": "2000kg", | ||||
|     "crew": "2/3", | ||||
|     "force": 2, | ||||
|     "range": "1080km", | ||||
|     "speed": "15 km/h", | ||||
|     "techAge": "Late Mechanical", | ||||
|     "tonnage": "10 Tons", | ||||
|     "cost": 25100, | ||||
|     "description": "The Cabin Cruiser serves for fishing and other leisure activities. It also can be fitted with weapons and used for security duties (shown in the example). The below-deck area is popular with crew, though a galley is not provided. A skipper, gunner, and three passengers are carried above deck. There are two bunks below-decks. The civilian version removes the weapon station and adds 0.5 tons of cargo and costs 13,600 Credits. Weapon is GP Machinegun (Damage 3d6+3, Auto, AV 2d6)." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Speedboat", | ||||
|     "agility": 0, | ||||
|     "armor": "6/8", | ||||
|     "cargo": "750kg", | ||||
|     "crew": "1/4", | ||||
|     "force": 0, | ||||
|     "range": "600km", | ||||
|     "speed": "120km/h", | ||||
|     "techAge": "Late Mechanical", | ||||
|     "tonnage": "2 Tons", | ||||
|     "cost": 14750, | ||||
|     "description": "The Speed Boat was designed for harbor and coastal patrol, as well as for leisure activities during the Late Mechanical Age. There is no below deck area. A skipper and four passengers are carried above deck. Has no weapon." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Mini-Sub", | ||||
|     "agility": -3, | ||||
|     "armor": "9/12 (pressure hull)", | ||||
|     "cargo": "1150kg", | ||||
|     "crew": "1/4", | ||||
|     "force": 1, | ||||
|     "range": "1080km", | ||||
|     "speed": "40km/h surface, 20km/h submerged", | ||||
|     "techAge": "Late Mechanical", | ||||
|     "tonnage": "10 Tons", | ||||
|     "cost": 358000, | ||||
|     "description": "This mini-sub can carry a pilot and four passengers in cramped seats. It has no airlock. Underwater, the mini-sub travels at 20kph up to two hours before having to surface and recharge its batteries. It has a maximum diving depth of 100 meters. Has no weapon." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Biplane", | ||||
|     "agility": 1, | ||||
|     "armor": "6/9", | ||||
|     "cargo": "Negligible", | ||||
|     "crew": "1/1", | ||||
|     "force": 0, | ||||
|     "range": "1000km", | ||||
|     "speed": "250km/h", | ||||
|     "techAge": "Late Mechanical", | ||||
|     "tonnage": "2 Tons", | ||||
|     "cost": 61000, | ||||
|     "description": "Biplanes are simple aircraft, using twin wings to generate lift. They have a maximum speed of 250km/h, and a Stall Speed of 80km/h. Armed variants carry machineguns, while unarmed variants gain 100kg of cargo space. Weapon is GP Machinegun (Damage 3d6+3, Auto, AV2d6)." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Propeller Plane", | ||||
|     "agility": 1, | ||||
|     "armor": "7/10", | ||||
|     "cargo": "500kg", | ||||
|     "crew": "2/3", | ||||
|     "force": 0, | ||||
|     "range": "2000km", | ||||
|     "speed": "400km/h", | ||||
|     "techAge": "Late Mechanical", | ||||
|     "tonnage": "10 Tons", | ||||
|     "cost": 166000, | ||||
|     "description": "The propeller-driven monoplane soon supersedes the biplane in the Late Mechanical Age. This variant is used for urgent errands on frontier planets, and can carry a pilot, co-pilot, and four passengers (one being a gunner). This aircraft has a stall speed of 120km/h and requires a landing strip. Weapon is GP Machinegun (Damage 3d6+3, Auto, AV2d6)." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Cargo Plane", | ||||
|     "agility": 0, | ||||
|     "armor": "8/11", | ||||
|     "cargo": "4000kg", | ||||
|     "crew": "2/20", | ||||
|     "force": 1, | ||||
|     "range": "2000km", | ||||
|     "speed": "400km/h", | ||||
|     "techAge": "Late Mechanical", | ||||
|     "tonnage": "20 Tons", | ||||
|     "cost": 250000, | ||||
|     "description": "This is a medium sized cargo plane driven by two, three, or four propellers. It is very reliable and rugged and can be found on any world where aircraft are useful and cheap to operate. Has no weapon." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Helicopter", | ||||
|     "agility": 1, | ||||
|     "armor": "7/10", | ||||
|     "cargo": "500kg", | ||||
|     "crew": "1/1", | ||||
|     "force": 1, | ||||
|     "range": "2100km", | ||||
|     "speed": "250km/h", | ||||
|     "techAge": "Late Mechanical", | ||||
|     "tonnage": "10 Tons", | ||||
|     "cost": 193500, | ||||
|     "description": "Introduced in the Late Mechanical Age, the Helicopter is the first heavier-than-air aircraft capable of hovering and vertical flight. It is popular on Earth-like worlds for search and rescue, law enforcement, and supply runs to remote settlements since it doesn’t need a landing strip. Has no weapon." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Jump Jet", | ||||
|     "agility": 1, | ||||
|     "armor": "7/10", | ||||
|     "cargo": "1250kg", | ||||
|     "crew": "2/3", | ||||
|     "force": 1, | ||||
|     "range": "2500km", | ||||
|     "speed": "800km/h", | ||||
|     "techAge": "Early Atomic", | ||||
|     "tonnage": "12 Tons", | ||||
|     "cost": 656000, | ||||
|     "description": "Jump Jets became feasible in the Early Atomic Age. They provide the high speed and high-altitude flight of normal jet aircraft, as well as the hovering, take-off, and landing capabilities of a helicopter. This version carries a pilot, flight engineer/electronics specialist, and three passengers. A commercial passenger version removes the advanced sensors and carries a pilot and 8 passengers with 1.5 tons of cargo. It costs 647000 Credits. Has no weapon." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Sky Car", | ||||
|     "agility": 2, | ||||
|     "armor": "7/9", | ||||
|     "cargo": "500kg", | ||||
|     "crew": "1/3", | ||||
|     "force": 1, | ||||
|     "range": "8000km", | ||||
|     "speed": "800km/h", | ||||
|     "techAge": "Early Space", | ||||
|     "tonnage": "5 Tons", | ||||
|     "cost": 170000, | ||||
|     "description": "This Space Age Skycar or “Buzzer” is a small, lightly armored vehicle. Skycars are often used by the authorities, emergency services, and the super-rich. This vehicle can hover in place. It holds a pilot and three passengers in cramped seats. Has no weapon (as per its listed stats, though an example later states it can be equipped with a Support Laser)." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Jet Fighter", | ||||
|     "agility": 1, | ||||
|     "armor": "7/10", | ||||
|     "cargo": "Nil", | ||||
|     "crew": "2/0", | ||||
|     "force": 1, | ||||
|     "range": "5000km", | ||||
|     "speed": "2000km/h", | ||||
|     "techAge": "Early Atomic", | ||||
|     "tonnage": "10 Tons", | ||||
|     "cost": 454000, | ||||
|     "description": "Jet fighters are a crucial defense against raiders and invasion. First introduced in the Early Atomic Age, they continue to improve until the Interstellar Age and are vital to planets with little or no space presence. This fighter has a crew of two: a pilot and electronics specialist, who both must wear oxygen masks. Jet fighters require extensive landing strips and maintenance to keep them flying. The aircraft’s stall speed is 300km/h, and they cannot reach their top speed without being at very high altitudes. Sensors: IR, Radar. Weapons include Light Autocannon (5d6, Auto, AV 3d6+1), 2x Anti-Tank Missile (5d6, Blast, AV 5d6), 2x Anti-Aircraft Missile (4d6, Blast, AV 3d6)." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Grav Car", | ||||
|     "agility": 2, | ||||
|     "armor": "7/9", | ||||
|     "cargo": "1750kg", | ||||
|     "crew": "1/5", | ||||
|     "force": 1, | ||||
|     "range": "Unlimited", | ||||
|     "speed": "1000km/h", | ||||
|     "techAge": "Late Galactic", | ||||
|     "tonnage": "5 Tons", | ||||
|     "cost": 170000, | ||||
|     "description": "The Grav Car is civilian transportation in the Late Galactic Age. It is built on a small, unarmored chassis. Its gravity drive gives it an impressive top speed and allows hovering or vertical flight. The grav car can reach orbit in 1d6 hours. Its cabin is sealed. Has no weapon." | ||||
|   } | ||||
| ] | ||||
							
								
								
									
										112
									
								
								assets/json_data/vehicleweapons.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,112 @@ | ||||
| [ | ||||
|   { | ||||
|     "name": "Autocannon, Light", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 10000, | ||||
|     "range": "heavy", | ||||
|     "damage": "5d6", | ||||
|     "mag": 100, | ||||
|     "ammo_cost": 1000, | ||||
|     "aspect": "Auto, AV 3d6+1.", | ||||
|     "description": "This weapon can fire bursts of shells at high rates of fire. It is common to find this mounted on aircraft as a dogfighting weapon, as well as on light armored vehicles as a support weapon for infantry." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Autocannon, Heavy", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 20000, | ||||
|     "range": "heavy", | ||||
|     "damage": "5d6", | ||||
|     "mag": 50, | ||||
|     "ammo_cost": 2000, | ||||
|     "aspect": "Auto, AV 3d6+2.", | ||||
|     "description": "A larger version of the light autocannon, often found on light armored vehicles, naval vessels, and ground attack aircraft." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Cannon, Light", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "cost": 15000, | ||||
|     "range": "heavy", | ||||
|     "damage": "5d6", | ||||
|     "mag": 40, | ||||
|     "ammo_cost": 2000, | ||||
|     "aspect": "AV 4d6+1.", | ||||
|     "description": "A multipurpose support weapon usually mounted on a vehicle or gun carriage." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Cannon, Heavy", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "cost": 30000, | ||||
|     "range": "heavy", | ||||
|     "damage": "6d6", | ||||
|     "mag": 20, | ||||
|     "ammo_cost": 3000, | ||||
|     "aspect": "AV 5d6.", | ||||
|     "description": "This cannon is a main battle tank’s primary weapon. It is often used as field artillery as well." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Heavy Flamer", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 1000, | ||||
|     "range": "Rifle", | ||||
|     "damage": "3d6", | ||||
|     "mag": 5, | ||||
|     "ammo_cost": 25, | ||||
|     "aspect": "AV 2d6, Fire. ", | ||||
|     "description": "A brutal close-in anti-infantry weapon used to flush out fortifications and defend against infantry attack in built-up areas. Vehicles that carry flamethrowers are usually the first to be targeted by enemy infantry." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Grenade Launcher", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "cost": 400, | ||||
|     "range": "Rifle", | ||||
|     "damage": "According to grenade", | ||||
|     "mag": 50, | ||||
|     "ammo_cost": 0, | ||||
|     "aspect": "", | ||||
|     "description": "A light support weapon mounted on many vehicles. This version has a 50 round belt magazine. Many vehicles mount smoke grenade launchers as defensive weapons." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Guided Anti-Tank Missile", | ||||
|     "tech_age": "Early Atomic", | ||||
|     "cost": 4000, | ||||
|     "range": "heavy", | ||||
|     "damage": "5d6", | ||||
|     "mag": 4, | ||||
|     "ammo_cost": 200, | ||||
|     "aspect": "Blast, AV 5d6. ", | ||||
|     "description": "This vehicle mounted version of the portable anti-tank missile launcher has a magazine of four missiles. A system like this is often mounted on a light vehicle to give it significant anti-armor punch." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Laser Cannon", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 10000, | ||||
|     "range": "heavy", | ||||
|     "damage": "5d6", | ||||
|     "mag": 100, | ||||
|     "ammo_cost": 100, | ||||
|     "aspect": "AV 4d6+2.", | ||||
|     "description": "The Early Space Age’s replacement for all but the heaviest cannons, this vehicle weapon is highly efficient, and can be recharged from any power grid. This makes it the darling of logistics officers in any military where it is found." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Laser, Support", | ||||
|     "tech_age": "Early Space", | ||||
|     "cost": 5000, | ||||
|     "range": "heavy", | ||||
|     "damage": "5d6", | ||||
|     "mag": 100, | ||||
|     "ammo_cost": 100, | ||||
|     "aspect": "AV 3d6+1.", | ||||
|     "description": "This vehicle mounted version of the portable support laser lacks auto fire capabilities but has a nearly inexhaustible ammunition supply." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Rocket Launcher", | ||||
|     "tech_age": "Early Atomic", | ||||
|     "cost": 2000, | ||||
|     "range": "heavy", | ||||
|     "damage": "4d6", | ||||
|     "mag": 1, | ||||
|     "ammo_cost": 150, | ||||
|     "aspect": "Greater Blast, AV 3d6.", | ||||
|     "description": "A portable rocket launcher utilizing unguided munitions. Rocket payloads are considered high explosive and are not particularly effective against armored vehicles. Vehicle-mounted versions usually have 12 or 24 shots before requiring reloading." | ||||
|   } | ||||
| ] | ||||
							
								
								
									
										255
									
								
								assets/json_data/weapon_distance.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,255 @@ | ||||
| [ | ||||
|  { | ||||
|     "name": "Anti-Materiel Rifle", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "range": "Rifle", | ||||
|     "cost": 1200, | ||||
|     "mag": 5, | ||||
|     "ammo_cost": 100, | ||||
|     "damage": "5d6", | ||||
|     "aspects": ["Scope", "AV 2d6"], | ||||
|     "description": "A rifle that fires re-purposed Heavy Machinegun or light cannon ammunition. Some are used as sniper rifles among elite soldiers." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Assault Rifle", | ||||
|     "tech_age": "Early Atomic", | ||||
|     "range": "Assault", | ||||
|     "cost": 300, | ||||
|     "mag": 30, | ||||
|     "ammo_cost": 15, | ||||
|     "damage": "3d6", | ||||
|     "aspects": ["Auto"], | ||||
|     "description": "The standard military weapon on mid-tech worlds. An automatic rifle that fires an intermediate caliber. At the Referee's discretion, its ammunition, or even magazines, may be interchangeable with those of a Carbine." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Bow", | ||||
|     "tech_age": "Early Primitive", | ||||
|     "range": "Rifle", | ||||
|     "cost": 60, | ||||
|     "mag": 1, | ||||
|     "ammo_cost": 10, | ||||
|     "damage": "2d6", | ||||
|     "aspects": [], | ||||
|     "description": "These are muscle-powered ranged weapons. Arrows for a bow or bolts for a crossbow are stored in a quiver of 20, which counts as one encumbrance item. A Bow may fire once per Action." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Crossbow", | ||||
|     "tech_age": "Late Primitive", | ||||
|     "range": "Rifle", | ||||
|     "cost": 75, | ||||
|     "mag": 1, | ||||
|     "ammo_cost": 10, | ||||
|     "damage": "3d6", | ||||
|     "aspects": [], | ||||
|     "description": "These are muscle-powered ranged weapons. Arrows for a bow or bolts for a crossbow are stored in a quiver of 20, which counts as one encumbrance item. A Crossbow requires 2 actions to reload." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Carbine", | ||||
|     "tech_age": "Early Atomic", | ||||
|     "range": "Assault", | ||||
|     "cost": 400, | ||||
|     "mag": 20, | ||||
|     "ammo_cost": 15, | ||||
|     "damage": "3d6", | ||||
|     "aspects": ["Auto"], | ||||
|     "description": "This weapon is popular among vehicle crews and light infantry due to its handiness and light weight. Carbines often fire regular assault rifle ammunition, but some fire pistol ammunition. Carbines count as one encumbrance item." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Gyrojet Rifle", | ||||
|     "tech_age": "Early Space", | ||||
|     "range": "Assault", | ||||
|     "cost": 900, | ||||
|     "mag": 20, | ||||
|     "ammo_cost": 15, | ||||
|     "damage": "3d6+2", | ||||
|     "aspects": ["Auto", "Zero-G"], | ||||
|     "description": "Designed for use in zero-G environment, gyrojet weapons fire small, self-propelled rocket rounds. They have minimal recoil and work in vacuum as well." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Gyrojet Pistol", | ||||
|     "tech_age": "Early Space", | ||||
|     "range": "Handgun", | ||||
|     "cost": 400, | ||||
|     "mag": 8, | ||||
|     "ammo_cost": 10, | ||||
|     "damage": "2d6+2", | ||||
|     "aspects": ["Zero-G"], | ||||
|     "description": "Designed for use in zero-G environment, gyrojet weapons fire small, self-propelled rocket rounds. They have minimal recoil and work in vacuum as well." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Light Machinegun", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "range": "Rifle", | ||||
|     "cost": 1200, | ||||
|     "mag": 100, | ||||
|     "ammo_cost": 125, | ||||
|     "damage": "3d6", | ||||
|     "aspects": ["Auto", "Bulky"], | ||||
|     "description": "This is the standard mid-tech squad-level support automatic weapon and is typically belt-fed. Light machine guns count as two items for encumbrance purposes." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Pulse Rifle", | ||||
|     "tech_age": "Late Space", | ||||
|     "range": "Assault", | ||||
|     "cost": 1000, | ||||
|     "mag": 40, | ||||
|     "ammo_cost": 30, | ||||
|     "damage": "3d6+3", | ||||
|     "aspects": ["Auto"], | ||||
|     "description": "An advanced automatic rifle that fires electrically ignited, caseless, high explosive, armor-piercing rounds. The ultimate in projectile weaponry. The pulse rifle provides a high rate of fire and a large magazine capacity." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Revolver", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "range": "Handgun", | ||||
|     "cost": 150, | ||||
|     "mag": 6, | ||||
|     "ammo_cost": 5, | ||||
|     "damage": "2d6+1", | ||||
|     "aspects": [], | ||||
|     "description": "A simple handgun that uses a manually loaded rotating drum to feed the action. Revolvers are reliable weapons." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Rifle", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "range": "Rifle", | ||||
|     "cost": 500, | ||||
|     "mag": 8, | ||||
|     "ammo_cost": 15, | ||||
|     "damage": "3d6+3", | ||||
|     "aspects": [], | ||||
|     "description": "The standard military firearm on low-tech worlds. Rifles use larger caliber projectiles and are common hunting weapons on the interstellar frontier." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Semi-Auto Pistol", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "range": "Handgun", | ||||
|     "cost": 150, | ||||
|     "mag": 16, | ||||
|     "ammo_cost": 7, | ||||
|     "damage": "2d6", | ||||
|     "aspects": [], | ||||
|     "description": "This is a handgun fed from a magazine. One pull of the trigger fires one bullet. At the Referee's discretion, its ammunition, or even magazines, may be interchangeable with those of a Submachine Gun." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Shotgun", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "range": "Assault", | ||||
|     "cost": 200, | ||||
|     "mag": 6, | ||||
|     "ammo_cost": 10, | ||||
|     "damage": "4d6", | ||||
|     "aspects": ["Scatter."], | ||||
|     "description": "A smoothbore weapon that fires shot pellets capable of hitting multiple targets at once. Shotguns are effective short-ranged weapons, but are not useful beyond Near Range, or against armored targets. These particulate stats represent a pump-action weapon with an internal magazine." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Sniper Rifle", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "range": "Rifle", | ||||
|     "cost": 800, | ||||
|     "mag": 5, | ||||
|     "ammo_cost": 30, | ||||
|     "damage": "3d6+3", | ||||
|     "aspects": ["Scope"], | ||||
|     "description": "This rifle is a higher quality version of the standard rifle, with better optics. It fires higher quality ammunition as well." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Stealth Pistol", | ||||
|     "tech_age": "Early Space", | ||||
|     "range": "Handgun", | ||||
|     "cost": 500, | ||||
|     "mag": 6, | ||||
|     "ammo_cost": 20, | ||||
|     "damage": "2d6", | ||||
|     "aspects": [], | ||||
|     "description": "This is an especially light, discreet, and concealable weapon made from advanced polymers. A stealth pistol can also be disguised as more innocuous items, like comms, digital recorders, and so on." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Submachine Gun", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "range": "Assault", | ||||
|     "cost": 500, | ||||
|     "mag": 30, | ||||
|     "ammo_cost": 20, | ||||
|     "damage": "2d6", | ||||
|     "aspects": ["Auto"], | ||||
|     "description": "This is a light automatic weapon that fires pistol rounds. It is intended for close quarters combat and counts as one item for encumbrance purposes." | ||||
|   }, | ||||
|    { | ||||
|     "name": "Blaster Pistol", | ||||
|     "tech_age": "Late Interstellar", | ||||
|     "range": "Handgun", | ||||
|     "cost": 5000, | ||||
|     "mag": 20, | ||||
|     "ammo_cost": 100, | ||||
|     "damage": "3d6+2", | ||||
|     "aspects": ["Laser", "AV 1d6"], | ||||
|     "description": "Blasters are advanced laser weapons with greater power and penetration. Their power packs are rechargeable from any major power source, such as a ship’s power plant [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Blaster Rifle", | ||||
|     "tech_age": "Late Interstellar", | ||||
|     "range": "Rifle", | ||||
|     "cost": 10000, | ||||
|     "mag": 50, | ||||
|     "ammo_cost": 500, | ||||
|     "damage": "4d6+3", | ||||
|     "aspects": ["Auto", "Laser", "AV 2d6"], | ||||
|     "description": "Blasters are advanced laser weapons with greater power and penetration. Their power packs are rechargeable from any major power source, such as a ship’s power plant [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Laser Pistol", | ||||
|     "tech_age": "Late Space", | ||||
|     "range": "Handgun", | ||||
|     "cost": 1500, | ||||
|     "mag": 10, | ||||
|     "ammo_cost": 50, | ||||
|     "damage": "3d6", | ||||
|     "aspects": ["Laser"], | ||||
|     "description": "A handgun-sized directed energy weapon. Its power pack is rechargeable from any major power source, such as a ship’s power plant [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Laser Rifle", | ||||
|     "tech_age": "Early Space", | ||||
|     "range": "Rifle", | ||||
|     "cost": 3500, | ||||
|     "mag": 20, | ||||
|     "ammo_cost": 100, | ||||
|     "damage": "4d6", | ||||
|     "aspects": ["Laser", "AV 1d6"], | ||||
|     "description": "A long-range version of the laser pistol, with better armor penetration, range, and even light anti-vehicular capabilities [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Plasma Rifle", | ||||
|     "tech_age": "Early Galactic", | ||||
|     "range": "Rifle", | ||||
|     "cost": 15000, | ||||
|     "mag": 10, | ||||
|     "ammo_cost": 500, | ||||
|     "damage": "6d6", | ||||
|     "aspects": ["Fire", "AV 3d6"], | ||||
|     "description": "The ultimate small arms, using miniaturized plasma gun technology to propel magnetically shaped pockets of super-hot ionized plasma at its targets [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Plasma Pistol", | ||||
|     "tech_age": "Late Galactic", | ||||
|     "range": "Handgun", | ||||
|     "cost": 10000, | ||||
|     "mag": 6, | ||||
|     "ammo_cost": 200, | ||||
|     "damage": "5d6", | ||||
|     "aspects": ["Fire", "AV 3d6"], | ||||
|     "description": "The ultimate small arms, using miniaturized plasma gun technology to propel magnetically shaped pockets of super-hot ionized plasma at its targets [1]." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Stunner", | ||||
|     "tech_age": "Early Interstellar", | ||||
|     "range": "Assault", | ||||
|     "cost": 600, | ||||
|     "mag": 10, | ||||
|     "ammo_cost": 10, | ||||
|     "damage": "3d6+1", | ||||
|     "aspects": ["Stun", "Zero-G"], | ||||
|     "description": "Firing a precise electromagnetic pulse, this non-lethal weapon disrupts the target's neural processes, leading to incapacitation and potentially unconsciousness. The stunner counts as one item for encumbrance purposes [2]." | ||||
|   } | ||||
| ] | ||||
							
								
								
									
										123
									
								
								assets/json_data/weapon_heavy.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,123 @@ | ||||
| [ | ||||
|   { | ||||
|     "name": "Anti-Aircraft Rocket Launcher", | ||||
|     "tech_age": "Early Atomic", | ||||
|     "range": "HeavyWeapon", | ||||
|     "cost": 10000, | ||||
|     "mag": 1, | ||||
|     "ammo_cost": 500, | ||||
|     "damage": "4d6", | ||||
|     "aspects": ["AV 3d6"], | ||||
|     "description": "This is a portable anti-air defense system designed to shoot down enemy aircraft with a self-guided rocket. It is not particularly effective against armored vehicles: reduce AV dice to 2d6." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Disposable Anti-Vehicle Rocket", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "range": "Assault", | ||||
|     "cost": 200, | ||||
|     "mag": 1, | ||||
|     "ammo_cost": null, | ||||
|     "damage": "4d6", | ||||
|     "aspects": ["Blast", "Bulky", "AV 4d6"], | ||||
|     "description": "A cheap, single-use, portable rocket launcher. These are often the only anti-tank weapons available to infantry. Note that this weapon cannot make attacks at Close range, despite using the Assault weapon range modifiers." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Flamer", | ||||
|     "tech_age": "Late Mech.", | ||||
|     "range": "Assault", | ||||
|     "cost": 1000, | ||||
|     "mag": 5, | ||||
|     "ammo_cost": 25, | ||||
|     "damage": "3d6", | ||||
|     "aspects": ["Bulky", "Fire"], | ||||
|     "description": "These terrifying weapons spew a stream of fire 3m wide at its far end. All targets within this cone suffer damage and catch fire (see the Environmental Hazards rules). The tank for the flamethrower itself holds enough fuel for five full attacks and counts as two items for encumbrance." | ||||
|   }, | ||||
|   { | ||||
|     "name": "General Purpose Machinegun", | ||||
|     "tech_age": "Late Mechanical", | ||||
|     "range": "Rifle", | ||||
|     "cost": 1500, | ||||
|     "mag": 100, | ||||
|     "ammo_cost": 150, | ||||
|     "damage": "3d6+3", | ||||
|     "aspects": ["Auto", "Bulky", "AV 2d6"], | ||||
|     "description": "A belt-fed medium machine gun. These are often mounted on vehicles but are also portable, although they are 3 items of encumbrance. Usually they are crew-served." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Grav Launcher", | ||||
|     "tech_age": "Late Interstellar", | ||||
|     "range": "HeavyWeapon", | ||||
|     "cost": 20000, | ||||
|     "mag": 1, | ||||
|     "ammo_cost": 100, | ||||
|     "damage": "6d6", | ||||
|     "aspects": ["Greater Blast", "Bulky", "Fire", "AV 5d6"], | ||||
|     "description": "Advanced micro-gravitic technology allows this weapon to guide a floating plasma bomb to its target without having to account for ballistic trajectories or inertia. Once launched, the bomb travels 50m per combat round. Its operator can move it in any direction and can also stop it and have it hover in mid-air. Detonation is on command. The bomb's internal battery allows for 10 rounds of gravitic flight." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Guided Anti-Tank Missile", | ||||
|     "tech_age": "Early Atomic", | ||||
|     "range": "HeavyWeapon", | ||||
|     "cost": 4000, | ||||
|     "mag": 1, | ||||
|     "ammo_cost": 200, | ||||
|     "damage": "5d6", | ||||
|     "aspects": ["Blast", "AV 5d6"], | ||||
|     "description": "This is a portable guided missile system, like the older AT-3 Sagger, or present-day Javelin and Eryx systems. These weapons must be fired from a fixed stationary position. However, the gunner need not be adjacent to the missile’s firing position." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Heavy Machinegun", | ||||
|     "tech_age": "Early Mechanical", | ||||
|     "range": "HeavyWeapon", | ||||
|     "cost": 2000, | ||||
|     "mag": 100, | ||||
|     "ammo_cost": 200, | ||||
|     "damage": "5d6", | ||||
|     "aspects": [], | ||||
|     "description": "This is the largest portable support weapon in the Mechanical and Atomic ages. These weapons are usually mounted on vehicles but can be mounted on fixed tripods. If they are carried, they count as four items of encumbrance and each ammo pack is two items of encumbrance." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Plasma Gun", | ||||
|     "tech_age": "Late Interstellar", | ||||
|     "range": "Rifle", | ||||
|     "cost": 20000, | ||||
|     "mag": 40, | ||||
|     "ammo_cost": 2500, | ||||
|     "damage": "6d6", | ||||
|     "aspects": ["Bulky", "Fire", "AV 5d6"], | ||||
|     "description": "A high-energy, anti-vehicular, portable weapon firing magnetically contained plasma bolts. It requires a backpack-mounted, rechargeable power pack that counts as two Items for encumbrance purposes." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Rocket Launcher", | ||||
|     "tech_age": "Early Atomic", | ||||
|     "range": "HeavyWeapon", | ||||
|     "cost": 2000, | ||||
|     "mag": 1, | ||||
|     "ammo_cost": 150, | ||||
|     "damage": "4d6", | ||||
|     "aspects": ["Greater Blast", "Bulky", "AV 3d6"], | ||||
|     "description": "A portable rocket launcher utilizing unguided munitions. Rocket launchers are unsafe to fire in enclosed spaces. Rocket payloads are considered high explosive and are not particularly effective against armored vehicles. Armor piercing rockets cost 500 Credits per rocket: trade the Greater Blast for AV 4d6, Blast. Rocket launchers can also be mounted on vehicles, where they usually have 12 or even 24 shots before requiring reloading." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Support Laser", | ||||
|     "tech_age": "Early Space", | ||||
|     "range": "HeavyWeapon", | ||||
|     "cost": 5000, | ||||
|     "mag": 20, | ||||
|     "ammo_cost": 200, | ||||
|     "damage": "5d6", | ||||
|     "aspects": ["Bulky", "AV 3d6+1"], | ||||
|     "description": "This is a heavy directed-energy portable weapon for fire support and light anti-vehicular duties. Its backpack mounted power pack counts as two Items for encumbrance purposes." | ||||
|   }, | ||||
|   { | ||||
|     "name": "Under-barrel Grenade Launcher", | ||||
|     "tech_age": "Early Atomic", | ||||
|     "range": "Rifle", | ||||
|     "cost": 1000, | ||||
|     "mag": 1, | ||||
|     "ammo_cost": null, | ||||
|     "damage": "By gren.", | ||||
|     "aspects": [], | ||||
|     "description": "This weapon greatly increases the firepower of an infantry team. The under-barrel grenade launcher can be mounted under most two-handed firearms in the Early Atomic Age or later. It is not Bulky and can fire a single grenade before reloading. In the Late Space Age, the weapon has a 3-shot magazine." | ||||
|   } | ||||
| ] | ||||
							
								
								
									
										105
									
								
								assets/json_data/weapon_melee.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,105 @@ | ||||
| [ | ||||
|   { | ||||
|     "cost": 25, | ||||
|     "damage": "2d6+1", | ||||
|     "description": "A one-handed ax made from metal or stone, anything from a stone-age tool through a medieval war ax to a modern fire ax [1].", | ||||
|     "name": "Axe", | ||||
|     "tech_age": "Early Primitive", | ||||
|     "aspects": [ | ||||
|       "Throwable" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|     "cost": 300, | ||||
|     "damage": "3d6+1", | ||||
|     "description": "A large two-handed sword that counts as 2 items for encumbrance purposes. A two-handed Great-Axe uses the same stats [1].", | ||||
|     "name": "Greatsword or Great-Axe", | ||||
|     "tech_age": "Late Primitive", | ||||
|     "aspects": [ | ||||
|       "Bulky", | ||||
|       "Two-Handed" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|     "cost": 10, | ||||
|     "damage": "2d6", | ||||
|     "description": "A straight, short knife [1].", | ||||
|     "name": "Dagger", | ||||
|     "tech_age": "Early Primitive", | ||||
|     "aspects": [ | ||||
|       "Throwable" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|     "cost": 50, | ||||
|     "damage": "2d6+1", | ||||
|     "description": "A simple yet sturdy broad blade used for cutting foliage and foes alike [1].", | ||||
|     "name": "Machete", | ||||
|     "tech_age": "Late Primitive", | ||||
|     "aspects": [] | ||||
|   }, | ||||
|   { | ||||
|     "cost": 500, | ||||
|     "damage": "1d6", | ||||
|     "description": "This slaver’s weapon consists of a stocky handle connected to a flexible strap. Each attack causes stun damage. Additionally, any target hit must roll Physical or fall to the ground, incapacitated for 2d6 combat rounds due to nightmarish pain. It also causes 1d6 physical damage per hit. The neural whip is ineffective against anyone in armor [1, 2].", | ||||
|     "name": "Neural Whip", | ||||
|     "tech_age": "Late Interstellar", | ||||
|     "aspects": [ | ||||
|       "Stun" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|     "cost": 20, | ||||
|     "damage": "1d6+1", | ||||
|     "description": "A personal shield, primarily used to protect its wielder by providing a -1D penalty against melee attacks. Anyone carrying a shield can penalize a number of enemy attacks per round equal to their Combat skill -1, with a minimum of 1 [2].", | ||||
|     "name": "Shield", | ||||
|     "tech_age": "Early Primitive", | ||||
|     "aspects": [] | ||||
|   }, | ||||
|   { | ||||
|     "cost": 15, | ||||
|     "damage": "3d6", | ||||
|     "description": "This weapon comes in all types of construction, from a sharpened shaft to a metal-tipped spear. Regardless of how it was made, a spear is a basic, effective, melee weapon [3].", | ||||
|     "name": "Spear", | ||||
|     "tech_age": "Early Primitive", | ||||
|     "aspects": [ | ||||
|       "Throwable" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|     "cost": 10, | ||||
|     "damage": "1d6+2", | ||||
|     "description": "A long wooden shaft used as a bludgeon, sometimes with metal fittings. It must be used with two hands [3].", | ||||
|     "name": "Staff", | ||||
|     "tech_age": "Early Primitive", | ||||
|     "aspects": [ | ||||
|       "Two-Handed" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|     "cost": 300, | ||||
|     "damage": "2d6", | ||||
|     "description": "A high-tech anti-riot weapon. The stun prod augments the basic baton’s blunt force with an electric charge capable of stunning opponents [3].", | ||||
|     "name": "Stun Prod", | ||||
|     "tech_age": "Early Space", | ||||
|     "aspects": [ | ||||
|       "Stun" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|     "cost": 150, | ||||
|     "damage": "3d6", | ||||
|     "description": "One of any variety of medium-sized one-handed blades, including the gladius, cutlass, longsword, and katana [3].", | ||||
|     "name": "Sword", | ||||
|     "tech_age": "Late Primitive", | ||||
|     "aspects": [] | ||||
|   }, | ||||
|   { | ||||
|     "cost": 1000, | ||||
|     "damage": "3d6+2", | ||||
|     "description": "An intermediate-sized, one-handed, bladed weapon that uses hypersonic micro-vibrations in the blade to create a deadly cutting edge [4].", | ||||
|     "name": "Vibroblade", | ||||
|     "tech_age": "Early Interstellar", | ||||
|     "aspects": [] | ||||
|   } | ||||
| ] | ||||
| @@ -28,21 +28,30 @@ Hooks.once("init", function () { | ||||
|     applications, | ||||
|     models, | ||||
|     documents, | ||||
|     utils: FTLNomadUtils, | ||||
|   } | ||||
|  | ||||
|   /* -------------------------------------------- */ | ||||
|   // Set an initiative formula for the system | ||||
|   CONFIG.Combat.initiative = { | ||||
|     formula: "2d6 + @skills.combat.value", | ||||
|     decimals: 1 | ||||
|   }; | ||||
|  | ||||
|   CONFIG.Actor.documentClass = documents.FTLNomadActor | ||||
|   CONFIG.Actor.dataModels = { | ||||
|     protagonist: models.FTLNomadCharacter, | ||||
|     character: models.FTLNomadCharacter, | ||||
|     vehicle: models.FTLNomadVehicle, | ||||
|     creature: models.FTLNomadCreature, | ||||
|     starship: models.FTLNomadStarship | ||||
|     starship: models.FTLNomadStarship, | ||||
|     robot: models.FTLNomadRobot | ||||
|   } | ||||
|  | ||||
|   CONFIG.Item.documentClass = documents.FTLNomadItem | ||||
|   CONFIG.Item.dataModels = { | ||||
|     psionic: models.FTLNomadPsionic, | ||||
|     weapon: models.FTLNomadWeapon, | ||||
|     armor:  models.FTLNomadArmor, | ||||
|     armor: models.FTLNomadArmor, | ||||
|     talent: models.FTLNomadTalent, | ||||
|     language: models.FTLNomadLanguage, | ||||
|     equipment: models.FTLNomadEquipment, | ||||
| @@ -52,22 +61,23 @@ Hooks.once("init", function () { | ||||
|   } | ||||
|  | ||||
|   // Register sheet application classes | ||||
|   Actors.unregisterSheet("core", ActorSheet) | ||||
|   Actors.registerSheet("fvtt-ftl-nomad", applications.FTLNomadCharacterSheet , { types: ["character"], makeDefault: true }) | ||||
|   Actors.registerSheet("fvtt-ftl-nomad", applications.FTLNomadVehicleSheet, { types: ["vehicle"], makeDefault: true }) | ||||
|   Actors.registerSheet("fvtt-ftl-nomad", applications.FTLNomadCreatureSheet, { types: ["creature"], makeDefault: true }) | ||||
|   Actors.registerSheet("fvtt-ftl-nomad", applications.FTLNomadStarshipSheet, { types: ["starship"], makeDefault: true }) | ||||
|   foundry.documents.collections.Actors.unregisterSheet("core", foundry.appv1.sheets.ActorSheet) | ||||
|   foundry.documents.collections.Actors.registerSheet("fvtt-ftl-nomad", applications.FTLNomadCharacterSheet, { types: ["character"], makeDefault: true }) | ||||
|   foundry.documents.collections.Actors.registerSheet("fvtt-ftl-nomad", applications.FTLNomadVehicleSheet, { types: ["vehicle"], makeDefault: true }) | ||||
|   foundry.documents.collections.Actors.registerSheet("fvtt-ftl-nomad", applications.FTLNomadCreatureSheet, { types: ["creature"], makeDefault: true }) | ||||
|   foundry.documents.collections.Actors.registerSheet("fvtt-ftl-nomad", applications.FTLNomadStarshipSheet, { types: ["starship"], makeDefault: true }) | ||||
|   foundry.documents.collections.Actors.registerSheet("fvtt-ftl-nomad", applications.FTLNomadRobotSheet, { types: ["robot"], makeDefault: true }) | ||||
|  | ||||
|   Items.unregisterSheet("core", ItemSheet) | ||||
|   Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadPsionicSheet, { types: ["psionic"], makeDefault: true }) | ||||
|   Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadTalentSheet, { types: ["talent"], makeDefault: true }) | ||||
|   Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadLanguageSheet, { types: ["language"], makeDefault: true }) | ||||
|   Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadWeaponSheet, { types: ["weapon"], makeDefault: true }) | ||||
|   Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadArmorSheet, { types: ["armor"], makeDefault: true }) | ||||
|   Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadEquipmentSheet, { types: ["equipment"], makeDefault: true }) | ||||
|   Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadImplantSheet, { types: ["implant"], makeDefault: true }) | ||||
|   Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadCreatureTraitSheet, { types: ["creature-trait"], makeDefault: true }) | ||||
|   Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadCreatureAbilitySheet, { types: ["creature-ability"], makeDefault: true }) | ||||
|   foundry.documents.collections.Items.unregisterSheet("core", foundry.appv1.sheets.ItemSheet) | ||||
|   foundry.documents.collections.Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadPsionicSheet, { types: ["psionic"], makeDefault: true }) | ||||
|   foundry.documents.collections.Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadTalentSheet, { types: ["talent"], makeDefault: true }) | ||||
|   foundry.documents.collections.Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadLanguageSheet, { types: ["language"], makeDefault: true }) | ||||
|   foundry.documents.collections.Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadWeaponSheet, { types: ["weapon"], makeDefault: true }) | ||||
|   foundry.documents.collections.Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadArmorSheet, { types: ["armor"], makeDefault: true }) | ||||
|   foundry.documents.collections.Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadEquipmentSheet, { types: ["equipment"], makeDefault: true }) | ||||
|   foundry.documents.collections.Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadImplantSheet, { types: ["implant"], makeDefault: true }) | ||||
|   foundry.documents.collections.Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadCreatureTraitSheet, { types: ["creature-trait"], makeDefault: true }) | ||||
|   foundry.documents.collections.Items.registerSheet("fvtt-ftl-nomad", applications.FTLNomadCreatureAbilitySheet, { types: ["creature-ability"], makeDefault: true }) | ||||
|  | ||||
|   // Other Document Configuration | ||||
|   CONFIG.ChatMessage.documentClass = documents.FTLNomadChatMessage | ||||
| @@ -88,7 +98,7 @@ Hooks.once("init", function () { | ||||
|  | ||||
|   FTLNomadUtils.registerSettings() | ||||
|   FTLNomadUtils.registerHandlebarsHelpers() | ||||
|    | ||||
|  | ||||
|   console.info("FTL Nomad | System Initialized") | ||||
| }) | ||||
|  | ||||
| @@ -115,7 +125,7 @@ Hooks.once("ready", function () { | ||||
|  | ||||
| }) | ||||
|  | ||||
| Hooks.on("renderChatMessage", (message, html, data) => { | ||||
| Hooks.on("renderChatMessageHTML", (message, html, data) => { | ||||
| }) | ||||
|  | ||||
|  | ||||
|   | ||||
							
								
								
									
										1089
									
								
								lang/en.json
									
									
									
									
									
								
							
							
						
						| @@ -11,4 +11,4 @@ export { default as FTLNomadImplantSheet } from "./sheets/implant-sheet.mjs" | ||||
| export { default as FTLNomadStarshipSheet } from "./sheets/starship-sheet.mjs" | ||||
| export { default as FTLNomadCreatureTraitSheet } from "./sheets/creature-trait-sheet.mjs" | ||||
| export { default as FTLNomadCreatureAbilitySheet } from "./sheets/creature-ability-sheet.mjs" | ||||
|  | ||||
| export { default as FTLNomadRobotSheet } from "./sheets/robot-sheet.mjs" | ||||
|   | ||||
| @@ -22,7 +22,7 @@ export default class FTLNomadArmorSheet extends FTLNomadItemSheet { | ||||
|   /** @override */ | ||||
|   async _prepareContext() { | ||||
|     const context = await super._prepareContext() | ||||
|     context.enrichedDescription = await TextEditor.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }) | ||||
|  | ||||
|     return context | ||||
|   } | ||||
|   | ||||
| @@ -66,7 +66,8 @@ export default class FTLNomadActorSheet extends HandlebarsApplicationMixin(found | ||||
|       actor: this.document, | ||||
|       system: this.document.system, | ||||
|       source: this.document.toObject(), | ||||
|       enrichedDescription: await TextEditor.enrichHTML(this.document.system.description, { async: true }), | ||||
|       isEncumbered: this.document.system.isEncumbered(), | ||||
|       enrichedDescription: await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }), | ||||
|       isEditMode: this.isEditMode, | ||||
|       isPlayMode: this.isPlayMode, | ||||
|       isEditable: this.isEditable, | ||||
| @@ -98,7 +99,7 @@ export default class FTLNomadActorSheet extends HandlebarsApplicationMixin(found | ||||
|         dragover: this._onDragOver.bind(this), | ||||
|         drop: this._onDrop.bind(this), | ||||
|       } | ||||
|       return new DragDrop(d) | ||||
|       return new foundry.applications.ux.DragDrop.implementation(d) | ||||
|     }) | ||||
|   } | ||||
|  | ||||
| @@ -139,7 +140,7 @@ export default class FTLNomadActorSheet extends HandlebarsApplicationMixin(found | ||||
|   _onDragOver(event) {} | ||||
|  | ||||
|   async _onDropItem(item) { | ||||
|     console.log("Dropped item", item)  | ||||
|     console.log("Dropped item", item) | ||||
|     let itemData = item.toObject() | ||||
|     await this.document.createEmbeddedDocuments("Item", [itemData], { renderSheet: false }) | ||||
|   } | ||||
|   | ||||
| @@ -64,7 +64,7 @@ export default class FTLNomadItemSheet extends HandlebarsApplicationMixin(foundr | ||||
|       item: this.document, | ||||
|       system: this.document.system, | ||||
|       source: this.document.toObject(), | ||||
|       enrichedDescription: await TextEditor.enrichHTML(this.document.system.description, { async: true }), | ||||
|       enrichedDescription: await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }), | ||||
|       isEditMode: this.isEditMode, | ||||
|       isPlayMode: this.isPlayMode, | ||||
|       isEditable: this.isEditable, | ||||
| @@ -94,7 +94,7 @@ export default class FTLNomadItemSheet extends HandlebarsApplicationMixin(foundr | ||||
|         dragover: this._onDragOver.bind(this), | ||||
|         drop: this._onDrop.bind(this), | ||||
|       } | ||||
|       return new DragDrop(d) | ||||
|       return new foundry.applications.ux.DragDrop.implementation(d) | ||||
|     }) | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -12,45 +12,39 @@ export default class FTLNomadCharacterSheet extends FTLNomadActorSheet { | ||||
|       contentClasses: ["character-content"], | ||||
|     }, | ||||
|     actions: { | ||||
|       setBP: FTLNomadCharacterSheet.#onSetBP, | ||||
|       createGear: FTLNomadCharacterSheet.#onCreateGear, | ||||
|       createEquipment: FTLNomadCharacterSheet.#onCreateEquipment, | ||||
|       createArmor: FTLNomadCharacterSheet.#onCreateArmor, | ||||
|       createWeapon: FTLNomadCharacterSheet.#onCreateWeapon, | ||||
|       createBond: FTLNomadCharacterSheet.#onCreateBond, | ||||
|       createInjury: FTLNomadCharacterSheet.#onCreateInjury, | ||||
|       createMentalDisorder: FTLNomadCharacterSheet.#onCreateMentalDisorder, | ||||
|       createMotivation: FTLNomadCharacterSheet.#onCreateMotivation, | ||||
|       createSkill: FTLNomadCharacterSheet.#onCreateSkill, | ||||
|       createRitual: FTLNomadCharacterSheet.#onCreateRitual, | ||||
|       createTome: FTLNomadCharacterSheet.#onCreateTome, | ||||
|       createTalent: FTLNomadCharacterSheet.#onCreateTalent, | ||||
|       createImplant: FTLNomadCharacterSheet.#onCreateImplant, | ||||
|       createPsionic: FTLNomadCharacterSheet.#onCreatePsionic, | ||||
|       createLanguage: FTLNomadCharacterSheet.#onCreateLanguage | ||||
|  | ||||
|     }, | ||||
|   } | ||||
|  | ||||
|   /** @override */ | ||||
|   static PARTS = { | ||||
|     main: { | ||||
|       template: "systems/fvtt-ftl-nomad/templates/protagonist-main.hbs", | ||||
|       template: "systems/fvtt-ftl-nomad/templates/character-main.hbs", | ||||
|     }, | ||||
|     tabs: { | ||||
|       template: "templates/generic/tab-navigation.hbs", | ||||
|     }, | ||||
|     skills: { | ||||
|       template: "systems/fvtt-ftl-nomad/templates/protagonist-skills.hbs", | ||||
|     }, | ||||
|     status: { | ||||
|       template: "systems/fvtt-ftl-nomad/templates/protagonist-status.hbs", | ||||
|     talents: { | ||||
|       template: "systems/fvtt-ftl-nomad/templates/character-talents.hbs", | ||||
|     }, | ||||
|     equipment: { | ||||
|       template: "systems/fvtt-ftl-nomad/templates/protagonist-equipment.hbs", | ||||
|       template: "systems/fvtt-ftl-nomad/templates/character-equipment.hbs", | ||||
|     }, | ||||
|     biography: { | ||||
|       template: "systems/fvtt-ftl-nomad/templates/protagonist-biography.hbs", | ||||
|       template: "systems/fvtt-ftl-nomad/templates/character-biography.hbs", | ||||
|     }, | ||||
|   } | ||||
|  | ||||
|   /** @override */ | ||||
|   tabGroups = { | ||||
|     sheet: "skills", | ||||
|     sheet: "talents", | ||||
|   } | ||||
|  | ||||
|   /** | ||||
| @@ -59,8 +53,7 @@ export default class FTLNomadCharacterSheet extends FTLNomadActorSheet { | ||||
|    */ | ||||
|   #getTabs() { | ||||
|     const tabs = { | ||||
|       skills: { id: "skills", group: "sheet", icon: "fa-solid fa-shapes", label: "FTLNOMAD.Label.skills" }, | ||||
|       status: { id: "status", group: "sheet", icon: "fa-solid fa-file-waveform", label: "FTLNOMAD.Label.status" }, | ||||
|       talents: { id: "talents", group: "sheet", icon: "fa-solid fa-compass", label: "FTLNOMAD.Label.talents" }, | ||||
|       equipment: { id: "equipment", group: "sheet", icon: "fa-solid fa-shapes", label: "FTLNOMAD.Label.equipment" }, | ||||
|       biography: { id: "biography", group: "sheet", icon: "fa-solid fa-book", label: "FTLNOMAD.Label.biography" }, | ||||
|     } | ||||
| @@ -76,17 +69,8 @@ export default class FTLNomadCharacterSheet extends FTLNomadActorSheet { | ||||
|     const context = await super._prepareContext() | ||||
|     context.tabs = this.#getTabs() | ||||
|  | ||||
|     context.enrichedDescription = await TextEditor.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedNotes = await TextEditor.enrichHTML(this.document.system.notes, { async: true }) | ||||
|      | ||||
|     context.tooltipsCharacteristic = { | ||||
|       str: game.i18n.localize("FTLNOMAD.Characteristic.Str"), | ||||
|       dex: game.i18n.localize("FTLNOMAD.Characteristic.Dex"), | ||||
|       con: game.i18n.localize("FTLNOMAD.Characteristic.Con"), | ||||
|       int: game.i18n.localize("FTLNOMAD.Characteristic.Int"), | ||||
|       pow: game.i18n.localize("FTLNOMAD.Characteristic.Pow"), | ||||
|       cha: game.i18n.localize("FTLNOMAD.Characteristic.Cha") | ||||
|     } | ||||
|     context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedNotes = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.notes, { async: true }) | ||||
|  | ||||
|     return context | ||||
|   } | ||||
| @@ -97,10 +81,16 @@ export default class FTLNomadCharacterSheet extends FTLNomadActorSheet { | ||||
|     switch (partId) { | ||||
|       case "main": | ||||
|         break | ||||
|       case "skills": | ||||
|         context.tab = context.tabs.skills | ||||
|         context.skills = doc.itemTypes.skill | ||||
|         context.skills.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|       case "talents": | ||||
|         context.tab = context.tabs.talents | ||||
|         context.talents = doc.itemTypes.talent | ||||
|         context.talents.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.implants = doc.itemTypes.implant | ||||
|         context.implants.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.psionics = doc.itemTypes.psionic | ||||
|         context.psionics.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.languages = doc.itemTypes.language | ||||
|         context.languages.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         break | ||||
|       case "equipment": | ||||
|         context.tab = context.tabs.equipment | ||||
| @@ -108,44 +98,21 @@ export default class FTLNomadCharacterSheet extends FTLNomadActorSheet { | ||||
|         context.weapons.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.armors = doc.itemTypes.armor | ||||
|         context.armors.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.gears = doc.itemTypes.gear | ||||
|         context.gears.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.rituals = doc.itemTypes.ritual | ||||
|         context.rituals.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.tomes = doc.itemTypes.tome | ||||
|         context.tomes.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         break | ||||
|       case "status": | ||||
|         context.tab = context.tabs.status | ||||
|         context.injuries = doc.itemTypes.injury | ||||
|         context.injuries.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.mentaldisorders = doc.itemTypes.mentaldisorder | ||||
|         context.mentaldisorders.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.motivations = doc.itemTypes.motivation | ||||
|         context.motivations.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.bonds = doc.itemTypes.bond | ||||
|         context.bonds.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.equipments = doc.itemTypes.equipment | ||||
|         context.equipments.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         break | ||||
|       case "biography": | ||||
|         context.tab = context.tabs.biography | ||||
|         context.enrichedDescription = await TextEditor.enrichHTML(doc.system.description, { async: true }) | ||||
|         context.enrichedNotes = await TextEditor.enrichHTML(doc.system.notes, { async: true }) | ||||
|         context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(doc.system.description, { async: true }) | ||||
|         context.enrichedNotes = await foundry.applications.ux.TextEditor.implementation.enrichHTML(doc.system.notes, { async: true }) | ||||
|         break | ||||
|     } | ||||
|     return context | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * Creates a new attack item directly from the sheet and embeds it into the document. | ||||
|    * @param {Event} event             The initiating click event. | ||||
|    * @param {HTMLElement} target      The current target of the event listener. | ||||
|    */ | ||||
|   static #onSetBP(event, target) { | ||||
|     this.document.system.setBP() | ||||
|   } | ||||
|    | ||||
|   static #onCreateGear(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newGear"), type: "gear" }]) | ||||
|  | ||||
|   static #onCreateEquipment(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newEquipment"), type: "equipment" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateWeapon(event, target) { | ||||
| @@ -156,33 +123,22 @@ export default class FTLNomadCharacterSheet extends FTLNomadActorSheet { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newArmor"), type: "armor" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateBond(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newBond"), type: "bond" }]) | ||||
|   static #onCreateTalent(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newTalent"), type: "talent" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateInjury(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newInjury"), type: "injury" }]) | ||||
|   static #onCreateImplant(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newImplant"), type: "implant" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateMentalDisorder(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newMentalDisorder"), type: "mentaldisorder" }]) | ||||
|   static #onCreatePsionic(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newPsionic"), type: "psionic" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateMotivation(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newMotivation"), type: "motivation" }]) | ||||
|   static #onCreateLanguage(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newLanguage"), type: "language" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateSkill(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newSkill"), type: "skill" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateRitual(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newRitual"), type: "ritual" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateTome(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newTome"), type: "tome" }]) | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * Handles the roll action triggered by user interaction. | ||||
| @@ -202,35 +158,16 @@ export default class FTLNomadCharacterSheet extends FTLNomadActorSheet { | ||||
|     const rollType = $(event.currentTarget).data("roll-type") | ||||
|     let item | ||||
|     let li | ||||
|     // Debug : console.log(">>>>", event, target, rollType) | ||||
|     // Deprecated : if (this.isEditMode) return | ||||
|     switch (rollType) { | ||||
|       case "resource":  | ||||
|         item = foundry.utils.duplicate(this.actor.system.resources) | ||||
|         item.name = game.i18n.localize(`FTLNOMAD.Label.Resources`) | ||||
|         item.targetScore = item.permanentRating | ||||
|         break | ||||
|       case "char": | ||||
|         let charId = $(event.currentTarget).data("char-id") | ||||
|         item = foundry.utils.duplicate(this.actor.system.characteristics[charId]) | ||||
|         item.name = game.i18n.localize(`FTLNOMAD.Label.${charId}Long`) | ||||
|         item.targetScore = item.value * 5 | ||||
|         break | ||||
|       case "skill": | ||||
|         li = $(event.currentTarget).parents(".item"); | ||||
|         item = this.actor.items.get(li.data("item-id")); | ||||
|         let skillId = $(event.currentTarget).data("skill-id"); | ||||
|         item = this.actor.system.skills[skillId]; | ||||
|         break | ||||
|       case "weapon": | ||||
|       case "damage": | ||||
|         li = $(event.currentTarget).parents(".item"); | ||||
|         item = this.actor.items.get(li.data("item-id")); | ||||
|         item.damageBonus = this.actor.system.damageBonus | ||||
|         break | ||||
|       case "san":  | ||||
|         item = foundry.utils.duplicate(this.actor.system.san) | ||||
|         item.name = game.i18n.localize("FTLNOMAD.Label.SAN") | ||||
|         item.targetScore = item.value | ||||
|         break; | ||||
|       default: | ||||
|         throw new Error(`Unknown roll type ${rollType}`) | ||||
|     } | ||||
|   | ||||
| @@ -22,7 +22,7 @@ export default class FTLNomadCreatureAbilitySheet extends FTLNomadItemSheet { | ||||
|   /** @override */ | ||||
|   async _prepareContext() { | ||||
|     const context = await super._prepareContext() | ||||
|     context.enrichedDescription = await TextEditor.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }) | ||||
|     return context | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -3,19 +3,17 @@ import FTLNomadActorSheet from "./base-actor-sheet.mjs" | ||||
| export default class FTLNomadCreatureSheet extends FTLNomadActorSheet { | ||||
|   /** @override */ | ||||
|   static DEFAULT_OPTIONS = { | ||||
|     classes: ["npc"], | ||||
|     classes: ["creature"], | ||||
|     position: { | ||||
|       width: 860, | ||||
|       height: 620, | ||||
|     }, | ||||
|     window: { | ||||
|       contentClasses: ["npc-content"], | ||||
|       contentClasses: ["creature-content"], | ||||
|     }, | ||||
|     actions: { | ||||
|       createArmor: FTLNomadCreatureSheet.#onCreateArmor, | ||||
|       createWeapon: FTLNomadCreatureSheet.#onCreateWeapon, | ||||
|       createSkill: FTLNomadCreatureSheet.#onCreateSkill, | ||||
|  | ||||
|       createTrait: FTLNomadCreatureSheet.#onCreateTrait, | ||||
|       createAbility: FTLNomadCreatureSheet.#onCreateAbility | ||||
|     }, | ||||
|   } | ||||
|  | ||||
| @@ -27,8 +25,8 @@ export default class FTLNomadCreatureSheet extends FTLNomadActorSheet { | ||||
|     tabs: { | ||||
|       template: "templates/generic/tab-navigation.hbs", | ||||
|     }, | ||||
|     skills: { | ||||
|       template: "systems/fvtt-ftl-nomad/templates/creature-skills.hbs", | ||||
|     traits: { | ||||
|       template: "systems/fvtt-ftl-nomad/templates/creature-sheet-trait.hbs", | ||||
|     }, | ||||
|     biography: { | ||||
|       template: "systems/fvtt-ftl-nomad/templates/creature-biography.hbs", | ||||
| @@ -37,7 +35,7 @@ export default class FTLNomadCreatureSheet extends FTLNomadActorSheet { | ||||
|  | ||||
|   /** @override */ | ||||
|   tabGroups = { | ||||
|     sheet: "skills", | ||||
|     sheet: "traits", | ||||
|   } | ||||
|  | ||||
|   /** | ||||
| @@ -46,7 +44,7 @@ export default class FTLNomadCreatureSheet extends FTLNomadActorSheet { | ||||
|    */ | ||||
|   #getTabs() { | ||||
|     const tabs = { | ||||
|       skills: { id: "skills", group: "sheet", icon: "fa-solid fa-shapes", label: "FTLNOMAD.Label.skills" }, | ||||
|       traits: { id: "traits", group: "sheet", icon: "fa-solid fa-shapes", label: "FTLNOMAD.Label.traits" }, | ||||
|       biography: { id: "biography", group: "sheet", icon: "fa-solid fa-book", label: "FTLNOMAD.Label.biography" }, | ||||
|     } | ||||
|     for (const v of Object.values(tabs)) { | ||||
| @@ -61,9 +59,9 @@ export default class FTLNomadCreatureSheet extends FTLNomadActorSheet { | ||||
|     const context = await super._prepareContext() | ||||
|     context.tabs = this.#getTabs() | ||||
|  | ||||
|     context.enrichedDescription = await TextEditor.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedNotes = await TextEditor.enrichHTML(this.document.system.notes, { async: true }) | ||||
|      | ||||
|     context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedNotes = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.notes, { async: true }) | ||||
|  | ||||
|     return context | ||||
|   } | ||||
|  | ||||
| @@ -73,19 +71,17 @@ export default class FTLNomadCreatureSheet extends FTLNomadActorSheet { | ||||
|     switch (partId) { | ||||
|       case "main": | ||||
|         break | ||||
|       case "skills": | ||||
|         context.tab = context.tabs.skills | ||||
|         context.skills = doc.itemTypes.skill | ||||
|         context.skills.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.weapons = doc.itemTypes.weapon | ||||
|         context.weapons.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.armors = doc.itemTypes.armor | ||||
|         context.armors.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|       case "traits": | ||||
|         context.tab = context.tabs.traits | ||||
|         context.abilities = doc.itemTypes["creature-ability"] | ||||
|         context.abilities.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.traits = doc.itemTypes["creature-trait"] | ||||
|         context.traits.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         break | ||||
|       case "biography": | ||||
|         context.tab = context.tabs.biography | ||||
|         context.enrichedDescription = await TextEditor.enrichHTML(doc.system.description, { async: true }) | ||||
|         context.enrichedNotes = await TextEditor.enrichHTML(doc.system.notes, { async: true }) | ||||
|         context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(doc.system.description, { async: true }) | ||||
|         context.enrichedNotes = await foundry.applications.ux.TextEditor.implementation.enrichHTML(doc.system.notes, { async: true }) | ||||
|         break | ||||
|     } | ||||
|     return context | ||||
| @@ -96,16 +92,12 @@ export default class FTLNomadCreatureSheet extends FTLNomadActorSheet { | ||||
|    * @param {Event} event             The initiating click event. | ||||
|    * @param {HTMLElement} target      The current target of the event listener. | ||||
|    */ | ||||
|   static #onCreateWeapon(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newWeapon"), type: "weapon" }]) | ||||
|   static #onCreateTrait(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newTrait"), type: "creature-trait" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateArmor(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newArmor"), type: "armor" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateSkill(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newSkill"), type: "skill" }]) | ||||
|   static #onCreateAbility(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newAbility"), type: "creature-ability" }]) | ||||
|   } | ||||
|  | ||||
|   /** | ||||
| @@ -125,30 +117,31 @@ export default class FTLNomadCreatureSheet extends FTLNomadActorSheet { | ||||
|   async _onRoll(event, target) { | ||||
|     const rollType = $(event.currentTarget).data("roll-type") | ||||
|     let item | ||||
|     let li | ||||
|     // Debug : console.log(">>>>", event, target, rollType) | ||||
|     // Deprecated : if (this.isEditMode) return | ||||
|     let formula | ||||
|     let roll | ||||
|     switch (rollType) { | ||||
|       case "char": | ||||
|         let charId = $(event.currentTarget).data("char-id") | ||||
|         item = foundry.utils.duplicate(this.actor.system.characteristics[charId]) | ||||
|         item.name = game.i18n.localize(`FTLNOMAD.Label.${charId}Long`) | ||||
|         item.targetScore = item.value * 5 | ||||
|         break | ||||
|       case "skill": | ||||
|         li = $(event.currentTarget).parents(".item"); | ||||
|         item = this.actor.items.get(li.data("item-id")); | ||||
|         let skillId = $(event.currentTarget).data("skill-id"); | ||||
|         item = this.actor.system.skills[skillId]; | ||||
|         await this.document.system.roll(rollType, item) | ||||
|         break | ||||
|       case "weapon": | ||||
|       case "damage": | ||||
|         li = $(event.currentTarget).parents(".item"); | ||||
|         item = this.actor.items.get(li.data("item-id")); | ||||
|         item.damageBonus = this.actor.system.damageBonus | ||||
|       case "creature-damage": | ||||
|         formula = this.actor.system.damage | ||||
|         // Rolll the damage | ||||
|         roll = new Roll(formula) | ||||
|         await roll.evaluate() | ||||
|         roll.toMessage( { flavor: `${this.actor.name} : Damage roll` }) | ||||
|         break | ||||
|       case "creature-number": | ||||
|         formula = this.actor.system.numberAppearing | ||||
|         // Rolll the damage | ||||
|         roll = new Roll(formula) | ||||
|         await roll.evaluate() | ||||
|         roll.toMessage({flavor: `${this.actor.name} : Number Appearing roll`}) | ||||
|         break | ||||
|       default: | ||||
|         throw new Error(`Unknown roll type ${rollType}`) | ||||
|     } | ||||
|     await this.document.system.roll(rollType, item) | ||||
|   } | ||||
|  | ||||
|   async _onDrop(event) { | ||||
| @@ -162,5 +155,4 @@ export default class FTLNomadCreatureSheet extends FTLNomadActorSheet { | ||||
|         return super._onDropItem(item) | ||||
|     } | ||||
|   } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -22,7 +22,7 @@ export default class FTLNomadCreatureTraitSheet extends FTLNomadItemSheet { | ||||
|   /** @override */ | ||||
|   async _prepareContext() { | ||||
|     const context = await super._prepareContext() | ||||
|     context.enrichedDescription = await TextEditor.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }) | ||||
|     return context | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -22,7 +22,7 @@ export default class FTLNomadEquipmentSheet extends FTLNomadItemSheet { | ||||
|   /** @override */ | ||||
|   async _prepareContext() { | ||||
|     const context = await super._prepareContext() | ||||
|     context.enrichedDescription = await TextEditor.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }) | ||||
|     return context | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -22,7 +22,7 @@ export default class FTLNomadLanguageSheet extends FTLNomadItemSheet { | ||||
|   /** @override */ | ||||
|   async _prepareContext() { | ||||
|     const context = await super._prepareContext() | ||||
|     context.enrichedDescription = await TextEditor.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }) | ||||
|     return context | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -22,7 +22,7 @@ export default class FTLNomadPsionicSheet extends FTLNomadItemSheet { | ||||
|   /** @override */ | ||||
|   async _prepareContext() { | ||||
|     const context = await super._prepareContext() | ||||
|     context.enrichedDescription = await TextEditor.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }) | ||||
|     return context | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										174
									
								
								module/applications/sheets/robot-sheet.mjs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,174 @@ | ||||
| import FTLNomadActorSheet from "./base-actor-sheet.mjs" | ||||
|  | ||||
| export default class FTLNomadRobotSheet extends FTLNomadActorSheet { | ||||
|   /** @override */ | ||||
|   static DEFAULT_OPTIONS = { | ||||
|     classes: ["robot"], | ||||
|     position: { | ||||
|       width: 860, | ||||
|       height: 620, | ||||
|     }, | ||||
|     window: { | ||||
|       contentClasses: ["robot-content"], | ||||
|     }, | ||||
|     actions: { | ||||
|       createEquipment: FTLNomadRobotSheet.#onCreateEquipment, | ||||
|       createArmor: FTLNomadRobotSheet.#onCreateArmor, | ||||
|       createWeapon: FTLNomadRobotSheet.#onCreateWeapon, | ||||
|       createTalent: FTLNomadRobotSheet.#onCreateTalent, | ||||
|       createImplant: FTLNomadRobotSheet.#onCreateImplant, | ||||
|       createPsionic: FTLNomadRobotSheet.#onCreatePsionic, | ||||
|       createLanguage: FTLNomadRobotSheet.#onCreateLanguage | ||||
|  | ||||
|     }, | ||||
|   } | ||||
|  | ||||
|   /** @override */ | ||||
|   static PARTS = { | ||||
|     main: { | ||||
|       template: "systems/fvtt-ftl-nomad/templates/robot-main.hbs", | ||||
|     }, | ||||
|     tabs: { | ||||
|       template: "templates/generic/tab-navigation.hbs", | ||||
|     }, | ||||
|     equipment: { | ||||
|       template: "systems/fvtt-ftl-nomad/templates/robot-equipment.hbs", | ||||
|     }, | ||||
|     description: { | ||||
|       template: "systems/fvtt-ftl-nomad/templates/robot-description.hbs", | ||||
|     }, | ||||
|   } | ||||
|  | ||||
|   /** @override */ | ||||
|   tabGroups = { | ||||
|     sheet: "equipment", | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * Prepare an array of form header tabs. | ||||
|    * @returns {Record<string, Partial<ApplicationTab>>} | ||||
|    */ | ||||
|   #getTabs() { | ||||
|     const tabs = { | ||||
|       equipment: { id: "equipment", group: "sheet", icon: "fa-solid fa-shapes", label: "FTLNOMAD.Label.equipment" }, | ||||
|       description: { | ||||
|         id: "description", group: "sheet", icon: "fa-solid fa-book", label: "FTLNOMAD.Label.description" | ||||
|       } | ||||
|     } | ||||
|     for (const v of Object.values(tabs)) { | ||||
|       v.active = this.tabGroups[v.group] === v.id | ||||
|       v.cssClass = v.active ? "active" : "" | ||||
|     } | ||||
|     return tabs | ||||
|   } | ||||
|  | ||||
|   /** @override */ | ||||
|   async _prepareContext() { | ||||
|     const context = await super._prepareContext() | ||||
|     context.tabs = this.#getTabs() | ||||
|  | ||||
|     context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedNotes = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.notes, { async: true }) | ||||
|  | ||||
|     return context | ||||
|   } | ||||
|  | ||||
|   /** @override */ | ||||
|   async _preparePartContext(partId, context) { | ||||
|     const doc = this.document | ||||
|     switch (partId) { | ||||
|       case "equipment": | ||||
|         context.tab = context.tabs.equipment | ||||
|         context.weapons = doc.itemTypes.weapon | ||||
|         context.weapons.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.armors = doc.itemTypes.armor | ||||
|         context.armors.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.equipments = doc.itemTypes.equipment | ||||
|         context.equipments.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         break | ||||
|       case "description": | ||||
|         context.tab = context.tabs.description | ||||
|         context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(doc.system.description, { async: true }) | ||||
|         context.enrichedNotes = await foundry.applications.ux.TextEditor.implementation.enrichHTML(doc.system.notes, { async: true }) | ||||
|         break | ||||
|     } | ||||
|     return context | ||||
|   } | ||||
|  | ||||
|  | ||||
|   static #onCreateEquipment(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newEquipment"), type: "equipment" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateWeapon(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newWeapon"), type: "weapon" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateArmor(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newArmor"), type: "armor" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateTalent(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newTalent"), type: "talent" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateImplant(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newImplant"), type: "implant" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreatePsionic(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newPsionic"), type: "psionic" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateLanguage(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newLanguage"), type: "language" }]) | ||||
|   } | ||||
|  | ||||
|  | ||||
|   /** | ||||
|    * Handles the roll action triggered by user interaction. | ||||
|    * | ||||
|    * @param {PointerEvent} event The event object representing the user interaction. | ||||
|    * @param {HTMLElement} target The target element that triggered the roll. | ||||
|    * | ||||
|    * @returns {Promise<void>} A promise that resolves when the roll action is complete. | ||||
|    * | ||||
|    * @throws {Error} Throws an error if the roll type is not recognized. | ||||
|    * | ||||
|    * @description This method checks the current mode (edit or not) and determines the type of roll | ||||
|    * (save, resource, or damage) based on the target element's data attributes. It retrieves the | ||||
|    * corresponding value from the document's system and performs the roll. | ||||
|    */ | ||||
|   async _onRoll(event, target) { | ||||
|     const rollType = $(event.currentTarget).data("roll-type") | ||||
|     let item | ||||
|     let li | ||||
|     switch (rollType) { | ||||
|       case "skill": | ||||
|         let skillId = $(event.currentTarget).data("skill-id"); | ||||
|         item = this.actor.system.skills[skillId]; | ||||
|         break | ||||
|       case "weapon": | ||||
|       case "damage": | ||||
|         li = $(event.currentTarget).parents(".item"); | ||||
|         item = this.actor.items.get(li.data("item-id")); | ||||
|         break | ||||
|       default: | ||||
|         throw new Error(`Unknown roll type ${rollType}`) | ||||
|     } | ||||
|     await this.document.system.roll(rollType, item) | ||||
|   } | ||||
|  | ||||
|   async _onDrop(event) { | ||||
|     if (!this.isEditable || !this.isEditMode) return | ||||
|     const data = TextEditor.getDragEventData(event) | ||||
|  | ||||
|     // Handle different data types | ||||
|     switch (data.type) { | ||||
|       case "Item": | ||||
|         const item = await fromUuid(data.uuid) | ||||
|         return super._onDropItem(item) | ||||
|     } | ||||
|   } | ||||
|  | ||||
| } | ||||
| @@ -3,16 +3,16 @@ import FTLNomadActorSheet from "./base-actor-sheet.mjs" | ||||
| export default class FTLNomadStarshipSheet extends FTLNomadActorSheet { | ||||
|   /** @override */ | ||||
|   static DEFAULT_OPTIONS = { | ||||
|     classes: ["vehicle"], | ||||
|     classes: ["starship"], | ||||
|     position: { | ||||
|       width: 680, | ||||
|       height: 540, | ||||
|       height: 640, | ||||
|     }, | ||||
|     window: { | ||||
|       contentClasses: ["vehicle-content"], | ||||
|       contentClasses: ["starship-content"], | ||||
|     }, | ||||
|     actions: { | ||||
|       createGear: FTLNomadStarshipSheet.#onCreateGear, | ||||
|       createEquipment: FTLNomadStarshipSheet.#onCreateEquipment, | ||||
|       createWeapon: FTLNomadStarshipSheet.#onCreateWeapon, | ||||
|     }, | ||||
|   } | ||||
| @@ -33,11 +33,11 @@ export default class FTLNomadStarshipSheet extends FTLNomadActorSheet { | ||||
|     }, | ||||
|   } | ||||
|  | ||||
|    /** @override */ | ||||
|    tabGroups = { | ||||
|   /** @override */ | ||||
|   tabGroups = { | ||||
|     sheet: "equipment", | ||||
|   } | ||||
|    | ||||
|  | ||||
|   /** | ||||
|    * Prepare an array of form header tabs. | ||||
|    * @returns {Record<string, Partial<ApplicationTab>>} | ||||
| @@ -59,9 +59,10 @@ export default class FTLNomadStarshipSheet extends FTLNomadActorSheet { | ||||
|     const context = await super._prepareContext() | ||||
|     context.tabs = this.#getTabs() | ||||
|  | ||||
|     context.enrichedDescription = await TextEditor.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedNotes = await TextEditor.enrichHTML(this.document.system.notes, { async: true }) | ||||
|      | ||||
|     context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedNotes = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.notes, { async: true }) | ||||
|     context.enrichedModifications = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.modifications, { async: true }) | ||||
|  | ||||
|     return context | ||||
|   } | ||||
|  | ||||
| @@ -77,30 +78,46 @@ export default class FTLNomadStarshipSheet extends FTLNomadActorSheet { | ||||
|       case "equipment": | ||||
|         context.tab = context.tabs.equipment | ||||
|         context.weapons = doc.itemTypes.weapon | ||||
|         context.gears = doc.itemTypes.gear | ||||
|         break   | ||||
|         context.weapons.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.equipments = doc.itemTypes.equipment | ||||
|         context.equipments.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         break | ||||
|       case "description": | ||||
|         context.tab = context.tabs.description | ||||
|         context.enrichedDescription = await TextEditor.enrichHTML(doc.system.description, { async: true }) | ||||
|         context.enrichedNotes = await TextEditor.enrichHTML(doc.system.notes, { async: true }) | ||||
|         context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(doc.system.description, { async: true }) | ||||
|         context.enrichedNotes = await foundry.applications.ux.TextEditor.implementation.enrichHTML(doc.system.notes, { async: true }) | ||||
|         break | ||||
|     } | ||||
|     return context | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * Creates a new attack item directly from the sheet and embeds it into the document. | ||||
|    * @param {Event} event             The initiating click event. | ||||
|    * @param {HTMLElement} target      The current target of the event listener. | ||||
|    */ | ||||
|   static #onCreateGear(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newGear"), type: "gear" }]) | ||||
|   static #onCreateEquipment(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newEquipment"), type: "equipment" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateWeapon(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newWeapon"), type: "weapon" }]) | ||||
|   } | ||||
|  | ||||
|   async _onRoll(event, target) { | ||||
|     let rollType = $(event.currentTarget).data("roll-type") | ||||
|     let item | ||||
|     console.log("rollType", rollType) | ||||
|     switch (rollType) { | ||||
|       case "damage": | ||||
|         let li = $(event.currentTarget).parents(".item"); | ||||
|         item = this.actor.items.get(li.data("item-id")); | ||||
|         break | ||||
|       case "starship-guns": | ||||
|         item = { name: "Starship Guns", type: "weapon", system: { damage: this.actor.system.guns, rangeType: "heavyweapon" } } | ||||
|         rollType = "damage" | ||||
|         break | ||||
|       default: | ||||
|         throw new Error(`Unknown roll type ${rollType}`) | ||||
|     } | ||||
|     await this.document.system.roll(rollType, item) | ||||
|   } | ||||
|  | ||||
|  | ||||
|   async _onDrop(event) { | ||||
|     if (!this.isEditable || !this.isEditMode) return | ||||
|   | ||||
| @@ -22,7 +22,7 @@ export default class FTLNomadTalentSheet extends FTLNomadItemSheet { | ||||
|   /** @override */ | ||||
|   async _prepareContext() { | ||||
|     const context = await super._prepareContext() | ||||
|     context.enrichedDescription = await TextEditor.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }) | ||||
|     return context | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -12,7 +12,7 @@ export default class FTLNomadVehicleSheet extends FTLNomadActorSheet { | ||||
|       contentClasses: ["vehicle-content"], | ||||
|     }, | ||||
|     actions: { | ||||
|       createGear: FTLNomadVehicleSheet.#onCreateGear, | ||||
|       createEquipment: FTLNomadVehicleSheet.#onCreateEquipment, | ||||
|       createWeapon: FTLNomadVehicleSheet.#onCreateWeapon, | ||||
|     }, | ||||
|   } | ||||
| @@ -33,11 +33,11 @@ export default class FTLNomadVehicleSheet extends FTLNomadActorSheet { | ||||
|     }, | ||||
|   } | ||||
|  | ||||
|    /** @override */ | ||||
|    tabGroups = { | ||||
|   /** @override */ | ||||
|   tabGroups = { | ||||
|     sheet: "equipment", | ||||
|   } | ||||
|    | ||||
|  | ||||
|   /** | ||||
|    * Prepare an array of form header tabs. | ||||
|    * @returns {Record<string, Partial<ApplicationTab>>} | ||||
| @@ -59,9 +59,9 @@ export default class FTLNomadVehicleSheet extends FTLNomadActorSheet { | ||||
|     const context = await super._prepareContext() | ||||
|     context.tabs = this.#getTabs() | ||||
|  | ||||
|     context.enrichedDescription = await TextEditor.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedNotes = await TextEditor.enrichHTML(this.document.system.notes, { async: true }) | ||||
|      | ||||
|     context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.description, { async: true }) | ||||
|     context.enrichedNotes = await foundry.applications.ux.TextEditor.implementation.enrichHTML(this.document.system.notes, { async: true }) | ||||
|  | ||||
|     return context | ||||
|   } | ||||
|  | ||||
| @@ -77,12 +77,14 @@ export default class FTLNomadVehicleSheet extends FTLNomadActorSheet { | ||||
|       case "equipment": | ||||
|         context.tab = context.tabs.equipment | ||||
|         context.weapons = doc.itemTypes.weapon | ||||
|         context.gears = doc.itemTypes.gear | ||||
|         break   | ||||
|         context.weapons.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         context.equipments = doc.itemTypes.equipment | ||||
|         context.equipments.sort((a, b) => a.name.localeCompare(b.name)) | ||||
|         break | ||||
|       case "description": | ||||
|         context.tab = context.tabs.description | ||||
|         context.enrichedDescription = await TextEditor.enrichHTML(doc.system.description, { async: true }) | ||||
|         context.enrichedNotes = await TextEditor.enrichHTML(doc.system.notes, { async: true }) | ||||
|         context.enrichedDescription = await foundry.applications.ux.TextEditor.implementation.enrichHTML(doc.system.description, { async: true }) | ||||
|         context.enrichedNotes = await foundry.applications.ux.TextEditor.implementation.enrichHTML(doc.system.notes, { async: true }) | ||||
|         break | ||||
|     } | ||||
|     return context | ||||
| @@ -93,14 +95,29 @@ export default class FTLNomadVehicleSheet extends FTLNomadActorSheet { | ||||
|    * @param {Event} event             The initiating click event. | ||||
|    * @param {HTMLElement} target      The current target of the event listener. | ||||
|    */ | ||||
|   static #onCreateGear(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newGear"), type: "gear" }]) | ||||
|   static #onCreateEquipment(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newEquipment"), type: "equipment" }]) | ||||
|   } | ||||
|  | ||||
|   static #onCreateWeapon(event, target) { | ||||
|     this.document.createEmbeddedDocuments("Item", [{ name: game.i18n.localize("FTLNOMAD.Label.newWeapon"), type: "weapon" }]) | ||||
|   } | ||||
|  | ||||
|   async _onRoll(event, target) { | ||||
|     const rollType = $(event.currentTarget).data("roll-type") | ||||
|     let item | ||||
|     let li | ||||
|     switch (rollType) { | ||||
|       case "damage": | ||||
|         li = $(event.currentTarget).parents(".item"); | ||||
|         item = this.actor.items.get(li.data("item-id")); | ||||
|         break | ||||
|       default: | ||||
|         throw new Error(`Unknown roll type ${rollType}`) | ||||
|     } | ||||
|     await this.document.system.roll(rollType, item) | ||||
|   } | ||||
|  | ||||
|  | ||||
|   async _onDrop(event) { | ||||
|     if (!this.isEditable || !this.isEditMode) return | ||||
|   | ||||
| @@ -2,15 +2,15 @@ export const SYSTEM_ID = "fvtt-ftl-nomad" | ||||
|  | ||||
| export const ASCII = ` | ||||
|  | ||||
| ░▒▓████████▓▒░▒▓████████▓▒░▒▓█▓▒░                           ░▒▓███████▓▒░ ░▒▓██████▓▒░░▒▓██████████████▓▒░ ░▒▓██████▓▒░░▒▓███████▓▒░   | ||||
| ░▒▓█▓▒░         ░▒▓█▓▒░   ░▒▓█▓▒░             ░▒▓██▓▒░      ░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░  | ||||
| ░▒▓█▓▒░         ░▒▓█▓▒░   ░▒▓█▓▒░             ░▒▓██▓▒░      ░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░  | ||||
| ░▒▓██████▓▒░    ░▒▓█▓▒░   ░▒▓█▓▒░                           ░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░░▒▓█▓▒░▒▓████████▓▒░▒▓█▓▒░░▒▓█▓▒░  | ||||
| ░▒▓█▓▒░         ░▒▓█▓▒░   ░▒▓█▓▒░             ░▒▓██▓▒░      ░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░  | ||||
| ░▒▓█▓▒░         ░▒▓█▓▒░   ░▒▓█▓▒░             ░▒▓██▓▒░      ░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░  | ||||
| ░▒▓█▓▒░         ░▒▓█▓▒░   ░▒▓████████▓▒░                    ░▒▓█▓▒░░▒▓█▓▒░░▒▓██████▓▒░░▒▓█▓▒░░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓███████▓▒░   | ||||
|                                                                                                                                        | ||||
|                                                                                                                                        | ||||
| ░▒▓████████▓▒░▒▓████████▓▒░▒▓█▓▒░                           ░▒▓███████▓▒░ ░▒▓██████▓▒░░▒▓██████████████▓▒░ ░▒▓██████▓▒░░▒▓███████▓▒░ | ||||
| ░▒▓█▓▒░         ░▒▓█▓▒░   ░▒▓█▓▒░             ░▒▓██▓▒░      ░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░ | ||||
| ░▒▓█▓▒░         ░▒▓█▓▒░   ░▒▓█▓▒░             ░▒▓██▓▒░      ░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░ | ||||
| ░▒▓██████▓▒░    ░▒▓█▓▒░   ░▒▓█▓▒░                           ░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░░▒▓█▓▒░▒▓████████▓▒░▒▓█▓▒░░▒▓█▓▒░ | ||||
| ░▒▓█▓▒░         ░▒▓█▓▒░   ░▒▓█▓▒░             ░▒▓██▓▒░      ░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░ | ||||
| ░▒▓█▓▒░         ░▒▓█▓▒░   ░▒▓█▓▒░             ░▒▓██▓▒░      ░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░ | ||||
| ░▒▓█▓▒░         ░▒▓█▓▒░   ░▒▓████████▓▒░                    ░▒▓█▓▒░░▒▓█▓▒░░▒▓██████▓▒░░▒▓█▓▒░░▒▓█▓▒░░▒▓█▓▒░▒▓█▓▒░░▒▓█▓▒░▒▓███████▓▒░ | ||||
|  | ||||
|  | ||||
|  | ||||
| ` | ||||
|  | ||||
| @@ -20,7 +20,7 @@ export const SKILLS = { | ||||
|   "social": { id: "social", label: "FTLNOMAD.Skill.Social" }, | ||||
|   "physical": { id: "physical", label: "FTLNOMAD.Skill.Physical" }, | ||||
|   "stealth": { id: "stealth", label: "FTLNOMAD.Skill.Stealth" }, | ||||
|   "vehicles": { id: "vehicle", label: "FTLNOMAD.Skill.Vehicles" }, | ||||
|   "vehicles": { id: "vehicles", label: "FTLNOMAD.Skill.Vehicles" }, | ||||
|   "technology": { id: "technology", label: "FTLNOMAD.Skill.Technology" } | ||||
| } | ||||
|  | ||||
| @@ -41,6 +41,13 @@ export const TECH_AGES = { | ||||
|   "cosmic": { id: "cosmic", label: "FTLNOMAD.TechAge.Cosmic", level: 13 } | ||||
| } | ||||
|  | ||||
| export const ROBOT_SIZES = { | ||||
|   "tiny": { id: "tiny", label: "FTLNOMAD.Robot.Size.Tiny" }, | ||||
|   "small": { id: "small", label: "FTLNOMAD.Robot.Size.Small" }, | ||||
|   "medium": { id: "medium", label: "FTLNOMAD.Robot.Size.Medium" }, | ||||
|   "large": { id: "large", label: "FTLNOMAD.Robot.Size.Large" }, | ||||
|   "giant": { id: "giant", label: "FTLNOMAD.Robot.Size.Giant" } | ||||
| } | ||||
|  | ||||
| export const WEAPON_TYPES = { | ||||
|   "melee": { id: "melee", label: "FTLNOMAD.Weapon.Types.Melee" }, | ||||
| @@ -50,43 +57,68 @@ export const WEAPON_TYPES = { | ||||
|   "grenade": { id: "grenade", label: "FTLNOMAD.Weapon.Types.Grenade" }, | ||||
|   "vehicle": { id: "vehicle", label: "FTLNOMAD.Weapon.Types.Vehicle" } | ||||
| } | ||||
|  | ||||
| export const WEAPON_RANGE = { | ||||
|   "melee": { id: "melee", label: "FTLNOMAD.Weapon.Range.Melee" }, | ||||
|   "handgun": { id: "handgun", label: "FTLNOMAD.Weapon.Range.Handgun" }, | ||||
|   "rifle": { id: "rifle", label: "FTLNOMAD.Weapon.Range.Rifle" }, | ||||
|   "longrange": { id: "longrange", label: "FTLNOMAD.Weapon.Range.LongRange" } | ||||
|   "handgun": { id: "handgun", label: "FTLNOMAD.Weapon.Range.Handgun", range: { close: 0, near: 0, far: -2 } }, | ||||
|   "assault": { id: "assault", label: "FTLNOMAD.Weapon.Range.Assault", range: { close: -2, near: 0, far: -1, distant: -2 } }, | ||||
|   "rifle": { id: "rifle", label: "FTLNOMAD.Weapon.Range.Rifle", range: { close: -3, near: 0, far: 0, distant: -1 } }, | ||||
|   "melee": { id: "melee", label: "FTLNOMAD.Weapon.Range.Melee", range: { close: 0 } }, | ||||
|   "heavyweapon": { id: "heavyweapon", label: "FTLNOMAD.Weapon.Range.HeavyWeapon", range: { near: -1, far: 0, distant: 0 } }, | ||||
|   "thrownweapon": { id: "thrownweapon", label: "FTLNOMAD.Weapon.Range.ThrownWeapon", range: { close: 0, near: -1 } } | ||||
| } | ||||
|  | ||||
| export const ATTACK_MODIFIERS = { | ||||
|   "two-attacks": -1, | ||||
|   "aiming": 1, | ||||
|   "dim": -1, | ||||
|   "darkness": -2, | ||||
|   "prone": -1, | ||||
|   "cover": -2, | ||||
|   "recoil-first": -1, | ||||
|   "recoil-third": -2, | ||||
|   "aware": -1 | ||||
| } | ||||
|  | ||||
| export const DAMAGE_MODIFIER_CHOICES = [ | ||||
|   { id: "-3", label: "-3", value: "-3" }, | ||||
|   { id: "-2", label: "-2", value: "-2" }, | ||||
|   { id: "-1", label: "-1", value: "-1" }, | ||||
|   { id: "0", label: "0", value: "0" }, | ||||
|   { id: "1", label: "+1", value: "1" }, | ||||
|   { id: "2", label: "+2", value: "2" }, | ||||
|   { id: "3", label: "+3", value: "3" } | ||||
| ] | ||||
|  | ||||
| export const TRIAGE_RESULTS = { | ||||
|   "none": { id: "none", dice:0, label: "FTLNOMAD.TriageResults.None" }, | ||||
|   "death": { id: "death", dice:3, label: "FTLNOMAD.TriageResults.Death" }, | ||||
|   "critical": { id: "critical", dice:4, label: "FTLNOMAD.TriageResults.Critical" }, | ||||
|   "severe": { id: "severe", dice:7, label: "FTLNOMAD.TriageResults.Severe" }, | ||||
|   "moderate": { id: "moderate", dice:10, label: "FTLNOMAD.TriageResults.Moderate" }, | ||||
|   "fleshwound": { id: "fleshwound", dice:12, label: "FTLNOMAD.TriageResults.FleshWound" } | ||||
|   "none": { id: "none", dice: 0, label: "FTLNOMAD.TriageResults.None" }, | ||||
|   "death": { id: "death", dice: 3, label: "FTLNOMAD.TriageResults.Death" }, | ||||
|   "critical": { id: "critical", dice: 4, label: "FTLNOMAD.TriageResults.Critical" }, | ||||
|   "severe": { id: "severe", dice: 7, label: "FTLNOMAD.TriageResults.Severe" }, | ||||
|   "moderate": { id: "moderate", dice: 10, label: "FTLNOMAD.TriageResults.Moderate" }, | ||||
|   "fleshwound": { id: "fleshwound", dice: 12, label: "FTLNOMAD.TriageResults.FleshWound" } | ||||
| } | ||||
|  | ||||
| export const CREATURE_TERRAIN_TYPES = { | ||||
|   "cave": { id: "cave", label: "FTLNOMAD.Creature.Terrain.Cave", niche:0, size: 0 }, | ||||
|   "coast": { id: "coast", label: "FTLNOMAD.Creature.Terrain.Coast", niche:1, size: 0 }, | ||||
|   "desert": { id: "desert", label: "FTLNOMAD.Creature.Terrain.Desert", niche:-1, size: -1 }, | ||||
|   "forest": { id: "forest", label: "FTLNOMAD.Creature.Terrain.Forest", niche:1, size: 1 }, | ||||
|   "jungle": { id: "jungle", label: "FTLNOMAD.Creature.Terrain.Jungle", niche:1, size: 1 }, | ||||
|   "mixed": { id: "mixed", label: "FTLNOMAD.Creature.Terrain.Mixed", niche:0, size: 0 }, | ||||
|   "mountain": { id: "mountain", label: "FTLNOMAD.Creature.Terrain.Mountain", niche:-1, size: -1 },  | ||||
|   "ocean": { id: "ocean", label: "FTLNOMAD.Creature.Terrain.Ocean", niche:-1, size: 1 }, | ||||
|   "river": { id: "river", label: "FTLNOMAD.Creature.Terrain.River", niche:1, size: 0 }, | ||||
|   "ruins": { id: "ruins", label: "FTLNOMAD.Creature.Terrain.Ruins", niche:0, size: 1 }, | ||||
|   "savannah": { id: "savannah", label: "FTLNOMAD.Creature.Terrain.Savannah", niche:0, size: 1 }, | ||||
|   "shallows": { id: "shallows", label: "FTLNOMAD.Creature.Terrain.Shallows", niche:1, size: 0 }, | ||||
|   "swamp": { id: "swamp", label: "FTLNOMAD.Creature.Terrain.Swamp", niche:1, size: 1 } | ||||
|   "cave": { id: "cave", label: "FTLNOMAD.Creature.Terrain.Cave", niche: 0, size: 0 }, | ||||
|   "coast": { id: "coast", label: "FTLNOMAD.Creature.Terrain.Coast", niche: 1, size: 0 }, | ||||
|   "desert": { id: "desert", label: "FTLNOMAD.Creature.Terrain.Desert", niche: -1, size: -1 }, | ||||
|   "forest": { id: "forest", label: "FTLNOMAD.Creature.Terrain.Forest", niche: 1, size: 1 }, | ||||
|   "jungle": { id: "jungle", label: "FTLNOMAD.Creature.Terrain.Jungle", niche: 1, size: 1 }, | ||||
|   "mixed": { id: "mixed", label: "FTLNOMAD.Creature.Terrain.Mixed", niche: 0, size: 0 }, | ||||
|   "mountain": { id: "mountain", label: "FTLNOMAD.Creature.Terrain.Mountain", niche: -1, size: -1 }, | ||||
|   "ocean": { id: "ocean", label: "FTLNOMAD.Creature.Terrain.Ocean", niche: -1, size: 1 }, | ||||
|   "river": { id: "river", label: "FTLNOMAD.Creature.Terrain.River", niche: 1, size: 0 }, | ||||
|   "ruins": { id: "ruins", label: "FTLNOMAD.Creature.Terrain.Ruins", niche: 0, size: 1 }, | ||||
|   "savannah": { id: "savannah", label: "FTLNOMAD.Creature.Terrain.Savannah", niche: 0, size: 1 }, | ||||
|   "shallows": { id: "shallows", label: "FTLNOMAD.Creature.Terrain.Shallows", niche: 1, size: 0 }, | ||||
|   "swamp": { id: "swamp", label: "FTLNOMAD.Creature.Terrain.Swamp", niche: 1, size: 1 } | ||||
| } | ||||
|  | ||||
| export const CREATURE_NICHES = { | ||||
|   "prey": { id: "prey", label: "FTLNOMAD.Creature.Niche.Prey" }, | ||||
|   "opportunist": { id: "opportunist", label: "FTLNOMAD.Creature.Niche.Opportunist" }, | ||||
|   "herbivore": { id: "herbivore", label: "FTLNOMAD.Creature.Niche.Herbivore" }, | ||||
|   "predator": { id: "predator", label: "FTLNOMAD.Creature.Niche.Predator" }  | ||||
|   "predator": { id: "predator", label: "FTLNOMAD.Creature.Niche.Predator" } | ||||
| } | ||||
|  | ||||
| export const CREATURE_SIZES = { | ||||
| @@ -98,18 +130,50 @@ export const CREATURE_SIZES = { | ||||
|   "titanic": { id: "titanic", label: "FTLNOMAD.Creature.Size.Titanic" } | ||||
| } | ||||
|  | ||||
| export const MODIFIER_CHOICES = { | ||||
|   "easy4": { id: "easy4", label: "FTLNOMAD.Label.Easy4", value: "4" }, | ||||
|   "easy3": { id: "easy3", label: "FTLNOMAD.Label.Easy3", value: "3" }, | ||||
|   "easy2": { id: "easy2", label: "FTLNOMAD.Label.Easy2", value: "2" }, | ||||
|   "easy": { id: "easy", label: "FTLNOMAD.Label.Easy", value: "1" }, | ||||
|   "moderate": { id: "moderate", label: "FTLNOMAD.Label.Moderate", value: "0" }, | ||||
|   "difficult": { id: "difficult", label: "FTLNOMAD.Label.Difficult", value: "-1" }, | ||||
|   "formidable": { id: "formidable", label: "FTLNOMAD.Label.Formidable", value: "-2" }, | ||||
|   "formidable3": { id: "formidable3", label: "FTLNOMAD.Label.Formidable3", value: "-3" }, | ||||
|   "impossible": { id: "impossible", label: "FTLNOMAD.Label.Impossible", value: "-4" }, | ||||
|   "impossible5": { id: "impossible5", label: "FTLNOMAD.Label.Impossible5", value: "-5" }, | ||||
|   "impossible6": { id: "impossible6", label: "FTLNOMAD.Label.Impossible6", value: "-6" } | ||||
| } | ||||
|  | ||||
| export const STARSHIP_HULL = { | ||||
|   "pod": { id: "pod", label: "FTLNOMAD.Starship.Hull.Pod" }, | ||||
|   "micro": { id: "micro", label: "FTLNOMAD.Starship.Hull.Micro" }, | ||||
|   "small": { id: "small", label: "FTLNOMAD.Starship.Hull.Small" }, | ||||
|   "scout": { id: "scout", label: "FTLNOMAD.Starship.Hull.Scout" }, | ||||
|   "picket": { id: "picket", label: "FTLNOMAD.Starship.Hull.Picket" }, | ||||
|   "destroyer": { id: "destroyer", label: "FTLNOMAD.Starship.Hull.Destroyer" }, | ||||
|   "cruiser": { id: "cruiser", label: "FTLNOMAD.Starship.Hull.Cruiser" }, | ||||
|   "battleship": { id: "battleship", label: "FTLNOMAD.Starship.Hull.Battleship" }, | ||||
|   "carrier": { id: "carrier", label: "FTLNOMAD.Starship.Hull.Carrier" } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Include all constant definitions within the SYSTEM global export | ||||
|  * @type {Object} | ||||
|  */ | ||||
| export const SYSTEM = { | ||||
|   id: SYSTEM_ID, | ||||
|   MODIFIER_CHOICES, | ||||
|   ATTACK_MODIFIERS, | ||||
|   TECH_AGES, | ||||
|   WEAPON_TYPES, | ||||
|   WEAPON_RANGE, | ||||
|   TRIAGE_RESULTS, | ||||
|   CREATURE_TERRAIN_TYPES, | ||||
|   CREATURE_SIZES, | ||||
|   CREATURE_NICHES, | ||||
|   STARSHIP_HULL, | ||||
|   SKILLS, | ||||
|   ROBOT_SIZES, | ||||
|   DAMAGE_MODIFIER_CHOICES, | ||||
|   ASCII | ||||
| } | ||||
|   | ||||
| @@ -40,14 +40,6 @@ export default class FTLNomadRoll extends Roll { | ||||
|     return this.options.help | ||||
|   } | ||||
|  | ||||
|   get gene() { | ||||
|     return this.options.gene | ||||
|   } | ||||
|  | ||||
|   get modifier() { | ||||
|     return this.options.modifier | ||||
|   } | ||||
|  | ||||
|   get resultType() { | ||||
|     return this.options.resultType | ||||
|   } | ||||
| @@ -68,41 +60,18 @@ export default class FTLNomadRoll extends Roll { | ||||
|     return this.options.weapon | ||||
|   } | ||||
|  | ||||
|   get isLowWP() { | ||||
|     return this.options.isLowWP | ||||
|   } | ||||
|  | ||||
|   get isZeroWP() { | ||||
|     return this.options.isZeroWP | ||||
|   } | ||||
|  | ||||
|   get isExhausted() { | ||||
|     return this.options.isExhausted | ||||
|   } | ||||
|  | ||||
|   get isNudgedRoll() { | ||||
|     return this.options.isNudgedRoll | ||||
|   } | ||||
|  | ||||
|   get wpCost()  { | ||||
|     return this.options.wpCost | ||||
|   } | ||||
|  | ||||
|   static updateResourceDialog(options) { | ||||
|     let rating = 0 | ||||
|     if (options.rollItem.enableHand) { | ||||
|       rating += options.rollItem.hand | ||||
|   static updateFullFormula(options) { | ||||
|     let fullFormula | ||||
|     if (options.rollType === "damage") { | ||||
|       fullFormula = `${options.formula} + ${options.damageModifier}D6 ` | ||||
|     } else { | ||||
|       let mod = options.rollItem?.value || 0 | ||||
|       fullFormula = `${options.formula} + ${options.skillModifier}D + ${mod} + ${options.rangeModifier}D + ${options.numericModifier}D` | ||||
|     } | ||||
|     if (options.rollItem.enableStowed) { | ||||
|       rating += options.rollItem.stowed | ||||
|     } | ||||
|     if (options.rollItem.enableStorage) { | ||||
|       rating += options.rollItem.storage | ||||
|     } | ||||
|     let multiplier = Number($(`.roll-skill-multiplier`).val()) | ||||
|     options.initialScore = rating | ||||
|     options.percentScore = rating * multiplier | ||||
|     $(".resource-score").text(`${rating} (${options.percentScore}%)`) | ||||
|     // Replace all the "+ -" with "-" | ||||
|     fullFormula = fullFormula.replace(/\+\s*\-/g, "- ") | ||||
|     $('#roll-dialog-full-formula').text(fullFormula) | ||||
|     options.fullFormula = fullFormula | ||||
|   } | ||||
|  | ||||
|   /** | ||||
| @@ -120,88 +89,26 @@ export default class FTLNomadRoll extends Roll { | ||||
|    * @returns {Promise<Object|null>} The roll result or null if the dialog was cancelled. | ||||
|    */ | ||||
|   static async prompt(options = {}) { | ||||
|     let formula = "1d100" | ||||
|     let hasModifier = true | ||||
|     let hasMultiplier = false | ||||
|     options.isNudge = true | ||||
|     let formula = "2d6" | ||||
|     let actor = game.actors.get(options.actorId) | ||||
|  | ||||
|     switch (options.rollType) { | ||||
|       case "skill": | ||||
|         console.log(options.rollItem) | ||||
|         options.initialScore = options.rollItem.system.computeScore() | ||||
|         break | ||||
|       case "san": | ||||
|       case "char": | ||||
|         options.initialScore = options.rollItem.targetScore | ||||
|         options.isNudge = (options.rollType !== "san") | ||||
|         break | ||||
|       case "resource": | ||||
|         hasModifier = false | ||||
|         hasMultiplier = true | ||||
|         options.initialScore = options.rollItem.targetScore | ||||
|         options.totalRating = options.rollItem.targetScore | ||||
|         options.percentScore = options.rollItem.targetScore * 5 | ||||
|         options.rollItem.enableHand = true | ||||
|         options.rollItem.enableStowed = true | ||||
|         options.rollItem.enableStorage = true | ||||
|         options.isNudge = false | ||||
|         break | ||||
|       case "damage": | ||||
|         let formula = options.rollItem.system.damage | ||||
|         if ( options.rollItem.system.weaponType === "melee" || options.rollItem.system.weaponType === "unarmed") { | ||||
|           formula += ` + ${options.rollItem.damageBonus}` | ||||
|         } | ||||
|         let damageRoll = new Roll(formula) | ||||
|         await damageRoll.evaluate() | ||||
|         await damageRoll.toMessage({ | ||||
|           flavor: `${options.rollItem.name} - Damage Roll` | ||||
|         }); | ||||
|         let isLethal = false | ||||
|         options.isNudge = false | ||||
|         if (options.rollItem.system.lethality > 0) { | ||||
|           let lethalityRoll = new Roll("1d100") | ||||
|           await lethalityRoll.evaluate() | ||||
|           isLethal = (lethalityRoll.total <= options.rollItem.system.lethality) | ||||
|           await lethalityRoll.toMessage({ | ||||
|             flavor: `${options.rollItem.name} - Lethality Roll : ${lethalityRoll.total} <= ${options.rollItem.system.lethality} => ${isLethal}` | ||||
|           }); | ||||
|         } | ||||
|         return | ||||
|         options.weapon = foundry.utils.duplicate(options.rollItem) | ||||
|         formula = options.weapon.system.damage | ||||
|         break | ||||
|       case "weapon": | ||||
|         let era = game.settings.get("fvtt-ftl-nomad", "settings-era") | ||||
|         if (era !== options.rollItem.system.settings) { | ||||
|           ui.notifications.error(game.i18n.localize("FTLNOMAD.Notifications.WrongEra")) | ||||
|           console.log("WP Wrong Era", era, options.rollItem.system.weaponType) | ||||
|           return | ||||
|         } | ||||
|         if (!SYSTEM.WEAPON_SKILL_MAPPING[era] || !SYSTEM.WEAPON_SKILL_MAPPING[era][options.rollItem.system.weaponType]) { | ||||
|           ui.notifications.error(game.i18n.localize("FTLNOMAD.Notifications.NoWeaponType")) | ||||
|           console.log("WP Not found", era, options.rollItem.system.weaponType) | ||||
|           return | ||||
|         } | ||||
|         options.weapon = options.rollItem | ||||
|         if (options.rollItem.system.hasDirectSkill) { | ||||
|           let skillName = options.rollItem.name | ||||
|           options.rollItem = {type: "skill", name: skillName, system: {base: 0, bonus:  options.weapon.system.directSkillValue} } | ||||
|           options.initialScore = options.weapon.system.directSkillValue | ||||
|         } else { | ||||
|           let skillName = game.i18n.localize(SYSTEM.WEAPON_SKILL_MAPPING[era][options.rollItem.system.weaponType]) | ||||
|           let actor = game.actors.get(options.actorId)           | ||||
|           options.rollItem = actor.items.find(i => i.type === "skill" && i.name.toLowerCase() === skillName.toLowerCase()) | ||||
|           if (!options.rollItem) { | ||||
|             ui.notifications.error(game.i18n.localize("FTLNOMAD.Notifications.NoWeaponSkill")) | ||||
|             return | ||||
|           } | ||||
|           options.initialScore = options.rollItem.system.computeScore() | ||||
|           console.log("WEAPON", skillName, era, options.rollItem) | ||||
|         } | ||||
|         options.weapon = foundry.utils.duplicate(options.rollItem) | ||||
|         options.rollItem = actor.system.skills.combat | ||||
|         break | ||||
|       default: | ||||
|         options.initialScore = 50 | ||||
|         break | ||||
|     } | ||||
|  | ||||
|     const rollModes = Object.fromEntries(Object.entries(CONFIG.Dice.rollModes).map(([key, value]) => [key, game.i18n.localize(value)])) | ||||
|     options.actor = actor | ||||
|     const rollModes = foundry.utils.duplicate(CONFIG.Dice.rollModes) | ||||
|     const fieldRollMode = new foundry.data.fields.StringField({ | ||||
|       choices: rollModes, | ||||
|       blank: false, | ||||
| @@ -209,37 +116,53 @@ export default class FTLNomadRoll extends Roll { | ||||
|     }) | ||||
|  | ||||
|     const choiceModifier = SYSTEM.MODIFIER_CHOICES | ||||
|     const choiceMultiplier = SYSTEM.MULTIPLIER_CHOICES | ||||
|     let choiceRangeModifier = {} | ||||
|     let rangeModifier = 0 | ||||
|     if (options.weapon) { | ||||
|       // Build the range modifiers | ||||
|       let range = SYSTEM.WEAPON_RANGE[options.weapon.system.rangeType] | ||||
|       for (let [key, value] of Object.entries(range.range)) { | ||||
|         choiceRangeModifier[key] = { label: `${key} (${value}D)`, value: value } | ||||
|       } | ||||
|     } | ||||
|     const choiceDamageModifier = SYSTEM.DAMAGE_MODIFIER_CHOICES | ||||
|     let damageModifier = "0" | ||||
|  | ||||
|     let modifier = "+0" | ||||
|     let multiplier = "5" | ||||
|     let modifier = "0" | ||||
|     options.skillModifier = 0 | ||||
|     options.numericModifier = 0 | ||||
|     options.rangeModifier = rangeModifier | ||||
|     options.damageModifier = damageModifier | ||||
|     let fullFormula = `${formula} + ${options.rollItem.value}` | ||||
|     if (options.isEncumbered) { | ||||
|       fullFormula += ` - 1D` | ||||
|     } | ||||
|     options.fullFormula = fullFormula | ||||
|     options.formula = formula | ||||
|  | ||||
|     let dialogContext = { | ||||
|       actorId: options.actorId, | ||||
|       actorName: options.actorName, | ||||
|       rollType: options.rollType, | ||||
|       rollItem: foundry.utils.duplicate(options.rollItem), // Object only, no class | ||||
|       fullFormula, | ||||
|       weapon: options?.weapon, | ||||
|       initialScore: options.initialScore, | ||||
|       targetScore: options.initialScore, | ||||
|       isLowWP: options.isLowWP, | ||||
|       isZeroWP: options.isZeroWP, | ||||
|       isExhausted: options.isExhausted, | ||||
|       enableHand: options.rollItem.enableHand, | ||||
|       enableStowed: options.rollItem.enableStowed, | ||||
|       enableStorage: options.rollItem.enableStorage, | ||||
|       isEncumbered: options.isEncumbered, | ||||
|       talents: options.talents, | ||||
|       rollModes, | ||||
|       fieldRollMode, | ||||
|       choiceModifier, | ||||
|       choiceMultiplier, | ||||
|       choiceRangeModifier, | ||||
|       choiceDamageModifier, | ||||
|       rangeModifier, | ||||
|       damageModifier, | ||||
|       formula, | ||||
|       hasTarget: options.hasTarget, | ||||
|       hasModifier, | ||||
|       hasMultiplier, | ||||
|       modifier, | ||||
|       multiplier | ||||
|     } | ||||
|     const content = await renderTemplate("systems/fvtt-ftl-nomad/templates/roll-dialog.hbs", dialogContext) | ||||
|     const content = await foundry.applications.handlebars.renderTemplate("systems/fvtt-ftl-nomad/templates/roll-dialog.hbs", dialogContext) | ||||
|  | ||||
|     const title = CthulhuEternalRoll.createTitle(options.rollType, options.rollTarget) | ||||
|     const title = FTLNomadRoll.createTitle(options.rollType, options.rollTarget) | ||||
|     const label = game.i18n.localize("FTLNOMAD.Roll.roll") | ||||
|     const rollContext = await foundry.applications.api.DialogV2.wait({ | ||||
|       window: { title: title }, | ||||
| @@ -258,24 +181,31 @@ export default class FTLNomadRoll extends Roll { | ||||
|         }, | ||||
|       ], | ||||
|       actions: { | ||||
|         "selectHand": (event, button, dialog) => { | ||||
|           options.rollItem.enableHand = !options.rollItem.enableHand | ||||
|           this.updateResourceDialog(options) | ||||
|         }, | ||||
|         "selectStowed": (event, button, dialog) => { | ||||
|           options.rollItem.enableStowed = !options.rollItem.enableStowed | ||||
|           this.updateResourceDialog(options) | ||||
|         }, | ||||
|         "selectStorage": (event, button, dialog) => { | ||||
|           options.rollItem.enableStorage = !options.rollItem.enableStorage | ||||
|           this.updateResourceDialog(options) | ||||
|         } | ||||
|       }, | ||||
|       rejectClose: false, // Click on Close button will not launch an error | ||||
|       render: (event, dialog) => { | ||||
|         $(".roll-skill-multiplier").change(event => { | ||||
|           options.multiplier = Number(event.target.value) | ||||
|           this.updateResourceDialog(options) | ||||
|         FTLNomadRoll.updateFullFormula(options) | ||||
|         $(".roll-skill-modifier").change(event => { | ||||
|           options.skillModifier = Number(event.target.value) | ||||
|           FTLNomadRoll.updateFullFormula(options) | ||||
|         }) | ||||
|         $(".roll-damage-modifier").change(event => { | ||||
|           options.damageModifier = Number(event.target.value) | ||||
|           FTLNomadRoll.updateFullFormula(options) | ||||
|         }) | ||||
|         $(".roll-skill-range-modifier").change(event => { | ||||
|           options.rangeModifier = Number(event.target.value) | ||||
|           FTLNomadRoll.updateFullFormula(options) | ||||
|         }) | ||||
|         $(".select-combat-option").change(event => { | ||||
|           let field = $(event.target).data("field") | ||||
|           let modifier = SYSTEM.ATTACK_MODIFIERS[field] | ||||
|           if (event.target.checked) { | ||||
|             options.numericModifier += modifier | ||||
|           } else { | ||||
|             options.numericModifier -= modifier | ||||
|           } | ||||
|           FTLNomadRoll.updateFullFormula(options) | ||||
|         }) | ||||
|       } | ||||
|     }) | ||||
| @@ -285,25 +215,50 @@ export default class FTLNomadRoll extends Roll { | ||||
|  | ||||
|     let rollData = foundry.utils.mergeObject(foundry.utils.duplicate(options), rollContext) | ||||
|     rollData.rollMode = rollContext.visibility | ||||
|  | ||||
|     // Update target score | ||||
|     console.log("Rolldata", rollData, options) | ||||
|     if (options.rollType === "resource") { | ||||
|       rollData.targetScore = options.initialScore * Number(rollContext.multiplier) | ||||
|     } else { | ||||
|       rollData.targetScore = Math.min(Math.max(options.initialScore + Number(rollData.modifier), 0), 100) | ||||
|       if (rollData.isLowWP || rollData.isExhausted) { | ||||
|         rollData.targetScore -= 20 | ||||
|       } | ||||
|       if (rollData.isZeroWP) { | ||||
|         rollData.targetScore = 0 | ||||
|       } | ||||
|       rollData.targetScore = Math.min(Math.max(rollData.targetScore, 0), 100) | ||||
|     } | ||||
|     rollData.targetScore = 8 | ||||
|  | ||||
|     if (Hooks.call("fvtt-ftl-nomad.preRoll", options, rollData) === false) return | ||||
|  | ||||
|     const roll = new this(formula, options.data, rollData) | ||||
|     options.numericModifier = Number(rollData.numericModifier) || 0 | ||||
|     options.skillModifier = Number(rollData.skillModifier) || 0 | ||||
|     options.rangeModifier = Number(rollData.rangeModifier) || 0 | ||||
|     options.finalModifier = options.numericModifier + options.skillModifier + options.rangeModifier | ||||
|     let mod = options.rollItem?.value || 0 | ||||
|  | ||||
|     // Build the dice formula | ||||
|     let diceFormula = "2d6" | ||||
|     if (options.rollType === "damage") { | ||||
|       let damageFormula = options.weapon.system.damage.toUpperCase().replace(/D/g, "d") | ||||
|       // Extract the mod (if present), like in 3d6+1 | ||||
|       let match = damageFormula.match(/([+-]\d+)$/) | ||||
|       if (match) { | ||||
|         mod += Number(match[1]) | ||||
|         damageFormula = damageFormula.replace(match[1], "") | ||||
|       } | ||||
|       // Replace the D6 by the correct number of D6 | ||||
|       damageFormula = damageFormula.replace(/(\d*)d6/gi, (match, p1) => { | ||||
|         let numDice = Number(p1) || 1 | ||||
|         let damageModifier = Number(options.damageModifier) | ||||
|         let postArg = "" | ||||
|         if (damageModifier && damageModifier !== 0) { | ||||
|           postArg = (damageModifier > 0) ? `kh${numDice}` : `kl${numDice}` | ||||
|           numDice += Math.abs(damageModifier) | ||||
|         } | ||||
|         return `${numDice}d6${postArg} + ${mod}` | ||||
|       }) | ||||
|       diceFormula = damageFormula | ||||
|     } else { | ||||
|       diceFormula = `${2 + Math.abs(options.finalModifier)}D6` | ||||
|       if (options.finalModifier > 0) { | ||||
|         diceFormula += `kh2 + ${mod}` | ||||
|       } else { | ||||
|         diceFormula += `kl2 + ${mod}` | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     console.log("FTLNomadRoll | Rolling ", diceFormula, options, rollData) | ||||
|     const roll = new this(diceFormula, options.data, rollData) | ||||
|     await roll.evaluate() | ||||
|  | ||||
|     roll.displayRollResult(roll, options, rollData) | ||||
| @@ -317,34 +272,15 @@ export default class FTLNomadRoll extends Roll { | ||||
|  | ||||
|     // Compute the result quality | ||||
|     let resultType = "failure" | ||||
|     let dec = Math.floor(this.total / 10) | ||||
|     let unit = this.total - (dec * 10) | ||||
|     if (this.total <= rollData.targetScore) { | ||||
|     if (this.total >= 8) { | ||||
|       resultType = "success" | ||||
|       // Detect if decimal == unit in the dire total result  | ||||
|       if (dec === unit || this.total === 1) { | ||||
|         resultType = "successCritical" | ||||
|       } | ||||
|     } else { | ||||
|       // Detect if decimal == unit in the dire total result  | ||||
|       if (dec === unit || this.total === 100) { | ||||
|         resultType = "failureCritical" | ||||
|       } | ||||
|       // Detect if decimal == unit in the dire total result | ||||
|     } | ||||
|  | ||||
|     this.options.resultType = resultType | ||||
|     if (this.options.isNudgedRoll) {  | ||||
|       this.options.isSuccess = resultType === "success" || resultType === "successCritical" | ||||
|       this.options.isFailure = resultType === "failure" || resultType === "failureCritical" | ||||
|       this.options.isCritical = false | ||||
|     } else { | ||||
|       this.options.isSuccess = resultType === "success" || resultType === "successCritical" | ||||
|       this.options.isFailure = resultType === "failure" || resultType === "failureCritical" | ||||
|       this.options.isCritical = resultType === "successCritical" || resultType === "failureCritical" | ||||
|     } | ||||
|     this.options.isLowWP = rollData.isLowWP | ||||
|     this.options.isZeroWP = rollData.isZeroWP | ||||
|     this.options.isExhausted = rollData.isExhausted | ||||
|     this.options.isSuccess = resultType === "success" | ||||
|     this.options.isFailure = resultType === "failure" | ||||
|     this.options.isEncumbered = rollData.isEncumbered | ||||
|     this.options.rollData = foundry.utils.duplicate(rollData) | ||||
|   } | ||||
|  | ||||
| @@ -357,14 +293,12 @@ export default class FTLNomadRoll extends Roll { | ||||
|    */ | ||||
|   static createTitle(type, target) { | ||||
|     switch (type) { | ||||
|       case "damage": | ||||
|         return `${game.i18n.localize("FTLNOMAD.Label.titleDamage")}` | ||||
|       case "skill": | ||||
|         return `${game.i18n.localize("FTLNOMAD.Label.titleSkill")}` | ||||
|       case "weapon": | ||||
|         return `${game.i18n.localize("FTLNOMAD.Label.titleWeapon")}` | ||||
|       case "char": | ||||
|         return `${game.i18n.localize("FTLNOMAD.Label.titleCharacteristic")}` | ||||
|       case "san": | ||||
|         return `${game.i18n.localize("FTLNOMAD.Label.titleSAN")}` | ||||
|       default: | ||||
|         return game.i18n.localize("FTLNOMAD.Label.titleStandard") | ||||
|     } | ||||
| @@ -373,7 +307,7 @@ export default class FTLNomadRoll extends Roll { | ||||
|   /** @override */ | ||||
|   async render(chatOptions = {}) { | ||||
|     let chatData = await this._getChatCardData(chatOptions.isPrivate) | ||||
|     return await renderTemplate(this.constructor.CHAT_TEMPLATE, chatData) | ||||
|     return await foundry.applications.handlebars.renderTemplate(this.constructor.CHAT_TEMPLATE, chatData) | ||||
|   } | ||||
|  | ||||
|   /** | ||||
| @@ -407,6 +341,8 @@ export default class FTLNomadRoll extends Roll { | ||||
|     cardData.diceTotal = this.dice.reduce((t, d) => t + d.total, 0) | ||||
|     cardData.isGM = game.user.isGM | ||||
|     cardData.formula = this.formula | ||||
|     cardData.fullFormula = this.options.fullFormula | ||||
|     cardData.numericModifier = this.options.numericModifier | ||||
|     cardData.total = this.total | ||||
|     cardData.actorId = this.actorId | ||||
|     cardData.actingCharName = this.actorName | ||||
| @@ -418,11 +354,7 @@ export default class FTLNomadRoll extends Roll { | ||||
|     cardData.realDamage = this.realDamage | ||||
|     cardData.isPrivate = isPrivate | ||||
|     cardData.weapon = this.weapon | ||||
|     cardData.isLowWP = this.isLowWP | ||||
|     cardData.isZeroWP = this.isZeroWP | ||||
|     cardData.isExhausted = this.isExhausted | ||||
|     cardData.isNudgedRoll = this.isNudgedRoll | ||||
|     cardData.wpCost = this.wpCost | ||||
|     cardData.isEncumbered = this.isEncumbered | ||||
|  | ||||
|     cardData.cssClass = cardData.css.join(" ") | ||||
|     cardData.tooltip = isPrivate ? "" : await this.getTooltip() | ||||
|   | ||||
| @@ -11,4 +11,4 @@ export { default as FTLNomadImplant } from "./implant.mjs" | ||||
| export { default as FTLNomadStarship } from "./starship.mjs" | ||||
| export { default as FTLNomadCreatureTrait } from "./creature-trait.mjs" | ||||
| export { default as FTLNomadCreatureAbility } from "./creature-ability.mjs" | ||||
|  | ||||
| export { default as FTLNomadRobot } from "./robot.mjs" | ||||
|   | ||||
| @@ -18,6 +18,7 @@ export default class FTLNomadProtagonist extends foundry.abstract.TypeDataModel | ||||
|     const skillField = (label) => { | ||||
|       const schema = { | ||||
|         value: new fields.NumberField({ ...requiredInteger, initial: 0, min: 0, max: 5 }), | ||||
|         label: new fields.StringField({ required: true, nullable: false, initial: label }) | ||||
|       } | ||||
|       return new fields.SchemaField(schema, { label }) | ||||
|     } | ||||
| @@ -34,8 +35,8 @@ export default class FTLNomadProtagonist extends foundry.abstract.TypeDataModel | ||||
|     schema.health = new fields.SchemaField({ | ||||
|       staminaValue: new fields.NumberField({ ...requiredInteger, initial: 1, min: 0 }), | ||||
|       staminaMax: new fields.NumberField({ ...requiredInteger, initial: 1, min: 0 }), | ||||
|       wounds: new fields.NumberField({ ...requiredInteger, initial: 1, min: 0 }), | ||||
|       triageResults:  new fields.StringField({ required: true, nullable: false, initial: "none", choices: SYSTEM.TRIAGE_RESULTS }) | ||||
|       wounds: new fields.NumberField({ ...requiredInteger, initial: 0, min: 0 }), | ||||
|       triageResults: new fields.StringField({ required: true, nullable: false, initial: "none", choices: SYSTEM.TRIAGE_RESULTS }) | ||||
|     }) | ||||
|  | ||||
|     schema.enc = new fields.SchemaField({ | ||||
| @@ -43,6 +44,10 @@ export default class FTLNomadProtagonist extends foundry.abstract.TypeDataModel | ||||
|       max: new fields.NumberField({ ...requiredInteger, initial: 0, min: 0 }) | ||||
|     }) | ||||
|  | ||||
|     schema.armor = new fields.SchemaField({ | ||||
|       value: new fields.NumberField({ ...requiredInteger, initial: 0, min: 0 }) | ||||
|     }) | ||||
|  | ||||
|     schema.credits = new fields.NumberField({ ...requiredInteger, initial: 0, min: 0 }) | ||||
|     schema.rank = new fields.SchemaField({ | ||||
|       experienced: new fields.NumberField({ ...requiredInteger, initial: 0, min: 0, max: 5 }), | ||||
| @@ -55,6 +60,7 @@ export default class FTLNomadProtagonist extends foundry.abstract.TypeDataModel | ||||
|     schema.biodata = new fields.SchemaField({ | ||||
|       age: new fields.NumberField({ ...requiredInteger, initial: 15, min: 6 }), | ||||
|       height: new fields.NumberField({ ...requiredInteger, initial: 170, min: 50 }), | ||||
|       weight: new fields.NumberField({ ...requiredInteger, initial: 70, min: 1 }), | ||||
|       gender: new fields.StringField({ required: true, nullable: false, initial: "" }), | ||||
|       home: new fields.StringField({ required: true, nullable: false, initial: "" }), | ||||
|       birthplace: new fields.StringField({ required: true, nullable: false, initial: "" }), | ||||
| @@ -71,8 +77,36 @@ export default class FTLNomadProtagonist extends foundry.abstract.TypeDataModel | ||||
|   prepareDerivedData() { | ||||
|     super.prepareDerivedData(); | ||||
|  | ||||
|     let encMax = 10 + (2 * this.skills.physical.value) | ||||
|     if (encMax !== this.enc.max) { | ||||
|       this.enc.max = encMax | ||||
|     } | ||||
|     let enc = 0 | ||||
|     let armor = 0 | ||||
|     for (let i of this.parent.items) { | ||||
|       if (i.system?.enc) { | ||||
|         enc += i.system.enc | ||||
|       } | ||||
|       if (i.system?.protection) { | ||||
|         armor += i.system.protection | ||||
|       } | ||||
|     } | ||||
|     if (enc !== this.enc.value) { | ||||
|       this.enc.value = enc | ||||
|     } | ||||
|     if (armor !== this.armor.value) { | ||||
|       this.armor.value = armor | ||||
|     } | ||||
|     let staminaMax = 14 + (3 * this.skills.physical.value) | ||||
|     if (staminaMax !== this.health.staminaMax) { | ||||
|       this.health.staminaMax = staminaMax | ||||
|     } | ||||
|   } | ||||
|    | ||||
|  | ||||
|   isEncumbered() { | ||||
|     return this.enc.value > this.enc.max | ||||
|   } | ||||
|  | ||||
|   /** */ | ||||
|   /** | ||||
|    * Rolls a dice for a character. | ||||
| @@ -84,12 +118,14 @@ export default class FTLNomadProtagonist extends foundry.abstract.TypeDataModel | ||||
|     let opponentTarget | ||||
|     const hasTarget = opponentTarget !== undefined | ||||
|  | ||||
|     let roll = await CthulhuEternalRoll.prompt({ | ||||
|     let roll = await FTLNomadRoll.prompt({ | ||||
|       rollType, | ||||
|       rollItem, | ||||
|       actorId: this.parent.id, | ||||
|       actorName: this.parent.name, | ||||
|       actorImage: this.parent.img, | ||||
|       talents: this.parent.items.filter(i => i.type === "talent" && i.system.isAdvantage), | ||||
|       isEncumbered: this.isEncumbered(), | ||||
|       hasTarget, | ||||
|       target: opponentTarget | ||||
|     }) | ||||
|   | ||||
| @@ -7,9 +7,11 @@ export default class FTLNomadCreature extends foundry.abstract.TypeDataModel { | ||||
|     const requiredInteger = { required: true, nullable: false, integer: true } | ||||
|     const schema = {} | ||||
|  | ||||
|     // Carac | ||||
|     const skillField = (label) => { | ||||
|       const schema = { | ||||
|         value: new fields.NumberField({ ...requiredInteger, initial: 0, min: 0, max: 5 }), | ||||
|         label: new fields.StringField({ required: true, nullable: false, initial: label }), | ||||
|         enabled: new fields.BooleanField({ required: true, initial: true }), | ||||
|       } | ||||
|       return new fields.SchemaField(schema, { label }) | ||||
| @@ -29,7 +31,7 @@ export default class FTLNomadCreature extends foundry.abstract.TypeDataModel { | ||||
|     schema.numberAppearing = new fields.StringField({ required: true, initial: "1d6" }) | ||||
|     schema.health = new fields.SchemaField({ | ||||
|       staminaValue: new fields.NumberField({ ...requiredInteger, initial: 1, min: 0 }), | ||||
|       staminaMax: new fields.NumberField({ ...requiredInteger, initial: 1, min: 0 }), | ||||
|       staminaMax: new fields.NumberField({ ...requiredInteger, initial: 1, min: 1 }), | ||||
|     }) | ||||
|     schema.damage = new fields.StringField({ required: true, initial: "1d6" }) | ||||
|  | ||||
| @@ -42,32 +44,29 @@ export default class FTLNomadCreature extends foundry.abstract.TypeDataModel { | ||||
|   /** @override */ | ||||
|   static LOCALIZATION_PREFIXES = ["FTLNOMAD.Creature"] | ||||
|  | ||||
|     /** */ | ||||
|     /** | ||||
|      * Rolls a dice for a character. | ||||
|      * @param {("save"|"resource|damage")} rollType The type of the roll. | ||||
|      * @param {number} rollItem The target value for the roll. Which caracteristic or resource. If the roll is a damage roll, this is the id of the item. | ||||
|      * @returns {Promise<null>} - A promise that resolves to null if the roll is cancelled. | ||||
|      */ | ||||
|     async roll(rollType, rollItem) { | ||||
|       let opponentTarget | ||||
|       const hasTarget = opponentTarget !== undefined | ||||
|    | ||||
|       let roll = await CthulhuEternalRoll.prompt({ | ||||
|         rollType, | ||||
|         rollItem, | ||||
|         isLowWP: false, | ||||
|         isZeroWP: false, | ||||
|         isExhausted: false, | ||||
|         actorId: this.parent.id, | ||||
|         actorName: this.parent.name, | ||||
|         actorImage: this.parent.img, | ||||
|         hasTarget, | ||||
|         target: opponentTarget | ||||
|       }) | ||||
|       if (!roll) return null | ||||
|    | ||||
|       await roll.toMessage({}, { rollMode: roll.options.rollMode }) | ||||
|     } | ||||
|    | ||||
|   isEncumbered() { | ||||
|     return false | ||||
|   } | ||||
|  | ||||
|   async roll(rollType, rollItem) { | ||||
|     let opponentTarget | ||||
|     const hasTarget = opponentTarget !== undefined | ||||
|  | ||||
|     let roll = await FTLNomadRoll.prompt({ | ||||
|       rollType, | ||||
|       rollItem, | ||||
|       actorId: this.parent.id, | ||||
|       actorName: this.parent.name, | ||||
|       actorImage: this.parent.img, | ||||
|       traits: this.parent.items.filter(i => i.type === "creature-trait" && i.system.isAdvantage), | ||||
|       abilities: this.parent.items.filter(i => i.type === "creature-ability" && i.system.isAdvantage), | ||||
|       isEncumbered: this.isEncumbered(), | ||||
|       hasTarget, | ||||
|       target: opponentTarget | ||||
|     }) | ||||
|     if (!roll) return null | ||||
|  | ||||
|     await roll.toMessage({}, { rollMode: roll.options.rollMode }) | ||||
|   } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -7,11 +7,11 @@ export default class FTLNomadEquipment extends foundry.abstract.TypeDataModel { | ||||
|     const requiredInteger = { required: true, nullable: false, integer: true } | ||||
|  | ||||
|     schema.description = new fields.HTMLField({ required: true, textSearch: true }) | ||||
|      | ||||
|  | ||||
|     schema.techAge = new fields.StringField({ required: true, choices: SYSTEM.TECH_AGES, initial : "lateatomic" }) | ||||
|  | ||||
|     schema.enc = new fields.NumberField({ ...requiredInteger, required: true, initial: 0, min: 0 }) | ||||
|     schema.cost = new fields.NumberField({ required: true, initial: 0, min: 0 }) | ||||
|     schema.cost = new fields.NumberField({ required: true, initial: 1, min: 0 }) | ||||
|  | ||||
|     return schema | ||||
|   } | ||||
|   | ||||
| @@ -7,11 +7,12 @@ export default class FTLNomadImplant extends foundry.abstract.TypeDataModel { | ||||
|     const requiredInteger = { required: true, nullable: false, integer: true } | ||||
|  | ||||
|     schema.description = new fields.HTMLField({ required: true, textSearch: true }) | ||||
|      | ||||
|  | ||||
|     schema.techAge = new fields.StringField({ required: true, choices: SYSTEM.TECH_AGES, initial : "lateatomic" }) | ||||
|  | ||||
|     schema.enc = new fields.NumberField({ ...requiredInteger, required: true, initial: 0, min: 0 }) | ||||
|     schema.cost = new fields.NumberField({ required: true, initial: 0, min: 0 }) | ||||
|     schema.loss = new fields.NumberField({ required: true, initial: 0, min: 0 }) | ||||
|  | ||||
|     return schema | ||||
|   } | ||||
|   | ||||
							
								
								
									
										84
									
								
								module/models/robot.mjs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,84 @@ | ||||
| import { SYSTEM } from "../config/system.mjs" | ||||
| import FTLNomadRoll from "../documents/roll.mjs" | ||||
|  | ||||
| export default class FTLNomadRobot extends foundry.abstract.TypeDataModel { | ||||
|   static defineSchema() { | ||||
|     const fields = foundry.data.fields | ||||
|     const requiredInteger = { required: true, nullable: false, integer: true } | ||||
|     const schema = {} | ||||
|  | ||||
|     schema.description = new fields.HTMLField({ required: true, textSearch: true }) | ||||
|     schema.notes = new fields.HTMLField({ required: true, textSearch: true }) | ||||
|     schema.name = new fields.StringField({ required: true, nullable: false, initial: "" }) | ||||
|  | ||||
|     schema.techAge = new fields.StringField({ required: true, choices: SYSTEM.TECH_AGES, initial: "lateatomic" }) | ||||
|     schema.robotSize = new fields.StringField({ required: true, choices: SYSTEM.ROBOT_SIZES, initial: "medium" }) | ||||
|     schema.durability = new fields.NumberField({ ...requiredInteger, initial: 1, min: 0 }) | ||||
|     schema.protection = new fields.NumberField({ ...requiredInteger, initial: 0, min: 0 }) | ||||
|     schema.brain = new fields.StringField({ required: true, nullable: false, initial: "" }) | ||||
|     schema.speed = new fields.StringField({ required: true, nullable: false, initial: "" }) | ||||
|  | ||||
|     // Carac | ||||
|     const skillField = (label) => { | ||||
|       const schema = { | ||||
|         value: new fields.NumberField({ ...requiredInteger, initial: 0, min: 0, max: 5 }), | ||||
|         label: new fields.StringField({ required: true, nullable: false, initial: label }) | ||||
|       } | ||||
|       return new fields.SchemaField(schema, { label }) | ||||
|     } | ||||
|  | ||||
|     schema.skills = new fields.SchemaField( | ||||
|       Object.values(SYSTEM.SKILLS).reduce((obj, characteristic) => { | ||||
|         obj[characteristic.id] = skillField(characteristic.label) | ||||
|         return obj | ||||
|       }, {}), | ||||
|     ) | ||||
|  | ||||
|     schema.enc = new fields.SchemaField({ | ||||
|       value: new fields.NumberField({ ...requiredInteger, initial: 0, min: 0 }), | ||||
|       max: new fields.NumberField({ ...requiredInteger, initial: 0, min: 0 }) | ||||
|     }) | ||||
|  | ||||
|     schema.cost = new fields.StringField({ required: true, initial: "0" }) | ||||
|  | ||||
|     return schema | ||||
|   } | ||||
|  | ||||
|   /** @override */ | ||||
|   static LOCALIZATION_PREFIXES = ["FTLNOMAD.Robot"] | ||||
|  | ||||
|   prepareDerivedData() { | ||||
|     super.prepareDerivedData(); | ||||
|  | ||||
|   } | ||||
|  | ||||
|   isEncumbered() { | ||||
|     return this.enc.value > this.enc.max | ||||
|   } | ||||
|  | ||||
|   /** */ | ||||
|   /** | ||||
|    * Rolls a dice for a character. | ||||
|    * @param {("save"|"resource|damage")} rollType The type of the roll. | ||||
|    * @param {number} rollItem The target value for the roll. Which caracteristic or resource. If the roll is a damage roll, this is the id of the item. | ||||
|    * @returns {Promise<null>} - A promise that resolves to null if the roll is cancelled. | ||||
|    */ | ||||
|   async roll(rollType, rollItem) { | ||||
|     let opponentTarget | ||||
|     const hasTarget = opponentTarget !== undefined | ||||
|  | ||||
|     let roll = await FTLNomadRoll.prompt({ | ||||
|       rollType, | ||||
|       rollItem, | ||||
|       actorId: this.parent.id, | ||||
|       actorName: this.parent.name, | ||||
|       actorImage: this.parent.img, | ||||
|       isEncumbered: this.isEncumbered(), | ||||
|       hasTarget, | ||||
|       target: opponentTarget | ||||
|     }) | ||||
|     if (!roll) return null | ||||
|  | ||||
|     await roll.toMessage({}, { rollMode: roll.options.rollMode }) | ||||
|   } | ||||
| } | ||||
| @@ -11,15 +11,17 @@ export default class FTLNomadStarship extends foundry.abstract.TypeDataModel { | ||||
|     schema.hullType = new fields.StringField({ required: true, initial: "small", choices: SYSTEM.STARSHIP_HULL }) | ||||
|     schema.endurance = new fields.StringField({ required: true, initial: "" }) | ||||
|     schema.armor = new fields.StringField({ required: true, initial: "" }) | ||||
|     schema.crew = new fields.NumberField({ ...requiredInteger, initial: 1, min: 1 }) | ||||
|     schema.cargo = new fields.NumberField({ ...requiredInteger, initial: 0, min: 0 }) | ||||
|     schema.crew = new fields.StringField({ required: true, initial: "" }) | ||||
|     schema.cargo = new fields.StringField({ required: true, initial: "" }) | ||||
|     schema.guns = new fields.StringField({ required: true, initial: "1d6" }) | ||||
|     schema.travelMultiplier = new fields.NumberField({ required: true, initial: 1, min: 0 }) | ||||
|     schema.cost = new fields.NumberField({ required: true, initial: 0, min: 0 }) | ||||
|     schema.monthlyCost = new fields.NumberField({ required: true, initial: 0, min: 0 }) | ||||
|      | ||||
|     schema.travelMultiplier = new fields.StringField({ required: true, initial: "" }) | ||||
|     schema.cost = new fields.StringField({ required: true, initial: "0" }) | ||||
|     schema.monthlyCost = new fields.StringField({ required: true, initial: "0" }) | ||||
|     schema.damages = new fields.StringField({ required: true, initial: "" }) | ||||
|     schema.techAge = new fields.StringField({ required: true, choices: SYSTEM.TECH_AGES, initial: "lateatomic" }) | ||||
|  | ||||
|     schema.description = new fields.HTMLField({ required: true, textSearch: true }) | ||||
|     schema.modifications = new fields.HTMLField({ required: true, textSearch: true })  | ||||
|     schema.modifications = new fields.HTMLField({ required: true, textSearch: true }) | ||||
|     schema.notes = new fields.HTMLField({ required: true, textSearch: true }) | ||||
|  | ||||
|     return schema | ||||
| @@ -28,4 +30,28 @@ export default class FTLNomadStarship extends foundry.abstract.TypeDataModel { | ||||
|   /** @override */ | ||||
|   static LOCALIZATION_PREFIXES = ["FTLNOMAD.Starship"] | ||||
|  | ||||
|   isEncumbered() { | ||||
|     return false | ||||
|   } | ||||
|  | ||||
|   async roll(rollType, rollItem) { | ||||
|     let opponentTarget | ||||
|     const hasTarget = opponentTarget !== undefined | ||||
|  | ||||
|     let roll = await FTLNomadRoll.prompt({ | ||||
|       rollType, | ||||
|       rollItem, | ||||
|       actorId: this.parent.id, | ||||
|       actorName: this.parent.name, | ||||
|       actorImage: this.parent.img, | ||||
|       isEncumbered: false, | ||||
|       hasTarget, | ||||
|       target: opponentTarget | ||||
|     }) | ||||
|     if (!roll) return null | ||||
|  | ||||
|     await roll.toMessage({}, { rollMode: roll.options.rollMode }) | ||||
|   } | ||||
|  | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -9,16 +9,17 @@ export default class FTLNomadVehicle extends foundry.abstract.TypeDataModel { | ||||
|  | ||||
|     schema.agility = new fields.NumberField({ ...requiredInteger, initial: 0, min: 0 }) | ||||
|     schema.armor = new fields.StringField({ required: true, initial: "" }) | ||||
|     schema.cargo = new fields.NumberField({ ...requiredInteger, initial: 0, min: 0 }) | ||||
|     schema.crew = new fields.NumberField({ ...requiredInteger, initial: 1, min: 1 }) | ||||
|     schema.cargo = new fields.StringField({ required: true, initial: "" }) | ||||
|     schema.crew = new fields.StringField({ required: true, initial: "" }) | ||||
|     schema.force = new fields.NumberField({ ...requiredInteger, initial: 1, min: 1 }) | ||||
|     schema.range = new fields.StringField({ required: true, initial: "1d6" }) | ||||
|     schema.speed = new fields.StringField({ required: true, initial: "1d6" }) | ||||
|     schema.techAge = new fields.StringField({ required: true, initial: "1d6" }) | ||||
|     schema.tonnage = new fields.NumberField({ required: true, initial: 1, min: 0 }) | ||||
|     schema.techAge = new fields.StringField({ required: true, choices: SYSTEM.TECH_AGES, initial: "lateatomic" }) | ||||
|     schema.tonnage = new fields.StringField({ required: true, initial: "" }) | ||||
|     schema.damages = new fields.StringField({ required: true, initial: "" }) | ||||
|  | ||||
|     schema.cost = new fields.StringField({ required: true, initial: "0" }) | ||||
|  | ||||
|     schema.cost = new fields.NumberField({ required: true, initial: 0, min: 0 }) | ||||
|      | ||||
|     schema.description = new fields.HTMLField({ required: true, textSearch: true }) | ||||
|     schema.notes = new fields.HTMLField({ required: true, textSearch: true }) | ||||
|  | ||||
| @@ -28,4 +29,27 @@ export default class FTLNomadVehicle extends foundry.abstract.TypeDataModel { | ||||
|   /** @override */ | ||||
|   static LOCALIZATION_PREFIXES = ["FTLNOMAD.Vehicle"] | ||||
|  | ||||
|   isEncumbered() { | ||||
|     return false | ||||
|   } | ||||
|  | ||||
|   async roll(rollType, rollItem) { | ||||
|     let opponentTarget | ||||
|     const hasTarget = opponentTarget !== undefined | ||||
|  | ||||
|     let roll = await FTLNomadRoll.prompt({ | ||||
|       rollType, | ||||
|       rollItem, | ||||
|       actorId: this.parent.id, | ||||
|       actorName: this.parent.name, | ||||
|       actorImage: this.parent.img, | ||||
|       isEncumbered: this.isEncumbered(), | ||||
|       hasTarget, | ||||
|       target: opponentTarget | ||||
|     }) | ||||
|     if (!roll) return null | ||||
|  | ||||
|     await roll.toMessage({}, { rollMode: roll.options.rollMode }) | ||||
|   } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -7,27 +7,27 @@ export default class FTLNomadWeapon extends foundry.abstract.TypeDataModel { | ||||
|     const requiredInteger = { required: true, nullable: false, integer: true } | ||||
|  | ||||
|     schema.description = new fields.HTMLField({ required: true, textSearch: true }) | ||||
|      | ||||
|     schema.techAge = new fields.StringField({ required: true, choices: SYSTEM.TECH_AGES, initial : "lateatomic" }) | ||||
|  | ||||
|     schema.techAge = new fields.StringField({ required: true, choices: SYSTEM.TECH_AGES, initial: "lateatomic" }) | ||||
|     schema.weaponType = new fields.StringField({ required: true, initial: "melee", choices: SYSTEM.WEAPON_TYPES }) | ||||
|     schema.rangeType = new fields.StringField({ required: true, initial: "melee", choices: SYSTEM.WEAPON_RANGE }) | ||||
|      | ||||
|     schema.damage = new fields.StringField({required: true, initial: "1d6"})   | ||||
|  | ||||
|     schema.damage = new fields.StringField({ required: true, initial: "1d6" }) | ||||
|     schema.magazine = new fields.NumberField({ required: true, initial: 1, min: 0 }) | ||||
|  | ||||
|     schema.range =  new fields.SchemaField({ | ||||
|     schema.range = new fields.SchemaField({ | ||||
|       close: new fields.NumberField({ ...requiredInteger, initial: 0 }), | ||||
|       near: new fields.NumberField({ ...requiredInteger, initial: 0 }), | ||||
|       far: new fields.NumberField({ ...requiredInteger, initial: 0 }), | ||||
|       dist: new fields.NumberField({ ...requiredInteger, initial: 0 }), | ||||
|     }) | ||||
|  | ||||
|     schema.enc = new fields.NumberField({ required: true, initial: 0, min: 0 }) | ||||
|     schema.aspect = new fields.StringField({ required: true, initial: ""}) | ||||
|      | ||||
|     schema.enc = new fields.NumberField({ required: true, initial: 1, min: 0 }) | ||||
|     schema.aspect = new fields.StringField({ required: true, initial: "" }) | ||||
|  | ||||
|     schema.cost = new fields.NumberField({ required: true, initial: 0, min: 0 }) | ||||
|     schema.ammoCost = new fields.NumberField({ required: true, initial: 0, min: 0 }) | ||||
|      | ||||
|  | ||||
|     return schema | ||||
|   } | ||||
|  | ||||
|   | ||||
							
								
								
									
										557
									
								
								module/utils.mjs
									
									
									
									
									
								
							
							
						
						| @@ -195,4 +195,561 @@ export default class FTLNomadUtils { | ||||
|     document.documentElement.style.setProperty('--background-image-base', `linear-gradient(rgba(255, 255, 255, 0.8), rgba(255, 255, 255, 0.8)), url("../assets/ui/${era}_background_main.webp")`); | ||||
|   } | ||||
|  | ||||
|   static getTechAgeKeyFromLabel(label) { | ||||
|     return Object.keys(SYSTEM.TECH_AGES).find(key => game.i18n.localize(SYSTEM.TECH_AGES[key].label) === label); | ||||
|   } | ||||
|  | ||||
|   static async importPsionics() { | ||||
|     // Create a psionics folder if it doesn't exist | ||||
|     const psionicsFolder = game.folders.getName("Psionics") || await Folder.create({ | ||||
|       name: "Psionics", type: "Item" | ||||
|     }) | ||||
|     if (!psionicsFolder) { | ||||
|       console.error("Failed to create Psionics folder"); | ||||
|       return; | ||||
|     } | ||||
|     // Load the psionics JSON file | ||||
|     const psionicsData = await fetch("systems/fvtt-ftl-nomad/assets/json_data/psionics.json") | ||||
|       .then(response => response.json()) | ||||
|       .catch(error => { | ||||
|         console.error("Failed to load psionics data:", error); | ||||
|         return []; | ||||
|       }); | ||||
|     // Import each psionic ability | ||||
|     for (const psionic of psionicsData) { | ||||
|       // Check if the psionic already exists | ||||
|       const existingPsionic = game.items.find(i => i.name === psionic.name && i.type === "psionic"); | ||||
|       if (existingPsionic) { | ||||
|         console.warn(`Psionic ${psionic.name} already exists, skipping import.`); | ||||
|         continue; | ||||
|       } | ||||
|       // Create the psionic item | ||||
|       await Item.create({ | ||||
|         name: psionic.name, | ||||
|         type: "psionic", | ||||
|         img: "systems/fvtt-ftl-nomad/assets/icons/icon_psionic.svg", | ||||
|         system: { | ||||
|           description: psionic.description, | ||||
|         }, | ||||
|         folder: psionicsFolder.id | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async importTalents() { | ||||
|     // Create a talents folder if it doesn't exist | ||||
|     const talentsFolder = game.folders.getName("Talents") || await Folder.create({ | ||||
|       name: "Talents", type: "Item" | ||||
|     }) | ||||
|     if (!talentsFolder) { | ||||
|       console.error("Failed to create Talents folder"); | ||||
|       return; | ||||
|     } | ||||
|     // Load the talents JSON file | ||||
|     const talentsData = await fetch("systems/fvtt-ftl-nomad/assets/json_data/talents.json") | ||||
|       .then(response => response.json()) | ||||
|       .catch(error => { | ||||
|         console.error("Failed to load talents data:", error); | ||||
|         return []; | ||||
|       }); | ||||
|     // Import each talent | ||||
|     for (const talent of talentsData) { | ||||
|       // Check if the talent already exists | ||||
|       const existingTalent = game.items.find(i => i.name === talent.name && i.type === "talent"); | ||||
|       if (existingTalent) { | ||||
|         console.warn(`Talent ${talent.name} already exists, skipping import.`); | ||||
|         continue; | ||||
|       } | ||||
|       talent.advantage = false; | ||||
|       if (talent.description.includes(" +1D")) { | ||||
|         talent.advantage = true | ||||
|       } | ||||
|       // Create the talent item | ||||
|       await Item.create({ | ||||
|         name: talent.name, | ||||
|         type: "talent", | ||||
|         img: "systems/fvtt-ftl-nomad/assets/icons/icon_talent.svg", | ||||
|         system: { | ||||
|           description: talent.description, | ||||
|           isAdvantage: talent.advantage, | ||||
|         }, | ||||
|         folder: talentsFolder.id | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async importImplants() { | ||||
|     // Create a implants folder if it doesn't exist | ||||
|     const implantsFolder = game.folders.getName("Implants") || await Folder.create({ | ||||
|       name: "Implants", type: "Item" | ||||
|     }) | ||||
|     if (!implantsFolder) { | ||||
|       console.error("Failed to create Implants folder"); | ||||
|       return; | ||||
|     } | ||||
|     // Load the implants JSON file | ||||
|     const implantsData = await fetch("systems/fvtt-ftl-nomad/assets/json_data/implants.json") | ||||
|       .then(response => response.json()) | ||||
|       .catch(error => { | ||||
|         console.error("Failed to load implants data:", error); | ||||
|         return []; | ||||
|       }); | ||||
|     // Import each implant | ||||
|     for (const implant of implantsData) { | ||||
|       // Check if the implant already exists | ||||
|       const existingImplant = game.items.find(i => i.name === implant.name && i.type === "implant"); | ||||
|       if (existingImplant) { | ||||
|         console.warn(`Implant ${implant.name} already exists, skipping import.`); | ||||
|         continue; | ||||
|       } | ||||
|       if (!Number(implant.cost)) { | ||||
|         implant.cost = 1000000; | ||||
|       } | ||||
|       // Create the implant item | ||||
|       await Item.create({ | ||||
|         name: implant.name, | ||||
|         type: "implant", | ||||
|         img: "systems/fvtt-ftl-nomad/assets/icons/icon_implant.svg", | ||||
|         system: { | ||||
|           cost: implant.cost, | ||||
|           loss: implant.loss, | ||||
|           techAge: this.getTechAgeKeyFromLabel(implant.tech_age), | ||||
|           description: implant.description, | ||||
|         }, | ||||
|         folder: implantsFolder.id | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async importDrugs() { | ||||
|     // Create a drugs folder if it doesn't exist | ||||
|     const drugsFolder = game.folders.getName("Drugs") || await Folder.create({ | ||||
|       name: "Drugs", type: "Item" | ||||
|     }) | ||||
|     if (!drugsFolder) { | ||||
|       console.error("Failed to create Drugs folder"); | ||||
|       return; | ||||
|     } | ||||
|     // Load the drugs JSON file | ||||
|     const drugsData = await fetch("systems/fvtt-ftl-nomad/assets/json_data/drugs.json") | ||||
|       .then(response => response.json()) | ||||
|       .catch(error => { | ||||
|         console.error("Failed to load drugs data:", error); | ||||
|         return []; | ||||
|       }); | ||||
|     // Import each drug | ||||
|     for (const drug of drugsData) { | ||||
|       // Check if the drug already exists | ||||
|       const existingDrug = game.items.find(i => i.name === drug.name && i.type === "drug"); | ||||
|       if (existingDrug) { | ||||
|         console.warn(`Drug ${drug.name} already exists, skipping import.`); | ||||
|         continue; | ||||
|       } | ||||
|       // Create the drug item | ||||
|       await Item.create({ | ||||
|         name: drug.name, | ||||
|         type: "equipment", | ||||
|         img: "systems/fvtt-ftl-nomad/assets/icons/icon_equipment.svg", | ||||
|         system: { | ||||
|           description: drug.description, | ||||
|           cost: drug.cost, | ||||
|           techAge: this.getTechAgeKeyFromLabel(drug.tech_age), | ||||
|         }, | ||||
|         folder: drugsFolder.id | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async importEquipments() { | ||||
|     // Create a equipments folder if it doesn't exist | ||||
|     const equipmentsFolder = game.folders.getName("Equipments") || await Folder.create({ | ||||
|       name: "Equipments", type: "Item" | ||||
|     }) | ||||
|     if (!equipmentsFolder) { | ||||
|       console.error("Failed to create Equipments folder"); | ||||
|       return; | ||||
|     } | ||||
|     // Load the equipments JSON file | ||||
|     const equipmentsData = await fetch("systems/fvtt-ftl-nomad/assets/json_data/exploration_equipment.json") | ||||
|       .then(response => response.json()) | ||||
|       .catch(error => { | ||||
|         console.error("Failed to load equipments data:", error); | ||||
|         return []; | ||||
|       }); | ||||
|     // Import each equipment | ||||
|     for (const equipment of equipmentsData) { | ||||
|       // Check if the equipment already exists | ||||
|       const existingEquipment = game.items.find(i => i.name === equipment.name && i.type === "equipment"); | ||||
|       if (existingEquipment) { | ||||
|         console.warn(`Equipment ${equipment.name} already exists, skipping import.`); | ||||
|         continue; | ||||
|       } | ||||
|       if (!Number(equipment.cost)) { | ||||
|         equipment.cost = 1000000; | ||||
|       } | ||||
|  | ||||
|       // Create the equipment item | ||||
|       await Item.create({ | ||||
|         name: equipment.name, | ||||
|         type: "equipment", | ||||
|         img: "systems/fvtt-ftl-nomad/assets/icons/icon_equipment.svg", | ||||
|         system: { | ||||
|           description: equipment.description, | ||||
|           cost: equipment.cost, | ||||
|           techAge: this.getTechAgeKeyFromLabel(equipment.tech_age), | ||||
|         }, | ||||
|         folder: equipmentsFolder.id | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async importDistanceWeapons() { | ||||
|     // Create a distance weapons folder if it doesn't exist | ||||
|     const distanceWeaponsFolder = game.folders.getName("Vehicle Weapons") || await Folder.create({ | ||||
|       name: "Vehicle Weapons", type: "Item" | ||||
|     }) | ||||
|     if (!distanceWeaponsFolder) { | ||||
|       console.error("Failed to create Distance Weapons folder"); | ||||
|       return; | ||||
|     } | ||||
|     // Load the distance weapons JSON file | ||||
|     const distanceWeaponsData = await fetch("systems/fvtt-ftl-nomad/assets/json_data/vehicle_weapons.json") | ||||
|       .then(response => response.json()) | ||||
|       .catch(error => { | ||||
|         console.error("Failed to load distance weapons data:", error); | ||||
|         return []; | ||||
|       }); | ||||
|  | ||||
|     // Import each distance weapon | ||||
|     for (const weapon of distanceWeaponsData) { | ||||
|       // Check if the weapon already exists | ||||
|       const existingWeapon = game.items.find(i => i.name === weapon.name && i.type === "weapon"); | ||||
|       if (existingWeapon) { | ||||
|         console.warn(`Weapon ${weapon.name} already exists, skipping import.`); | ||||
|         continue; | ||||
|       } | ||||
|       // Create the weapon item | ||||
|       await Item.create({ | ||||
|         name: weapon.name, | ||||
|         type: "weapon", | ||||
|         img: "systems/fvtt-ftl-nomad/assets/icons/icon_weapon.svg", | ||||
|         system: { | ||||
|           description: weapon.description, | ||||
|           damage: weapon.damage, | ||||
|           techAge: this.getTechAgeKeyFromLabel(weapon.tech_age), | ||||
|           weaponType: SYSTEM.WEAPON_TYPES.vehicle.id, | ||||
|           rangeType: weapon.range.toLowerCase(), | ||||
|           enc: 0, | ||||
|           aspect: weapon.aspects.join(", "), | ||||
|           cost: weapon.cost || 0, | ||||
|           ammoCost: weapon.ammo_cost || 0, | ||||
|           magazine: weapon.mag || 1, | ||||
|         }, | ||||
|         folder: distanceWeaponsFolder.id | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async importStarships() { | ||||
|     // Create a starships folder if it doesn't exist | ||||
|     const starshipsFolder = game.folders.getName("Starships") || await Folder.create({ | ||||
|       name: "Starships", type: "Actor" | ||||
|     }) | ||||
|     if (!starshipsFolder) { | ||||
|       console.error("Failed to create Starships folder"); | ||||
|       return; | ||||
|     } | ||||
|     // Load the starships JSON file | ||||
|     const starshipsData = await fetch("systems/fvtt-ftl-nomad/assets/json_data/starships.json") | ||||
|       .then(response => response.json()) | ||||
|       .catch(error => { | ||||
|         console.error("Failed to load starships data:", error); | ||||
|         return []; | ||||
|       }); | ||||
|  | ||||
|     // Import each starship | ||||
|     for (const starship of starshipsData) { | ||||
|       // Check if the starship already exists | ||||
|       const existingStarship = game.actors.find(a => a.name === starship.name && a.type === "starship"); | ||||
|       if (existingStarship) { | ||||
|         console.warn(`Starship ${starship.name} already exists, skipping import.`); | ||||
|         continue; | ||||
|       } | ||||
|       // Create the starship actor | ||||
|       await Actor.create({ | ||||
|         name: starship.name, | ||||
|         type: "starship", | ||||
|         img: "systems/fvtt-ftl-nomad/assets/icons/icon_starship.svg", | ||||
|         system: { | ||||
|           description: starship.description, | ||||
|           agility: starship.agility, | ||||
|           hullType: starship.hullType.toLowerCase(), | ||||
|           endurance: starship.endurance, | ||||
|           armor: starship.armor, | ||||
|           crew: starship.crew, | ||||
|           cargo: starship.cargo, | ||||
|           guns: starship.guns, | ||||
|           travelMultiplier: starship.travelMultiplier, | ||||
|           cost: starship.cost || 0, | ||||
|           monthlyCost: starship.monthlyCost || 0, | ||||
|           damages: starship.damages, | ||||
|         }, | ||||
|         folder: starshipsFolder.id | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async importVehicles() { | ||||
|     // Create a vehicles folder if it doesn't exist | ||||
|     const vehiclesFolder = game.folders.getName("Vehicles") || await Folder.create({ | ||||
|       name: "Vehicles", type: "Actor" | ||||
|     }) | ||||
|     if (!vehiclesFolder) { | ||||
|       console.error("Failed to create Vehicles folder"); | ||||
|       return; | ||||
|     } | ||||
|     // Load the vehicles JSON file | ||||
|     const vehiclesData = await fetch("systems/fvtt-ftl-nomad/assets/json_data/vehicles.json") | ||||
|       .then(response => response.json()) | ||||
|       .catch(error => { | ||||
|         console.error("Failed to load vehicles data:", error); | ||||
|         return []; | ||||
|       }); | ||||
|  | ||||
|     // Import each vehicle | ||||
|     for (const vehicle of vehiclesData) { | ||||
|       // Check if the vehicle already exists | ||||
|       const existingVehicle = game.items.find(i => i.name === vehicle.name && i.type === "vehicle"); | ||||
|       if (existingVehicle) { | ||||
|         console.warn(`Vehicle ${vehicle.name} already exists, skipping import.`); | ||||
|         continue; | ||||
|       } | ||||
|       // Create the vehicle item | ||||
|       await Actor.create({ | ||||
|         name: vehicle.name, | ||||
|         type: "vehicle", | ||||
|         img: "systems/fvtt-ftl-nomad/assets/icons/icon_vehicle.svg", | ||||
|         system: { | ||||
|           description: vehicle.description, | ||||
|           agility: vehicle.agility, | ||||
|           armor: vehicle.armor, | ||||
|           cargo: vehicle.cargo, | ||||
|           crew: vehicle.crew, | ||||
|           force: vehicle.force, | ||||
|           range: vehicle.range, | ||||
|           speed: vehicle.speed, | ||||
|           techAge: this.getTechAgeKeyFromLabel(vehicle.tech_age), | ||||
|           tonnage: vehicle.tonnage, | ||||
|           damages: vehicle.damages, | ||||
|           cost: vehicle.cost || 0, | ||||
|         }, | ||||
|         folder: vehiclesFolder.id | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async importArmors() { | ||||
|     // Create an armors folder if it doesn't exist | ||||
|     const armorsFolder = game.folders.getName("Armors") || await Folder.create({ | ||||
|       name: "Armors", type: "Item" | ||||
|     }) | ||||
|     if (!armorsFolder) { | ||||
|       console.error("Failed to create Armors folder"); | ||||
|       return; | ||||
|     } | ||||
|     // Load the armors JSON file | ||||
|     const armorsData = await fetch("systems/fvtt-ftl-nomad/assets/json_data/armors.json") | ||||
|       .then(response => response.json()) | ||||
|       .catch(error => { | ||||
|         console.error("Failed to load armors data:", error); | ||||
|         return []; | ||||
|       }); | ||||
|  | ||||
|     // Import each armor | ||||
|     for (const armor of armorsData) { | ||||
|       // Check if the armor already exists | ||||
|       const existingArmor = game.items.find(i => i.name === armor.name && i.type === "armor"); | ||||
|       if (existingArmor) { | ||||
|         console.warn(`Armor ${armor.name} already exists, skipping import.`); | ||||
|         continue; | ||||
|       } | ||||
|       // Create the armor item | ||||
|       await Item.create({ | ||||
|         name: armor.name, | ||||
|         type: "armor", | ||||
|         img: "systems/fvtt-ftl-nomad/assets/icons/icon_armor.svg", | ||||
|         system: { | ||||
|           description: armor.description, | ||||
|           enc: armor.enc || 0, | ||||
|           techAge: this.getTechAgeKeyFromLabel(armor.tech_age), | ||||
|           cost: armor.cost || 0, | ||||
|           protection: armor.protection || 0, | ||||
|         }, | ||||
|         folder: armorsFolder.id | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async importGrenadeWeapons() { | ||||
|     // Create a grenade weapons folder if it doesn't exist | ||||
|     const grenadeWeaponsFolder = game.folders.getName("Grenade") || await Folder.create({ | ||||
|       name: "Grenade", type: "Item" | ||||
|     }) | ||||
|     if (!grenadeWeaponsFolder) { | ||||
|       console.error("Failed to create Grenade Weapons folder"); | ||||
|       return; | ||||
|     } | ||||
|     // Load the grenade weapons JSON file | ||||
|     const grenadeWeaponsData = await fetch("systems/fvtt-ftl-nomad/assets/json_data/grenades.json") | ||||
|       .then(response => response.json()) | ||||
|       .catch(error => { | ||||
|         console.error("Failed to load grenade weapons data:", error); | ||||
|         return []; | ||||
|       }); | ||||
|  | ||||
|     // Import each grenade weapon | ||||
|     for (const weapon of grenadeWeaponsData) { | ||||
|       // Check if the weapon already exists | ||||
|       const existingWeapon = game.items.find(i => i.name === weapon.name && i.type === "weapon"); | ||||
|       if (existingWeapon) { | ||||
|         console.warn(`Weapon ${weapon.name} already exists, skipping import.`); | ||||
|         continue; | ||||
|       } | ||||
|       // Create the weapon item | ||||
|       await Item.create({ | ||||
|         name: weapon.name, | ||||
|         type: "weapon", | ||||
|         img: "systems/fvtt-ftl-nomad/assets/icons/icon_weapon.svg", | ||||
|         system: { | ||||
|           description: weapon.description, | ||||
|           damage: weapon.damage, | ||||
|           techAge: this.getTechAgeKeyFromLabel(weapon.tech_age), | ||||
|           weaponType: SYSTEM.WEAPON_TYPES.grenade.id, | ||||
|           rangeType: SYSTEM.WEAPON_RANGE.thrownweapon.id, | ||||
|           enc: 0, | ||||
|           aspect: weapon.aspects.join(", "), | ||||
|           cost: weapon.cost || 0, | ||||
|           ammoCost: 0, | ||||
|         }, | ||||
|         folder: grenadeWeaponsFolder.id | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async fixWeaponsEnc() { | ||||
|     // Iterate over all items of type "weapon" from compendium | ||||
|     const weapons = game.items.filter(i => i.type === "weapon"); | ||||
|     for (const weapon of weapons) { | ||||
|       // If the weapon's encumbrance is less than 1, set it to 1 | ||||
|       if (weapon.system.enc < 1) { | ||||
|         await weapon.update({ "system.enc": 1 }); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async fixEquipmentEnc() { | ||||
|     // Iterate over all items of type "equipment" from compendium | ||||
|     const equipments = game.items.filter(i => i.type === "equipment"); | ||||
|     for (const equipment of equipments) { | ||||
|       // If the equipment's encumbrance is less than 1, set it to | ||||
|       if (equipment.system.enc < 1) { | ||||
|         await equipment.update({ "system.enc": 1 }); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async importMeleeWeapons() { | ||||
|     // Create a melee weapons folder if it doesn't exist | ||||
|     const meleeWeaponsFolder = game.folders.getName("Melee Weapons") || await Folder.create({ | ||||
|       name: "Melee Weapons", type: "Item" | ||||
|     }) | ||||
|     if (!meleeWeaponsFolder) { | ||||
|       console.error("Failed to create Melee Weapons folder"); | ||||
|       return; | ||||
|     } | ||||
|     // Load the melee weapons JSON file | ||||
|     const meleeWeaponsData = await fetch("systems/fvtt-ftl-nomad/assets/json_data/weapon_melee.json") | ||||
|       .then(response => response.json()) | ||||
|       .catch(error => { | ||||
|         console.error("Failed to load melee weapons data:", error); | ||||
|         return []; | ||||
|       }); | ||||
|  | ||||
|     console.log("Melee Weapons Data", meleeWeaponsData); | ||||
|     // Import each melee weapon | ||||
|     for (const weapon of meleeWeaponsData) { | ||||
|       // Check if the weapon already exists | ||||
|       const existingWeapon = game.items.find(i => i.name === weapon.name && i.type === "weapon"); | ||||
|       if (existingWeapon) { | ||||
|         console.warn(`Weapon ${weapon.name} already exists, skipping import.`); | ||||
|         continue; | ||||
|       } | ||||
|       // Create the weapon item | ||||
|       await Item.create({ | ||||
|         name: weapon.name, | ||||
|         type: "weapon", | ||||
|         img: "systems/fvtt-ftl-nomad/assets/icons/icon_weapon.svg", | ||||
|         system: { | ||||
|           description: weapon.description, | ||||
|           damage: weapon.damage, | ||||
|           techAge: this.getTechAgeKeyFromLabel(weapon.tech_age), | ||||
|           weaponType: SYSTEM.WEAPON_TYPES.melee.id, | ||||
|           rangeType: SYSTEM.WEAPON_RANGE.melee.id, | ||||
|           enc: 0, | ||||
|           aspect: weapon.aspects.join(", "), | ||||
|           cost: weapon.cost || 0, | ||||
|           ammoCost: 0, | ||||
|         }, | ||||
|         folder: meleeWeaponsFolder.id | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static async importVehicleWeapons() { | ||||
|     // Create a melee weapons folder if it doesn't exist | ||||
|     const meleeWeaponsFolder = game.folders.getName("Vehicle Weapons") || await Folder.create({ | ||||
|       name: "Vehicle Weapons", type: "Item" | ||||
|     }) | ||||
|     if (!meleeWeaponsFolder) { | ||||
|       console.error("Failed to create Vehicle Weapons folder"); | ||||
|       return; | ||||
|     } | ||||
|     // Load the melee weapons JSON file | ||||
|     const meleeWeaponsData = await fetch("systems/fvtt-ftl-nomad/assets/json_data/vehicleweapons.json") | ||||
|       .then(response => response.json()) | ||||
|       .catch(error => { | ||||
|         console.error("Failed to load vehicle weapons data:", error); | ||||
|         return []; | ||||
|       }); | ||||
|  | ||||
|     console.log("vehicle Weapons Data", meleeWeaponsData); | ||||
|     // Import each melee weapon | ||||
|     for (const weapon of meleeWeaponsData) { | ||||
|       // Check if the weapon already exists | ||||
|       const existingWeapon = game.items.find(i => i.name === weapon.name && i.type === "weapon"); | ||||
|       if (existingWeapon) { | ||||
|         console.warn(`Weapon ${weapon.name} already exists, skipping import.`); | ||||
|       } | ||||
|       // Create the weapon item | ||||
|       await Item.create({ | ||||
|         name: weapon.name, | ||||
|         type: "weapon", | ||||
|         img: "systems/fvtt-ftl-nomad/assets/icons/icon_weapon.svg", | ||||
|         system: { | ||||
|           description: weapon.description, | ||||
|           damage: weapon.damage, | ||||
|           techAge: this.getTechAgeKeyFromLabel(weapon.tech_age), | ||||
|           weaponType: "vehicle", //SYSTEM.WEAPON_TYPES.melee.id, | ||||
|           rangeType: SYSTEM.WEAPON_RANGE.melee.id, | ||||
|           enc: 0, | ||||
|           aspect: weapon.aspects, | ||||
|           cost: weapon.cost || 0, | ||||
|           ammoCost: weapon.ammo_cost || 0, | ||||
|           magazine: weapon.mag || 1, | ||||
|         }, | ||||
|         folder: meleeWeaponsFolder.id | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|  | ||||
| } | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								packs/ftl-nomad-items/000090.ldb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										0
									
								
								packs/ftl-nomad-items/000093.log
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										1
									
								
								packs/ftl-nomad-items/CURRENT
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| MANIFEST-000091 | ||||
							
								
								
									
										0
									
								
								packs/ftl-nomad-items/LOCK
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										7
									
								
								packs/ftl-nomad-items/LOG
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | ||||
| 2025/10/15-21:50:26.265715 7f189ffff6c0 Recovering log #88 | ||||
| 2025/10/15-21:50:26.325136 7f189ffff6c0 Delete type=3 #86 | ||||
| 2025/10/15-21:50:26.325207 7f189ffff6c0 Delete type=0 #88 | ||||
| 2025/10/15-22:19:29.836460 7f189e7fc6c0 Level-0 table #94: started | ||||
| 2025/10/15-22:19:29.836492 7f189e7fc6c0 Level-0 table #94: 0 bytes OK | ||||
| 2025/10/15-22:19:29.881737 7f189e7fc6c0 Delete type=0 #92 | ||||
| 2025/10/15-22:19:29.944478 7f189e7fc6c0 Manual compaction at level-0 from '!folders!AuBtSOj1mJmh88qx' @ 72057594037927935 : 1 .. '!items!zv9dwgL3p7ThQn7j' @ 0 : 0; will stop at (end) | ||||
							
								
								
									
										14
									
								
								packs/ftl-nomad-items/LOG.old
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | ||||
| 2025/10/15-13:50:39.028219 7f189f7fe6c0 Recovering log #84 | ||||
| 2025/10/15-13:50:39.038979 7f189f7fe6c0 Delete type=3 #82 | ||||
| 2025/10/15-13:50:39.039047 7f189f7fe6c0 Delete type=0 #84 | ||||
| 2025/10/15-15:26:39.732570 7f189e7fc6c0 Level-0 table #89: started | ||||
| 2025/10/15-15:26:39.736109 7f189e7fc6c0 Level-0 table #89: 12073 bytes OK | ||||
| 2025/10/15-15:26:39.742312 7f189e7fc6c0 Delete type=0 #87 | ||||
| 2025/10/15-15:26:39.748688 7f189e7fc6c0 Manual compaction at level-0 from '!folders!AuBtSOj1mJmh88qx' @ 72057594037927935 : 1 .. '!items!zv9dwgL3p7ThQn7j' @ 0 : 0; will stop at '!items!zoE4ITyfl3od6hYD' @ 721 : 1 | ||||
| 2025/10/15-15:26:39.748706 7f189e7fc6c0 Compacting 1@0 + 1@1 files | ||||
| 2025/10/15-15:26:39.753893 7f189e7fc6c0 Generated table #90@0: 316 keys, 126470 bytes | ||||
| 2025/10/15-15:26:39.753929 7f189e7fc6c0 Compacted 1@0 + 1@1 files => 126470 bytes | ||||
| 2025/10/15-15:26:39.760361 7f189e7fc6c0 compacted to: files[ 0 1 0 0 0 0 0 ] | ||||
| 2025/10/15-15:26:39.760492 7f189e7fc6c0 Delete type=2 #73 | ||||
| 2025/10/15-15:26:39.760637 7f189e7fc6c0 Delete type=2 #89 | ||||
| 2025/10/15-15:26:39.774069 7f189e7fc6c0 Manual compaction at level-0 from '!items!zoE4ITyfl3od6hYD' @ 721 : 1 .. '!items!zv9dwgL3p7ThQn7j' @ 0 : 0; will stop at (end) | ||||
							
								
								
									
										
											BIN
										
									
								
								packs/ftl-nomad-items/MANIFEST-000091
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										0
									
								
								packs/ftl-nomad-items/lost/000042.log
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										0
									
								
								packs/ftl-nomad-items/lost/000065.log
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								packs/ftl-nomad-vehicles/000068.ldb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										0
									
								
								packs/ftl-nomad-vehicles/000071.log
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										1
									
								
								packs/ftl-nomad-vehicles/CURRENT
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| MANIFEST-000069 | ||||
							
								
								
									
										0
									
								
								packs/ftl-nomad-vehicles/LOCK
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										7
									
								
								packs/ftl-nomad-vehicles/LOG
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | ||||
| 2025/10/15-21:50:26.444225 7f18a4ffa6c0 Recovering log #66 | ||||
| 2025/10/15-21:50:26.495317 7f18a4ffa6c0 Delete type=3 #64 | ||||
| 2025/10/15-21:50:26.495386 7f18a4ffa6c0 Delete type=0 #66 | ||||
| 2025/10/15-22:19:30.067316 7f189e7fc6c0 Level-0 table #72: started | ||||
| 2025/10/15-22:19:30.067349 7f189e7fc6c0 Level-0 table #72: 0 bytes OK | ||||
| 2025/10/15-22:19:30.103968 7f189e7fc6c0 Delete type=0 #70 | ||||
| 2025/10/15-22:19:30.104104 7f189e7fc6c0 Manual compaction at level-0 from '!actors!0FQ6XaRi24OorI21' @ 72057594037927935 : 1 .. '!folders!vRnrOJqSMlxbSgyX' @ 0 : 0; will stop at (end) | ||||
							
								
								
									
										14
									
								
								packs/ftl-nomad-vehicles/LOG.old
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | ||||
| 2025/10/15-13:50:39.042685 7f18a4ffa6c0 Recovering log #62 | ||||
| 2025/10/15-13:50:39.052767 7f18a4ffa6c0 Delete type=3 #60 | ||||
| 2025/10/15-13:50:39.052843 7f18a4ffa6c0 Delete type=0 #62 | ||||
| 2025/10/15-15:26:39.665250 7f189e7fc6c0 Level-0 table #67: started | ||||
| 2025/10/15-15:26:39.668457 7f189e7fc6c0 Level-0 table #67: 12881 bytes OK | ||||
| 2025/10/15-15:26:39.674498 7f189e7fc6c0 Delete type=0 #65 | ||||
| 2025/10/15-15:26:39.688322 7f189e7fc6c0 Manual compaction at level-0 from '!actors!0FQ6XaRi24OorI21' @ 72057594037927935 : 1 .. '!folders!vRnrOJqSMlxbSgyX' @ 0 : 0; will stop at '!folders!HobFutlnH6HPTEfT' @ 93 : 1 | ||||
| 2025/10/15-15:26:39.688334 7f189e7fc6c0 Compacting 1@0 + 1@1 files | ||||
| 2025/10/15-15:26:39.692347 7f189e7fc6c0 Generated table #68@0: 78 keys, 62826 bytes | ||||
| 2025/10/15-15:26:39.692380 7f189e7fc6c0 Compacted 1@0 + 1@1 files => 62826 bytes | ||||
| 2025/10/15-15:26:39.698356 7f189e7fc6c0 compacted to: files[ 0 1 0 0 0 0 0 ] | ||||
| 2025/10/15-15:26:39.698484 7f189e7fc6c0 Delete type=2 #51 | ||||
| 2025/10/15-15:26:39.698645 7f189e7fc6c0 Delete type=2 #67 | ||||
| 2025/10/15-15:26:39.709845 7f189e7fc6c0 Manual compaction at level-0 from '!folders!HobFutlnH6HPTEfT' @ 93 : 1 .. '!folders!vRnrOJqSMlxbSgyX' @ 0 : 0; will stop at (end) | ||||
							
								
								
									
										
											BIN
										
									
								
								packs/ftl-nomad-vehicles/MANIFEST-000069
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										0
									
								
								packs/ftl-nomad-vehicles/lost/000020.log
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										0
									
								
								packs/ftl-nomad-vehicles/lost/000043.log
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -19,6 +19,7 @@ | ||||
|  | ||||
|     .character-left { | ||||
|       min-width: 180px; | ||||
|       max-width: 180px; | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|  | ||||
| @@ -39,11 +40,22 @@ | ||||
|         align-items: center; | ||||
|         input { | ||||
|           flex: none; | ||||
|           width: 2rem; | ||||
|           margin-left: 4px; | ||||
|           width: 2.5rem; | ||||
|           margin-left: 2px; | ||||
|           margin-right: 4px; | ||||
|         } | ||||
|         .damage-bonus { | ||||
|           font-size: calc(var(--font-size-standard) * 0.8); | ||||
|         .form-group { | ||||
|           display: flex; | ||||
|           flex-direction: row; | ||||
|           align-items: center; | ||||
|           margin-bottom: 4px; | ||||
|  | ||||
|           label { | ||||
|             font-weight: bold; | ||||
|             margin-right: 4px; | ||||
|             min-width: 4rem; | ||||
|             max-width: 4rem; | ||||
|           } | ||||
|         } | ||||
|         .hp-separator { | ||||
|           font-size: calc(var(--font-size-standard) * 1.2); | ||||
| @@ -59,7 +71,7 @@ | ||||
|           flex: none; | ||||
|         } | ||||
|       } | ||||
|       .character-dmax-edit { | ||||
|       /*.character-dmax-edit { | ||||
|         input { | ||||
|           display: flex; | ||||
|           width: 60px; | ||||
| @@ -69,7 +81,7 @@ | ||||
|           padding: 0 5px 0 5px; | ||||
|           text-align: center; | ||||
|         } | ||||
|       } | ||||
|       }*/ | ||||
|     } | ||||
|  | ||||
|     .character-right { | ||||
| @@ -77,6 +89,63 @@ | ||||
|       flex-direction: column; | ||||
|       gap: 5px; | ||||
|  | ||||
|       .character-spec { | ||||
|         .form-group { | ||||
|           display: flex; | ||||
|           flex-direction: row; | ||||
|           align-items: center; | ||||
|           margin-bottom: 4px; | ||||
|  | ||||
|           label { | ||||
|             font-weight: bold; | ||||
|             margin-right: 4px; | ||||
|             min-width: 4rem; | ||||
|             max-width: 4rem; | ||||
|           } | ||||
|           input { | ||||
|             min-width: 20rem; | ||||
|             max-width: 20rem; | ||||
|           } | ||||
|         } | ||||
|         .hero-armor { | ||||
|           display: grid; | ||||
|           grid-template-columns: repeat(2, 1fr); | ||||
|           gap: 4px; | ||||
|           label { | ||||
|             min-width: 6rem; | ||||
|             max-width: 6rem; | ||||
|           } | ||||
|           input { | ||||
|             max-width: 5.2rem; | ||||
|             min-width: 5.2rem; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .encumbrance { | ||||
|         display: flex; | ||||
|         gap: 4px; | ||||
|         .form-group { | ||||
|           display: flex; | ||||
|           flex-direction: row; | ||||
|           margin-bottom: 4px; | ||||
|         } | ||||
|         label { | ||||
|           min-width: 3rem; | ||||
|           max-width: 3rem; | ||||
|         } | ||||
|         input { | ||||
|           max-width: 3rem; | ||||
|           min-width: 3rem; | ||||
|         } | ||||
|         .character-credit { | ||||
|           input { | ||||
|             max-width: 9rem; | ||||
|             min-width: 9rem; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .character-name { | ||||
|         display: flex; | ||||
|         input { | ||||
| @@ -85,87 +154,6 @@ | ||||
|           width: 400px; | ||||
|         } | ||||
|       } | ||||
|       .san { | ||||
|         align-content: flex-start; | ||||
|         input { | ||||
|           min-width: 2.2rem; | ||||
|           max-width: 2.2rem; | ||||
|           margin-bottom: 4px; | ||||
|         } | ||||
|         select { | ||||
|           min-width: 6rem; | ||||
|           max-width: 6rem; | ||||
|         } | ||||
|         .rollable:hover, | ||||
|         .rollable:focus { | ||||
|           text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|           cursor: pointer; | ||||
|           font-size: 0.9rem; | ||||
|         } | ||||
|         .button { | ||||
|           min-width: 4rem; | ||||
|           max-width: 4rem; | ||||
|         } | ||||
|         .san-checkbox { | ||||
|           min-width: 1rem; | ||||
|           max-width: 1rem; | ||||
|         } | ||||
|         .label-short-field { | ||||
|           font-size: 0.9rem; | ||||
|           max-width: 3rem; | ||||
|           min-width: 3rem; | ||||
|           flex-grow: 1; | ||||
|         } | ||||
|         .label-recovery { | ||||
|           margin-left: 4px; | ||||
|         } | ||||
|         .label-field { | ||||
|           font-size: 0.9rem; | ||||
|           max-width: 6rem; | ||||
|           min-width: 6rem; | ||||
|           flex-grow: 1; | ||||
|         } | ||||
|         .label-bp { | ||||
|           flex-grow: 1; | ||||
|           max-width: 3rem; | ||||
|           min-width: 3rem; | ||||
|           margin-left: 4px; | ||||
|         } | ||||
|         .label-insanity { | ||||
|           flex-grow: 1; | ||||
|           margin-left: 4px; | ||||
|           max-width: 8rem; | ||||
|           min-width: 8rem; | ||||
|         } | ||||
|         .spacing { | ||||
|           margin-left: 4px; | ||||
|         } | ||||
|         .d100 { | ||||
|           flex: 0; | ||||
|         } | ||||
|       } | ||||
|       .willpower { | ||||
|         input { | ||||
|           min-width: 2.4rem; | ||||
|           max-width: 2.4rem; | ||||
|         } | ||||
|         input[type="checkbox"] { | ||||
|           min-width: 1rem; | ||||
|           max-width: 1rem; | ||||
|         } | ||||
|         .label-field { | ||||
|           flex-grow: 1; | ||||
|           margin-left: 4px; | ||||
|           max-width: 5rem; | ||||
|           min-width: 5rem; | ||||
|           font-size: 0.9rem; | ||||
|         } | ||||
|         .checkbox { | ||||
|           flex-grow: 0; | ||||
|           min-width: 1rem; | ||||
|           max-width: 1rem; | ||||
|         } | ||||
|       } | ||||
|       label { | ||||
|         min-width: 120px; | ||||
|       } | ||||
| @@ -177,27 +165,32 @@ | ||||
|   } | ||||
|  | ||||
|   .character-pc-edit { | ||||
|     min-width: 650px; | ||||
|     min-width: 500px; | ||||
|   } | ||||
|  | ||||
|   .character-characteristics { | ||||
|   .character-skills { | ||||
|     background-color: var(--color-light-1); | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     gap: 5px; | ||||
|     flex: 1; | ||||
|  | ||||
|     .character-characteristic { | ||||
|     .character-skill { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       .icon-skill { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin-right: 4px; | ||||
|       } | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .rollable { | ||||
|         min-width: 3rem; | ||||
|         max-width: 3rem; | ||||
|         min-width: 4.5rem; | ||||
|         max-width: 4.5rem; | ||||
|       } | ||||
|       .char-text { | ||||
|         margin-left: 0.5rem; | ||||
| @@ -218,33 +211,59 @@ | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .character-characteristic-play { | ||||
|   .character-skill-play { | ||||
|     min-width: 225px; | ||||
|   } | ||||
|  | ||||
|   .character-characteristic-edit { | ||||
|     min-width: 400px; | ||||
|   .character-skill-edit { | ||||
|     min-width: 225px; | ||||
|   } | ||||
| } | ||||
|  | ||||
| .character-biography { | ||||
|   background-color: var(--color-light-1); | ||||
|   prose-mirror.inactive { | ||||
|     min-height: 40px; | ||||
|     min-height: 16rem; | ||||
|   } | ||||
|   prose-mirror.active { | ||||
|     min-height: 150px; | ||||
|     min-height: 16rem; | ||||
|   } | ||||
|   .field-label { | ||||
|     margin-left: 8px; | ||||
|   } | ||||
|  | ||||
|   .adapted { | ||||
|   .rank { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(5, 1fr); | ||||
|     gap: 8px; | ||||
|     label { | ||||
|       min-width: 6rem; | ||||
|     } | ||||
|     input { | ||||
|       max-width: 4rem; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .biodata { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 8px; | ||||
|     label { | ||||
|       min-width: 20rem; | ||||
|     .form-group { | ||||
|       display: flex; | ||||
|       flex-direction: row; | ||||
|       align-items: center; | ||||
|       margin-bottom: 4px; | ||||
|  | ||||
|       label { | ||||
|         font-weight: bold; | ||||
|         margin-right: 4px; | ||||
|         min-width: 5rem; | ||||
|         max-width: 5rem; | ||||
|       } | ||||
|       input { | ||||
|         min-width: 18rem; | ||||
|         max-width: 18rem; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -275,7 +294,7 @@ | ||||
|   } | ||||
| } | ||||
|  | ||||
| .tab.character-skills { | ||||
| .tab.character-skills .main-div { | ||||
|   background-color: var(--color-light-1); | ||||
|   display: grid; | ||||
|   grid-template-columns: 1fr; | ||||
| @@ -323,7 +342,7 @@ | ||||
|   } | ||||
| } | ||||
|  | ||||
| .tab.character-status { | ||||
| .tab.character-status .main-div { | ||||
|   background-color: var(--color-light-1); | ||||
|   display: grid; | ||||
|   grid-template-columns: 1fr; | ||||
| @@ -455,7 +474,7 @@ | ||||
|   } | ||||
| } | ||||
|  | ||||
| .tab.character-equipment { | ||||
| .tab.character-talents .main-div { | ||||
|   background-color: var(--color-light-1); | ||||
|   display: grid; | ||||
|   grid-template-columns: 1fr; | ||||
| @@ -465,6 +484,175 @@ | ||||
|       padding-left: 5px; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .talents { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .talent { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .damage { | ||||
|         min-width: 6rem; | ||||
|         max-width: 6rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 10rem; | ||||
|         max-width: 10rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .languages { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .language { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .damage { | ||||
|         min-width: 6rem; | ||||
|         max-width: 6rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 10rem; | ||||
|         max-width: 10rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .psionics { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .psionic { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .damage { | ||||
|         min-width: 6rem; | ||||
|         max-width: 6rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 10rem; | ||||
|         max-width: 10rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| .tab.character-equipment .main-div { | ||||
|   background-color: var(--color-light-1); | ||||
|   display: grid; | ||||
|   grid-template-columns: 1fr; | ||||
|   legend { | ||||
|     a { | ||||
|       font-size: calc(var(--font-size-standard) * 1.4); | ||||
|       padding-left: 5px; | ||||
|     } | ||||
|   } | ||||
|   .encumbrance { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(4, 1fr); | ||||
|     gap: 4px; | ||||
|     input { | ||||
|       max-width: 4rem; | ||||
|     } | ||||
|     .encumbered { | ||||
|       color: red; | ||||
|       font-weight: bold; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .implants { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .implant { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .damage { | ||||
|         min-width: 6rem; | ||||
|         max-width: 6rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 10rem; | ||||
|         max-width: 10rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .weapons { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
| @@ -500,9 +688,10 @@ | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .armors { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(3, 1fr); | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .armor { | ||||
|       display: flex; | ||||
| @@ -510,23 +699,18 @@ | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .protection { | ||||
|         min-width: 5rem; | ||||
|         max-width: 5rem; | ||||
|         min-width: 6rem; | ||||
|         max-width: 6rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 8rem; | ||||
|         max-width: 8rem; | ||||
|         min-width: 10rem; | ||||
|         max-width: 10rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
| @@ -536,11 +720,11 @@ | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .gears { | ||||
|   .equipments { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(3, 1fr); | ||||
|     gap: 4px; | ||||
|     .gear { | ||||
|     .equipment { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
| @@ -568,70 +752,6 @@ | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .rituals { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .ritual { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 20rem; | ||||
|       max-width: 20rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 17rem; | ||||
|         max-width: 17rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .tomes { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .tome { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 20rem; | ||||
|       max-width: 20rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 17rem; | ||||
|         max-width: 17rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   prose-mirror.inactive { | ||||
|     min-height: 40px; | ||||
|   } | ||||
|   | ||||
| @@ -19,6 +19,7 @@ | ||||
|  | ||||
|     .creature-left { | ||||
|       min-width: 180px; | ||||
|       max-width: 180px; | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|  | ||||
| @@ -37,13 +38,15 @@ | ||||
|       .creature-hp { | ||||
|         gap: 2px; | ||||
|         align-items: center; | ||||
|         input { | ||||
|           flex: none; | ||||
|           width: 2rem; | ||||
|           margin-left: 4px; | ||||
|         label { | ||||
|           font-weight: bold; | ||||
|           margin-right: 4px; | ||||
|           min-width: 4rem; | ||||
|           max-width: 4rem; | ||||
|         } | ||||
|         .damage-bonus { | ||||
|           font-size: calc(var(--font-size-standard) * 0.8); | ||||
|         input { | ||||
|           max-width: 3rem; | ||||
|           min-width: 3rem; | ||||
|         } | ||||
|         .hp-separator { | ||||
|           font-size: calc(var(--font-size-standard) * 1.2); | ||||
| @@ -77,6 +80,29 @@ | ||||
|       flex-direction: column; | ||||
|       gap: 5px; | ||||
|  | ||||
|       .creature-spec { | ||||
|         label { | ||||
|           max-width: 8rem; | ||||
|         } | ||||
|         select { | ||||
|           max-width: 10rem; | ||||
|           min-width: 10rem; | ||||
|         } | ||||
|         input { | ||||
|           max-width: 6rem; | ||||
|         } | ||||
|         .dice-2d6 { | ||||
|           max-width: 1.5rem; | ||||
|         } | ||||
|         .rollable:hover, | ||||
|         .rollable:focus { | ||||
|           text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|           cursor: pointer; | ||||
|         } | ||||
|         .rollable { | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .creature-name { | ||||
|         display: flex; | ||||
|         input { | ||||
| @@ -85,87 +111,6 @@ | ||||
|           width: 400px; | ||||
|         } | ||||
|       } | ||||
|       .san { | ||||
|         align-content: flex-start; | ||||
|         input { | ||||
|           min-width: 2.2rem; | ||||
|           max-width: 2.2rem; | ||||
|           margin-bottom: 4px; | ||||
|         } | ||||
|         select { | ||||
|           min-width: 6rem; | ||||
|           max-width: 6rem; | ||||
|         } | ||||
|         .rollable:hover, | ||||
|         .rollable:focus { | ||||
|           text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|           cursor: pointer; | ||||
|           font-size: 0.9rem; | ||||
|         } | ||||
|         .button { | ||||
|           min-width: 4rem; | ||||
|           max-width: 4rem; | ||||
|         } | ||||
|         .san-checkbox { | ||||
|           min-width: 1rem; | ||||
|           max-width: 1rem; | ||||
|         } | ||||
|         .label-short-field { | ||||
|           font-size: 0.9rem; | ||||
|           max-width: 3rem; | ||||
|           min-width: 3rem; | ||||
|           flex-grow: 1; | ||||
|         } | ||||
|         .label-recovery { | ||||
|           margin-left: 4px; | ||||
|         } | ||||
|         .label-field { | ||||
|           font-size: 0.9rem; | ||||
|           max-width: 6rem; | ||||
|           min-width: 6rem; | ||||
|           flex-grow: 1; | ||||
|         } | ||||
|         .label-bp { | ||||
|           flex-grow: 1; | ||||
|           max-width: 3rem; | ||||
|           min-width: 3rem; | ||||
|           margin-left: 4px; | ||||
|         } | ||||
|         .label-insanity { | ||||
|           flex-grow: 1; | ||||
|           margin-left: 4px; | ||||
|           max-width: 8rem; | ||||
|           min-width: 8rem; | ||||
|         } | ||||
|         .spacing { | ||||
|           margin-left: 4px; | ||||
|         } | ||||
|         .d100 { | ||||
|           flex: 0; | ||||
|         } | ||||
|       } | ||||
|       .willpower { | ||||
|         input { | ||||
|           min-width: 2.4rem; | ||||
|           max-width: 2.4rem; | ||||
|         } | ||||
|         input[type="checkbox"] { | ||||
|           min-width: 1rem; | ||||
|           max-width: 1rem; | ||||
|         } | ||||
|         .label-field { | ||||
|           flex-grow: 1; | ||||
|           margin-left: 4px; | ||||
|           max-width: 5rem; | ||||
|           min-width: 5rem; | ||||
|           font-size: 0.9rem; | ||||
|         } | ||||
|         .checkbox { | ||||
|           flex-grow: 0; | ||||
|           min-width: 1rem; | ||||
|           max-width: 1rem; | ||||
|         } | ||||
|       } | ||||
|       label { | ||||
|         min-width: 120px; | ||||
|       } | ||||
| @@ -177,27 +122,32 @@ | ||||
|   } | ||||
|  | ||||
|   .creature-pc-edit { | ||||
|     min-width: 650px; | ||||
|     min-width: 500px; | ||||
|   } | ||||
|  | ||||
|   .creature-characteristics { | ||||
|   .creature-skills { | ||||
|     background-color: var(--color-light-1); | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     gap: 5px; | ||||
|     flex: 1; | ||||
|  | ||||
|     .creature-characteristic { | ||||
|     .creature-skill { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       .icon-skill { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin-right: 4px; | ||||
|       } | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .rollable { | ||||
|         min-width: 3rem; | ||||
|         max-width: 3rem; | ||||
|         min-width: 4.5rem; | ||||
|         max-width: 4.5rem; | ||||
|       } | ||||
|       .char-text { | ||||
|         margin-left: 0.5rem; | ||||
| @@ -218,12 +168,12 @@ | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .creature-characteristic-play { | ||||
|   .creature-skill-play { | ||||
|     min-width: 225px; | ||||
|   } | ||||
|  | ||||
|   .creature-characteristic-edit { | ||||
|     min-width: 400px; | ||||
|   .creature-skill-edit { | ||||
|     min-width: 225px; | ||||
|   } | ||||
| } | ||||
|  | ||||
| @@ -239,12 +189,34 @@ | ||||
|     margin-left: 8px; | ||||
|   } | ||||
|  | ||||
|   .adapted { | ||||
|   .rank { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(5, 1fr); | ||||
|     gap: 8px; | ||||
|     label { | ||||
|       min-width: 6rem; | ||||
|     } | ||||
|     input { | ||||
|       max-width: 4rem; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .section-editor { | ||||
|     min-height: 10rem; | ||||
|     .editor { | ||||
|       min-height: 10rem; | ||||
|     } | ||||
|     .editor-content { | ||||
|       min-height: 10rem; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .biodata { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 8px; | ||||
|     label { | ||||
|       min-width: 20rem; | ||||
|       min-width: 12rem; | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -275,7 +247,7 @@ | ||||
|   } | ||||
| } | ||||
|  | ||||
| .tab.creature-skills { | ||||
| .tab.creature-skills .main-div { | ||||
|   background-color: var(--color-light-1); | ||||
|   display: grid; | ||||
|   grid-template-columns: 1fr; | ||||
| @@ -285,79 +257,6 @@ | ||||
|       padding-left: 5px; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .armors { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(3, 1fr); | ||||
|     gap: 4px; | ||||
|     .armor { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .protection { | ||||
|         min-width: 5rem; | ||||
|         max-width: 5rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 8rem; | ||||
|         max-width: 8rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .weapons { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .weapon { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .damage { | ||||
|         min-width: 6rem; | ||||
|         max-width: 6rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 10rem; | ||||
|         max-width: 10rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .skills { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(3, 1fr); | ||||
| @@ -396,7 +295,7 @@ | ||||
|   } | ||||
| } | ||||
|  | ||||
| .tab.creature-status { | ||||
| .tab.creature-status .main-div { | ||||
|   background-color: var(--color-light-1); | ||||
|   display: grid; | ||||
|   grid-template-columns: 1fr; | ||||
| @@ -528,7 +427,7 @@ | ||||
|   } | ||||
| } | ||||
|  | ||||
| .tab.creature-equipment { | ||||
| .tab.creature-traits .main-div { | ||||
|   background-color: var(--color-light-1); | ||||
|   display: grid; | ||||
|   grid-template-columns: 1fr; | ||||
| @@ -539,11 +438,246 @@ | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .gears { | ||||
|   .traits { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .trait { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .damage { | ||||
|         min-width: 6rem; | ||||
|         max-width: 6rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 10rem; | ||||
|         max-width: 10rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .abilities { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .ability { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .damage { | ||||
|         min-width: 6rem; | ||||
|         max-width: 6rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 10rem; | ||||
|         max-width: 10rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .psionics { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .psionic { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .damage { | ||||
|         min-width: 6rem; | ||||
|         max-width: 6rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 10rem; | ||||
|         max-width: 10rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| .tab.creature-equipment .main-div { | ||||
|   background-color: var(--color-light-1); | ||||
|   display: grid; | ||||
|   grid-template-columns: 1fr; | ||||
|   legend { | ||||
|     a { | ||||
|       font-size: calc(var(--font-size-standard) * 1.4); | ||||
|       padding-left: 5px; | ||||
|     } | ||||
|   } | ||||
|   .encumbrance { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(4, 1fr); | ||||
|     gap: 4px; | ||||
|     input { | ||||
|       max-width: 4rem; | ||||
|     } | ||||
|     .encumbered { | ||||
|       color: red; | ||||
|       font-weight: bold; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .implants { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .implant { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .damage { | ||||
|         min-width: 6rem; | ||||
|         max-width: 6rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 10rem; | ||||
|         max-width: 10rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .weapons { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .weapon { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .damage { | ||||
|         min-width: 6rem; | ||||
|         max-width: 6rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 10rem; | ||||
|         max-width: 10rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .armors { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .armor { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13rem; | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .protection { | ||||
|         min-width: 6rem; | ||||
|         max-width: 6rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 10rem; | ||||
|         max-width: 10rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .equipments { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(3, 1fr); | ||||
|     gap: 4px; | ||||
|     .gear { | ||||
|     .equipment { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
| @@ -571,70 +705,6 @@ | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .rituals { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .ritual { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 20rem; | ||||
|       max-width: 20rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 17rem; | ||||
|         max-width: 17rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .tomes { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .tome { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 20rem; | ||||
|       max-width: 20rem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         font-size: 0.7rem; | ||||
|         min-width: 1.8rem; | ||||
|         max-width: 1.8rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 17rem; | ||||
|         max-width: 17rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   prose-mirror.inactive { | ||||
|     min-height: 40px; | ||||
|   } | ||||
|   | ||||
| @@ -5,6 +5,7 @@ | ||||
|   @import "mixins.less"; | ||||
|   @import "character.less"; | ||||
|   @import "vehicle.less"; | ||||
|   @import "robot.less"; | ||||
|   @import "starship.less"; | ||||
|   @import "creature.less"; | ||||
|   @import "weapon.less"; | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| :root { | ||||
|   --font-size-standard: 0.9rem; | ||||
|   --font-size-result: 1.4rem; | ||||
|   --background-image-base: linear-gradient(rgba(255, 255, 255, 0.8), rgba(255, 255, 255, 0.8)),  | ||||
|   --background-image-base: linear-gradient(rgba(255, 255, 255, 0.8), rgba(255, 255, 255, 0.8)), | ||||
|                 url("../assets/ui/ftl_nomad_background_01.webp"); | ||||
|   --font-primary: "Atkinson"; | ||||
|   --font-secondary: "Atkinson"; | ||||
| @@ -15,6 +15,11 @@ | ||||
|   /*--img-icon-color-filter: invert(60%) sepia(12%) saturate(6853%) hue-rotate(81deg) brightness(113%) contrast(104%);*/ | ||||
| } | ||||
|  | ||||
| body { | ||||
|   --color-text-secondary: rgb(22, 3, 3); | ||||
|   --color-text-emphatic: rgb(9, 97, 14); | ||||
| } | ||||
|  | ||||
| .d100 { | ||||
|   width: 18px; | ||||
|   height: 18px; | ||||
|   | ||||
| @@ -2,7 +2,8 @@ | ||||
|   font-family: var(--font-primary); | ||||
|   font-size: calc(var(--font-size-standard) * 1); | ||||
|   color: var(--color-dark-1); | ||||
|   background-image: var(--background-image-base); | ||||
|   background-color: var(--color-light-1); | ||||
|   /*background-image: var(--background-image-base);*/ | ||||
|   background-repeat: no-repeat; | ||||
|   background-size: 100% 100%; | ||||
|  | ||||
| @@ -39,11 +40,11 @@ | ||||
|     input, | ||||
|     select { | ||||
|       text-align: center; | ||||
|       font-size: calc(var(--font-size-standard) * 1.0); | ||||
|       font-size: calc(var(--font-size-standard) * 1); | ||||
|     } | ||||
|     select { | ||||
|       font-family: var(--font-secondary); | ||||
|       font-size: calc(var(--font-size-standard) * 1.0); | ||||
|       font-size: calc(var(--font-size-standard) * 1); | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -58,21 +59,87 @@ | ||||
| .character-sheet-common { | ||||
|   label { | ||||
|     font-family: var(--font-secondary); | ||||
|     font-size: calc(var(--font-size-standard) * 1.0); | ||||
|     font-size: calc(var(--font-size-standard) * 1); | ||||
|   } | ||||
| } | ||||
|  | ||||
| .vehicle-sheet-common { | ||||
|   label { | ||||
|     font-family: var(--font-secondary); | ||||
|     font-size: calc(var(--font-size-standard) * 1.0); | ||||
|     font-size: calc(var(--font-size-standard) * 1); | ||||
|   } | ||||
|   .form-group { | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|     align-items: center; | ||||
|     margin-bottom: 4px; | ||||
|  | ||||
|     label { | ||||
|       font-weight: bold; | ||||
|       margin-right: 8px; | ||||
|       min-width: 4rem; | ||||
|       max-width: 4rem; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| .starship-sheet-common { | ||||
|   label { | ||||
|     font-family: var(--font-secondary); | ||||
|     font-size: calc(var(--font-size-standard) * 1); | ||||
|   } | ||||
|   .form-group { | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|     align-items: center; | ||||
|     margin-bottom: 4px; | ||||
|  | ||||
|     label { | ||||
|       font-weight: bold; | ||||
|       margin-right: 8px; | ||||
|       min-width: 5rem; | ||||
|       max-width: 5rem; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| .robot-sheet-common { | ||||
|   label { | ||||
|     font-family: var(--font-secondary); | ||||
|     font-size: calc(var(--font-size-standard) * 1); | ||||
|   } | ||||
|   .form-group { | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|     align-items: center; | ||||
|     margin-bottom: 4px; | ||||
|  | ||||
|     label { | ||||
|       font-weight: bold; | ||||
|       margin-right: 8px; | ||||
|       min-width: 5rem; | ||||
|       max-width: 5rem; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| .creature-sheet-common { | ||||
|   label { | ||||
|     font-family: var(--font-secondary); | ||||
|     font-size: calc(var(--font-size-standard) * 1.0); | ||||
|     font-size: calc(var(--font-size-standard) * 1); | ||||
|   } | ||||
|   .form-group { | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|     align-items: center; | ||||
|     margin-bottom: 4px; | ||||
|  | ||||
|     label { | ||||
|       font-weight: bold; | ||||
|       margin-right: 8px; | ||||
|       min-width: 12rem; | ||||
|       max-width: 12rem; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| @@ -80,13 +147,26 @@ | ||||
|   .form-fields { | ||||
|     padding-top: 4px; | ||||
|   } | ||||
|   .form-group { | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|     align-items: center; | ||||
|     margin-bottom: 4px; | ||||
|  | ||||
|     label { | ||||
|       font-weight: bold; | ||||
|       margin-right: 8px; | ||||
|       min-width: 12rem; | ||||
|       max-width: 12rem; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   label { | ||||
|     font-family: var(--font-secondary); | ||||
|     font-size: calc(var(--font-size-standard) * 1.0); | ||||
|     font-size: calc(var(--font-size-standard) * 1); | ||||
|     flex: 50%; | ||||
|   } | ||||
|    | ||||
|  | ||||
|   .align-top { | ||||
|     align-self: flex-start; | ||||
|     padding: 0.1rem; | ||||
| @@ -99,5 +179,5 @@ | ||||
|  | ||||
|   .shift-right { | ||||
|     margin-left: 2rem; | ||||
|   }   | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										335
									
								
								styles/robot.less
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,335 @@ | ||||
| .robot-content { | ||||
|   .sheet-common(); | ||||
|   .robot-sheet-common(); | ||||
|   overflow: scroll; | ||||
| } | ||||
|  | ||||
| .sheet-tabs { | ||||
|   background-color: var(--color-light-1); | ||||
| } | ||||
|  | ||||
| .robot-main { | ||||
|   background-color: var(--color-light-1); | ||||
|   display: flex; | ||||
|  | ||||
|   .robot-pc { | ||||
|     display: flex; | ||||
|     gap: 4px; | ||||
|     flex: 1; | ||||
|  | ||||
|     .robot-left { | ||||
|       min-width: 180px; | ||||
|       max-width: 180px; | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|  | ||||
|       .robot-left-image { | ||||
|         display: flex; | ||||
|         justify-content: center; | ||||
|         align-items: center; | ||||
|         padding-bottom: 8px; | ||||
|         .robot-img { | ||||
|           height: 140px; | ||||
|           width: auto; | ||||
|           border: none; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .robot-cost { | ||||
|         label { | ||||
|           min-width: 3rem; | ||||
|           max-width: 3rem; | ||||
|         } | ||||
|         input { | ||||
|           min-width: 6rem; | ||||
|           max-width: 6rem; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     .robot-right { | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       gap: 5px; | ||||
|  | ||||
|       .robot-name { | ||||
|         display: flex; | ||||
|         input { | ||||
|           font-family: var(--font-title); | ||||
|           font-size: calc(var(--font-size-standard) * 1.4); | ||||
|           width: 380px; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .robot-brain { | ||||
|         min-width: 25rem; | ||||
|         max-width: 25rem; | ||||
|         textarea { | ||||
|           min-width: 23rem; | ||||
|           max-width: 23rem; | ||||
|           background-color: rgba(0, 0, 0, 0.2); | ||||
|           color: var(--color-dark-3); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .main-stats { | ||||
|         min-width: 25rem; | ||||
|         max-width: 25rem; | ||||
|       } | ||||
|       .cargo, | ||||
|       .capacity { | ||||
|         label { | ||||
|           min-width: 4rem; | ||||
|           max-width: 4rem; | ||||
|         } | ||||
|         input { | ||||
|           max-width: 6rem; | ||||
|           margin-right: 0.5rem; | ||||
|         } | ||||
|         .cargo-content { | ||||
|           display: flex; | ||||
|           align-items: center; | ||||
|           gap: 4px; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .robot-infos { | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         gap: 4px; | ||||
|  | ||||
|         label { | ||||
|           min-width: 120px; | ||||
|         } | ||||
|  | ||||
|         .robot-hp { | ||||
|           display: flex; | ||||
|           gap: 2px; | ||||
|           align-items: center; | ||||
|           .robot-hp-value { | ||||
|             .form-fields input { | ||||
|               flex: none; | ||||
|               width: 50px; | ||||
|               margin-left: 4px; | ||||
|               font-size: calc(var(--font-size-standard) * 1.4); | ||||
|             } | ||||
|           } | ||||
|           .robot-hp-max { | ||||
|             clear: both; | ||||
|             display: flex; | ||||
|             flex-direction: row; | ||||
|             flex-wrap: wrap; | ||||
|             margin: 3px 0; | ||||
|             align-items: center; | ||||
|             input { | ||||
|               width: 50px; | ||||
|               text-align: center; | ||||
|               font-size: calc(var(--font-size-standard) * 1.4); | ||||
|             } | ||||
|           } | ||||
|           .hp-separator { | ||||
|             font-size: calc(var(--font-size-standard) * 1.2); | ||||
|             display: flex; | ||||
|             align-items: center; | ||||
|             justify-content: center; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .robot-skills { | ||||
|     background-color: var(--color-light-1); | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     gap: 5px; | ||||
|     flex: 1; | ||||
|  | ||||
|     .robot-skill { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       .icon-skill { | ||||
|         width: 24px; | ||||
|         height: 24px; | ||||
|         margin-right: 4px; | ||||
|       } | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .rollable { | ||||
|         min-width: 4.5rem; | ||||
|         max-width: 4.5rem; | ||||
|       } | ||||
|       .char-text { | ||||
|         margin-left: 0.5rem; | ||||
|       } | ||||
|       .d100 { | ||||
|         flex: 0; | ||||
|         max-width: 0.6rem; | ||||
|       } | ||||
|       .form-group { | ||||
|         flex: 0; | ||||
|         padding-left: 5px; | ||||
|         .form-fields { | ||||
|           font-size: 1.1rem; | ||||
|           flex: none; | ||||
|           width: 40px; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .robot-skill-play { | ||||
|     min-width: 225px; | ||||
|   } | ||||
|  | ||||
|   .robot-skill-edit { | ||||
|     min-width: 225px; | ||||
|   } | ||||
|  | ||||
|   .robot-pc-play { | ||||
|     min-width: 500px; | ||||
|   } | ||||
|  | ||||
|   .robot-pc-edit { | ||||
|     min-width: 500px; | ||||
|   } | ||||
| } | ||||
|  | ||||
| .robot-description { | ||||
|   .text-center { | ||||
|     text-align: left; | ||||
|     align-content: center; | ||||
|   } | ||||
|   background-color: var(--color-light-1); | ||||
|   prose-mirror.inactive { | ||||
|     min-height: 40px; | ||||
|   } | ||||
|   prose-mirror.active { | ||||
|     min-height: 150px; | ||||
|   } | ||||
|   .field-label { | ||||
|     margin-left: 8px; | ||||
|   } | ||||
|   .robot-brain { | ||||
|     input { | ||||
|       min-width: 32rem; | ||||
|       max-width: 32rem; | ||||
|     } | ||||
|   } | ||||
|   .section-editor { | ||||
|     min-height: 10rem; | ||||
|     .editor { | ||||
|       min-height: 10rem; | ||||
|     } | ||||
|     .editor-content { | ||||
|       min-height: 10rem; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .biodata { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 8px; | ||||
|     label { | ||||
|       min-width: 3rem; | ||||
|     } | ||||
|     .feature { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 18rem; | ||||
|       max-width: 18rem; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| .tab.robot-equipment .main-div { | ||||
|   background-color: var(--color-light-1); | ||||
|   display: grid; | ||||
|   grid-template-columns: 1fr; | ||||
|   legend { | ||||
|     a { | ||||
|       font-size: calc(var(--font-size-standard) * 1.4); | ||||
|       padding-left: 5px; | ||||
|     } | ||||
|   } | ||||
|   .weapons { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 4px; | ||||
|     .weapon { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13srem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         min-width: 2rem; | ||||
|         max-width: 2rem; | ||||
|       } | ||||
|       .damage { | ||||
|         min-width: 5rem; | ||||
|         max-width: 5rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 8rem; | ||||
|         max-width: 8rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 32px; | ||||
|         height: 32px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .equipments { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(3, 1fr); | ||||
|     gap: 4px; | ||||
|     .equipment { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 13rem; | ||||
|       max-width: 13srem; | ||||
|       .rollable:hover, | ||||
|       .rollable:focus { | ||||
|         text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|         cursor: pointer; | ||||
|       } | ||||
|       .controls { | ||||
|         min-width: 2rem; | ||||
|         max-width: 2rem; | ||||
|       } | ||||
|       .damage { | ||||
|         min-width: 5rem; | ||||
|         max-width: 5rem; | ||||
|       } | ||||
|       .name { | ||||
|         min-width: 8rem; | ||||
|         max-width: 8rem; | ||||
|       } | ||||
|       .item-img { | ||||
|         width: 32px; | ||||
|         height: 32px; | ||||
|         margin: 4px 0 0 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   prose-mirror.inactive { | ||||
|     min-height: 40px; | ||||
|   } | ||||
|   prose-mirror.active { | ||||
|     min-height: 150px; | ||||
|   } | ||||
| } | ||||
| @@ -1,4 +1,4 @@ | ||||
| .application.dialog.fvtt-cthulhu-eternal { | ||||
| .application.dialog.fvtt-ftl-nomad { | ||||
|   color: var(--color-dark-1); | ||||
|   background-color: var(--color-light-1); | ||||
|  | ||||
| @@ -31,7 +31,7 @@ | ||||
|     border: none; | ||||
|     background-color: rgba(0, 0, 0, 0.1); | ||||
|     color: var(--color-dark-2); | ||||
|     width: 4rem; | ||||
|     width: 10rem; | ||||
|     text-align: center; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| .starship-content { | ||||
|   .sheet-common(); | ||||
|   .character-sheet-common(); | ||||
|   .starship-sheet-common(); | ||||
|   overflow: scroll; | ||||
| } | ||||
|  | ||||
| @@ -34,6 +34,14 @@ | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .spec { | ||||
|         label { | ||||
|           max-width: 4rem; | ||||
|         } | ||||
|         input { | ||||
|           max-width: 4rem; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     .starship-right { | ||||
| @@ -50,6 +58,59 @@ | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .capacity { | ||||
|         .dice-2d6 { | ||||
|           max-width: 1.5rem; | ||||
|           margin-left: 0.2rem; | ||||
|         } | ||||
|         .rollable:hover, | ||||
|         .rollable:focus { | ||||
|           text-shadow: 0 0 8px var(--color-shadow-primary); | ||||
|           cursor: pointer; | ||||
|         } | ||||
|         .hull-field { | ||||
|           select { | ||||
|             min-width: 10rem; | ||||
|             max-width: 10rem; | ||||
|           } | ||||
|         } | ||||
|         label { | ||||
|           margin-left: 0.2rem; | ||||
|           max-width: 7rem; | ||||
|           min-width: 7rem; | ||||
|         } | ||||
|         .flexrow-guns { | ||||
|           display: flex; | ||||
|           align-items: center; | ||||
|           align-self: center; | ||||
|           gap: 4px; | ||||
|           .starship-guns-label { | ||||
|             min-width: 7.4rem; | ||||
|             max-width: 7.4rem; | ||||
|             font-weight: bold; | ||||
|           } | ||||
|         } | ||||
|         input { | ||||
|           max-width: 5rem; | ||||
|         } | ||||
|         select { | ||||
|           max-width: 5rem; | ||||
|         } | ||||
|       } | ||||
|       .cargo { | ||||
|         label { | ||||
|           margin-left: 0.2rem; | ||||
|           min-width: 4rem; | ||||
|           max-width: 4rem; | ||||
|         } | ||||
|         input { | ||||
|           max-width: 6rem; | ||||
|         } | ||||
|         select { | ||||
|           max-width: 5rem; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .starship-infos { | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
| @@ -70,7 +131,7 @@ | ||||
|               margin-left: 4px; | ||||
|               font-size: calc(var(--font-size-standard) * 1.4); | ||||
|             } | ||||
|           }           | ||||
|           } | ||||
|           .starship-hp-max { | ||||
|             clear: both; | ||||
|             display: flex; | ||||
| @@ -100,11 +161,11 @@ | ||||
|   } | ||||
|  | ||||
|   .starship-pc-edit { | ||||
|     min-width: 650px; | ||||
|     min-width: 500px; | ||||
|   } | ||||
| } | ||||
|  | ||||
| .starship-biography { | ||||
| .starship-description { | ||||
|   background-color: var(--color-light-1); | ||||
|   prose-mirror.inactive { | ||||
|     min-height: 40px; | ||||
| @@ -115,24 +176,32 @@ | ||||
|   .field-label { | ||||
|     margin-left: 8px; | ||||
|   } | ||||
|    | ||||
|    | ||||
|  | ||||
|   .section-editor { | ||||
|     min-height: 10rem; | ||||
|     .editor { | ||||
|       min-height: 10rem; | ||||
|     } | ||||
|     .editor-content { | ||||
|       min-height: 10rem; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .biodata { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 8px; | ||||
|     label { | ||||
|       min-width: 3.0rem; | ||||
|       min-width: 3rem; | ||||
|     } | ||||
|     .feature { | ||||
|         display: flex; | ||||
|         align-items: center; | ||||
|         gap: 4px; | ||||
|         min-width: 18rem; | ||||
|         max-width: 18rem;   | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 18rem; | ||||
|       max-width: 18rem; | ||||
|     } | ||||
|   } | ||||
|  | ||||
| } | ||||
|  | ||||
| .tab.starship-equipment { | ||||
|   | ||||
| @@ -33,7 +33,6 @@ | ||||
|           border: none; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     .vehicle-right { | ||||
| @@ -50,6 +49,23 @@ | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .cargo, | ||||
|       .capacity { | ||||
|         label { | ||||
|           min-width: 4rem; | ||||
|           max-width: 4rem; | ||||
|         } | ||||
|         input { | ||||
|           max-width: 6rem; | ||||
|           margin-right: 0.5rem; | ||||
|         } | ||||
|         .cargo-content { | ||||
|           display: flex; | ||||
|           align-items: center; | ||||
|           gap: 4px; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .vehicle-infos { | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
| @@ -70,7 +86,7 @@ | ||||
|               margin-left: 4px; | ||||
|               font-size: calc(var(--font-size-standard) * 1.4); | ||||
|             } | ||||
|           }           | ||||
|           } | ||||
|           .vehicle-hp-max { | ||||
|             clear: both; | ||||
|             display: flex; | ||||
| @@ -100,11 +116,15 @@ | ||||
|   } | ||||
|  | ||||
|   .vehicle-pc-edit { | ||||
|     min-width: 650px; | ||||
|     min-width: 500px; | ||||
|   } | ||||
| } | ||||
|  | ||||
| .vehicle-biography { | ||||
| .vehicle-description { | ||||
|   .text-center { | ||||
|     text-align: left; | ||||
|     align-content: center; | ||||
|   } | ||||
|   background-color: var(--color-light-1); | ||||
|   prose-mirror.inactive { | ||||
|     min-height: 40px; | ||||
| @@ -115,27 +135,35 @@ | ||||
|   .field-label { | ||||
|     margin-left: 8px; | ||||
|   } | ||||
|    | ||||
|    | ||||
|  | ||||
|   .section-editor { | ||||
|     min-height: 10rem; | ||||
|     .editor { | ||||
|       min-height: 10rem; | ||||
|     } | ||||
|     .editor-content { | ||||
|       min-height: 10rem; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .biodata { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(2, 1fr); | ||||
|     gap: 8px; | ||||
|     label { | ||||
|       min-width: 3.0rem; | ||||
|       min-width: 3rem; | ||||
|     } | ||||
|     .feature { | ||||
|         display: flex; | ||||
|         align-items: center; | ||||
|         gap: 4px; | ||||
|         min-width: 18rem; | ||||
|         max-width: 18rem;   | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|       min-width: 18rem; | ||||
|       max-width: 18rem; | ||||
|     } | ||||
|   } | ||||
|  | ||||
| } | ||||
|  | ||||
| .tab.vehicle-equipment { | ||||
| .tab.vehicle-equipment .main-div { | ||||
|   background-color: var(--color-light-1); | ||||
|   display: grid; | ||||
|   grid-template-columns: 1fr; | ||||
| @@ -179,11 +207,12 @@ | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   .gears { | ||||
|  | ||||
|   .equipments { | ||||
|     display: grid; | ||||
|     grid-template-columns: repeat(3, 1fr); | ||||
|     gap: 4px; | ||||
|     .gear { | ||||
|     .equipment { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|   | ||||
							
								
								
									
										37
									
								
								system.json
									
									
									
									
									
								
							
							
						
						| @@ -2,11 +2,11 @@ | ||||
|   "id": "fvtt-ftl-nomad", | ||||
|   "title": "Faster Than Light : Nomad", | ||||
|   "description": "Faster Than Light : Nomad is a system for Foundry VTT, based on the FTL system. It is a system for playing in a universe of space opera, where the players are the crew of a spaceship, traveling from planet to planet, from system to system, in search of adventure and fortune.", | ||||
|   "manifest": "https://www.uberwald.me/gitea/public/fvtt-ftl-nomad/raw/branch/main/system.json", | ||||
|   "manifest": "https://www.uberwald.me/gitea/uberwald/fvtt-ftl-nomad/releases/download/latest/system.json", | ||||
|   "download": "#{DOWNLOAD}#", | ||||
|   "url": "https://www.uberwald.me/gitea/public/fvtt-ftl-nomad", | ||||
|   "license": "LICENSE", | ||||
|   "version": "12.0.0", | ||||
|   "version": "13.0.0", | ||||
|   "authors": [ | ||||
|     { | ||||
|       "name": "Uberwald", | ||||
| @@ -20,8 +20,8 @@ | ||||
|     } | ||||
|   }, | ||||
|   "compatibility": { | ||||
|     "minimum": "12", | ||||
|     "verified": "12" | ||||
|     "minimum": "13", | ||||
|     "verified": "13" | ||||
|   }, | ||||
|   "esmodules": ["fvtt-ftl-nomad.mjs"], | ||||
|   "styles": ["css/fvtt-ftl-nomad.css"], | ||||
| @@ -37,7 +37,8 @@ | ||||
|       "character": { "htmlFields": ["description", "notes"] }, | ||||
|       "creature": { "htmlFields": ["description", "notes"] }, | ||||
|       "vehicle": { "htmlFields": ["description", "notes"] }, | ||||
|       "starship": { "htmlFields": ["description", "notes"] } | ||||
|       "starship": { "htmlFields": ["description", "notes"] }, | ||||
|       "robot": { "htmlFields": ["description", "notes"] } | ||||
|     }, | ||||
|     "Item": { | ||||
|       "talent": { "htmlFields": ["description"] }, | ||||
| @@ -52,6 +53,30 @@ | ||||
|     } | ||||
|   }, | ||||
|   "packs": [ | ||||
|     { | ||||
|       "label": "FTL Nomad - Items", | ||||
|       "type": "Item", | ||||
|       "name": "ftl-nomad-items", | ||||
|       "path": "packs/ftl-nomad-items", | ||||
|       "system": "fvtt-ftl-nomad", | ||||
|       "flags": {}, | ||||
|       "ownership": { | ||||
|         "PLAYER": "NONE", | ||||
|         "ASSISTANT": "OWNER" | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "label": "FTL Nomad - Vehicles", | ||||
|       "type": "Actor", | ||||
|       "name": "ftl-nomad-vehicles", | ||||
|       "path": "packs/ftl-nomad-vehicles", | ||||
|       "system": "fvtt-ftl-nomad", | ||||
|       "flags": {}, | ||||
|       "ownership": { | ||||
|         "PLAYER": "NONE", | ||||
|         "ASSISTANT": "OWNER" | ||||
|       } | ||||
|     } | ||||
|   ], | ||||
|   "grid": { | ||||
|     "distance": 10, | ||||
| @@ -59,5 +84,5 @@ | ||||
|   }, | ||||
|   "primaryTokenAttribute": "hp", | ||||
|   "socket": true, | ||||
|   "background": "systems/fvtt-ftl-nomad/assets/ui/background_01.webp" | ||||
|   "background": "systems/fvtt-ftl-nomad/assets/ui/ftl_nomad_background_01.webp" | ||||
| } | ||||
|   | ||||
| @@ -1,79 +1,34 @@ | ||||
| <section class="tab protagonist-{{tab.id}} {{tab.cssClass}}" data-tab="{{tab.id}}" data-group="{{tab.group}}"> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.resources"}}</legend> | ||||
|      <div class="resources"> | ||||
|       {{formField systemFields.resources.fields.permanentRating value=system.resources.permanentRating name="system.resources.permanentRating" localize=true}} | ||||
|       <div> | ||||
|         <span class="label-field">{{localize "FTLNOMAD.Label.resourceChecks"}}</span> | ||||
|         {{#each system.resources.checks as |check idx|}} | ||||
|         <input class="san-checkbox" type="checkbox" data-action="updateCheckResource" data-index="{{@index}}" | ||||
|           data-name="checks" {{#if check}} checked {{/if}}   {{#if (gte @index @root.system.resources.nbValidChecks)}} disabled {{/if}}> | ||||
|         {{/each}} | ||||
|       </div> | ||||
|       <button class="resource-roll rollable" data-roll-type="resource"><img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" />Resource roll</button> | ||||
|       {{formField systemFields.resources.fields.hand value=system.resources.hand name="system.resources.hand" localize=true disabled=true}} | ||||
|       {{formField systemFields.resources.fields.stowed value=system.resources.stowed name="system.resources.stowed" localize=true disabled=true}} | ||||
|       {{formField systemFields.resources.fields.storage value=system.resources.storage name="system.resources.storage" localize=true disabled=true}} | ||||
|       </div> | ||||
| <section class="tab character-{{tab.id}} {{tab.cssClass}}" data-tab="{{tab.id}}" data-group="{{tab.group}}"> | ||||
| <div class="main-div"> | ||||
|     <fieldset class="rank"> | ||||
|     <legend>Rank</legend> | ||||
|     {{formField systemFields.rank.fields.experienced value=system.rank.experienced type="number" rootId=partId disabled=isPlayMode}} | ||||
|     {{formField systemFields.rank.fields.expert value=system.rank.expert type="number" rootId=partId disabled=isPlayMode}} | ||||
|     {{formField systemFields.rank.fields.veteran value=system.rank.veteran type="number" rootId=partId disabled=isPlayMode}} | ||||
|     {{formField systemFields.rank.fields.elite value=system.rank.elite type="number" rootId=partId disabled=isPlayMode}} | ||||
|     {{formField systemFields.rank.fields.legend value=system.rank.legend type="number" rootId=partId disabled=isPlayMode}} | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset> | ||||
|   <fieldset class="biodata"> | ||||
|     <legend>{{localize "FTLNOMAD.Label.biodata"}}</legend> | ||||
|     <div class="adapted"> | ||||
|       {{formField systemFields.biodata.fields.adaptedToViolence value=system.biodata.adaptedToViolence name="system.biodata.adaptedToViolence" localize=true}}       | ||||
|       {{formField systemFields.biodata.fields.adaptedToHelplessness value=system.biodata.adaptedToHelplessness name="system.biodata.adaptedToHelplessness" localize=true}}             | ||||
|     </div> | ||||
|     <div class="biodata"> | ||||
|       {{formField systemFields.biodata.fields.harshness value=system.biodata.harshness name="system.biodata.harshness" localize=true}}       | ||||
|       {{formField systemFields.biodata.fields.age value=system.biodata.age name="system.biodata.age" localize=true classes="field-label"}} | ||||
|       {{formField systemFields.biodata.fields.gender value=system.biodata.gender name="system.biodata.gender" localize=true classes="field-label"}} | ||||
|       {{formField systemFields.biodata.fields.height value=system.biodata.height name="system.biodata.height" localize=true classes="field-label"}} | ||||
|       {{formField systemFields.biodata.fields.eyes value=system.biodata.eyes name="system.biodata.eyes" localize=true classes="field-label"}} | ||||
|       {{formField systemFields.biodata.fields.hair value=system.biodata.hair name="system.biodata.hair" localize=true classes="field-label"}} | ||||
|       {{formField systemFields.biodata.fields.home value=system.biodata.home name="system.biodata.home" localize=true classes="field-label"}} | ||||
|       {{formField systemFields.biodata.fields.birthplace value=system.biodata.birthplace name="system.biodata.birthplace" localize=true classes="field-label"}} | ||||
|     </div> | ||||
|     {{formField systemFields.biodata.fields.gender value=system.biodata.gender rootId=partId disabled=isPlayMode classes="short"}} | ||||
|     {{formField systemFields.biodata.fields.age value=system.biodata.age rootId=partId disabled=isPlayMode classes="short"}} | ||||
|     {{formField systemFields.biodata.fields.height value=system.biodata.height rootId=partId disabled=isPlayMode classes="short"}} | ||||
|     {{formField systemFields.biodata.fields.weight value=system.biodata.weight rootId=partId disabled=isPlayMode classes="short"}} | ||||
|     {{formField systemFields.biodata.fields.eyes value=system.biodata.eyes rootId=partId disabled=isPlayMode classes="short"}} | ||||
|     {{formField systemFields.biodata.fields.hair value=system.biodata.hair rootId=partId disabled=isPlayMode classes="short"}} | ||||
|     {{formField systemFields.biodata.fields.home value=system.biodata.home rootId=partId disabled=isPlayMode}} | ||||
|     {{formField systemFields.biodata.fields.birthplace value=system.biodata.birthplace rootId=partId disabled=isPlayMode}} | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.distinguishingFeatures"}}</legend> | ||||
|     <div class="features"> | ||||
|       <div class="feature">  | ||||
|         <label>{{localize "FTLNOMAD.Label.strShort"}}</label> | ||||
|         {{formInput systemFields.characteristics.fields.str.fields.feature value=system.characteristics.str.feature localize=true}}  | ||||
|       </div> | ||||
|       <div class="feature">  | ||||
|         <label>{{localize "FTLNOMAD.Label.dexShort"}}</label> | ||||
|         {{formInput systemFields.characteristics.fields.dex.fields.feature value=system.characteristics.dex.feature localize=true}}  | ||||
|       </div> | ||||
|       <div class="feature">  | ||||
|         <label>{{localize "FTLNOMAD.Label.conShort"}}</label> | ||||
|         {{formInput systemFields.characteristics.fields.con.fields.feature value=system.characteristics.con.feature localize=true}}  | ||||
|       </div> | ||||
|       <div class="feature">  | ||||
|         <label>{{localize "FTLNOMAD.Label.intShort"}}</label> | ||||
|         {{formInput systemFields.characteristics.fields.int.fields.feature value=system.characteristics.int.feature localize=true}}  | ||||
|       </div> | ||||
|       <div class="feature">  | ||||
|         <label>{{localize "FTLNOMAD.Label.powShort"}}</label> | ||||
|         {{formInput systemFields.characteristics.fields.pow.fields.feature value=system.characteristics.pow.feature localize=true}}  | ||||
|       </div> | ||||
|       <div class="feature">  | ||||
|         <label>{{localize "FTLNOMAD.Label.chaShort"}}</label> | ||||
|         {{formInput systemFields.characteristics.fields.cha.fields.feature value=system.characteristics.cha.feature localize=true}}  | ||||
|       </div> | ||||
|     </div> | ||||
|   </fieldset>  | ||||
|  | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.description"}}</legend> | ||||
|     {{formInput systemFields.description enriched=enrichedDescription value=system.description name="system.description" toggled=true}} | ||||
|     {{formInput systemFields.description enriched=enrichedDescription value=system.description name="system.description" toggled=true class="character-description"}} | ||||
|   </fieldset> | ||||
|    | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.notes"}}</legend> | ||||
|     {{formInput systemFields.notes enriched=enrichedNotes value=system.notes name="system.notes" toggled=true}} | ||||
|     {{formInput systemFields.notes enriched=enrichedNotes value=system.notes name="system.notes" toggled=true class="character-notes"}} | ||||
|   </fieldset> | ||||
| </div> | ||||
| </section> | ||||
| @@ -1,22 +1,22 @@ | ||||
| <section class="tab protagonist-{{tab.id}} {{tab.cssClass}}" data-tab="{{tab.id}}" data-group="{{tab.group}}"> | ||||
| <section class="tab character-{{tab.id}} {{tab.cssClass}}" data-tab="{{tab.id}}" data-group="{{tab.group}}"> | ||||
| <div class="main-div"> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.weapons"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize " FTLNOMAD.Tooltip.addWeapon"}}" data-tooltip-direction="UP"><i | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addWeapon"}}" data-tooltip-direction="UP"><i | ||||
|           class="fas fa-plus" data-action="createWeapon"></i></a>{{/if}} | ||||
|     </legend> | ||||
|     <div class="weapons"> | ||||
|       {{#each weapons as |item|}} | ||||
|       {{!log 'weapon' this}} | ||||
|       <div class="weapon item" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}" data-drag="true"> | ||||
|  | ||||
|         <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|         <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|         <i class="fa-regular fa-dice"></i> | ||||
|         <div class="name rollable" data-roll-type="weapon" data-tooltip="{{{item.system.description}}}"> | ||||
|           {{item.name}} | ||||
|         </div> | ||||
|  | ||||
|         <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|         <i class="fa-regular fa-dice"></i> | ||||
|         <a class="damage rollable" data-item-id="{{item.id}}" data-action="roll" data-roll-type="damage" | ||||
|           data-roll-value="{{item.system.damage}}"> | ||||
|           {{localize "FTLNOMAD.Label.damageShort"}} : | ||||
| @@ -34,13 +34,12 @@ | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.armors"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize " FTLNOMAD.Tooltip.addArmor"}}" data-tooltip-direction="UP"><i | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addArmor"}}" data-tooltip-direction="UP"><i | ||||
|           class="fas fa-plus" data-action="createArmor"></i></a>{{/if}} | ||||
|     </legend> | ||||
|     <div class="armors"> | ||||
|       {{#each armors as |item|}} | ||||
|       {{!log 'armor' this}} | ||||
|       <div class="armor" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"> | ||||
|       <div class="armor item" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"> | ||||
|         <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|         <div class="name" data-tooltip="{{{item.system.description}}}"> | ||||
|           {{item.name}} | ||||
| @@ -58,14 +57,46 @@ | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.gears"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize " FTLNOMAD.Tooltip.addGear"}}" data-tooltip-direction="UP"><i | ||||
|           class="fas fa-plus" data-action="createGear"></i></a>{{/if}} | ||||
|     <legend>{{localize "FTLNOMAD.Label.implants"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addTalent"}}" data-tooltip-direction="UP"><i | ||||
|           class="fas fa-plus" data-action="createImplant"></i></a>{{/if}} | ||||
|     </legend> | ||||
|     <div class="gears"> | ||||
|       {{#each gears as |item|}} | ||||
|     <div class="implants"> | ||||
|       {{#each implants as |item|}} | ||||
|       {{!log 'weapon' this}} | ||||
|       <div class="implant item" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}" data-drag="true"> | ||||
|  | ||||
|         <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|         <div class="name" data-roll-type="weapon" data-tooltip="{{{item.system.description}}}"> | ||||
|           {{item.name}} | ||||
|         </div> | ||||
|  | ||||
|         <a class="damage rollable" data-item-id="{{item.id}}" data-action="roll" data-roll-type="damage" | ||||
|           data-roll-value="{{item.system.damage}}"> | ||||
|           {{#if item.system.isAdvantage}} | ||||
|           <i data-tooltip="Provides advantage" class="fas fa-circle-chevron-up"></i> | ||||
|           {{else}} | ||||
|           {{/if}} | ||||
|         <div class="controls"> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|         </div> | ||||
|       </div> | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.equipments"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addEquipment"}}" data-tooltip-direction="UP"><i | ||||
|           class="fas fa-plus" data-action="createEquipment"></i></a>{{/if}} | ||||
|     </legend> | ||||
|     <div class="equipments"> | ||||
|       {{#each equipments as |item|}} | ||||
|       {{!log 'armor' this}} | ||||
|       <div class="gear" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"> | ||||
|       <div class="equipment" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"> | ||||
|         <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|         <div class="name" data-tooltip="{{{item.system.description}}}"> | ||||
|           {{item.name}} | ||||
| @@ -80,56 +111,5 @@ | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|   {{#if (count tomes)}} | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.Tomes"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addTome"}}" data-tooltip-direction="UP"><i | ||||
|           class="fas fa-plus" data-action="createTome"></i></a>{{/if}} | ||||
|     </legend> | ||||
|     <div class="tomes"> | ||||
|       {{#each tomes as |item|}} | ||||
|       <div class="tome" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"> | ||||
|         <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|         <div class="name" data-tooltip="{{{item.system.description}}}"> | ||||
|           {{item.name}} | ||||
|         </div> | ||||
|         <div class="controls"> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|         </div> | ||||
|       </div> | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|   {{/if}} | ||||
|  | ||||
|   {{#if (count rituals)}} | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.Rituals"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addRitual"}}" data-tooltip-direction="UP"><i | ||||
|           class="fas fa-plus" data-action="createRitual"></i></a>{{/if}} | ||||
|     </legend> | ||||
|     <div class="rituals"> | ||||
|       {{#each rituals as |item|}} | ||||
|       <div class="ritual" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"> | ||||
|         <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|         <div class="name" data-tooltip="{{{item.system.description}}}"> | ||||
|           {{item.name}} | ||||
|         </div> | ||||
|         <div class="controls"> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|         </div> | ||||
|       </div> | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|   {{/if}} | ||||
|  | ||||
|  | ||||
| </div> | ||||
| </section> | ||||
| @@ -1,155 +1,265 @@ | ||||
| <section class="protagonist-main protagonist-main-{{ifThen isPlayMode 'play' 'edit'}}"> | ||||
|   {{!log "protagonist-main" this}} | ||||
|  | ||||
| <section | ||||
|   class="character-main character-main-{{ifThen isPlayMode 'play' 'edit'}}" | ||||
| > | ||||
|   {{!log "character-main" this}} | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.protagonist"}}</legend> | ||||
|     <div class="protagonist-pc protagonist-pc-{{ifThen isPlayMode 'play' 'edit'}}"> | ||||
|       <div class="protagonist-left"> | ||||
|         <div class="protagonist-left-image"> | ||||
|           <img class="protagonist-img" src="{{actor.img}}" data-edit="img" data-action="editImage" | ||||
|             data-tooltip="{{actor.name}}" /> | ||||
|     <legend>{{localize "FTLNOMAD.Label.character"}}</legend> | ||||
|     <div class="character-pc character-pc-{{ifThen isPlayMode 'play' 'edit'}}"> | ||||
|       <div class="character-left"> | ||||
|         <div class="character-left-image"> | ||||
|           <img | ||||
|             class="character-img" | ||||
|             src="{{actor.img}}" | ||||
|             data-edit="img" | ||||
|             data-action="editImage" | ||||
|             data-tooltip="{{actor.name}}" | ||||
|           /> | ||||
|         </div> | ||||
|         <fieldset class="protagonist-hp">  | ||||
|           <legend>{{localize "FTLNOMAD.Label.HP"}}</legend> | ||||
|         <fieldset class="character-hp"> | ||||
|           <legend>{{localize "FTLNOMAD.Label.Stamina"}}</legend> | ||||
|           <div class="flexrow"> | ||||
|             {{formField systemFields.hp.fields.value value=system.hp.value}} | ||||
|             <span class="hp-separator">/</span> | ||||
|             {{formField systemFields.hp.fields.max value=system.hp.max rootId=partId disabled=true}} | ||||
|             {{formField | ||||
|               systemFields.health.fields.staminaValue | ||||
|               value=system.health.staminaValue | ||||
|             }} | ||||
|             {{formField | ||||
|               systemFields.health.fields.staminaMax | ||||
|               value=system.health.staminaMax | ||||
|               rootId=partId | ||||
|               disabled=true | ||||
|             }} | ||||
|           </div> | ||||
|           <div class="flexrow "> | ||||
|             {{formField systemFields.damageBonus value=system.damageBonus classes="damage-bonus"}} | ||||
|           <div class="flexrow"> | ||||
|             {{formField | ||||
|               systemFields.health.fields.wounds | ||||
|               value=system.health.wounds | ||||
|             }} | ||||
|           </div> | ||||
|         </fieldset> | ||||
|  | ||||
|       </div> | ||||
|       <div class="protagonist-right"> | ||||
|         <div class="protagonist-name"> | ||||
|           {{formInput fields.name value=source.name rootId=partId disabled=isPlayMode}} | ||||
|           <a class="control" data-action="toggleSheet" data-tooltip="FTLNOMAD.ToggleSheet" | ||||
|             data-tooltip-direction="UP"> | ||||
|       <div class="character-right"> | ||||
|         <div class="character-name"> | ||||
|           {{formInput | ||||
|             fields.name | ||||
|             value=source.name | ||||
|             rootId=partId | ||||
|             disabled=isPlayMode | ||||
|           }} | ||||
|           <a | ||||
|             class="control" | ||||
|             data-action="toggleSheet" | ||||
|             data-tooltip="FTLNOMAD.ToggleSheet" | ||||
|             data-tooltip-direction="UP" | ||||
|           > | ||||
|             <i class="fa-solid fa-user-{{ifThen isPlayMode 'lock' 'pen'}}"></i> | ||||
|           </a> | ||||
|         </div> | ||||
|  | ||||
|  | ||||
|         <fieldset class="san"> | ||||
|           <legend>{{localize "FTLNOMAD.Label.SAN"}}</legend> | ||||
|           <div class="flexrow"> | ||||
|              | ||||
|             <label class="label-field rollable" data-roll-type="san" ><img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" />  | ||||
|             {{localize "FTLNOMAD.Label.current"}}</label> | ||||
|             {{formInput systemFields.san.fields.value value=system.san.value}} | ||||
|              | ||||
|             <span class="label-field label-recovery">{{localize "FTLNOMAD.Label.recovery"}}</span>  | ||||
|             {{formInput systemFields.san.fields.recovery value=system.san.recovery disabled=true}} | ||||
|  | ||||
|             <span class="label-bp" data-tooltip='{{localize "FTLNOMAD.Tooltip.sanBPShort"}}'>{{localize | ||||
|               "FTLNOMAD.Label.sanBPShort"}}</span>  | ||||
|             {{formInput systemFields.san.fields.breakingPoint value=system.san.breakingPoint disabled=true}} | ||||
|  | ||||
|         <fieldset class="character-spec"> | ||||
|           {{formField | ||||
|             systemFields.concept | ||||
|             value=system.concept | ||||
|             rootId=partId | ||||
|             disabled=isPlayMode | ||||
|           }} | ||||
|           {{formField | ||||
|             systemFields.species | ||||
|             value=system.species | ||||
|             rootId=partId | ||||
|             disabled=isPlayMode | ||||
|           }} | ||||
|           {{formField | ||||
|             systemFields.archetype | ||||
|             value=system.archetype | ||||
|             rootId=partId | ||||
|             disabled=isPlayMode | ||||
|           }} | ||||
|           <div class="hero-armor"> | ||||
|             {{formField | ||||
|               systemFields.heroPoints | ||||
|               value=system.heroPoints | ||||
|               rootId=partId | ||||
|             }} | ||||
|             {{formField | ||||
|               systemFields.armor.fields.value | ||||
|               value=system.armor.value | ||||
|               rootId=partId | ||||
|               disabled=isPlayMode | ||||
|             }} | ||||
|           </div> | ||||
|  | ||||
|           <div class="flexrow"> | ||||
|              | ||||
|             <div class="flexrow"> | ||||
|              <span class="label-short-field">{{localize "FTLNOMAD.Label.max"}}</span>  | ||||
|               {{formInput systemFields.san.fields.max value=system.san.max rootId=partId disabled=true}} | ||||
|             </div> | ||||
|  | ||||
|             <span class="label-insanity" > | ||||
|               {{localize "FTLNOMAD.Label.tempInsanity"}} | ||||
|             </span> | ||||
|             {{formInput systemFields.san.fields.insanity value=system.san.insanity localize=true}} | ||||
|  | ||||
|             {{#if (not isPlayMode)}} | ||||
|               <button data-action="setBP" class="button" data-tooltip="{{localize "FTLNOMAD.Tooltip.setBP"}}">{{localize "FTLNOMAD.Label.setBP"}}</button> | ||||
|             {{/if}} | ||||
|  | ||||
|           </div> | ||||
|           <div class="flexrow"> | ||||
|  | ||||
|             <span class="label-field">{{localize "FTLNOMAD.Label.violence"}}</span> | ||||
|             {{#each system.san.violence as |violence idx|}} | ||||
|             <input class="san-checkbox" type="checkbox" data-action="updateCheckboxArray" data-index="{{@index}}" | ||||
|               data-name="violence" {{#if violence}} checked {{/if}}> | ||||
|             {{/each}} | ||||
|              | ||||
|             <span class="label-field">{{localize "FTLNOMAD.Label.helplessness"}}</span> | ||||
|             {{#each system.san.helplessness as |helplessness idx|}} | ||||
|             <input class="san-checkbox" type="checkbox" data-action="updateCheckboxArray" data-index="{{@index}}" | ||||
|               data-name="helplessness" {{#if helplessness}} checked {{/if}}> | ||||
|             {{/each}} | ||||
|  | ||||
|           </div> | ||||
|  | ||||
|         </fieldset> | ||||
|  | ||||
|         <fieldset class="willpower"> | ||||
|           <legend>{{localize "FTLNOMAD.Label.willpower"}}</legend> | ||||
|           <div class="flexrow"> | ||||
|             <label class="label-field">{{localize "FTLNOMAD.Label.current"}}</label> | ||||
|             {{formInput systemFields.wp.fields.value value=system.wp.value}} | ||||
|             <label class="label-field">{{localize "FTLNOMAD.Label.max"}}</label> | ||||
|             {{formInput systemFields.wp.fields.max value=system.wp.max rootId=partId disabled=true}} | ||||
|             <label class="label-field">{{localize "FTLNOMAD.Label.exhausted"}}</label> | ||||
|             {{formInput systemFields.wp.fields.exhausted value=system.wp.exhausted classes="checkbox"}} | ||||
|           </div> | ||||
|         <fieldset class="encumbrance"> | ||||
|           {{#if isEncumbered}} | ||||
|             {{formField | ||||
|               systemFields.enc.fields.value | ||||
|               value=system.enc.value | ||||
|               rootId=partId | ||||
|               disabled=true | ||||
|               classes="encumbered" | ||||
|             }} | ||||
|           {{else}} | ||||
|             {{formField | ||||
|               systemFields.enc.fields.value | ||||
|               value=system.enc.value | ||||
|               rootId=partId | ||||
|               disabled=true | ||||
|             }} | ||||
|           {{/if}} | ||||
|           {{formField | ||||
|             systemFields.enc.fields.max | ||||
|             value=system.enc.max | ||||
|             rootId=partId | ||||
|             disabled=isPlayMode | ||||
|           }} | ||||
|           <span class="character-credit"> | ||||
|             {{formField | ||||
|               systemFields.credits | ||||
|               value=system.credits | ||||
|               rootId=partId | ||||
|               disabled=isPlayMode | ||||
|             }} | ||||
|           </span> | ||||
|         </fieldset> | ||||
|  | ||||
|       </div> | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset class="protagonist-characteristics protagonist-characteristics-{{ifThen isPlayMode 'play' 'edit'}}"> | ||||
|     <legend>{{localize "FTLNOMAD.Label.characteristics"}}</legend> | ||||
|     <div class="protagonist-characteristic" > | ||||
|       <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|       <label class="rollable" data-roll-type="char" data-char-id="str" data-tooltip="{{system.characteristics.str.feature}}">{{localize | ||||
|         "FTLNOMAD.Label.strShort"}}</label> | ||||
|       {{formField systemFields.characteristics.fields.str.fields.value value=system.characteristics.str.value | ||||
|       rootId=partId disabled=isPlayMode }} | ||||
|       <label class="char-text">{{mul system.characteristics.str.value 5}}</label> | ||||
|   <fieldset | ||||
|     class="character-skills character-skills-{{ifThen | ||||
|         isPlayMode | ||||
|         'play' | ||||
|         'edit' | ||||
|       }}" | ||||
|   > | ||||
|     <legend>{{localize "FTLNOMAD.Label.skills"}}</legend> | ||||
|     <div class="character-skill"> | ||||
|       <img | ||||
|         src="systems/fvtt-ftl-nomad/assets/icons/icon_combat.svg" | ||||
|         class="icon-skill" | ||||
|       /> | ||||
|       <label | ||||
|         class="rollable" | ||||
|         data-roll-type="skill" | ||||
|         data-skill-id="combat" | ||||
|       >{{localize "FTLNOMAD.Label.combat"}}</label> | ||||
|       {{formInput | ||||
|         systemFields.skills.fields.combat.fields.value | ||||
|         value=system.skills.combat.value | ||||
|         rootId=partId | ||||
|         disabled=isPlayMode | ||||
|         type="number" | ||||
|       }} | ||||
|     </div> | ||||
|     <div class="protagonist-characteristic"> | ||||
|       <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|       <label class="rollable" data-roll-type="char" data-char-id="dex" data-tooltip="{{system.characteristics.dex.feature}}">{{localize | ||||
|         "FTLNOMAD.Label.dexShort"}}</label> | ||||
|       {{formField systemFields.characteristics.fields.dex.fields.value value=system.characteristics.dex.value | ||||
|       rootId=partId disabled=isPlayMode }} | ||||
|       <label class="char-text">{{mul system.characteristics.dex.value 5}}</label> | ||||
|     <div class="character-skill"> | ||||
|       <img | ||||
|         src="systems/fvtt-ftl-nomad/assets/icons/icon_knowledge.svg" | ||||
|         class="icon-skill" | ||||
|       /> | ||||
|       <label | ||||
|         class="rollable" | ||||
|         data-roll-type="skill" | ||||
|         data-skill-id="knowledge" | ||||
|       >{{localize "FTLNOMAD.Label.knowledge"}}</label> | ||||
|       {{formInput | ||||
|         systemFields.skills.fields.knowledge.fields.value | ||||
|         value=system.skills.knowledge.value | ||||
|         rootId=partId | ||||
|         disabled=isPlayMode | ||||
|         type="number" | ||||
|       }} | ||||
|     </div> | ||||
|     <div class="protagonist-characteristic" > | ||||
|       <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|       <label class="rollable" data-roll-type="char" data-char-id="con" data-tooltip="{{system.characteristics.con.feature}}">{{localize | ||||
|         "FTLNOMAD.Label.conShort"}}</label> | ||||
|       {{formField systemFields.characteristics.fields.con.fields.value value=system.characteristics.con.value | ||||
|       rootId=partId disabled=isPlayMode }} | ||||
|        <label class="char-text">{{mul system.characteristics.con.value 5}}</label> | ||||
|     <div class="character-skill"> | ||||
|       <img | ||||
|         src="systems/fvtt-ftl-nomad/assets/icons/icon_social.svg" | ||||
|         class="icon-skill" | ||||
|       /> | ||||
|       <label | ||||
|         class="rollable" | ||||
|         data-roll-type="skill" | ||||
|         data-skill-id="social" | ||||
|       >{{localize "FTLNOMAD.Label.social"}}</label> | ||||
|       {{formInput | ||||
|         systemFields.skills.fields.social.fields.value | ||||
|         value=system.skills.social.value | ||||
|         rootId=partId | ||||
|         disabled=isPlayMode | ||||
|         type="number" | ||||
|       }} | ||||
|     </div> | ||||
|     <div class="protagonist-characteristic" > | ||||
|       <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|       <label class="rollable" data-roll-type="char" data-char-id="int" data-tooltip="{{system.characteristics.int.feature}}">{{localize | ||||
|         "FTLNOMAD.Label.intShort"}}</label> | ||||
|       {{formField systemFields.characteristics.fields.int.fields.value value=system.characteristics.int.value | ||||
|       rootId=partId disabled=isPlayMode }} | ||||
|        <label class="char-text">{{mul system.characteristics.int.value 5}}</label> | ||||
|     <div class="character-skill"> | ||||
|       <img | ||||
|         src="systems/fvtt-ftl-nomad/assets/icons/icon_physical.svg" | ||||
|         class="icon-skill" | ||||
|       /> | ||||
|       <label | ||||
|         class="rollable" | ||||
|         data-roll-type="skill" | ||||
|         data-skill-id="physical" | ||||
|       >{{localize "FTLNOMAD.Label.physical"}}</label> | ||||
|       {{formInput | ||||
|         systemFields.skills.fields.physical.fields.value | ||||
|         value=system.skills.physical.value | ||||
|         rootId=partId | ||||
|         disabled=isPlayMode | ||||
|         type="number" | ||||
|       }} | ||||
|     </div> | ||||
|     <div class="protagonist-characteristic" > | ||||
|       <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|       <label class="rollable" data-roll-type="char" data-char-id="pow" data-tooltip="{{system.characteristics.pow.feature}}">{{localize | ||||
|         "FTLNOMAD.Label.powShort"}}</label> | ||||
|       {{formField systemFields.characteristics.fields.pow.fields.value value=system.characteristics.pow.value | ||||
|       rootId=partId disabled=isPlayMode }} | ||||
|        <label class="char-text">{{mul system.characteristics.pow.value 5}}</label> | ||||
|     <div class="character-skill"> | ||||
|       <img | ||||
|         src="systems/fvtt-ftl-nomad/assets/icons/icon_stealth.svg" | ||||
|         class="icon-skill" | ||||
|       /> | ||||
|       <label | ||||
|         class="rollable" | ||||
|         data-roll-type="skill" | ||||
|         data-skill-id="stealth" | ||||
|       >{{localize "FTLNOMAD.Label.stealth"}}</label> | ||||
|       {{formInput | ||||
|         systemFields.skills.fields.stealth.fields.value | ||||
|         value=system.skills.stealth.value | ||||
|         rootId=partId | ||||
|         disabled=isPlayMode | ||||
|         type="number" | ||||
|       }} | ||||
|     </div> | ||||
|     <div class="protagonist-characteristic" > | ||||
|       <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|       <label class="rollable" data-roll-type="char" data-char-id="cha" data-tooltip="{{system.characteristics.cha.feature}}">{{localize | ||||
|         "FTLNOMAD.Label.chaShort"}}</label> | ||||
|       {{formField systemFields.characteristics.fields.cha.fields.value value=system.characteristics.cha.value | ||||
|       rootId=partId disabled=isPlayMode }} | ||||
|        <label class="char-text">{{mul system.characteristics.cha.value 5}}</label> | ||||
|     <div class="character-skill"> | ||||
|       <img | ||||
|         src="systems/fvtt-ftl-nomad/assets/icons/icon_vehicles.svg" | ||||
|         class="icon-skill" | ||||
|       /> | ||||
|       <label | ||||
|         class="rollable" | ||||
|         data-roll-type="skill" | ||||
|         data-skill-id="vehicles" | ||||
|       >{{localize "FTLNOMAD.Label.vehicles"}}</label> | ||||
|       {{formInput | ||||
|         systemFields.skills.fields.vehicles.fields.value | ||||
|         value=system.skills.vehicles.value | ||||
|         rootId=partId | ||||
|         disabled=isPlayMode | ||||
|         type="number" | ||||
|       }} | ||||
|     </div> | ||||
|     <div class="character-skill"> | ||||
|       <img | ||||
|         src="systems/fvtt-ftl-nomad/assets/icons/icon_technology.svg" | ||||
|         class="icon-skill" | ||||
|       /> | ||||
|       <label | ||||
|         class="rollable" | ||||
|         data-roll-type="skill" | ||||
|         data-skill-id="technology" | ||||
|       >{{localize "FTLNOMAD.Label.technology"}}</label> | ||||
|       {{formInput | ||||
|         systemFields.skills.fields.technology.fields.value | ||||
|         value=system.skills.technology.value | ||||
|         rootId=partId | ||||
|         disabled=isPlayMode | ||||
|         type="number" | ||||
|       }} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
| </section> | ||||
| @@ -1,25 +0,0 @@ | ||||
| <section class="tab protagonist-{{tab.id}} {{tab.cssClass}}" data-tab="{{tab.id}}" data-group="{{tab.group}}"> | ||||
|   {{log this}} | ||||
|   <fieldset> | ||||
|     <legend data-tooltip="{{localize "FTLNOMAD.Tooltip.skills"}}" data-tooltip-direction="UP">{{localize "FTLNOMAD.Label.skills"}}</legend> | ||||
|     <div class="skills"> | ||||
|         {{#each skills as |item|}} | ||||
|           <div class="skill item" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"> | ||||
|             <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|             <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|             <div class="name rollable" data-roll-type="skill" data-tooltip="{{{item.description}}}" data-tooltip-direction="UP">{{item.name}} | ||||
|             </div> | ||||
|             <div class="score" > | ||||
|               {{item.system.skillTotal}} | ||||
|             </div> | ||||
|             <div class="controls"> | ||||
|               <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|               <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|             </div> | ||||
|           </div> | ||||
|         {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|  | ||||
| </section> | ||||
| @@ -1,95 +0,0 @@ | ||||
| <section class="tab protagonist-{{tab.id}} {{tab.cssClass}}" data-tab="{{tab.id}}" data-group="{{tab.group}}"> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.bonds"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addBond"}}" data-tooltip-direction="UP"><i class="fas fa-plus" data-action="createBond"></i></a>{{/if}}</legend> | ||||
|     <div class="bonds"> | ||||
|       {{#each bonds as |item|}} | ||||
|         {{!log 'weapon' this}} | ||||
|         <div class="bond" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}" > | ||||
|           <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|           <div class="name" data-tooltip="{{{item.system.description}}}"> | ||||
|             {{item.name}} | ||||
|           </div> | ||||
|           <div class="type"> | ||||
|             {{upperFirst item.system.bondType}} | ||||
|           </div> | ||||
|           <div class="level"> | ||||
|             {{item.system.value}} | ||||
|           </div> | ||||
|           <div class="controls"> | ||||
|             <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|             <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|           </div> | ||||
|         </div> | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.injuries"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addInjury"}}" data-tooltip-direction="UP"><i class="fas fa-plus" data-action="createInjury"></i></a>{{/if}}</legend> | ||||
|     <div class="injuries"> | ||||
|       {{#each injuries as |item|}} | ||||
|         <div class="injury" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"> | ||||
|           <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|           <div class="name" data-tooltip="{{{item.system.description}}}"> | ||||
|             {{item.name}} | ||||
|           </div> | ||||
|           <div class="controls"> | ||||
|             <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|             <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|           </div> | ||||
|         </div> | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.mentaldisorders"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addMentalDisorder"}}" data-tooltip-direction="UP"><i class="fas fa-plus" data-action="createMentalDisorder"></i></a>{{/if}}</legend> | ||||
|     <div class="mentaldisorders"> | ||||
|       {{#each mentaldisorders as |item|}} | ||||
|         {{!log 'armor' this}} | ||||
|         <div class="mentaldisorder" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"> | ||||
|           <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|           <div class="name" data-tooltip="{{{item.system.description}}}"> | ||||
|             {{item.name}} | ||||
|           </div> | ||||
|           <div class="cured"> | ||||
|             {{#if item.system.cured}} | ||||
|               {{localize 'FTLNOMAD.Label.Cured'}} | ||||
|             {{else}} | ||||
|               {{localize 'FTLNOMAD.Label.Uncured'}} | ||||
|             {{/if}} | ||||
|           </div> | ||||
|           <div class="controls"> | ||||
|             <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|             <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|           </div> | ||||
|         </div> | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.motivations"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addMotivation"}}" data-tooltip-direction="UP"><i class="fas fa-plus" data-action="createMotivation"></i></a>{{/if}}</legend> | ||||
|     <div class="motivations"> | ||||
|       {{#each motivations as |item|}} | ||||
|         {{!log 'armor' this}} | ||||
|         <div class="motivation" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"> | ||||
|           <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|           <div class="name" data-tooltip="{{{item.system.description}}}"> | ||||
|             {{item.name}} | ||||
|           </div> | ||||
|           <div class="controls"> | ||||
|             <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|             <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|           </div> | ||||
|         </div> | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
| </section> | ||||
							
								
								
									
										100
									
								
								templates/character-talents.hbs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,100 @@ | ||||
| <section class="tab character-{{tab.id}} {{tab.cssClass}}" data-tab="{{tab.id}}" data-group="{{tab.group}}"> | ||||
| <div class="main-div"> | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.talents"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addTalent"}}" data-tooltip-direction="UP"><i | ||||
|           class="fas fa-plus" data-action="createTalent"></i></a>{{/if}} | ||||
|     </legend> | ||||
|     <div class="talents"> | ||||
|       {{#each talents as |item|}} | ||||
|       {{!log 'weapon' this}} | ||||
|       <div class="talent item" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}" data-drag="true"> | ||||
|  | ||||
|         <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|         <div class="name" data-roll-type="weapon" data-tooltip="{{{item.system.description}}}"> | ||||
|           {{item.name}} | ||||
|         </div> | ||||
|  | ||||
|         <a class="damage rollable" data-item-id="{{item.id}}" data-action="roll" data-roll-type="damage" | ||||
|           data-roll-value="{{item.system.damage}}"> | ||||
|           {{#if item.system.isAdvantage}} | ||||
|           <i data-tooltip="Provides advantage" class="fas fa-circle-chevron-up"></i> | ||||
|           {{else}} | ||||
|           {{/if}} | ||||
|         <div class="controls"> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|         </div> | ||||
|       </div> | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.psionics"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addImplant"}}" data-tooltip-direction="UP"><i | ||||
|           class="fas fa-plus" data-action="createPsionic"></i></a>{{/if}} | ||||
|     </legend> | ||||
|     <div class="psionics"> | ||||
|       {{#each psionics as |item|}} | ||||
|       {{!log 'weapon' this}} | ||||
|       <div class="psionic item" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}" data-drag="true"> | ||||
|  | ||||
|         <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|         <div class="name" data-roll-type="weapon" data-tooltip="{{{item.system.description}}}"> | ||||
|           {{item.name}} | ||||
|         </div> | ||||
|  | ||||
|         <a class="damage rollable" data-item-id="{{item.id}}" data-action="roll" data-roll-type="damage" | ||||
|           data-roll-value="{{item.system.damage}}"> | ||||
|           {{#if item.system.isAdvantage}} | ||||
|           <i data-tooltip="Provides advantage" class="fas fa-circle-chevron-up"></i> | ||||
|           {{else}} | ||||
|           {{/if}} | ||||
|         <div class="controls"> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|         </div> | ||||
|       </div> | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.languages"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addLanguage"}}" data-tooltip-direction="UP"><i | ||||
|           class="fas fa-plus" data-action="createLanguage"></i></a>{{/if}} | ||||
|     </legend> | ||||
|     <div class="languages"> | ||||
|       {{#each languages as |item|}} | ||||
|       {{!log 'weapon' this}} | ||||
|       <div class="language item" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}" data-drag="true"> | ||||
|  | ||||
|         <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|         <div class="name" data-roll-type="weapon" data-tooltip="{{{item.system.description}}}"> | ||||
|           {{item.name}} | ||||
|         </div> | ||||
|  | ||||
|         <a class="damage rollable" data-item-id="{{item.id}}" data-action="roll" data-roll-type="damage" | ||||
|           data-roll-value="{{item.system.damage}}"> | ||||
|           {{#if item.system.isAdvantage}} | ||||
|           <i data-tooltip="Provides advantage" class="fas fa-circle-chevron-up"></i> | ||||
|           {{else}} | ||||
|           {{/if}} | ||||
|         <div class="controls"> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|         </div> | ||||
|       </div> | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
| </div> | ||||
|  | ||||
| </section> | ||||
| @@ -6,87 +6,69 @@ | ||||
|     </div> | ||||
|     <div class="intro-right"> | ||||
|       <ul> | ||||
|         {{#if (eq rollType "char")}} | ||||
|         <li><strong>{{localize "FTLNOMAD.Label.charRoll"}}</strong></li> | ||||
|         {{/if}} | ||||
|  | ||||
|         {{#if (eq rollType "skill")}} | ||||
|         <li><strong>{{localize "FTLNOMAD.Label.skillRoll"}}</strong></li> | ||||
|           <li><strong>{{localize "FTLNOMAD.Label.skillRoll"}}</strong></li> | ||||
|         {{/if}} | ||||
|  | ||||
|         {{#if isNudgedRoll}} | ||||
|         <li><strong>{{localize "FTLNOMAD.Label.nudgedRoll"}} : {{wpCost}} WP spent</strong></li> | ||||
|         {{/if}} | ||||
|  | ||||
|         {{#if weapon}} | ||||
|         <li><strong>Weapon : {{weapon.name}}</strong></li> | ||||
|         {{/if}} | ||||
|          | ||||
|         {{#if (eq rollType "resource")}} | ||||
|         <li><strong>{{rollItem.name}} : {{initialScore}}</strong></li> | ||||
|         {{#if (eq rollType "damage")}} | ||||
|           <li class="result-success">Weapon Damage</li> | ||||
|           <li><strong>{{weapon.name}}</strong></li> | ||||
|         {{else}} | ||||
|         <li><strong>{{rollItem.name}} : {{initialScore}}%</strong></li> | ||||
|         {{/if}} | ||||
|  | ||||
|         {{#if isZeroWP}} | ||||
|         <li class="red-warning">Zero WP : Automatic failure (ie 0%)</li> | ||||
|         {{else}} | ||||
|         {{#if isLowWP}} | ||||
|         <li class="orange-warning">Low WP : -20%</li> | ||||
|         {{/if}} | ||||
|         {{/if}} | ||||
|  | ||||
|         {{#if isExhausted}} | ||||
|         <li class="orange-warning">Exhausted : -20%</li> | ||||
|         {{/if}} | ||||
|  | ||||
|         {{#if  (eq rollType "resource")}} | ||||
|         <li>{{localize "FTLNOMAD.Label.multiplier"}} : {{multiplier}}</li> | ||||
|         {{else}} | ||||
|         <li>{{localize "FTLNOMAD.Label.modifier"}} : {{modifier}}%</li> | ||||
|         {{/if}} | ||||
|    | ||||
|         <li>{{localize "FTLNOMAD.Label.targetScore"}} : {{targetScore}}%</li> | ||||
|         {{#if isSuccess}} | ||||
|         {{#if isCritical}} | ||||
|         <li class="result-critical-success">{{localize "FTLNOMAD.Label.criticalSuccess"}}</li> | ||||
|         {{else}} | ||||
|         <li class="result-success"> | ||||
|           {{localize "FTLNOMAD.Label.success"}} | ||||
|           {{#if isNudge}} | ||||
|             <a class="nudge-roll"><i class="fa-solid fa-circle-sort-down"></i></a> | ||||
|           {{#if weapon}} | ||||
|             <li><strong>Weapon : {{weapon.name}}</strong></li> | ||||
|             <li><strong>Aspects : {{weapon.system.aspect}}</strong></li> | ||||
|           {{/if}} | ||||
|         </li> | ||||
|         {{/if}} | ||||
|         {{/if}} | ||||
|         {{#if isFailure}} | ||||
|         {{#if isCritical}} | ||||
|         <li class="result-critical-failure">{{localize "FTLNOMAD.Label.criticalFailure"}}</li> | ||||
|         {{else}} | ||||
|         <li class="result-failure"> | ||||
|           {{localize "FTLNOMAD.Label.failure"}} | ||||
|           {{#if isNudge}} | ||||
|             <a class="nudge-roll"><i class="fa-solid fa-circle-sort-down"></i></a> | ||||
|  | ||||
|         <li><strong>{{localize rollItem.label}} : {{fullFormula}}</strong></li> | ||||
|  | ||||
|         {{#if (eq rollType "damage")}}{{else}} | ||||
|           {{#if isEncumbered}} | ||||
|             <li class="red-warning">Encumbered : -1D</li> | ||||
|           {{/if}} | ||||
|         </li> | ||||
|           <li>{{localize "FTLNOMAD.Label.modifier"}} | ||||
|             : | ||||
|             {{skillModifier}}D | ||||
|             {{rangeModifier}}D | ||||
|             {{numericModifier}}D</li> | ||||
|         {{/if}} | ||||
|  | ||||
|         {{#if (eq rollType "damage")}}{{else}} | ||||
|           {{#if isSuccess}} | ||||
|             <li class="result-success"> | ||||
|               {{localize "FTLNOMAD.Label.success"}} | ||||
|             </li> | ||||
|           {{/if}} | ||||
|  | ||||
|           {{#if isFailure}} | ||||
|             <li class="result-failure"> | ||||
|               {{localize "FTLNOMAD.Label.failure"}} | ||||
|             </li> | ||||
|           {{/if}} | ||||
|         {{/if}} | ||||
|  | ||||
|       </ul> | ||||
|     </div> | ||||
|   </div> | ||||
|  | ||||
|   {{#if isDamage}} | ||||
|   <div> | ||||
|     {{#if (and isGM hasTarget)}} | ||||
|     {{{localize "FTLNOMAD.Roll.displayArmor" targetName=targetName targetArmor=targetArmor | ||||
|     realDamage=realDamage}}} | ||||
|     {{/if}} | ||||
|   </div> | ||||
|     <div> | ||||
|       {{#if (and isGM hasTarget)}} | ||||
|         {{{localize | ||||
|           "FTLNOMAD.Roll.displayArmor" | ||||
|           targetName=targetName | ||||
|           targetArmor=targetArmor | ||||
|           realDamage=realDamage | ||||
|         }}} | ||||
|       {{/if}} | ||||
|     </div> | ||||
|   {{/if}} | ||||
|   {{#unless isPrivate}} | ||||
|   <div class="dice-result"> | ||||
|     <h4 class="dice-total">{{total}}</h4> | ||||
|     <div class="dice-formula">{{formula}}</div> | ||||
|     {{{tooltip}}} | ||||
|   </div> | ||||
|     <div class="dice-result"> | ||||
|       <h4 class="dice-total">{{total}}</h4> | ||||
|       <div class="dice-formula">{{formula}}</div> | ||||
|       {{{tooltip}}} | ||||
|     </div> | ||||
|   {{/unless}} | ||||
| </div> | ||||
| @@ -1,12 +1,33 @@ | ||||
| <section class="tab creature-{{tab.id}} {{tab.cssClass}}" data-tab="{{tab.id}}" data-group="{{tab.group}}"> | ||||
| <section | ||||
|   class="tab creature-{{tab.id}} {{tab.cssClass}}" | ||||
|   data-tab="{{tab.id}}" | ||||
|   data-group="{{tab.group}}" | ||||
| > | ||||
|  | ||||
|   <div class="main-div"> | ||||
|  | ||||
|     <fieldset class="section-editor"> | ||||
|       <legend>{{localize "FTLNOMAD.Label.description"}}</legend> | ||||
|       {{formInput | ||||
|         systemFields.description | ||||
|         enriched=enrichedDescription | ||||
|         value=system.description | ||||
|         name="system.description" | ||||
|         toggled=true | ||||
|       }} | ||||
|     </fieldset> | ||||
|  | ||||
|     <fieldset class="section-editor"> | ||||
|       <legend>{{localize "FTLNOMAD.Label.notes"}}</legend> | ||||
|       {{formInput | ||||
|         systemFields.notes | ||||
|         enriched=enrichedNotes | ||||
|         value=system.notes | ||||
|         name="system.notes" | ||||
|         toggled=true | ||||
|       }} | ||||
|     </fieldset> | ||||
|  | ||||
|   </div> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.description"}}</legend> | ||||
|     {{formInput systemFields.description enriched=enrichedDescription value=system.description name="system.description" toggled=true}} | ||||
|   </fieldset> | ||||
|    | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.notes"}}</legend> | ||||
|     {{formInput systemFields.notes enriched=enrichedNotes value=system.notes name="system.notes" toggled=true}} | ||||
|   </fieldset> | ||||
| </section> | ||||
| @@ -1,4 +1,5 @@ | ||||
| <section class="creature-main creature-main-{{ifThen isPlayMode 'play' 'edit'}}"> | ||||
|   {{!log "creature-main" this}} | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.creature"}}</legend> | ||||
|     <div class="creature-pc creature-pc-{{ifThen isPlayMode 'play' 'edit'}}"> | ||||
| @@ -8,88 +9,98 @@ | ||||
|             data-tooltip="{{actor.name}}" /> | ||||
|         </div> | ||||
|  | ||||
|         <fieldset class="creature-hp"> | ||||
|           <legend>{{localize "FTLNOMAD.Label.Stamina"}}</legend> | ||||
|           <div class="flexrow"> | ||||
|             {{formField systemFields.health.fields.staminaValue value=system.health.staminaValue}} | ||||
|           </div> | ||||
|           <div class="flexrow"> | ||||
|             {{formField systemFields.health.fields.staminaMax value=system.health.staminaMax rootId=partId}} | ||||
|           </div> | ||||
|         </fieldset> | ||||
|  | ||||
|       </div> | ||||
|       <div class="creature-right"> | ||||
|         <div class="creature-name"> | ||||
|           {{formInput fields.name value=source.name rootId=partId disabled=isPlayMode}} | ||||
|           <a class="control" data-action="toggleSheet" data-tooltip="FTLNOMAD.ToggleSheet" | ||||
|             data-tooltip-direction="UP"> | ||||
|           <a class="control" data-action="toggleSheet" data-tooltip="FTLNOMAD.ToggleSheet" data-tooltip-direction="UP"> | ||||
|             <i class="fa-solid fa-user-{{ifThen isPlayMode 'lock' 'pen'}}"></i> | ||||
|           </a> | ||||
|         </div> | ||||
|         <fieldset class="creature-hp">  | ||||
|           <legend>{{localize "FTLNOMAD.Label.HP"}}</legend> | ||||
|           <div class="flexrow"> | ||||
|             {{formField systemFields.hp.fields.value value=system.hp.value}} | ||||
|             <span class="hp-separator">/</span> | ||||
|             {{formField systemFields.hp.fields.max value=system.hp.max }} | ||||
|           </div> | ||||
|           <div class="flexrow "> | ||||
|             {{formField systemFields.damageBonus value=system.damageBonus classes="damage-bonus"}} | ||||
|           </div> | ||||
|         </fieldset> | ||||
|  | ||||
|         <fieldset class="willpower"> | ||||
|           <legend>{{localize "FTLNOMAD.Label.willpower"}}</legend> | ||||
|           <div class="flexrow"> | ||||
|             <label class="label-field">{{localize "FTLNOMAD.Label.current"}}</label> | ||||
|             {{formInput systemFields.wp.fields.value value=system.wp.value}} | ||||
|             <label class="label-field">{{localize "FTLNOMAD.Label.max"}}</label> | ||||
|             {{formInput systemFields.wp.fields.max value=system.wp.max rootId=partId }} | ||||
|             <label class="label-field">{{localize "FTLNOMAD.Label.exhausted"}}</label> | ||||
|             {{formInput systemFields.wp.fields.exhausted value=system.wp.exhausted classes="checkbox"}} | ||||
|         <fieldset class="creature-spec"> | ||||
|           <legend>{{localize "FTLNOMAD.Label.Details"}}</legend> | ||||
|           {{formField systemFields.terrain value=system.terrain rootId=partId disabled=isPlayMode localize=true}} | ||||
|           {{formField systemFields.niche value=system.niche rootId=partId disabled=isPlayMode localize=true}} | ||||
|           {{formField systemFields.size value=system.size rootId=partId disabled=isPlayMode localize=true}} | ||||
|           <div class="flexrow rollable" data-roll-type="creature-number"> | ||||
|             <i class="fa-regular fa-dice dice-2d6"></i> | ||||
|             {{formField systemFields.numberAppearing value=system.numberAppearing rootId=partId disabled=isPlayMode | ||||
|             localize=true}} | ||||
|           </div> | ||||
|           <div class="flexrow rollable" data-roll-type="creature-damage"> | ||||
|             <i class="fa-regular fa-dice dice-2d6"></i> | ||||
|             {{formField systemFields.damage value=system.damage rootId=partId disabled=isPlayMode localize=true}} | ||||
|           </div> | ||||
|         </fieldset> | ||||
|  | ||||
|       </div> | ||||
|  | ||||
|        | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset class="creature-characteristics creature-characteristics-{{ifThen isPlayMode 'play' 'edit'}}"> | ||||
|     <legend>{{localize "FTLNOMAD.Label.characteristics"}}</legend> | ||||
|     <div class="creature-characteristic" > | ||||
|       <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|       <label class="rollable" data-roll-type="char" data-char-id="str" data-tooltip="{{system.characteristics.str.feature}}">{{localize | ||||
|         "FTLNOMAD.Label.strShort"}}</label> | ||||
|       {{formField systemFields.characteristics.fields.str.fields.value value=system.characteristics.str.value | ||||
|       rootId=partId disabled=isPlayMode }} | ||||
|       <label class="char-text">{{mul system.characteristics.str.value 5}}</label> | ||||
|   <fieldset class="creature-skills creature-skills-{{ifThen isPlayMode 'play' 'edit'}}"> | ||||
|     <legend>{{localize "FTLNOMAD.Label.skills"}}</legend> | ||||
|     <div class="creature-skill"> | ||||
|       <img src="systems/fvtt-ftl-nomad/assets/icons/icon_combat.svg" class="icon-skill" /> | ||||
|       <label class="rollable" data-roll-type="skill" data-skill-id="combat">{{localize | ||||
|         "FTLNOMAD.Label.combat"}}</label> | ||||
|       {{formInput systemFields.skills.fields.combat.fields.value value=system.skills.combat.value rootId=partId | ||||
|       disabled=isPlayMode type="number" }} | ||||
|     </div> | ||||
|     <div class="creature-characteristic"> | ||||
|       <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|       <label class="rollable" data-roll-type="char" data-char-id="dex" data-tooltip="{{system.characteristics.dex.feature}}">{{localize | ||||
|         "FTLNOMAD.Label.dexShort"}}</label> | ||||
|       {{formField systemFields.characteristics.fields.dex.fields.value value=system.characteristics.dex.value | ||||
|       rootId=partId disabled=isPlayMode }} | ||||
|       <label class="char-text">{{mul system.characteristics.dex.value 5}}</label> | ||||
|     <div class="creature-skill"> | ||||
|       <img src="systems/fvtt-ftl-nomad/assets/icons/icon_knowledge.svg" class="icon-skill" /> | ||||
|       <label class="rollable" data-roll-type="skill" data-skill-id="knowledge">{{localize | ||||
|         "FTLNOMAD.Label.knowledge"}}</label> | ||||
|       {{formInput systemFields.skills.fields.knowledge.fields.value value=system.skills.knowledge.value | ||||
|       rootId=partId disabled=isPlayMode type="number"}} | ||||
|     </div> | ||||
|     <div class="creature-characteristic" > | ||||
|       <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|       <label class="rollable" data-roll-type="char" data-char-id="con" data-tooltip="{{system.characteristics.con.feature}}">{{localize | ||||
|         "FTLNOMAD.Label.conShort"}}</label> | ||||
|       {{formField systemFields.characteristics.fields.con.fields.value value=system.characteristics.con.value | ||||
|       rootId=partId disabled=isPlayMode }} | ||||
|        <label class="char-text">{{mul system.characteristics.con.value 5}}</label> | ||||
|     <div class="creature-skill"> | ||||
|       <img src="systems/fvtt-ftl-nomad/assets/icons/icon_social.svg" class="icon-skill" /> | ||||
|       <label class="rollable" data-roll-type="skill" data-skill-id="social">{{localize | ||||
|         "FTLNOMAD.Label.social"}}</label> | ||||
|       {{formInput systemFields.skills.fields.social.fields.value value=system.skills.social.value | ||||
|       rootId=partId disabled=isPlayMode type="number"}} | ||||
|     </div> | ||||
|     <div class="creature-characteristic" > | ||||
|       <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|       <label class="rollable" data-roll-type="char" data-char-id="int" data-tooltip="{{system.characteristics.int.feature}}">{{localize | ||||
|         "FTLNOMAD.Label.intShort"}}</label> | ||||
|       {{formField systemFields.characteristics.fields.int.fields.value value=system.characteristics.int.value | ||||
|       rootId=partId disabled=isPlayMode }} | ||||
|        <label class="char-text">{{mul system.characteristics.int.value 5}}</label> | ||||
|     <div class="creature-skill"> | ||||
|       <img src="systems/fvtt-ftl-nomad/assets/icons/icon_physical.svg" class="icon-skill" /> | ||||
|       <label class="rollable" data-roll-type="skill" data-skill-id="physical">{{localize | ||||
|         "FTLNOMAD.Label.physical"}}</label> | ||||
|       {{formInput systemFields.skills.fields.physical.fields.value value=system.skills.physical.value | ||||
|       rootId=partId disabled=isPlayMode type="number"}} | ||||
|     </div> | ||||
|     <div class="creature-characteristic" > | ||||
|       <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|       <label class="rollable" data-roll-type="char" data-char-id="pow" data-tooltip="{{system.characteristics.pow.feature}}">{{localize | ||||
|         "FTLNOMAD.Label.powShort"}}</label> | ||||
|       {{formField systemFields.characteristics.fields.pow.fields.value value=system.characteristics.pow.value | ||||
|       rootId=partId disabled=isPlayMode }} | ||||
|        <label class="char-text">{{mul system.characteristics.pow.value 5}}</label> | ||||
|     <div class="creature-skill"> | ||||
|       <img src="systems/fvtt-ftl-nomad/assets/icons/icon_stealth.svg" class="icon-skill" /> | ||||
|       <label class="rollable" data-roll-type="skill" data-skill-id="stealth">{{localize | ||||
|         "FTLNOMAD.Label.stealth"}}</label> | ||||
|       {{formInput systemFields.skills.fields.stealth.fields.value value=system.skills.stealth.value | ||||
|       rootId=partId disabled=isPlayMode type="number"}} | ||||
|     </div> | ||||
|     <div class="creature-skill"> | ||||
|       <img src="systems/fvtt-ftl-nomad/assets/icons/icon_vehicles.svg" class="icon-skill" /> | ||||
|       <label class="rollable" data-roll-type="skill" data-skill-id="vehicles">{{localize | ||||
|         "FTLNOMAD.Label.vehicles"}}</label> | ||||
|       {{formInput systemFields.skills.fields.vehicles.fields.value value=system.skills.vehicles.value | ||||
|       rootId=partId disabled=isPlayMode type="number"}} | ||||
|     </div> | ||||
|     <div class="creature-skill"> | ||||
|       <img src="systems/fvtt-ftl-nomad/assets/icons/icon_technology.svg" class="icon-skill" /> | ||||
|       <label class="rollable" data-roll-type="skill" data-skill-id="technology">{{localize | ||||
|         "FTLNOMAD.Label.technology"}}</label> | ||||
|       {{formInput systemFields.skills.fields.technology.fields.value value=system.skills.technology.value | ||||
|       rootId=partId disabled=isPlayMode type="number"}} | ||||
|     </div> | ||||
|  | ||||
|   </fieldset> | ||||
|  | ||||
|  | ||||
|  | ||||
| </section> | ||||
							
								
								
									
										50
									
								
								templates/creature-sheet-trait.hbs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,50 @@ | ||||
| <section class="tab creature-{{tab.id}} {{tab.cssClass}}" data-tab="{{tab.id}}" data-group="{{tab.group}}"> | ||||
| <div class="main-div"> | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.traits"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addTrait"}}" data-tooltip-direction="UP"><i | ||||
|           class="fas fa-plus" data-action="createTrait"></i></a>{{/if}} | ||||
|     </legend> | ||||
|     <div class="traits"> | ||||
|       {{#each traits as |item|}} | ||||
|       <div class="trait item" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}" data-drag="true"> | ||||
|         <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|         <div class="name" data-tooltip="{{{item.system.description}}}"> | ||||
|           {{item.name}} | ||||
|         </div> | ||||
|         <div class="controls"> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|         </div> | ||||
|       </div> | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.abilities"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize "FTLNOMAD.Tooltip.addAbility"}}" data-tooltip-direction="UP"><i | ||||
|           class="fas fa-plus" data-action="createAbility"></i></a>{{/if}} | ||||
|     </legend> | ||||
|     <div class="abilities"> | ||||
|       {{#each abilities as |item|}} | ||||
|       {{!log 'armor' this}} | ||||
|       <div class="ability" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"> | ||||
|         <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|         <div class="name" data-tooltip="{{{item.system.description}}}"> | ||||
|           {{item.name}} | ||||
|         </div> | ||||
|         <div class="controls"> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|         </div> | ||||
|       </div> | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
| </div> | ||||
| </section> | ||||
| @@ -1,57 +0,0 @@ | ||||
| <section class="tab creature-{{tab.id}} {{tab.cssClass}}" data-tab="{{tab.id}}" data-group="{{tab.group}}"> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.weapons"}}{{#if isEditMode}} | ||||
|       <a class="action" data-tooltip="{{localize " FTLNOMAD.Tooltip.addWeapon"}}" data-tooltip-direction="UP"><i | ||||
|           class="fas fa-plus" data-action="createWeapon"></i></a>{{/if}} | ||||
|     </legend> | ||||
|     <div class="weapons"> | ||||
|       {{#each weapons as |item|}} | ||||
|       {{!log 'weapon' this}} | ||||
|       <div class="weapon item" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}" data-drag="true"> | ||||
|  | ||||
|         <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|         <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|         <div class="name rollable" data-roll-type="weapon" data-tooltip="{{{item.system.description}}}"> | ||||
|           {{item.name}} | ||||
|         </div> | ||||
|  | ||||
|         <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|         <a class="damage rollable" data-item-id="{{item.id}}" data-action="roll" data-roll-type="damage" | ||||
|           data-roll-value="{{item.system.damage}}"> | ||||
|           {{localize "FTLNOMAD.Label.damageShort"}} : | ||||
|           {{item.system.damage}}</a> | ||||
|         <div class="controls"> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|           <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" | ||||
|             data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|         </div> | ||||
|       </div> | ||||
|       {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend data-tooltip="{{localize "FTLNOMAD.Tooltip.skills"}}" data-tooltip-direction="UP">{{localize "FTLNOMAD.Label.skills"}}</legend> | ||||
|     <div class="skills"> | ||||
|         {{#each skills as |item|}} | ||||
|           <div class="skill item" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"> | ||||
|             <img class="item-img" src="{{item.img}}" data-tooltip="{{item.name}}" /> | ||||
|             <img src="systems/fvtt-cthulhu-eternal/assets/ui/d100.svg" class="d100" /> | ||||
|             <div class="name rollable" data-roll-type="skill" data-tooltip="{{{item.description}}}" data-tooltip-direction="UP">{{item.name}} | ||||
|             </div> | ||||
|             <div class="score" > | ||||
|               {{item.system.skillTotal}} | ||||
|             </div> | ||||
|             <div class="controls"> | ||||
|               <a data-tooltip="{{localize 'FTLNOMAD.Edit'}}" data-action="edit" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"><i class="fas fa-edit"></i></a> | ||||
|               <a data-tooltip="{{localize 'FTLNOMAD.Delete'}}" data-action="delete" data-item-id="{{item.id}}" data-item-uuid="{{item.uuid}}"><i class="fas fa-trash"></i></a> | ||||
|             </div> | ||||
|           </div> | ||||
|         {{/each}} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|  | ||||
| </section> | ||||
| @@ -10,12 +10,8 @@ | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.description"}}</legend> | ||||
|     {{formInput | ||||
|       systemFields.description | ||||
|       enriched=description | ||||
|       value=system.description | ||||
|       name="system.description" | ||||
|       toggled="false" | ||||
|     }} | ||||
|     {{formInput systemFields.description enriched=enrichedDescription value=system.description name="system.description" | ||||
|     toggled=true}} | ||||
|   </fieldset> | ||||
|  | ||||
| </section> | ||||
| @@ -12,7 +12,7 @@ | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.description"}}</legend> | ||||
|     {{formInput systemFields.description enriched=description value=system.description name="system.description" | ||||
|     {{formInput systemFields.description enriched=enrichedDescription value=system.description name="system.description" | ||||
|     toggled=true}} | ||||
|   </fieldset> | ||||
|  | ||||
|   | ||||
| @@ -2,17 +2,17 @@ | ||||
|   <div class="header"> | ||||
|     <img class="item-img" src="{{item.img}}" data-edit="img" data-action="editImage" data-tooltip="{{item.name}}" /> | ||||
|     {{formInput fields.name value=source.name}} | ||||
|   </div>   | ||||
|   </div> | ||||
|  | ||||
|   <fieldset> | ||||
|     {{formField systemFields.techAge value=system.techAge localize=true}} | ||||
|     {{formField systemFields.enc value=system.enc}} | ||||
|     {{formField systemFields.loss value=system.loss}} | ||||
|     {{formField systemFields.cost value=system.cost}} | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.description"}}</legend> | ||||
|     {{formInput systemFields.description enriched=description value=system.description name="system.description" | ||||
|     {{formInput systemFields.description enriched=enrichedDescription value=system.description name="system.description" | ||||
|     toggled=true}} | ||||
|   </fieldset> | ||||
|  | ||||
|   | ||||
| @@ -2,11 +2,11 @@ | ||||
|   <div class="header"> | ||||
|     <img class="item-img era-icon-color" src="{{item.img}}" data-edit="img" data-action="editImage" data-tooltip="{{item.name}}" /> | ||||
|     {{formInput fields.name value=source.name}} | ||||
|   </div>   | ||||
|   </div> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.description"}}</legend> | ||||
|     {{formInput systemFields.description enriched=description value=system.description name="system.description" toggled=true}} | ||||
|     {{formInput systemFields.description enriched=enrichedDescription value=system.description name="system.description" toggled=true}} | ||||
|   </fieldset> | ||||
|  | ||||
| </section> | ||||
							
								
								
									
										53
									
								
								templates/robot-description.hbs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,53 @@ | ||||
| <section | ||||
|   class="tab robot-{{tab.id}} {{tab.cssClass}}" | ||||
|   data-tab="{{tab.id}}" | ||||
|   data-group="{{tab.group}}" | ||||
| > | ||||
|   <fieldset> | ||||
|     <legend>Tech Age</legend> | ||||
|     <div class="flexrow"> | ||||
|       {{formField | ||||
|         systemFields.techAge | ||||
|         value=system.techAge | ||||
|         localize=true | ||||
|         disabled=isPlayMode | ||||
|       }} | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset> | ||||
|     <legend>{{localize "FTLNOMAD.Label.damages"}}</legend> | ||||
|     <div class="text-center"> | ||||
|       <textarea | ||||
|         class="form-control" | ||||
|         cols="64" | ||||
|         rows="5" | ||||
|         name="system.damages" | ||||
|         data-tooltip="{{localize 'FTLNOMAD.Tooltip.damages'}}" | ||||
|       >{{system.damages}}</textarea> | ||||
|     </div> | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset class="section-editor"> | ||||
|     <legend>{{localize "FTLNOMAD.Label.description"}}</legend> | ||||
|     {{formInput | ||||
|       systemFields.description | ||||
|       enriched=enrichedDescription | ||||
|       value=system.description | ||||
|       name="system.description" | ||||
|       toggled=true | ||||
|     }} | ||||
|   </fieldset> | ||||
|  | ||||
|   <fieldset class="section-editor"> | ||||
|     <legend>{{localize "FTLNOMAD.Label.notes"}}</legend> | ||||
|     {{formInput | ||||
|       systemFields.notes | ||||
|       enriched=enrichedNotes | ||||
|       value=system.notes | ||||
|       name="system.notes" | ||||
|       toggled=true | ||||
|     }} | ||||
|   </fieldset> | ||||
|  | ||||
| </section> | ||||