mirror of
				https://github.com/SquidDev-CC/CC-Tweaked
				synced 2025-10-23 09:57:39 +00:00 
			
		
		
		
	Compare commits
	
		
			447 Commits
		
	
	
		
			v1.21.1-1.
			...
			v1.97.2
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | ed9e823f3f | ||
|   | 2b8fea5b35 | ||
|   | b3e6b4c29a | ||
|   | c94ff68e7f | ||
|   | 3fb96a3438 | ||
|   | 7a2b7c0701 | ||
|   | fa7a731ad1 | ||
|   | 8be48ebcf0 | ||
|   | 123f3abcaa | ||
|   | 9f2e43f00a | ||
|   | 9fb35b2799 | ||
|   | 272572dbf2 | ||
|   | 702e4fc787 | ||
|   | 511327effe | ||
|   | 940e35caea | ||
|   | fca01693dd | ||
|   | 8846a3f36f | ||
|   | 8002c9e932 | ||
|   | 6fd8331e94 | ||
|   | 7a667b9028 | ||
|   | 67b7cd7a14 | ||
|   | 960d79803d | ||
|   | 43ef36fa15 | ||
|   | f037d9539e | ||
|   | a3b2f7b5bd | ||
|   | 0859145d18 | ||
|   | f6e4122f7f | ||
|   | 8e4023fdbc | ||
|   | f10818ec6b | ||
|   | bc2358489b | ||
|   | 6e5e682e3a | ||
|   | 02c63ce7e8 | ||
|   | fd24d89f89 | ||
|   | 685d33a2dd | ||
|   | ca40c9cf19 | ||
|   | 8985ea9560 | ||
|   | 1832028dfd | ||
|   | 316889dab7 | ||
|   | 96c80a6549 | ||
|   | fb274e2ed6 | ||
|   | ebe847c04b | ||
|   | fdbf41b3c4 | ||
|   | 9dcb9f9537 | ||
|   | 1395c877af | ||
|   | 969451f18c | ||
|   | 259c8d57f5 | ||
|   | 2ee07283fa | ||
|   | c85a80617a | ||
|   | 5aa8611e43 | ||
|   | 86aecd945e | ||
|   | a7a9f4fa13 | ||
|   | 22e4c0603b | ||
|   | f3b731460e | ||
|   | d787c6e6c1 | ||
|   | 11272b8d00 | ||
|   | 1c254e3bb0 | ||
|   | 7ade380fad | ||
|   | 2a377a37ee | ||
|   | 861a9e199d | ||
|   | d4f1e34023 | ||
|   | 6d25278a5c | ||
|   | abfb05f04c | ||
|   | 9d8e76c5b4 | ||
|   | 89662bf54a | ||
|   | 520635dfec | ||
|   | ee82a8d75f | ||
|   | 967f00cd1b | ||
|   | dda5236d84 | ||
|   | 427ca2f108 | ||
|   | e30f2a86f0 | ||
|   | 121ef6e976 | ||
|   | e4271ff6f7 | ||
|   | 8d3b94734d | ||
|   | 9e57150384 | ||
|   | 0dc34a7dbe | ||
|   | fbe28dc51a | ||
|   | 14e013d075 | ||
|   | 9bc4e9530c | ||
|   | cbff505297 | ||
|   | c4aef2d4c7 | ||
|   | 2ed6692f76 | ||
|   | f075eabc32 | ||
|   | 3d1c9d1667 | ||
|   | 0028ad8f54 | ||
|   | 4c805f1c27 | ||
|   | 505543ad94 | ||
|   | 644471c6bf | ||
|   | bf87d7faa1 | ||
|   | be95b65488 | ||
|   | 7e9f3bd61a | ||
|   | 4a74aa8166 | ||
|   | b4ce4361ce | ||
|   | 42a3dd24ad | ||
|   | 9129da2e3d | ||
|   | 668cdcdd39 | ||
|   | 2543ebee10 | ||
|   | 104a317163 | ||
|   | 921802e6c9 | ||
|   | 56cfc1a452 | ||
|   | a14b98be22 | ||
|   | 76b5c05acc | ||
|   | 8b3f1a448f | ||
|   | 3de515c617 | ||
|   | ab199cd9b0 | ||
|   | fc4bc9aa05 | ||
|   | 9662a106f0 | ||
|   | 06b0538b76 | ||
|   | fb128152a5 | ||
|   | f596af059d | ||
|   | 52bb06d250 | ||
|   | f3d22444d3 | ||
|   | 84a1f63445 | ||
|   | dbe98ac270 | ||
|   | a00775dcbb | ||
|   | 2279f5044d | ||
|   | e8d90b94ec | ||
|   | 0bf7f3de89 | ||
|   | 59266fe6e9 | ||
|   | a198a5241d | ||
|   | b429095f88 | ||
|   | a6c7236fa0 | ||
|   | 69deb0c947 | ||
|   | b0d7f690eb | ||
|   | 62f93f1b4b | ||
|   | 7fe322a93a | ||
|   | de97fa35a8 | ||
|   | 25ce9b9bea | ||
|   | 59048b3f30 | ||
|   | a960db7ced | ||
|   | 6a4f09e7b1 | ||
|   | 19d1cc0320 | ||
|   | 1a52275dcb | ||
|   | 2fc2cf2e29 | ||
|   | 80c2fc68aa | ||
|   | 7c0664b9f2 | ||
|   | 4d1a950fbf | ||
|   | 135a3f56a5 | ||
|   | 1f117b7c47 | ||
|   | c820e051b3 | ||
|   | 01d7aaf062 | ||
|   | 1e75e4ff6e | ||
|   | f967a70121 | ||
|   | ac5150a664 | ||
|   | 0318e62524 | ||
|   | ca69d755e6 | ||
|   | 3ba7acc414 | ||
|   | 4c27eea73b | ||
|   | 225a64dd6b | ||
|   | 1831e81dd4 | ||
|   | a4dd6c24e5 | ||
|   | 936742895b | ||
|   | b5d1e618b9 | ||
|   | 10aa6c5297 | ||
|   | 010ebacd1a | ||
|   | 9e002beed7 | ||
|   | 6029defb20 | ||
|   | c16aa5f247 | ||
|   | 8d27bdca7b | ||
|   | 41aa8fa163 | ||
|   | e40fb67b50 | ||
|   | 96d46ffd2c | ||
|   | 652b7ebba6 | ||
|   | f0d7a1165d | ||
|   | 4a20eea852 | ||
|   | ccdd2bf477 | ||
|   | 8d8f41a50b | ||
|   | 72c78db28b | ||
|   | bf8a992273 | ||
|   | 9c214fcd8a | ||
|   | 082cc9d2d8 | ||
|   | bff2e81713 | ||
|   | 63aa3d8877 | ||
|   | 74c5d7e719 | ||
|   | 78a685c2cb | ||
|   | a475f55b9c | ||
|   | 9353ded2d5 | ||
|   | d197268e53 | ||
|   | 9648265056 | ||
|   | 264dbc570f | ||
|   | e4ad8bc117 | ||
|   | 1b31972092 | ||
|   | 6cd9fa9fb2 | ||
|   | d10f297ca0 | ||
|   | 9ae5636d41 | ||
|   | a60daf78cf | ||
|   | 76e0507c3b | ||
|   | d8e2fab2b1 | ||
|   | 5c5e510075 | ||
|   | 5a309258af | ||
|   | b0782ec38b | ||
|   | 46846a4fde | ||
|   | d955443b21 | ||
|   | 57e6c49844 | ||
|   | 66dbab7a6b | ||
|   | 558976e4ca | ||
|   | 3a7470a108 | ||
|   | cf2b332c3c | ||
|   | 8984ebcf80 | ||
|   | a28e7e2db3 | ||
|   | 3860e2466c | ||
|   | 5b31c2536a | ||
|   | 98dc7a6e58 | ||
|   | 5e0ceda7ce | ||
|   | 7fe3ac9222 | ||
|   | a65b8ed04c | ||
|   | 3e14b84c23 | ||
|   | 1710ad9861 | ||
|   | 9c48c99be7 | ||
|   | d10d1b45fe | ||
|   | 70a1cf5c5a | ||
|   | b546a10bd6 | ||
|   | 2f35bbb538 | ||
|   | 891dde43a9 | ||
|   | 094e0d4f33 | ||
|   | df40adce20 | ||
|   | e12ce95b2d | ||
|   | bcc0effd00 | ||
|   | 64f3aa2dba | ||
|   | 515ccfebd3 | ||
|   | 0b6dbe7778 | ||
|   | 19054684c6 | ||
|   | a72a5e6deb | ||
|   | eb2d9482a2 | ||
|   | 95794fdaf3 | ||
|   | af477a401a | ||
|   | 27531c161d | ||
|   | a8f208c394 | ||
|   | 148d9e9dda | ||
|   | 67fc21271b | ||
|   | ce28e2659a | ||
|   | 47e4826f2b | ||
|   | 9bde2d8d13 | ||
|   | 46acb6866e | ||
|   | c0506108f9 | ||
|   | 0dd7c3695f | ||
|   | 2a290be248 | ||
|   | c4a6c16330 | ||
|   | f8b3b544d1 | ||
|   | 53ae689468 | ||
|   | 73e460e533 | ||
|   | a4e68d637a | ||
|   | 1f1b20c81e | ||
|   | 20a9ff7a3f | ||
|   | 58cd5e3df7 | ||
|   | 5922d7548c | ||
|   | 386e01364e | ||
|   | 70eff6aa63 | ||
|   | 45ac601946 | ||
|   | d6c5a5bd27 | ||
|   | 02c7903cb7 | ||
|   | 37e856efdc | ||
|   | 8a38097887 | ||
|   | 884b417ec2 | ||
|   | 1d1d6227eb | ||
|   | d5eb82db60 | ||
|   | 3575654d02 | ||
|   | 79fcc7241b | ||
|   | b203d3aa0a | ||
|   | f1176af9d1 | ||
|   | 717686d855 | ||
|   | 92811ca0bc | ||
|   | f8a58dbcb1 | ||
|   | 98aabe2cfb | ||
|   | f938ed9779 | ||
|   | 03c9274b27 | ||
|   | 92c94ac039 | ||
|   | 5c6fd80b0c | ||
|   | f1ec59df15 | ||
|   | 624b23c7ac | ||
|   | 9b77a4aaee | ||
|   | d8b0037cfa | ||
|   | 91a9e7fdf7 | ||
|   | 2c3573719b | ||
|   | 46bc42d5a7 | ||
|   | 1b9e93a80f | ||
|   | 7b93f91ffa | ||
|   | 31a1027401 | ||
|   | e0fc994819 | ||
|   | 43408bf085 | ||
|   | d28f42e8b7 | ||
|   | ac452582c1 | ||
|   | 7e65c6b25c | ||
|   | 457a863842 | ||
|   | eef36e1358 | ||
|   | feda08b915 | ||
|   | 0240ce50ce | ||
|   | 592b83e784 | ||
|   | 5d91491ec7 | ||
|   | 7326d1110d | ||
|   | 0aa6ac05a0 | ||
|   | 27a2c063b9 | ||
|   | 89a195ec06 | ||
|   | 0e5fd4e8e0 | ||
|   | aa4ec53bb6 | ||
|   | 6b49327462 | ||
|   | 74ad934889 | ||
|   | 08b3dbbad5 | ||
|   | 54eac0a2f8 | ||
|   | 51ca5e620c | ||
|   | 076d572831 | ||
|   | 75f2b91fef | ||
|   | 79534e5630 | ||
|   | d483a91459 | ||
|   | 71d764f122 | ||
|   | 2e527eb11e | ||
|   | 19273b3696 | ||
|   | 4643641d51 | ||
|   | 63cd9c5bc7 | ||
|   | e9c11ff325 | ||
|   | 452464aa01 | ||
|   | 8885462175 | ||
|   | b8bd64913b | ||
|   | 00b458c39a | ||
|   | 88722d484f | ||
|   | 6d103e2114 | ||
|   | 42f23d56ae | ||
|   | 89d5211bd7 | ||
|   | 83e70377f7 | ||
|   | f6a26f75c3 | ||
|   | 664df62d5d | ||
|   | 1348ee0588 | ||
|   | 145dce7653 | ||
|   | 7f2651c23e | ||
|   | 05464107a8 | ||
|   | 86705787f0 | ||
|   | b34d8387d9 | ||
|   | 4d00969ef0 | ||
|   | 01d3d12992 | ||
|   | 5e31dcde83 | ||
|   | 5184883af1 | ||
|   | 0c45112262 | ||
|   | 0bf1672f45 | ||
|   | e1b8ac1f84 | ||
|   | deea552d99 | ||
|   | 54229c2ce1 | ||
|   | 1346a26179 | ||
|   | 14df44f09d | ||
|   | 1a21529499 | ||
|   | 2546990f41 | ||
|   | b39ca02464 | ||
|   | 628618105c | ||
|   | 075ba03f5d | ||
|   | 10bf84b631 | ||
|   | c346e22a45 | ||
|   | 418e1335b1 | ||
|   | d52372df31 | ||
|   | cc72e1c2bd | ||
|   | d28afcc6a9 | ||
|   | 220ed21e6e | ||
|   | 09d465774d | ||
|   | dfc8f48f12 | ||
|   | abe2ec4686 | ||
|   | 5333cda44e | ||
|   | dc88fbeb12 | ||
|   | c5eb7a9501 | ||
|   | bc969db2be | ||
|   | 50b7646178 | ||
|   | 3fa6b5bc9d | ||
|   | 33e65e39e3 | ||
|   | 0380e60590 | ||
|   | 2918892ee2 | ||
|   | e43dd9f7c6 | ||
|   | 244fd95034 | ||
|   | 14e98e2fcb | ||
|   | dcf5d59109 | ||
|   | b95083c77e | ||
|   | c54c8c3ea6 | ||
|   | f79c67e243 | ||
|   | 6ec7ebe439 | ||
|   | 76fe33760d | ||
|   | cb549d8f43 | ||
|   | 7b400fdcdd | ||
|   | ab70d918b5 | ||
|   | c64644b9ec | ||
|   | afb12eb342 | ||
|   | 64f5ca02b3 | ||
|   | 84bca21b0c | ||
|   | b6757c416f | ||
|   | dc9d3f2d15 | ||
|   | 807825d74e | ||
|   | 14c17676c6 | ||
|   | 324519575c | ||
|   | fbcf26bdc9 | ||
|   | 30ab6bd045 | ||
|   | 2ea816b78b | ||
|   | 56dcc57755 | ||
|   | a4830aff86 | ||
|   | 621bc526be | ||
|   | 605e1f6b9b | ||
|   | cb66ef7e30 | ||
|   | fd2f6a38c1 | ||
|   | 229821d398 | ||
|   | 1f2e0c444d | ||
|   | 429baa350c | ||
|   | 7fe62485fa | ||
|   | fe10c68099 | ||
|   | 06092cfddd | ||
|   | 9967dc5740 | ||
|   | b920e04c59 | ||
|   | baa1b5a5c9 | ||
|   | e84663a5c5 | ||
|   | 1c46220c42 | ||
|   | 889b445855 | ||
|   | 19553a981e | ||
|   | 15f4dbd061 | ||
|   | 98e12c7c3e | ||
|   | dfad319864 | ||
|   | 111b58f533 | ||
|   | 9345652808 | ||
|   | d40163b409 | ||
|   | 96400966d7 | ||
|   | db83bd4f64 | ||
|   | 579a38d366 | ||
|   | 0e6797c7da | ||
|   | c935577768 | ||
|   | c2b4077aa1 | ||
|   | ac020859f3 | ||
|   | 238be8955b | ||
|   | f997b02b8f | ||
|   | 799bb77847 | ||
|   | 2f66792a0d | ||
|   | 6cae8e211e | ||
|   | 48b1b2f01d | ||
|   | 8c45fd362a | ||
|   | 07b13dd2b4 | ||
|   | 45e84e1ede | ||
|   | 725dfa764f | ||
|   | c221502ec9 | ||
|   | 234f18c228 | ||
|   | 006ad109cb | ||
|   | 0db080154c | ||
|   | 200311033f | ||
|   | 3192dc81ac | ||
|   | b11d4bb209 | ||
|   | 2a716244e9 | ||
|   | 19b7ed538a | ||
|   | b0d9dc0b88 | ||
|   | e6094a59fa | ||
|   | e8d7e6a562 | ||
|   | 536c2d9b2d | ||
|   | f15a278f3b | ||
|   | 26b73c2ff3 | ||
|   | c1e08fc3c7 | ||
|   | b9ec6f236d | ||
|   | b1fff97bff | ||
|   | c81bc70475 | ||
|   | 55a7ee4acf | 
| @@ -1,7 +1,3 @@ | ||||
| # SPDX-FileCopyrightText: 2017 The CC: Tweaked Developers | ||||
| # | ||||
| # SPDX-License-Identifier: CC0-1.0 | ||||
|  | ||||
| root = true | ||||
|  | ||||
| [*] | ||||
| @@ -12,30 +8,5 @@ charset = utf-8 | ||||
| trim_trailing_whitespace = true | ||||
| insert_final_newline = true | ||||
|  | ||||
| ij_continuation_indent_size = 4 | ||||
| ij_any_do_while_brace_force = if_multiline | ||||
| ij_any_if_brace_force = if_multiline | ||||
| ij_any_for_brace_force = if_multiline | ||||
| ij_any_spaces_within_array_initializer_braces = true | ||||
|  | ||||
| [*.md] | ||||
| trim_trailing_whitespace = false | ||||
|  | ||||
| [*.sexp] | ||||
| indent_size = 2 | ||||
|  | ||||
| [*.yml] | ||||
| indent_size = 2 | ||||
|  | ||||
| [{*.kt,*.kts}] | ||||
| ij_kotlin_code_style_defaults = KOTLIN_OFFICIAL | ||||
| ij_kotlin_continuation_indent_size = 4 | ||||
| ij_kotlin_spaces_around_equality_operators = true | ||||
|  | ||||
| ij_kotlin_allow_trailing_comma = true | ||||
| ij_kotlin_allow_trailing_comma_on_call_site = true | ||||
|  | ||||
| # Prefer to handle these manually | ||||
| ij_kotlin_method_parameters_wrap = off | ||||
| ij_kotlin_call_parameters_wrap = off | ||||
| ij_kotlin_extends_list_wrap = off | ||||
|   | ||||
| @@ -1,6 +0,0 @@ | ||||
| # SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| # | ||||
| # SPDX-License-Identifier: CC0-1.0 | ||||
|  | ||||
| # Reformat everything | ||||
| f478c4ffc4fb9fc2200ec9b0bc751d047057ce81 | ||||
							
								
								
									
										21
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										21
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							| @@ -1,21 +0,0 @@ | ||||
| # SPDX-FileCopyrightText: 2020 The CC: Tweaked Developers | ||||
| # | ||||
| # SPDX-License-Identifier: CC0-1.0 | ||||
|  | ||||
| # Ignore changes in generated files | ||||
| projects/*/src/generated/** linguist-generated | ||||
| projects/common/src/testMod/resources/data/cctest/structures/* linguist-generated | ||||
|  | ||||
| * text=auto | ||||
|  | ||||
| *.gradle eol=lf diff=java | ||||
| *.java   eol=lf diff=java | ||||
| *.kt     eol=lf diff=java | ||||
| *.kts    eol=lf diff=java | ||||
| *.lua    eol=lf | ||||
| *.md     eol=lf diff=markdown | ||||
| *.txt    eol=lf | ||||
|  | ||||
| *.png binary | ||||
| *.jar binary | ||||
| *.dfpwm binary | ||||
							
								
								
									
										16
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| --- | ||||
| name: Bug report | ||||
| about: Report some misbehaviour in the mod | ||||
| labels: bug | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| ## Before reporting | ||||
|  - Search for the bug on the issue tracker. Make sure to look at closed issues too! | ||||
| --> | ||||
|  | ||||
| ## Useful information to include: | ||||
|  - Minecraft version | ||||
|  - CC: Restitched version | ||||
|  - Logs: These will be located in the `logs/` directory of your Minecraft instance. Please upload them as a gist or directly into this editor. | ||||
|  - Detailed reproduction steps: sometimes I can spot a bug pretty easily, but often it's much more obscure. The more information I have to help reproduce it, the quicker it'll get fixed. | ||||
							
								
								
									
										34
									
								
								.github/ISSUE_TEMPLATE/bug_report.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										34
									
								
								.github/ISSUE_TEMPLATE/bug_report.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,34 +0,0 @@ | ||||
| name: Bug report | ||||
| description: Report some misbehaviour in the mod | ||||
| labels: [ bug ] | ||||
| type: bug | ||||
| body: | ||||
| - type: dropdown | ||||
|   id: mc-version | ||||
|   attributes: | ||||
|     label: Minecraft Version | ||||
|     description: | | ||||
|         What version of Minecraft are you using? If your version is not listed, please try to reproduce on one of the supported versions. | ||||
|     options: | ||||
|       - 1.20.1 | ||||
|       - 1.21.x | ||||
|   validations: | ||||
|     required: true | ||||
| - type: input | ||||
|   id: version | ||||
|   attributes: | ||||
|     label: Version | ||||
|     description: "What version of CC: Tweaked are you using?" | ||||
|     placeholder: "e.g. 1.96.0" | ||||
|   validations: | ||||
|     required: true | ||||
| - type: textarea | ||||
|   id: details | ||||
|   attributes: | ||||
|     label: Details | ||||
|     description: | | ||||
|       Description of the bug. Please include the following: | ||||
|       - Logs: These will be located in the `logs/` directory of your Minecraft instance. This is always useful, even if it doesn't include errors, so please upload this! | ||||
|       - Detailed reproduction steps: sometimes I can spot a bug pretty easily, but often it's much more obscure. The more information I have to help reproduce it, the quicker it'll get fixed. | ||||
|  | ||||
|        | ||||
							
								
								
									
										5
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,5 +0,0 @@ | ||||
| blank_issues_enabled: false | ||||
| contact_links: | ||||
| - name: GitHub Discussions | ||||
|   url: https://github.com/cc-tweaked/CC-Tweaked/discussions | ||||
|   about: Ask questions on GitHub Discussions. | ||||
							
								
								
									
										1
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
								
							| @@ -2,7 +2,6 @@ | ||||
| name: Feature request | ||||
| about: Suggest an idea or improvement | ||||
| labels: enhancement | ||||
| type: feature | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
|   | ||||
							
								
								
									
										12
									
								
								.github/ISSUE_TEMPLATE/peripheral_shoutout.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								.github/ISSUE_TEMPLATE/peripheral_shoutout.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| --- | ||||
| name: Peripheral Shoutout | ||||
| about: Made a Peripheral mod for CC:R? Let us know so we can give it a shoutout | ||||
| labels: peripheralShoutout | ||||
| --- | ||||
|  | ||||
| ## What to include? | ||||
|  - Link to the mod's Icon | ||||
|  - Link to the mod | ||||
|  - Mod Name | ||||
|  - Basic description of the mod | ||||
|  - Link to CC:R Peripheral Documentation for the mod | ||||
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE/something_else.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/ISSUE_TEMPLATE/something_else.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +0,0 @@ | ||||
| --- | ||||
| name: Something else | ||||
| about: An issue about something else. | ||||
| --- | ||||
							
								
								
									
										107
									
								
								.github/workflows/main-ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										107
									
								
								.github/workflows/main-ci.yml
									
									
									
									
										vendored
									
									
								
							| @@ -8,101 +8,52 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|     - name: 📥 Clone repository | ||||
|       uses: actions/checkout@v4 | ||||
|     - uses: actions/checkout@v2 | ||||
|     - name: Checkout submodules | ||||
|       run: git submodule update --init --recursive | ||||
|  | ||||
|     - name: 📥 Set up Java | ||||
|       uses: actions/setup-java@v4 | ||||
|     - name: Set up Java 16 | ||||
|       uses: actions/setup-java@v1 | ||||
|       with: | ||||
|         java-version: 21 | ||||
|         distribution: 'temurin' | ||||
|         java-version: 16 | ||||
|  | ||||
|     - name: 📥 Setup Gradle | ||||
|       uses: gradle/actions/setup-gradle@v3 | ||||
|     - name: Cache gradle dependencies | ||||
|       uses: actions/cache@v2 | ||||
|       with: | ||||
|         cache-read-only: ${{ !startsWith(github.ref, 'refs/heads/mc-') }} | ||||
|         path: ~/.gradle/caches | ||||
|         key: ${{ runner.os }}-gradle-${{ hashFiles('gradle.properties') }} | ||||
|         restore-keys: | | ||||
|           ${{ runner.os }}-gradle- | ||||
|  | ||||
|     - name: Disable Gradle daemon | ||||
|       run: | | ||||
|         mkdir -p ~/.gradle | ||||
|         echo "org.gradle.daemon=false" >> ~/.gradle/gradle.properties | ||||
|  | ||||
|     - name: ⚒️ Build | ||||
|       run: ./gradlew assemble || ./gradlew assemble | ||||
|  | ||||
|     - name: 📦 Prepare Jars | ||||
|     - name: Build with Gradle | ||||
|       run: | | ||||
|         # Find the main jar and append the git hash onto it. | ||||
|         mkdir -p jars | ||||
|         find projects/forge/build/libs projects/fabric/build/libs -type f -regex '.*[0-9.]+\(-SNAPSHOT\)?\.jar$' -exec bash -c 'cp {} "jars/$(basename {} .jar)-$(git rev-parse HEAD).jar"' \; | ||||
|         ./gradlew assemble || ./gradlew assemble | ||||
|         ./gradlew build | ||||
|  | ||||
|     - name: 📤 Upload Jar | ||||
|       uses: actions/upload-artifact@v4 | ||||
|     - name: Upload Jar | ||||
|       uses: actions/upload-artifact@v2 | ||||
|       with: | ||||
|         name: CC-Tweaked | ||||
|         path: ./jars | ||||
|         name: cc-restitched | ||||
|         path: build/libs | ||||
|  | ||||
|     - name: Cache pre-commit | ||||
|       uses: actions/cache@v4 | ||||
|       with: | ||||
|         path: ~/.cache/pre-commit | ||||
|         key: pre-commit-3|${{ env.pythonLocation }}|${{ hashFiles('.pre-commit-config.yaml') }} | ||||
|  | ||||
|     - name: 💡 Lint | ||||
|       run: | | ||||
|         pipx install pre-commit | ||||
|         pre-commit run --show-diff-on-failure --color=always | ||||
|  | ||||
|     - name: 🧪 Run tests | ||||
|       run: ./gradlew test validateMixinNames checkChangelog | ||||
|  | ||||
|     - name: 📥 Download assets for game tests | ||||
|       run: ./gradlew downloadAssets || ./gradlew downloadAssets | ||||
|  | ||||
|     - name: 🧪 Run integration tests | ||||
|       run: ./gradlew runGametest | ||||
|  | ||||
|     - name: 🧪 Parse test reports | ||||
|     - name: Parse test reports | ||||
|       run: ./tools/parse-reports.py | ||||
|       if: ${{ failure() }} | ||||
|  | ||||
|   build-core: | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         include: | ||||
|         - name: Windows | ||||
|           uses: windows-latest | ||||
|  | ||||
|         - name: macOS | ||||
|           uses: macos-latest | ||||
|  | ||||
|     name: Test on ${{ matrix.name }} | ||||
|     runs-on: ${{ matrix.uses }} | ||||
|  | ||||
|     steps: | ||||
|     - name: 📥 Clone repository | ||||
|       uses: actions/checkout@v4 | ||||
|  | ||||
|     - name: 📥 Set up Java | ||||
|       uses: actions/setup-java@v4 | ||||
|     - name: Cache pre-commit | ||||
|       uses: actions/cache@v2 | ||||
|       with: | ||||
|         java-version: 21 | ||||
|         distribution: 'temurin' | ||||
|         path: ~/.cache/pre-commit | ||||
|         key: ${{ runner.os }}-pre-commit-${{ hashFiles('config/pre-commit/config.yml') }} | ||||
|         restore-keys: | | ||||
|           ${{ runner.os }}-pre-commit- | ||||
|  | ||||
|     - name: 📥 Setup Gradle | ||||
|       uses: gradle/actions/setup-gradle@v3 | ||||
|       with: | ||||
|         cache-read-only: ${{ !startsWith(github.ref, 'refs/heads/mc-') }} | ||||
|  | ||||
|     - name: ⚒️ Build | ||||
|     - name: Run linters | ||||
|       run: | | ||||
|         ./gradlew --configure-on-demand :core:assemble | ||||
|  | ||||
|     - name: 🧪 Run tests | ||||
|       run: | | ||||
|         ./gradlew --configure-on-demand :core:test | ||||
|  | ||||
|     - name: 🧪 Parse test reports | ||||
|       run: python3 ./tools/parse-reports.py | ||||
|       if: ${{ failure() }} | ||||
|         pip install pre-commit | ||||
|         pre-commit run --config config/pre-commit/config.yml --show-diff-on-failure --all --color=always | ||||
|   | ||||
							
								
								
									
										35
									
								
								.github/workflows/make-doc.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										35
									
								
								.github/workflows/make-doc.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,35 +0,0 @@ | ||||
| name: Build documentation | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|     - mc-* | ||||
|  | ||||
| jobs: | ||||
|   make_doc: | ||||
|     name: Build | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|     - name: 📥 Clone repository | ||||
|       uses: actions/checkout@v4 | ||||
|  | ||||
|     - name: 📥 Set up Java | ||||
|       uses: actions/setup-java@v4 | ||||
|       with: | ||||
|         java-version: 21 | ||||
|         distribution: 'temurin' | ||||
|  | ||||
|     - name: 📥 Setup Gradle | ||||
|       uses: gradle/actions/setup-gradle@v3 | ||||
|       with: | ||||
|         cache-read-only: ${{ !startsWith(github.ref, 'refs/heads/mc-') }} | ||||
|  | ||||
|     - name: ⚒️ Generate documentation | ||||
|       run: ./gradlew docWebsite --no-daemon | ||||
|  | ||||
|     - name: 📤 Upload Jar | ||||
|       uses: actions/upload-artifact@v4 | ||||
|       with: | ||||
|         name: Documentation | ||||
|         path: ./projects/web/build/site/ | ||||
							
								
								
									
										31
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										31
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,39 +1,24 @@ | ||||
| # SPDX-FileCopyrightText: 2017 The CC: Tweaked Developers | ||||
| # | ||||
| # SPDX-License-Identifier: CC0-1.0 | ||||
|  | ||||
| # Build directories | ||||
| /classes | ||||
| /logs | ||||
| /build | ||||
| /projects/*/logs | ||||
| /projects/fabric/fabricloader.log | ||||
| /projects/*/build | ||||
| /projects/*/src/test/generated_tests/ | ||||
| /buildSrc/build | ||||
| /out | ||||
| /buildSrc/out | ||||
| /jars | ||||
| /doc/out/ | ||||
| /node_modules | ||||
| .jqwik-database | ||||
|  | ||||
| # Runtime directories | ||||
| /run | ||||
| /projects/*/run | ||||
| /run-* | ||||
| /test-files | ||||
|  | ||||
| # Autogenerated by IDE | ||||
| /bin | ||||
| /.settings | ||||
| .classpath | ||||
|  | ||||
| *.ipr | ||||
| *.iws | ||||
| *.iml | ||||
| .idea | ||||
| .gradle | ||||
| .kotlin | ||||
| *.DS_Store | ||||
|  | ||||
| /.classpath | ||||
| /.project | ||||
| /.settings | ||||
| /.vscode | ||||
| .project | ||||
| *.launch | ||||
|  | ||||
| /projects/*/src/generated/resources/.cache | ||||
|   | ||||
							
								
								
									
										3
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| [submodule "src/main/resources/resourcepacks/overhaul"] | ||||
| 	path = src/main/resources/resourcepacks/overhaul | ||||
| 	url = https://github.com/3prm3/cc-pack | ||||
							
								
								
									
										19
									
								
								.gitpod.Dockerfile
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								.gitpod.Dockerfile
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| FROM gitpod/workspace-base | ||||
|  | ||||
| USER gitpod | ||||
|  | ||||
| # Install custom tools, runtime, etc. using apt-get | ||||
| # For example, the command below would install "bastet" - a command line tetris clone: | ||||
| # | ||||
| # RUN sudo apt-get -q update && \ | ||||
| #     sudo apt-get install -yq bastet && \ | ||||
| #     sudo rm -rf /var/lib/apt/lists/* | ||||
| # | ||||
| # More information: https://www.gitpod.io/docs/config-docker/ | ||||
|  | ||||
| # Install Java 8 and 16 | ||||
| RUN sudo apt-get -q update && \ | ||||
|     sudo apt install -yq openjdk-8-jdk openjdk-16-jdk | ||||
|  | ||||
| # This is so that you can use java 8 until such a time as you switch to java 16 | ||||
| RUN sudo update-java-alternatives --set java-1.16.0-openjdk-amd64 | ||||
							
								
								
									
										18
									
								
								.gitpod.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								.gitpod.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| image: | ||||
|   file: .gitpod.Dockerfile | ||||
|  | ||||
| ports: | ||||
|   - port: 25565 | ||||
|     onOpen: notify | ||||
|  | ||||
| vscode: | ||||
|   extensions: | ||||
|     - ms-azuretools.vscode-docker | ||||
|     - redhat.java | ||||
|     - richardwillis.vscode-gradle | ||||
|     - vscjava.vscode-java-debug | ||||
|     - vscode.github | ||||
|  | ||||
|  | ||||
| tasks: | ||||
|   - init: ./gradlew | ||||
							
								
								
									
										34
									
								
								.luacheckrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								.luacheckrc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| std = "max" | ||||
|  | ||||
| ignore = { | ||||
|     -- Allow access to undefined globals or their fields. In the future we'll | ||||
|     -- define all of CC's globals within this file | ||||
|     '113', '143', | ||||
|  | ||||
|     -- FIXME: Ignore unused arguments and loop variables | ||||
|     '212', '213', | ||||
|  | ||||
|     -- Disable line is too long for now. It would be possible to clean | ||||
|     -- this up in the future. | ||||
|     '631', | ||||
| } | ||||
|  | ||||
| -- Only run the linter on ROM and bios for now, as the treasure disks | ||||
| -- are largely unsupported. | ||||
| include_files = { | ||||
|     'src/main/resources/assets/computercraft/lua/rom', | ||||
|     'src/main/resources/assets/computercraft/lua/bios.lua' | ||||
| } | ||||
|  | ||||
| files['src/main/resources/assets/computercraft/lua/bios.lua'] = { | ||||
|     -- Allow declaring and mutating globals | ||||
|     allow_defined_top = true, | ||||
|     ignore = { '112', '121', '122', '131', '142' }, | ||||
| } | ||||
|  | ||||
| files['src/main/resources/assets/computercraft/lua/rom/apis'] = { | ||||
|     -- APIs may define globals on the top level. We'll ignore unused globals, | ||||
|     -- as obviously they may be used outside that API. | ||||
|     allow_defined_top = true, | ||||
|     ignore = { '131' }, | ||||
| } | ||||
| @@ -1,64 +0,0 @@ | ||||
| # SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| # | ||||
| # SPDX-License-Identifier: CC0-1.0 | ||||
|  | ||||
| # See https://pre-commit.com for more information | ||||
| # See https://pre-commit.com/hooks.html for more hooks | ||||
| repos: | ||||
| - repo: https://github.com/pre-commit/pre-commit-hooks | ||||
|   rev: v4.4.0 | ||||
|   hooks: | ||||
|   - id: trailing-whitespace | ||||
|   - id: end-of-file-fixer | ||||
|   - id: check-merge-conflict | ||||
|  | ||||
|   # Quick syntax checkers | ||||
|   - id: check-xml | ||||
|   - id: check-yaml | ||||
|   - id: check-toml | ||||
|   - id: check-json | ||||
|     exclude: "tsconfig\\.json$" | ||||
|  | ||||
| - repo: https://github.com/editorconfig-checker/editorconfig-checker.python | ||||
|   rev: 2.7.2 | ||||
|   hooks: | ||||
|   - id: editorconfig-checker | ||||
|     args: ['-disable-indentation'] | ||||
|     exclude: "^(.*\\.(bat)|LICENSE)$" | ||||
|  | ||||
| - repo: https://github.com/fsfe/reuse-tool | ||||
|   rev: v5.0.2 | ||||
|   hooks: | ||||
|   - id: reuse | ||||
|  | ||||
| - repo: local | ||||
|   hooks: | ||||
|   - id: license | ||||
|     name: Spotless | ||||
|     files: ".*\\.(java|kt|kts)$" | ||||
|     language: system | ||||
|     entry: ./gradlew spotlessApply | ||||
|     pass_filenames: false | ||||
|     require_serial: true | ||||
|   - id: checkstyle | ||||
|     name: Check Java codestyle | ||||
|     files: ".*\\.java$" | ||||
|     language: system | ||||
|     entry: ./gradlew checkstyle | ||||
|     pass_filenames: false | ||||
|     require_serial: true | ||||
|   - id: illuaminate | ||||
|     name: Check Lua code | ||||
|     files: ".*\\.(lua|java|md)" | ||||
|     language: system | ||||
|     entry: ./gradlew lintLua | ||||
|     pass_filenames: false | ||||
|     require_serial: true | ||||
|  | ||||
| exclude: | | ||||
|   (?x)^( | ||||
|     projects/[a-z]+/src/generated| | ||||
|     projects/[a-z]+/src/examples/generatedResources| | ||||
|     projects/core/src/test/resources/test-rom/data/json-parsing/| | ||||
|     .*\.dfpwm | ||||
|   ) | ||||
							
								
								
									
										15
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| { | ||||
| 	"files.exclude": { | ||||
| 		// Default Java Dev | ||||
|         "**/.classpath": true, | ||||
|         "**/.project": true, | ||||
|         "**/.settings": true, | ||||
|         "**/.factorypath": true, | ||||
|  | ||||
| 		// Custom Hidden Files | ||||
| 		"**/.bin": true, | ||||
| 		"**/.editorconfig": true, | ||||
|     }, | ||||
|  | ||||
| 	"java.configuration.updateBuildConfiguration": "automatic" | ||||
| } | ||||
| @@ -1,138 +0,0 @@ | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2014 Coraline Ada Ehmke and contributors | ||||
|  | ||||
| SPDX-License-Identifier: CC-BY-4.0 | ||||
| --> | ||||
|  | ||||
| # Contributor Covenant Code of Conduct | ||||
|  | ||||
| ## Our Pledge | ||||
|  | ||||
| We as members, contributors, and leaders pledge to make participation in our | ||||
| community a harassment-free experience for everyone, regardless of age, body | ||||
| size, visible or invisible disability, ethnicity, sex characteristics, gender | ||||
| identity and expression, level of experience, education, socio-economic status, | ||||
| nationality, personal appearance, race, caste, color, religion, or sexual | ||||
| identity and orientation. | ||||
|  | ||||
| We pledge to act and interact in ways that contribute to an open, welcoming, | ||||
| diverse, inclusive, and healthy community. | ||||
|  | ||||
| ## Our Standards | ||||
|  | ||||
| Examples of behavior that contributes to a positive environment for our | ||||
| community include: | ||||
|  | ||||
| * Demonstrating empathy and kindness toward other people | ||||
| * Being respectful of differing opinions, viewpoints, and experiences | ||||
| * Giving and gracefully accepting constructive feedback | ||||
| * Accepting responsibility and apologizing to those affected by our mistakes, | ||||
|   and learning from the experience | ||||
| * Focusing on what is best not just for us as individuals, but for the overall | ||||
|   community | ||||
|  | ||||
| Examples of unacceptable behavior include: | ||||
|  | ||||
| * The use of sexualized language or imagery, and sexual attention or advances of | ||||
|   any kind | ||||
| * Trolling, insulting or derogatory comments, and personal or political attacks | ||||
| * Public or private harassment | ||||
| * Publishing others' private information, such as a physical or email address, | ||||
|   without their explicit permission | ||||
| * Other conduct which could reasonably be considered inappropriate in a | ||||
|   professional setting | ||||
|  | ||||
| ## Enforcement Responsibilities | ||||
|  | ||||
| Community leaders are responsible for clarifying and enforcing our standards of | ||||
| acceptable behavior and will take appropriate and fair corrective action in | ||||
| response to any behavior that they deem inappropriate, threatening, offensive, | ||||
| or harmful. | ||||
|  | ||||
| Community leaders have the right and responsibility to remove, edit, or reject | ||||
| comments, commits, code, wiki edits, issues, and other contributions that are | ||||
| not aligned to this Code of Conduct, and will communicate reasons for moderation | ||||
| decisions when appropriate. | ||||
|  | ||||
| ## Scope | ||||
|  | ||||
| This Code of Conduct applies within all community spaces, and also applies when | ||||
| an individual is officially representing the community in public spaces. | ||||
| Examples of representing our community include using an official e-mail address, | ||||
| posting via an official social media account, or acting as an appointed | ||||
| representative at an online or offline event. | ||||
|  | ||||
| ## Enforcement | ||||
|  | ||||
| Instances of abusive, harassing, or otherwise unacceptable behavior may be | ||||
| reported to the community leaders responsible for enforcement at | ||||
| "conduct AT squiddev DOT cc". All complaints will be reviewed and investigated | ||||
| promptly and fairly. | ||||
|  | ||||
| All community leaders are obligated to respect the privacy and security of the | ||||
| reporter of any incident. | ||||
|  | ||||
| ## Enforcement Guidelines | ||||
|  | ||||
| Community leaders will follow these Community Impact Guidelines in determining | ||||
| the consequences for any action they deem in violation of this Code of Conduct: | ||||
|  | ||||
| ### 1. Correction | ||||
|  | ||||
| **Community Impact**: Use of inappropriate language or other behavior deemed | ||||
| unprofessional or unwelcome in the community. | ||||
|  | ||||
| **Consequence**: A private, written warning from community leaders, providing | ||||
| clarity around the nature of the violation and an explanation of why the | ||||
| behavior was inappropriate. A public apology may be requested. | ||||
|  | ||||
| ### 2. Warning | ||||
|  | ||||
| **Community Impact**: A violation through a single incident or series of | ||||
| actions. | ||||
|  | ||||
| **Consequence**: A warning with consequences for continued behavior. No | ||||
| interaction with the people involved, including unsolicited interaction with | ||||
| those enforcing the Code of Conduct, for a specified period of time. This | ||||
| includes avoiding interactions in community spaces as well as external channels | ||||
| like social media. Violating these terms may lead to a temporary or permanent | ||||
| ban. | ||||
|  | ||||
| ### 3. Temporary Ban | ||||
|  | ||||
| **Community Impact**: A serious violation of community standards, including | ||||
| sustained inappropriate behavior. | ||||
|  | ||||
| **Consequence**: A temporary ban from any sort of interaction or public | ||||
| communication with the community for a specified period of time. No public or | ||||
| private interaction with the people involved, including unsolicited interaction | ||||
| with those enforcing the Code of Conduct, is allowed during this period. | ||||
| Violating these terms may lead to a permanent ban. | ||||
|  | ||||
| ### 4. Permanent Ban | ||||
|  | ||||
| **Community Impact**: Demonstrating a pattern of violation of community | ||||
| standards, including sustained inappropriate behavior, harassment of an | ||||
| individual, or aggression toward or disparagement of classes of individuals. | ||||
|  | ||||
| **Consequence**: A permanent ban from any sort of public interaction within the | ||||
| community. | ||||
|  | ||||
| ## Attribution | ||||
|  | ||||
| This Code of Conduct is adapted from the [Contributor Covenant][homepage], | ||||
| version 2.1, available at | ||||
| [https://www.contributor-covenant.org/version/2/1/code_of_conduct.html][v2.1]. | ||||
|  | ||||
| Community Impact Guidelines were inspired by | ||||
| [Mozilla's code of conduct enforcement ladder][Mozilla CoC]. | ||||
|  | ||||
| For answers to common questions about this code of conduct, see the FAQ at | ||||
| [https://www.contributor-covenant.org/faq][FAQ]. Translations are available at | ||||
| [https://www.contributor-covenant.org/translations][translations]. | ||||
|  | ||||
| [homepage]: https://www.contributor-covenant.org | ||||
| [v2.1]: https://www.contributor-covenant.org/version/2/1/code_of_conduct.html | ||||
| [Mozilla CoC]: https://github.com/mozilla/diversity | ||||
| [FAQ]: https://www.contributor-covenant.org/faq | ||||
| [translations]: https://www.contributor-covenant.org/translations | ||||
							
								
								
									
										112
									
								
								CONTRIBUTING.md
									
									
									
									
									
								
							
							
						
						
									
										112
									
								
								CONTRIBUTING.md
									
									
									
									
									
								
							| @@ -1,112 +0,0 @@ | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2020 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| # Contributing to CC: Tweaked | ||||
| As with many open source projects, CC: Tweaked thrives on contributions from other people! This document (hopefully) | ||||
| provides an introduction as to how to get started with helping out. | ||||
|  | ||||
| If you've any other questions, [just ask the community][community] or [open an issue][new-issue]. | ||||
|  | ||||
| ## Table of Contents | ||||
|  - [Reporting issues](#reporting-issues) | ||||
|  - [Translations](#translations) | ||||
|  - [Setting up a development environment](#setting-up-a-development-environment) | ||||
|  - [Developing CC: Tweaked](#developing-cc-tweaked) | ||||
|  - [Writing documentation](#writing-documentation) | ||||
|  | ||||
| ## Reporting issues | ||||
| If you have a bug, suggestion, or other feedback, the best thing to do is [file an issue][new-issue]. When doing so, do | ||||
| use the issue templates - they provide a useful hint on what information to provide. | ||||
|  | ||||
| ## Translations | ||||
| Translations are managed through [CrowdIn], an online interface for managing language strings. | ||||
|  | ||||
| ## Setting up a development environment | ||||
| In order to develop CC: Tweaked, you'll need to download the source code and then run it. | ||||
|  | ||||
|  - Make sure you've got the following software installed: | ||||
|    - Java Development Kit 21 (JDK). This can be downloaded from [Adoptium]. | ||||
|    - [Git](https://git-scm.com/). | ||||
|    - [NodeJS 20 or later][node]. | ||||
|  | ||||
|  - Download CC: Tweaked's source code: | ||||
|    ``` | ||||
|    git clone https://github.com/cc-tweaked/CC-Tweaked.git | ||||
|    cd CC-Tweaked | ||||
|    ``` | ||||
|  | ||||
|  - Build CC: Tweaked with `./gradlew build`. This will be very slow the first time it runs, as it needs to download a | ||||
|    lot of dependencies (and decompile Minecraft several times). Subsequent runs should be much faster! | ||||
|  | ||||
|  - You're now ready to start developing CC: Tweaked. Running `./gradlew :forge:runClient` or | ||||
|    `./gradle :fabric:runClient` will start Minecraft under Forge and Fabric respectively. | ||||
|  | ||||
| If you want to run CC:T in a normal Minecraft instance, run `./gradlew assemble` and copy the `.jar` from | ||||
| `projects/forge/build/libs` (for Forge) or `projects/fabric/build/libs` (for Fabric). | ||||
|  | ||||
| ## Developing CC: Tweaked | ||||
| Before making any major changes to CC: Tweaked, I'd recommend starting opening an issue or starting a discussion on | ||||
| GitHub first. It's often helpful to discuss features before spending time developing them! | ||||
|  | ||||
| Once you're ready to start programming, have a read of the [the architecture document][architecture] first. While it's | ||||
| not a comprehensive document, it gives a good hint of where you should start looking to make your changes. As always, if | ||||
| you're not sure, [do ask the community][community]! | ||||
|  | ||||
| ### Testing | ||||
| When making larger changes, it may be useful to write a test to make sure your code works as expected. | ||||
|  | ||||
| CC: Tweaked has several test suites, each designed to test something different: | ||||
|  | ||||
|  - In order to test CraftOS and its builtin APIs, we have a test suite written in Lua located at | ||||
|    `projects/core/src/test/resources/test-rom/`. These don't rely on any Minecraft code, which means they can run on | ||||
|    emulators, acting as a sort of compliance test. | ||||
|  | ||||
|    These tests are written using a test system called "mcfly", heavily inspired by [busted]. Groups of tests go inside | ||||
|    `describe` blocks, and a single test goes inside `it`. Assertions are generally written using `expect` (inspired by | ||||
|    Hamcrest and the like). For instance, `expect(foo):eq("bar")` asserts that your variable `foo` is equal to the | ||||
|    expected value `"bar"`. | ||||
|  | ||||
|    These tests can be run with `./gradlew :core:test`. | ||||
|  | ||||
|  - In-game functionality, such as the behaviour of blocks and items, is tested using [Minecraft's gametest | ||||
|    system][mc-test] (`projects/common/src/testMod`). These tests spin up a server, spawn a structure for each test, and | ||||
|    then run some code on the blocks defined in that structure. | ||||
|  | ||||
|    These tests can be run with `./gradlew runGametest` (or `./gradle :forge:runGametest`/`./gradlew :fabric:runGametest` | ||||
|    for a single loader). | ||||
|  | ||||
| For more information, [see the architecture document][architecture]. | ||||
|  | ||||
| ## Writing documentation | ||||
| When writing documentation for [CC: Tweaked's documentation website][docs], it may be useful to build the documentation | ||||
| and preview it yourself before submitting a PR. | ||||
|  | ||||
| You'll first need to [set up a development environment as above](#setting-up-a-development-environment). | ||||
|  | ||||
| Once this is set up, you can now run `./gradlew docWebsite`. This generates documentation from our Lua and Java code, | ||||
| writing the resulting HTML into `./projects/web/build/site`, which can then be opened in a browser. When iterating on | ||||
| documentation, you can instead run `./gradlew docWebsite -t`, which will rebuild documentation every time you change a | ||||
| file. | ||||
|  | ||||
| Documentation is built using [illuaminate] which, while not currently documented (somewhat ironic), is largely the same | ||||
| as [ldoc][ldoc]. Documentation comments are written in Markdown, though note that we do not support many GitHub-specific | ||||
| markdown features. If you can, do check what the documentation looks like locally! | ||||
|  | ||||
| When writing long-form documentation (such as the guides in [doc/guides](doc/guides)), I find it useful to tell a | ||||
| narrative. Think of what you want the user to learn or achieve, then start introducing a simple concept, and then talk | ||||
| about how you can build on that until you've covered everything! | ||||
|  | ||||
| [new-issue]: https://github.com/cc-tweaked/CC-Tweaked/issues/new/choose "Create a new issue" | ||||
| [community]: README.md#community "Get in touch with the community." | ||||
| [Adoptium]: https://adoptium.net/temurin/releases?version=17 "Download OpenJDK 17" | ||||
| [illuaminate]: https://github.com/SquidDev/illuaminate/ "Illuaminate on GitHub" | ||||
| [docs]: https://tweaked.cc/ "CC: Tweaked documentation" | ||||
| [ldoc]: http://stevedonovan.github.io/ldoc/ "ldoc, a Lua documentation generator." | ||||
| [mc-test]: https://www.youtube.com/watch?v=vXaWOJTCYNg | ||||
| [busted]: https://github.com/Olivine-Labs/busted "busted: Elegant Lua unit testing." | ||||
| [node]: https://nodejs.org/en/ "Node.js" | ||||
| [architecture]: projects/ARCHITECTURE.md | ||||
| [Crowdin]: https://crowdin.com/project/cc-tweaked/ | ||||
| @@ -1,73 +0,0 @@ | ||||
| Apache License | ||||
| Version 2.0, January 2004 | ||||
| http://www.apache.org/licenses/ | ||||
|  | ||||
| TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION | ||||
|  | ||||
| 1. Definitions. | ||||
|  | ||||
| "License" shall mean the terms and conditions for use, reproduction, and distribution as defined by Sections 1 through 9 of this document. | ||||
|  | ||||
| "Licensor" shall mean the copyright owner or entity authorized by the copyright owner that is granting the License. | ||||
|  | ||||
| "Legal Entity" shall mean the union of the acting entity and all other entities that control, are controlled by, or are under common control with that entity. For the purposes of this definition, "control" means (i) the power, direct or indirect, to cause the direction or management of such entity, whether by contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the outstanding shares, or (iii) beneficial ownership of such entity. | ||||
|  | ||||
| "You" (or "Your") shall mean an individual or Legal Entity exercising permissions granted by this License. | ||||
|  | ||||
| "Source" form shall mean the preferred form for making modifications, including but not limited to software source code, documentation source, and configuration files. | ||||
|  | ||||
| "Object" form shall mean any form resulting from mechanical transformation or translation of a Source form, including but not limited to compiled object code, generated documentation, and conversions to other media types. | ||||
|  | ||||
| "Work" shall mean the work of authorship, whether in Source or Object form, made available under the License, as indicated by a copyright notice that is included in or attached to the work (an example is provided in the Appendix below). | ||||
|  | ||||
| "Derivative Works" shall mean any work, whether in Source or Object form, that is based on (or derived from) the Work and for which the editorial revisions, annotations, elaborations, or other modifications represent, as a whole, an original work of authorship. For the purposes of this License, Derivative Works shall not include works that remain separable from, or merely link (or bind by name) to the interfaces of, the Work and Derivative Works thereof. | ||||
|  | ||||
| "Contribution" shall mean any work of authorship, including the original version of the Work and any modifications or additions to that Work or Derivative Works thereof, that is intentionally submitted to Licensor for inclusion in the Work by the copyright owner or by an individual or Legal Entity authorized to submit on behalf of the copyright owner. For the purposes of this definition, "submitted" means any form of electronic, verbal, or written communication sent to the Licensor or its representatives, including but not limited to communication on electronic mailing lists, source code control systems, and issue tracking systems that are managed by, or on behalf of, the Licensor for the purpose of discussing and improving the Work, but excluding communication that is conspicuously marked or otherwise designated in writing by the copyright owner as "Not a Contribution." | ||||
|  | ||||
| "Contributor" shall mean Licensor and any individual or Legal Entity on behalf of whom a Contribution has been received by Licensor and subsequently incorporated within the Work. | ||||
|  | ||||
| 2. Grant of Copyright License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable copyright license to reproduce, prepare Derivative Works of, publicly display, publicly perform, sublicense, and distribute the Work and such Derivative Works in Source or Object form. | ||||
|  | ||||
| 3. Grant of Patent License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this section) patent license to make, have made, use, offer to sell, sell, import, and otherwise transfer the Work, where such license applies only to those patent claims licensable by such Contributor that are necessarily infringed by their Contribution(s) alone or by combination of their Contribution(s) with the Work to which such Contribution(s) was submitted. If You institute patent litigation against any entity (including a cross-claim or counterclaim in a lawsuit) alleging that the Work or a Contribution incorporated within the Work constitutes direct or contributory patent infringement, then any patent licenses granted to You under this License for that Work shall terminate as of the date such litigation is filed. | ||||
|  | ||||
| 4. Redistribution. You may reproduce and distribute copies of the Work or Derivative Works thereof in any medium, with or without modifications, and in Source or Object form, provided that You meet the following conditions: | ||||
|  | ||||
|      (a) You must give any other recipients of the Work or Derivative Works a copy of this License; and | ||||
|  | ||||
|      (b) You must cause any modified files to carry prominent notices stating that You changed the files; and | ||||
|  | ||||
|      (c) You must retain, in the Source form of any Derivative Works that You distribute, all copyright, patent, trademark, and attribution notices from the Source form of the Work, excluding those notices that do not pertain to any part of the Derivative Works; and | ||||
|  | ||||
|      (d) If the Work includes a "NOTICE" text file as part of its distribution, then any Derivative Works that You distribute must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be construed as modifying the License. | ||||
|  | ||||
|      You may add Your own copyright statement to Your modifications and may provide additional or different license terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated in this License. | ||||
|  | ||||
| 5. Submission of Contributions. Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the Licensor shall be under the terms and conditions of this License, without any additional terms or conditions. Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement you may have executed with Licensor regarding such Contributions. | ||||
|  | ||||
| 6. Trademarks. This License does not grant permission to use the trade names, trademarks, service marks, or product names of the Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing the content of the NOTICE file. | ||||
|  | ||||
| 7. Disclaimer of Warranty. Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides its Contributions) on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or redistributing the Work and assume any risks associated with Your exercise of permissions under this License. | ||||
|  | ||||
| 8. Limitation of Liability. In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even if such Contributor has been advised of the possibility of such damages. | ||||
|  | ||||
| 9. Accepting Warranty or Additional Liability. While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional liability. | ||||
|  | ||||
| END OF TERMS AND CONDITIONS | ||||
|  | ||||
| APPENDIX: How to apply the Apache License to your work. | ||||
|  | ||||
| To apply the Apache License to your work, attach the following boilerplate notice, with the fields enclosed by brackets "[]" replaced with your own identifying information. (Don't include the brackets!)  The text should be enclosed in the appropriate comment syntax for the file format. We also recommend that a file or class name and description of purpose be included on the same "printed page" as the copyright notice for easier identification within third-party archives. | ||||
|  | ||||
| Copyright [yyyy] [name of copyright owner] | ||||
|  | ||||
| Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| you may not use this file except in compliance with the License. | ||||
| You may obtain a copy of the License at | ||||
|  | ||||
| http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  | ||||
| Unless required by applicable law or agreed to in writing, software | ||||
| distributed under the License is distributed on an "AS IS" BASIS, | ||||
| WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| See the License for the specific language governing permissions and | ||||
| limitations under the License. | ||||
| @@ -1,93 +0,0 @@ | ||||
| Creative Commons Attribution 3.0 Unported | ||||
|  | ||||
|  CREATIVE COMMONS CORPORATION IS NOT A LAW FIRM AND DOES NOT PROVIDE LEGAL SERVICES. DISTRIBUTION OF THIS LICENSE DOES NOT CREATE AN ATTORNEY-CLIENT RELATIONSHIP. CREATIVE COMMONS PROVIDES THIS INFORMATION ON AN "AS-IS" BASIS. CREATIVE COMMONS MAKES NO WARRANTIES REGARDING THE INFORMATION PROVIDED, AND DISCLAIMS LIABILITY FOR DAMAGES RESULTING FROM ITS USE. | ||||
|  | ||||
| License | ||||
|  | ||||
| THE WORK (AS DEFINED BELOW) IS PROVIDED UNDER THE TERMS OF THIS CREATIVE COMMONS PUBLIC LICENSE ("CCPL" OR "LICENSE"). THE WORK IS PROTECTED BY COPYRIGHT AND/OR OTHER APPLICABLE LAW. ANY USE OF THE WORK OTHER THAN AS AUTHORIZED UNDER THIS LICENSE OR COPYRIGHT LAW IS PROHIBITED. | ||||
|  | ||||
| BY EXERCISING ANY RIGHTS TO THE WORK PROVIDED HERE, YOU ACCEPT AND AGREE TO BE BOUND BY THE TERMS OF THIS LICENSE. TO THE EXTENT THIS LICENSE MAY BE CONSIDERED TO BE A CONTRACT, THE LICENSOR GRANTS YOU THE RIGHTS CONTAINED HERE IN CONSIDERATION OF YOUR ACCEPTANCE OF SUCH TERMS AND CONDITIONS. | ||||
|  | ||||
| 1. Definitions | ||||
|  | ||||
|      a. "Adaptation" means a work based upon the Work, or upon the Work and other pre-existing works, such as a translation, adaptation, derivative work, arrangement of music or other alterations of a literary or artistic work, or phonogram or performance and includes cinematographic adaptations or any other form in which the Work may be recast, transformed, or adapted including in any form recognizably derived from the original, except that a work that constitutes a Collection will not be considered an Adaptation for the purpose of this License. For the avoidance of doubt, where the Work is a musical work, performance or phonogram, the synchronization of the Work in timed-relation with a moving image ("synching") will be considered an Adaptation for the purpose of this License. | ||||
|  | ||||
|      b. "Collection" means a collection of literary or artistic works, such as encyclopedias and anthologies, or performances, phonograms or broadcasts, or other works or subject matter other than works listed in Section 1(f) below, which, by reason of the selection and arrangement of their contents, constitute intellectual creations, in which the Work is included in its entirety in unmodified form along with one or more other contributions, each constituting separate and independent works in themselves, which together are assembled into a collective whole. A work that constitutes a Collection will not be considered an Adaptation (as defined above) for the purposes of this License. | ||||
|  | ||||
|      c. "Distribute" means to make available to the public the original and copies of the Work or Adaptation, as appropriate, through sale or other transfer of ownership. | ||||
|  | ||||
|      d. "Licensor" means the individual, individuals, entity or entities that offer(s) the Work under the terms of this License. | ||||
|  | ||||
|      e. "Original Author" means, in the case of a literary or artistic work, the individual, individuals, entity or entities who created the Work or if no individual or entity can be identified, the publisher; and in addition (i) in the case of a performance the actors, singers, musicians, dancers, and other persons who act, sing, deliver, declaim, play in, interpret or otherwise perform literary or artistic works or expressions of folklore; (ii) in the case of a phonogram the producer being the person or legal entity who first fixes the sounds of a performance or other sounds; and, (iii) in the case of broadcasts, the organization that transmits the broadcast. | ||||
|  | ||||
|      f. "Work" means the literary and/or artistic work offered under the terms of this License including without limitation any production in the literary, scientific and artistic domain, whatever may be the mode or form of its expression including digital form, such as a book, pamphlet and other writing; a lecture, address, sermon or other work of the same nature; a dramatic or dramatico-musical work; a choreographic work or entertainment in dumb show; a musical composition with or without words; a cinematographic work to which are assimilated works expressed by a process analogous to cinematography; a work of drawing, painting, architecture, sculpture, engraving or lithography; a photographic work to which are assimilated works expressed by a process analogous to photography; a work of applied art; an illustration, map, plan, sketch or three-dimensional work relative to geography, topography, architecture or science; a performance; a broadcast; a phonogram; a compilation of data to the extent it is protected as a copyrightable work; or a work performed by a variety or circus performer to the extent it is not otherwise considered a literary or artistic work. | ||||
|  | ||||
|      g. "You" means an individual or entity exercising rights under this License who has not previously violated the terms of this License with respect to the Work, or who has received express permission from the Licensor to exercise rights under this License despite a previous violation. | ||||
|  | ||||
|      h. "Publicly Perform" means to perform public recitations of the Work and to communicate to the public those public recitations, by any means or process, including by wire or wireless means or public digital performances; to make available to the public Works in such a way that members of the public may access these Works from a place and at a place individually chosen by them; to perform the Work to the public by any means or process and the communication to the public of the performances of the Work, including by public digital performance; to broadcast and rebroadcast the Work by any means including signs, sounds or images. | ||||
|  | ||||
|      i. "Reproduce" means to make copies of the Work by any means including without limitation by sound or visual recordings and the right of fixation and reproducing fixations of the Work, including storage of a protected performance or phonogram in digital form or other electronic medium. | ||||
|  | ||||
| 2. Fair Dealing Rights. Nothing in this License is intended to reduce, limit, or restrict any uses free from copyright or rights arising from limitations or exceptions that are provided for in connection with the copyright protection under copyright law or other applicable laws. | ||||
|  | ||||
| 3. License Grant. Subject to the terms and conditions of this License, Licensor hereby grants You a worldwide, royalty-free, non-exclusive, perpetual (for the duration of the applicable copyright) license to exercise the rights in the Work as stated below: | ||||
|  | ||||
|      a. to Reproduce the Work, to incorporate the Work into one or more Collections, and to Reproduce the Work as incorporated in the Collections; | ||||
|  | ||||
|      b. to create and Reproduce Adaptations provided that any such Adaptation, including any translation in any medium, takes reasonable steps to clearly label, demarcate or otherwise identify that changes were made to the original Work. For example, a translation could be marked "The original work was translated from English to Spanish," or a modification could indicate "The original work has been modified."; | ||||
|  | ||||
|      c. to Distribute and Publicly Perform the Work including as incorporated in Collections; and, | ||||
|  | ||||
|      d. to Distribute and Publicly Perform Adaptations. | ||||
|  | ||||
|      e. For the avoidance of doubt: | ||||
|  | ||||
|           i. Non-waivable Compulsory License Schemes. In those jurisdictions in which the right to collect royalties through any statutory or compulsory licensing scheme cannot be waived, the Licensor reserves the exclusive right to collect such royalties for any exercise by You of the rights granted under this License; | ||||
|  | ||||
|           ii. Waivable Compulsory License Schemes. In those jurisdictions in which the right to collect royalties through any statutory or compulsory licensing scheme can be waived, the Licensor waives the exclusive right to collect such royalties for any exercise by You of the rights granted under this License; and, | ||||
|  | ||||
|           iii. Voluntary License Schemes. The Licensor waives the right to collect royalties, whether individually or, in the event that the Licensor is a member of a collecting society that administers voluntary licensing schemes, via that society, from any exercise by You of the rights granted under this License. | ||||
|  | ||||
| The above rights may be exercised in all media and formats whether now known or hereafter devised. The above rights include the right to make such modifications as are technically necessary to exercise the rights in other media and formats. Subject to Section 8(f), all rights not expressly granted by Licensor are hereby reserved. | ||||
|  | ||||
| 4. Restrictions. The license granted in Section 3 above is expressly made subject to and limited by the following restrictions: | ||||
|  | ||||
|      a. You may Distribute or Publicly Perform the Work only under the terms of this License. You must include a copy of, or the Uniform Resource Identifier (URI) for, this License with every copy of the Work You Distribute or Publicly Perform. You may not offer or impose any terms on the Work that restrict the terms of this License or the ability of the recipient of the Work to exercise the rights granted to that recipient under the terms of the License. You may not sublicense the Work. You must keep intact all notices that refer to this License and to the disclaimer of warranties with every copy of the Work You Distribute or Publicly Perform. When You Distribute or Publicly Perform the Work, You may not impose any effective technological measures on the Work that restrict the ability of a recipient of the Work from You to exercise the rights granted to that recipient under the terms of the License. This Section 4(a) applies to the Work as incorporated in a Collection, but this does not require the Collection apart from the Work itself to be made subject to the terms of this License. If You create a Collection, upon notice from any Licensor You must, to the extent practicable, remove from the Collection any credit as required by Section 4(b), as requested. If You create an Adaptation, upon notice from any Licensor You must, to the extent practicable, remove from the Adaptation any credit as required by Section 4(b), as requested. | ||||
|  | ||||
|      b. If You Distribute, or Publicly Perform the Work or any Adaptations or Collections, You must, unless a request has been made pursuant to Section 4(a), keep intact all copyright notices for the Work and provide, reasonable to the medium or means You are utilizing: (i) the name of the Original Author (or pseudonym, if applicable) if supplied, and/or if the Original Author and/or Licensor designate another party or parties (e.g., a sponsor institute, publishing entity, journal) for attribution ("Attribution Parties") in Licensor's copyright notice, terms of service or by other reasonable means, the name of such party or parties; (ii) the title of the Work if supplied; (iii) to the extent reasonably practicable, the URI, if any, that Licensor specifies to be associated with the Work, unless such URI does not refer to the copyright notice or licensing information for the Work; and (iv) , consistent with Section 3(b), in the case of an Adaptation, a credit identifying the use of the Work in the Adaptation (e.g., "French translation of the Work by Original Author," or "Screenplay based on original Work by Original Author"). The credit required by this Section 4 (b) may be implemented in any reasonable manner; provided, however, that in the case of a Adaptation or Collection, at a minimum such credit will appear, if a credit for all contributing authors of the Adaptation or Collection appears, then as part of these credits and in a manner at least as prominent as the credits for the other contributing authors. For the avoidance of doubt, You may only use the credit required by this Section for the purpose of attribution in the manner set out above and, by exercising Your rights under this License, You may not implicitly or explicitly assert or imply any connection with, sponsorship or endorsement by the Original Author, Licensor and/or Attribution Parties, as appropriate, of You or Your use of the Work, without the separate, express prior written permission of the Original Author, Licensor and/or Attribution Parties. | ||||
|  | ||||
|      c. Except as otherwise agreed in writing by the Licensor or as may be otherwise permitted by applicable law, if You Reproduce, Distribute or Publicly Perform the Work either by itself or as part of any Adaptations or Collections, You must not distort, mutilate, modify or take other derogatory action in relation to the Work which would be prejudicial to the Original Author's honor or reputation. Licensor agrees that in those jurisdictions (e.g. Japan), in which any exercise of the right granted in Section 3(b) of this License (the right to make Adaptations) would be deemed to be a distortion, mutilation, modification or other derogatory action prejudicial to the Original Author's honor and reputation, the Licensor will waive or not assert, as appropriate, this Section, to the fullest extent permitted by the applicable national law, to enable You to reasonably exercise Your right under Section 3(b) of this License (right to make Adaptations) but not otherwise. | ||||
|  | ||||
| 5. Representations, Warranties and Disclaimer | ||||
|  | ||||
| UNLESS OTHERWISE MUTUALLY AGREED TO BY THE PARTIES IN WRITING, LICENSOR OFFERS THE WORK AS-IS AND MAKES NO REPRESENTATIONS OR WARRANTIES OF ANY KIND CONCERNING THE WORK, EXPRESS, IMPLIED, STATUTORY OR OTHERWISE, INCLUDING, WITHOUT LIMITATION, WARRANTIES OF TITLE, MERCHANTIBILITY, FITNESS FOR A PARTICULAR PURPOSE, NONINFRINGEMENT, OR THE ABSENCE OF LATENT OR OTHER DEFECTS, ACCURACY, OR THE PRESENCE OF ABSENCE OF ERRORS, WHETHER OR NOT DISCOVERABLE. SOME JURISDICTIONS DO NOT ALLOW THE EXCLUSION OF IMPLIED WARRANTIES, SO SUCH EXCLUSION MAY NOT APPLY TO YOU. | ||||
|  | ||||
| 6. Limitation on Liability. EXCEPT TO THE EXTENT REQUIRED BY APPLICABLE LAW, IN NO EVENT WILL LICENSOR BE LIABLE TO YOU ON ANY LEGAL THEORY FOR ANY SPECIAL, INCIDENTAL, CONSEQUENTIAL, PUNITIVE OR EXEMPLARY DAMAGES ARISING OUT OF THIS LICENSE OR THE USE OF THE WORK, EVEN IF LICENSOR HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. | ||||
|  | ||||
| 7. Termination | ||||
|  | ||||
|      a. This License and the rights granted hereunder will terminate automatically upon any breach by You of the terms of this License. Individuals or entities who have received Adaptations or Collections from You under this License, however, will not have their licenses terminated provided such individuals or entities remain in full compliance with those licenses. Sections 1, 2, 5, 6, 7, and 8 will survive any termination of this License. | ||||
|  | ||||
|      b. Subject to the above terms and conditions, the license granted here is perpetual (for the duration of the applicable copyright in the Work). Notwithstanding the above, Licensor reserves the right to release the Work under different license terms or to stop distributing the Work at any time; provided, however that any such election will not serve to withdraw this License (or any other license that has been, or is required to be, granted under the terms of this License), and this License will continue in full force and effect unless terminated as stated above. | ||||
|  | ||||
| 8. Miscellaneous | ||||
|  | ||||
|      a. Each time You Distribute or Publicly Perform the Work or a Collection, the Licensor offers to the recipient a license to the Work on the same terms and conditions as the license granted to You under this License. | ||||
|  | ||||
|      b. Each time You Distribute or Publicly Perform an Adaptation, Licensor offers to the recipient a license to the original Work on the same terms and conditions as the license granted to You under this License. | ||||
|  | ||||
|      c. If any provision of this License is invalid or unenforceable under applicable law, it shall not affect the validity or enforceability of the remainder of the terms of this License, and without further action by the parties to this agreement, such provision shall be reformed to the minimum extent necessary to make such provision valid and enforceable. | ||||
|  | ||||
|      d. No term or provision of this License shall be deemed waived and no breach consented to unless such waiver or consent shall be in writing and signed by the party to be charged with such waiver or consent. This License constitutes the entire agreement between the parties with respect to the Work licensed here. There are no understandings, agreements or representations with respect to the Work not specified here. Licensor shall not be bound by any additional provisions that may appear in any communication from You. | ||||
|  | ||||
|      e. This License may not be modified without the mutual written agreement of the Licensor and You. | ||||
|  | ||||
|      f. The rights granted under, and the subject matter referenced, in this License were drafted utilizing the terminology of the Berne Convention for the Protection of Literary and Artistic Works (as amended on September 28, 1979), the Rome Convention of 1961, the WIPO Copyright Treaty of 1996, the WIPO Performances and Phonograms Treaty of 1996 and the Universal Copyright Convention (as revised on July 24, 1971). These rights and subject matter take effect in the relevant jurisdiction in which the License terms are sought to be enforced according to the corresponding provisions of the implementation of those treaty provisions in the applicable national law. If the standard suite of rights granted under applicable copyright law includes additional rights not granted under this License, such additional rights are deemed to be included in the License; this License is not intended to restrict the license of any rights under applicable law. | ||||
|  | ||||
| Creative Commons Notice | ||||
|  | ||||
| Creative Commons is not a party to this License, and makes no warranty whatsoever in connection with the Work. Creative Commons will not be liable to You or any party on any legal theory for any damages whatsoever, including without limitation any general, special, incidental or consequential damages arising in connection to this license. Notwithstanding the foregoing two (2) sentences, if Creative Commons has expressly identified itself as the Licensor hereunder, it shall have all rights and obligations of Licensor. | ||||
|  | ||||
| Except for the limited purpose of indicating to the public that the Work is licensed under the CCPL, Creative Commons does not authorize the use by either party of the trademark "Creative Commons" or any related trademark or logo of Creative Commons without the prior written consent of Creative Commons. Any permitted use will be in compliance with Creative Commons' then-current trademark usage guidelines, as may be published on its website or otherwise made available upon request from time to time. For the avoidance of doubt, this trademark restriction does not form part of this License. | ||||
|  | ||||
| Creative Commons may be contacted at http://creativecommons.org/. | ||||
| @@ -1,156 +0,0 @@ | ||||
| Creative Commons Attribution 4.0 International | ||||
|  | ||||
|  Creative Commons Corporation (“Creative Commons”) is not a law firm and does not provide legal services or legal advice. Distribution of Creative Commons public licenses does not create a lawyer-client or other relationship. Creative Commons makes its licenses and related information available on an “as-is” basis. Creative Commons gives no warranties regarding its licenses, any material licensed under their terms and conditions, or any related information. Creative Commons disclaims all liability for damages resulting from their use to the fullest extent possible. | ||||
|  | ||||
| Using Creative Commons Public Licenses | ||||
|  | ||||
| Creative Commons public licenses provide a standard set of terms and conditions that creators and other rights holders may use to share original works of authorship and other material subject to copyright and certain other rights specified in the public license below. The following considerations are for informational purposes only, are not exhaustive, and do not form part of our licenses. | ||||
|  | ||||
| Considerations for licensors: Our public licenses are intended for use by those authorized to give the public permission to use material in ways otherwise restricted by copyright and certain other rights. Our licenses are irrevocable. Licensors should read and understand the terms and conditions of the license they choose before applying it. Licensors should also secure all rights necessary before applying our licenses so that the public can reuse the material as expected. Licensors should clearly mark any material not subject to the license. This includes other CC-licensed material, or material used under an exception or limitation to copyright. More considerations for licensors. | ||||
|  | ||||
| Considerations for the public: By using one of our public licenses, a licensor grants the public permission to use the licensed material under specified terms and conditions. If the licensor’s permission is not necessary for any reason–for example, because of any applicable exception or limitation to copyright–then that use is not regulated by the license. Our licenses grant only permissions under copyright and certain other rights that a licensor has authority to grant. Use of the licensed material may still be restricted for other reasons, including because others have copyright or other rights in the material. A licensor may make special requests, such as asking that all changes be marked or described. Although not required by our licenses, you are encouraged to respect those requests where reasonable. More considerations for the public. | ||||
|  | ||||
| Creative Commons Attribution 4.0 International Public License | ||||
|  | ||||
| By exercising the Licensed Rights (defined below), You accept and agree to be bound by the terms and conditions of this Creative Commons Attribution 4.0 International Public License ("Public License"). To the extent this Public License may be interpreted as a contract, You are granted the Licensed Rights in consideration of Your acceptance of these terms and conditions, and the Licensor grants You such rights in consideration of benefits the Licensor receives from making the Licensed Material available under these terms and conditions. | ||||
|  | ||||
| Section 1 – Definitions. | ||||
|  | ||||
|      a.	Adapted Material means material subject to Copyright and Similar Rights that is derived from or based upon the Licensed Material and in which the Licensed Material is translated, altered, arranged, transformed, or otherwise modified in a manner requiring permission under the Copyright and Similar Rights held by the Licensor. For purposes of this Public License, where the Licensed Material is a musical work, performance, or sound recording, Adapted Material is always produced where the Licensed Material is synched in timed relation with a moving image. | ||||
|  | ||||
|      b.	Adapter's License means the license You apply to Your Copyright and Similar Rights in Your contributions to Adapted Material in accordance with the terms and conditions of this Public License. | ||||
|  | ||||
|      c.	Copyright and Similar Rights means copyright and/or similar rights closely related to copyright including, without limitation, performance, broadcast, sound recording, and Sui Generis Database Rights, without regard to how the rights are labeled or categorized. For purposes of this Public License, the rights specified in Section 2(b)(1)-(2) are not Copyright and Similar Rights. | ||||
|  | ||||
|      d.	Effective Technological Measures means those measures that, in the absence of proper authority, may not be circumvented under laws fulfilling obligations under Article 11 of the WIPO Copyright Treaty adopted on December 20, 1996, and/or similar international agreements. | ||||
|  | ||||
|      e.	Exceptions and Limitations means fair use, fair dealing, and/or any other exception or limitation to Copyright and Similar Rights that applies to Your use of the Licensed Material. | ||||
|  | ||||
|      f.	Licensed Material means the artistic or literary work, database, or other material to which the Licensor applied this Public License. | ||||
|  | ||||
|      g.	Licensed Rights means the rights granted to You subject to the terms and conditions of this Public License, which are limited to all Copyright and Similar Rights that apply to Your use of the Licensed Material and that the Licensor has authority to license. | ||||
|  | ||||
|      h.	Licensor means the individual(s) or entity(ies) granting rights under this Public License. | ||||
|  | ||||
|      i.	Share means to provide material to the public by any means or process that requires permission under the Licensed Rights, such as reproduction, public display, public performance, distribution, dissemination, communication, or importation, and to make material available to the public including in ways that members of the public may access the material from a place and at a time individually chosen by them. | ||||
|  | ||||
|      j.	Sui Generis Database Rights means rights other than copyright resulting from Directive 96/9/EC of the European Parliament and of the Council of 11 March 1996 on the legal protection of databases, as amended and/or succeeded, as well as other essentially equivalent rights anywhere in the world. | ||||
|  | ||||
|      k.	You means the individual or entity exercising the Licensed Rights under this Public License. Your has a corresponding meaning. | ||||
|  | ||||
| Section 2 – Scope. | ||||
|  | ||||
|      a.	License grant. | ||||
|  | ||||
|           1. Subject to the terms and conditions of this Public License, the Licensor hereby grants You a worldwide, royalty-free, non-sublicensable, non-exclusive, irrevocable license to exercise the Licensed Rights in the Licensed Material to: | ||||
|  | ||||
|                A. reproduce and Share the Licensed Material, in whole or in part; and | ||||
|  | ||||
|                B. produce, reproduce, and Share Adapted Material. | ||||
|  | ||||
|           2. Exceptions and Limitations. For the avoidance of doubt, where Exceptions and Limitations apply to Your use, this Public License does not apply, and You do not need to comply with its terms and conditions. | ||||
|  | ||||
|           3. Term. The term of this Public License is specified in Section 6(a). | ||||
|  | ||||
|           4. Media and formats; technical modifications allowed. The Licensor authorizes You to exercise the Licensed Rights in all media and formats whether now known or hereafter created, and to make technical modifications necessary to do so. The Licensor waives and/or agrees not to assert any right or authority to forbid You from making technical modifications necessary to exercise the Licensed Rights, including technical modifications necessary to circumvent Effective Technological Measures. For purposes of this Public License, simply making modifications authorized by this Section 2(a)(4) never produces Adapted Material. | ||||
|  | ||||
|           5. Downstream recipients. | ||||
|  | ||||
|                A. Offer from the Licensor – Licensed Material. Every recipient of the Licensed Material automatically receives an offer from the Licensor to exercise the Licensed Rights under the terms and conditions of this Public License. | ||||
|  | ||||
|                B. No downstream restrictions. You may not offer or impose any additional or different terms or conditions on, or apply any Effective Technological Measures to, the Licensed Material if doing so restricts exercise of the Licensed Rights by any recipient of the Licensed Material. | ||||
|  | ||||
|           6.  No endorsement. Nothing in this Public License constitutes or may be construed as permission to assert or imply that You are, or that Your use of the Licensed Material is, connected with, or sponsored, endorsed, or granted official status by, the Licensor or others designated to receive attribution as provided in Section 3(a)(1)(A)(i). | ||||
|  | ||||
| b. Other rights. | ||||
|  | ||||
|           1. Moral rights, such as the right of integrity, are not licensed under this Public License, nor are publicity, privacy, and/or other similar personality rights; however, to the extent possible, the Licensor waives and/or agrees not to assert any such rights held by the Licensor to the limited extent necessary to allow You to exercise the Licensed Rights, but not otherwise. | ||||
|  | ||||
|           2. Patent and trademark rights are not licensed under this Public License. | ||||
|  | ||||
|           3. To the extent possible, the Licensor waives any right to collect royalties from You for the exercise of the Licensed Rights, whether directly or through a collecting society under any voluntary or waivable statutory or compulsory licensing scheme. In all other cases the Licensor expressly reserves any right to collect such royalties. | ||||
|  | ||||
| Section 3 – License Conditions. | ||||
|  | ||||
| Your exercise of the Licensed Rights is expressly made subject to the following conditions. | ||||
|  | ||||
|      a.	Attribution. | ||||
|  | ||||
|           1. If You Share the Licensed Material (including in modified form), You must: | ||||
|  | ||||
|                A. retain the following if it is supplied by the Licensor with the Licensed Material: | ||||
|  | ||||
|                     i. identification of the creator(s) of the Licensed Material and any others designated to receive attribution, in any reasonable manner requested by the Licensor (including by pseudonym if designated); | ||||
|  | ||||
|                     ii. a copyright notice; | ||||
|  | ||||
|                     iii. a notice that refers to this Public License; | ||||
|  | ||||
|                     iv.	a notice that refers to the disclaimer of warranties; | ||||
|  | ||||
|                     v. a URI or hyperlink to the Licensed Material to the extent reasonably practicable; | ||||
|  | ||||
|                B. indicate if You modified the Licensed Material and retain an indication of any previous modifications; and | ||||
|  | ||||
|                C. indicate the Licensed Material is licensed under this Public License, and include the text of, or the URI or hyperlink to, this Public License. | ||||
|  | ||||
|           2. You may satisfy the conditions in Section 3(a)(1) in any reasonable manner based on the medium, means, and context in which You Share the Licensed Material. For example, it may be reasonable to satisfy the conditions by providing a URI or hyperlink to a resource that includes the required information. | ||||
|  | ||||
|           3. If requested by the Licensor, You must remove any of the information required by Section 3(a)(1)(A) to the extent reasonably practicable. | ||||
|  | ||||
|           4. If You Share Adapted Material You produce, the Adapter's License You apply must not prevent recipients of the Adapted Material from complying with this Public License. | ||||
|  | ||||
| Section 4 – Sui Generis Database Rights. | ||||
|  | ||||
| Where the Licensed Rights include Sui Generis Database Rights that apply to Your use of the Licensed Material: | ||||
|  | ||||
|      a.	for the avoidance of doubt, Section 2(a)(1) grants You the right to extract, reuse, reproduce, and Share all or a substantial portion of the contents of the database; | ||||
|  | ||||
|      b.	if You include all or a substantial portion of the database contents in a database in which You have Sui Generis Database Rights, then the database in which You have Sui Generis Database Rights (but not its individual contents) is Adapted Material; and | ||||
|  | ||||
|      c.	You must comply with the conditions in Section 3(a) if You Share all or a substantial portion of the contents of the database. | ||||
| For the avoidance of doubt, this Section 4 supplements and does not replace Your obligations under this Public License where the Licensed Rights include other Copyright and Similar Rights. | ||||
|  | ||||
| Section 5 – Disclaimer of Warranties and Limitation of Liability. | ||||
|  | ||||
|      a.	Unless otherwise separately undertaken by the Licensor, to the extent possible, the Licensor offers the Licensed Material as-is and as-available, and makes no representations or warranties of any kind concerning the Licensed Material, whether express, implied, statutory, or other. This includes, without limitation, warranties of title, merchantability, fitness for a particular purpose, non-infringement, absence of latent or other defects, accuracy, or the presence or absence of errors, whether or not known or discoverable. Where disclaimers of warranties are not allowed in full or in part, this disclaimer may not apply to You. | ||||
|  | ||||
|      b.	To the extent possible, in no event will the Licensor be liable to You on any legal theory (including, without limitation, negligence) or otherwise for any direct, special, indirect, incidental, consequential, punitive, exemplary, or other losses, costs, expenses, or damages arising out of this Public License or use of the Licensed Material, even if the Licensor has been advised of the possibility of such losses, costs, expenses, or damages. Where a limitation of liability is not allowed in full or in part, this limitation may not apply to You. | ||||
|  | ||||
|      c.	The disclaimer of warranties and limitation of liability provided above shall be interpreted in a manner that, to the extent possible, most closely approximates an absolute disclaimer and waiver of all liability. | ||||
|  | ||||
| Section 6 – Term and Termination. | ||||
|  | ||||
|      a.	This Public License applies for the term of the Copyright and Similar Rights licensed here. However, if You fail to comply with this Public License, then Your rights under this Public License terminate automatically. | ||||
|  | ||||
|      b.	Where Your right to use the Licensed Material has terminated under Section 6(a), it reinstates: | ||||
|  | ||||
|           1. automatically as of the date the violation is cured, provided it is cured within 30 days of Your discovery of the violation; or | ||||
|  | ||||
|           2. upon express reinstatement by the Licensor. | ||||
|  | ||||
|      c.	For the avoidance of doubt, this Section 6(b) does not affect any right the Licensor may have to seek remedies for Your violations of this Public License. | ||||
|  | ||||
|      d.	For the avoidance of doubt, the Licensor may also offer the Licensed Material under separate terms or conditions or stop distributing the Licensed Material at any time; however, doing so will not terminate this Public License. | ||||
|  | ||||
|      e.	Sections 1, 5, 6, 7, and 8 survive termination of this Public License. | ||||
|  | ||||
| Section 7 – Other Terms and Conditions. | ||||
|  | ||||
|      a.	The Licensor shall not be bound by any additional or different terms or conditions communicated by You unless expressly agreed. | ||||
|  | ||||
|      b.	Any arrangements, understandings, or agreements regarding the Licensed Material not stated herein are separate from and independent of the terms and conditions of this Public License. | ||||
|  | ||||
| Section 8 – Interpretation. | ||||
|  | ||||
|      a.	For the avoidance of doubt, this Public License does not, and shall not be interpreted to, reduce, limit, restrict, or impose conditions on any use of the Licensed Material that could lawfully be made without permission under this Public License. | ||||
|  | ||||
|      b.	To the extent possible, if any provision of this Public License is deemed unenforceable, it shall be automatically reformed to the minimum extent necessary to make it enforceable. If the provision cannot be reformed, it shall be severed from this Public License without affecting the enforceability of the remaining terms and conditions. | ||||
|  | ||||
|      c.	No term or condition of this Public License will be waived and no failure to comply consented to unless expressly agreed to by the Licensor. | ||||
|  | ||||
|      d.	Nothing in this Public License constitutes or may be interpreted as a limitation upon, or waiver of, any privileges and immunities that apply to the Licensor or You, including from the legal processes of any jurisdiction or authority. | ||||
|  | ||||
| Creative Commons is not a party to its public licenses. Notwithstanding, Creative Commons may elect to apply one of its public licenses to material it publishes and in those instances will be considered the “Licensor.” Except for the limited purpose of indicating that material is shared under a Creative Commons public license or as otherwise permitted by the Creative Commons policies published at creativecommons.org/policies, Creative Commons does not authorize the use of the trademark “Creative Commons” or any other trademark or logo of Creative Commons without its prior written consent including, without limitation, in connection with any unauthorized modifications to any of its public licenses or any other arrangements, understandings, or agreements concerning use of licensed material. For the avoidance of doubt, this paragraph does not form part of the public licenses. | ||||
|  | ||||
| Creative Commons may be contacted at creativecommons.org. | ||||
| @@ -1,121 +0,0 @@ | ||||
| Creative Commons Legal Code | ||||
|  | ||||
| CC0 1.0 Universal | ||||
|  | ||||
|     CREATIVE COMMONS CORPORATION IS NOT A LAW FIRM AND DOES NOT PROVIDE | ||||
|     LEGAL SERVICES. DISTRIBUTION OF THIS DOCUMENT DOES NOT CREATE AN | ||||
|     ATTORNEY-CLIENT RELATIONSHIP. CREATIVE COMMONS PROVIDES THIS | ||||
|     INFORMATION ON AN "AS-IS" BASIS. CREATIVE COMMONS MAKES NO WARRANTIES | ||||
|     REGARDING THE USE OF THIS DOCUMENT OR THE INFORMATION OR WORKS | ||||
|     PROVIDED HEREUNDER, AND DISCLAIMS LIABILITY FOR DAMAGES RESULTING FROM | ||||
|     THE USE OF THIS DOCUMENT OR THE INFORMATION OR WORKS PROVIDED | ||||
|     HEREUNDER. | ||||
|  | ||||
| Statement of Purpose | ||||
|  | ||||
| The laws of most jurisdictions throughout the world automatically confer | ||||
| exclusive Copyright and Related Rights (defined below) upon the creator | ||||
| and subsequent owner(s) (each and all, an "owner") of an original work of | ||||
| authorship and/or a database (each, a "Work"). | ||||
|  | ||||
| Certain owners wish to permanently relinquish those rights to a Work for | ||||
| the purpose of contributing to a commons of creative, cultural and | ||||
| scientific works ("Commons") that the public can reliably and without fear | ||||
| of later claims of infringement build upon, modify, incorporate in other | ||||
| works, reuse and redistribute as freely as possible in any form whatsoever | ||||
| and for any purposes, including without limitation commercial purposes. | ||||
| These owners may contribute to the Commons to promote the ideal of a free | ||||
| culture and the further production of creative, cultural and scientific | ||||
| works, or to gain reputation or greater distribution for their Work in | ||||
| part through the use and efforts of others. | ||||
|  | ||||
| For these and/or other purposes and motivations, and without any | ||||
| expectation of additional consideration or compensation, the person | ||||
| associating CC0 with a Work (the "Affirmer"), to the extent that he or she | ||||
| is an owner of Copyright and Related Rights in the Work, voluntarily | ||||
| elects to apply CC0 to the Work and publicly distribute the Work under its | ||||
| terms, with knowledge of his or her Copyright and Related Rights in the | ||||
| Work and the meaning and intended legal effect of CC0 on those rights. | ||||
|  | ||||
| 1. Copyright and Related Rights. A Work made available under CC0 may be | ||||
| protected by copyright and related or neighboring rights ("Copyright and | ||||
| Related Rights"). Copyright and Related Rights include, but are not | ||||
| limited to, the following: | ||||
|  | ||||
|   i. the right to reproduce, adapt, distribute, perform, display, | ||||
|      communicate, and translate a Work; | ||||
|  ii. moral rights retained by the original author(s) and/or performer(s); | ||||
| iii. publicity and privacy rights pertaining to a person's image or | ||||
|      likeness depicted in a Work; | ||||
|  iv. rights protecting against unfair competition in regards to a Work, | ||||
|      subject to the limitations in paragraph 4(a), below; | ||||
|   v. rights protecting the extraction, dissemination, use and reuse of data | ||||
|      in a Work; | ||||
|  vi. database rights (such as those arising under Directive 96/9/EC of the | ||||
|      European Parliament and of the Council of 11 March 1996 on the legal | ||||
|      protection of databases, and under any national implementation | ||||
|      thereof, including any amended or successor version of such | ||||
|      directive); and | ||||
| vii. other similar, equivalent or corresponding rights throughout the | ||||
|      world based on applicable law or treaty, and any national | ||||
|      implementations thereof. | ||||
|  | ||||
| 2. Waiver. To the greatest extent permitted by, but not in contravention | ||||
| of, applicable law, Affirmer hereby overtly, fully, permanently, | ||||
| irrevocably and unconditionally waives, abandons, and surrenders all of | ||||
| Affirmer's Copyright and Related Rights and associated claims and causes | ||||
| of action, whether now known or unknown (including existing as well as | ||||
| future claims and causes of action), in the Work (i) in all territories | ||||
| worldwide, (ii) for the maximum duration provided by applicable law or | ||||
| treaty (including future time extensions), (iii) in any current or future | ||||
| medium and for any number of copies, and (iv) for any purpose whatsoever, | ||||
| including without limitation commercial, advertising or promotional | ||||
| purposes (the "Waiver"). Affirmer makes the Waiver for the benefit of each | ||||
| member of the public at large and to the detriment of Affirmer's heirs and | ||||
| successors, fully intending that such Waiver shall not be subject to | ||||
| revocation, rescission, cancellation, termination, or any other legal or | ||||
| equitable action to disrupt the quiet enjoyment of the Work by the public | ||||
| as contemplated by Affirmer's express Statement of Purpose. | ||||
|  | ||||
| 3. Public License Fallback. Should any part of the Waiver for any reason | ||||
| be judged legally invalid or ineffective under applicable law, then the | ||||
| Waiver shall be preserved to the maximum extent permitted taking into | ||||
| account Affirmer's express Statement of Purpose. In addition, to the | ||||
| extent the Waiver is so judged Affirmer hereby grants to each affected | ||||
| person a royalty-free, non transferable, non sublicensable, non exclusive, | ||||
| irrevocable and unconditional license to exercise Affirmer's Copyright and | ||||
| Related Rights in the Work (i) in all territories worldwide, (ii) for the | ||||
| maximum duration provided by applicable law or treaty (including future | ||||
| time extensions), (iii) in any current or future medium and for any number | ||||
| of copies, and (iv) for any purpose whatsoever, including without | ||||
| limitation commercial, advertising or promotional purposes (the | ||||
| "License"). The License shall be deemed effective as of the date CC0 was | ||||
| applied by Affirmer to the Work. Should any part of the License for any | ||||
| reason be judged legally invalid or ineffective under applicable law, such | ||||
| partial invalidity or ineffectiveness shall not invalidate the remainder | ||||
| of the License, and in such case Affirmer hereby affirms that he or she | ||||
| will not (i) exercise any of his or her remaining Copyright and Related | ||||
| Rights in the Work or (ii) assert any associated claims and causes of | ||||
| action with respect to the Work, in either case contrary to Affirmer's | ||||
| express Statement of Purpose. | ||||
|  | ||||
| 4. Limitations and Disclaimers. | ||||
|  | ||||
|  a. No trademark or patent rights held by Affirmer are waived, abandoned, | ||||
|     surrendered, licensed or otherwise affected by this document. | ||||
|  b. Affirmer offers the Work as-is and makes no representations or | ||||
|     warranties of any kind concerning the Work, express, implied, | ||||
|     statutory or otherwise, including without limitation warranties of | ||||
|     title, merchantability, fitness for a particular purpose, non | ||||
|     infringement, or the absence of latent or other defects, accuracy, or | ||||
|     the present or absence of errors, whether or not discoverable, all to | ||||
|     the greatest extent permissible under applicable law. | ||||
|  c. Affirmer disclaims responsibility for clearing rights of other persons | ||||
|     that may apply to the Work or any use thereof, including without | ||||
|     limitation any person's Copyright and Related Rights in the Work. | ||||
|     Further, Affirmer disclaims responsibility for obtaining any necessary | ||||
|     consents, permissions or other rights required for any use of the | ||||
|     Work. | ||||
|  d. Affirmer understands and acknowledges that Creative Commons is not a | ||||
|     party to this document and has no duty or obligation with respect to | ||||
|     this CC0 or use of the Work. | ||||
| @@ -1,9 +0,0 @@ | ||||
| MIT License | ||||
|  | ||||
| Copyright (c) <year> <copyright holders> | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: | ||||
|  | ||||
| The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. | ||||
|  | ||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. | ||||
| @@ -1,373 +0,0 @@ | ||||
| Mozilla Public License Version 2.0 | ||||
| ================================== | ||||
|  | ||||
| 1. Definitions | ||||
| -------------- | ||||
|  | ||||
| 1.1. "Contributor" | ||||
|     means each individual or legal entity that creates, contributes to | ||||
|     the creation of, or owns Covered Software. | ||||
|  | ||||
| 1.2. "Contributor Version" | ||||
|     means the combination of the Contributions of others (if any) used | ||||
|     by a Contributor and that particular Contributor's Contribution. | ||||
|  | ||||
| 1.3. "Contribution" | ||||
|     means Covered Software of a particular Contributor. | ||||
|  | ||||
| 1.4. "Covered Software" | ||||
|     means Source Code Form to which the initial Contributor has attached | ||||
|     the notice in Exhibit A, the Executable Form of such Source Code | ||||
|     Form, and Modifications of such Source Code Form, in each case | ||||
|     including portions thereof. | ||||
|  | ||||
| 1.5. "Incompatible With Secondary Licenses" | ||||
|     means | ||||
|  | ||||
|     (a) that the initial Contributor has attached the notice described | ||||
|         in Exhibit B to the Covered Software; or | ||||
|  | ||||
|     (b) that the Covered Software was made available under the terms of | ||||
|         version 1.1 or earlier of the License, but not also under the | ||||
|         terms of a Secondary License. | ||||
|  | ||||
| 1.6. "Executable Form" | ||||
|     means any form of the work other than Source Code Form. | ||||
|  | ||||
| 1.7. "Larger Work" | ||||
|     means a work that combines Covered Software with other material, in | ||||
|     a separate file or files, that is not Covered Software. | ||||
|  | ||||
| 1.8. "License" | ||||
|     means this document. | ||||
|  | ||||
| 1.9. "Licensable" | ||||
|     means having the right to grant, to the maximum extent possible, | ||||
|     whether at the time of the initial grant or subsequently, any and | ||||
|     all of the rights conveyed by this License. | ||||
|  | ||||
| 1.10. "Modifications" | ||||
|     means any of the following: | ||||
|  | ||||
|     (a) any file in Source Code Form that results from an addition to, | ||||
|         deletion from, or modification of the contents of Covered | ||||
|         Software; or | ||||
|  | ||||
|     (b) any new file in Source Code Form that contains any Covered | ||||
|         Software. | ||||
|  | ||||
| 1.11. "Patent Claims" of a Contributor | ||||
|     means any patent claim(s), including without limitation, method, | ||||
|     process, and apparatus claims, in any patent Licensable by such | ||||
|     Contributor that would be infringed, but for the grant of the | ||||
|     License, by the making, using, selling, offering for sale, having | ||||
|     made, import, or transfer of either its Contributions or its | ||||
|     Contributor Version. | ||||
|  | ||||
| 1.12. "Secondary License" | ||||
|     means either the GNU General Public License, Version 2.0, the GNU | ||||
|     Lesser General Public License, Version 2.1, the GNU Affero General | ||||
|     Public License, Version 3.0, or any later versions of those | ||||
|     licenses. | ||||
|  | ||||
| 1.13. "Source Code Form" | ||||
|     means the form of the work preferred for making modifications. | ||||
|  | ||||
| 1.14. "You" (or "Your") | ||||
|     means an individual or a legal entity exercising rights under this | ||||
|     License. For legal entities, "You" includes any entity that | ||||
|     controls, is controlled by, or is under common control with You. For | ||||
|     purposes of this definition, "control" means (a) the power, direct | ||||
|     or indirect, to cause the direction or management of such entity, | ||||
|     whether by contract or otherwise, or (b) ownership of more than | ||||
|     fifty percent (50%) of the outstanding shares or beneficial | ||||
|     ownership of such entity. | ||||
|  | ||||
| 2. License Grants and Conditions | ||||
| -------------------------------- | ||||
|  | ||||
| 2.1. Grants | ||||
|  | ||||
| Each Contributor hereby grants You a world-wide, royalty-free, | ||||
| non-exclusive license: | ||||
|  | ||||
| (a) under intellectual property rights (other than patent or trademark) | ||||
|     Licensable by such Contributor to use, reproduce, make available, | ||||
|     modify, display, perform, distribute, and otherwise exploit its | ||||
|     Contributions, either on an unmodified basis, with Modifications, or | ||||
|     as part of a Larger Work; and | ||||
|  | ||||
| (b) under Patent Claims of such Contributor to make, use, sell, offer | ||||
|     for sale, have made, import, and otherwise transfer either its | ||||
|     Contributions or its Contributor Version. | ||||
|  | ||||
| 2.2. Effective Date | ||||
|  | ||||
| The licenses granted in Section 2.1 with respect to any Contribution | ||||
| become effective for each Contribution on the date the Contributor first | ||||
| distributes such Contribution. | ||||
|  | ||||
| 2.3. Limitations on Grant Scope | ||||
|  | ||||
| The licenses granted in this Section 2 are the only rights granted under | ||||
| this License. No additional rights or licenses will be implied from the | ||||
| distribution or licensing of Covered Software under this License. | ||||
| Notwithstanding Section 2.1(b) above, no patent license is granted by a | ||||
| Contributor: | ||||
|  | ||||
| (a) for any code that a Contributor has removed from Covered Software; | ||||
|     or | ||||
|  | ||||
| (b) for infringements caused by: (i) Your and any other third party's | ||||
|     modifications of Covered Software, or (ii) the combination of its | ||||
|     Contributions with other software (except as part of its Contributor | ||||
|     Version); or | ||||
|  | ||||
| (c) under Patent Claims infringed by Covered Software in the absence of | ||||
|     its Contributions. | ||||
|  | ||||
| This License does not grant any rights in the trademarks, service marks, | ||||
| or logos of any Contributor (except as may be necessary to comply with | ||||
| the notice requirements in Section 3.4). | ||||
|  | ||||
| 2.4. Subsequent Licenses | ||||
|  | ||||
| No Contributor makes additional grants as a result of Your choice to | ||||
| distribute the Covered Software under a subsequent version of this | ||||
| License (see Section 10.2) or under the terms of a Secondary License (if | ||||
| permitted under the terms of Section 3.3). | ||||
|  | ||||
| 2.5. Representation | ||||
|  | ||||
| Each Contributor represents that the Contributor believes its | ||||
| Contributions are its original creation(s) or it has sufficient rights | ||||
| to grant the rights to its Contributions conveyed by this License. | ||||
|  | ||||
| 2.6. Fair Use | ||||
|  | ||||
| This License is not intended to limit any rights You have under | ||||
| applicable copyright doctrines of fair use, fair dealing, or other | ||||
| equivalents. | ||||
|  | ||||
| 2.7. Conditions | ||||
|  | ||||
| Sections 3.1, 3.2, 3.3, and 3.4 are conditions of the licenses granted | ||||
| in Section 2.1. | ||||
|  | ||||
| 3. Responsibilities | ||||
| ------------------- | ||||
|  | ||||
| 3.1. Distribution of Source Form | ||||
|  | ||||
| All distribution of Covered Software in Source Code Form, including any | ||||
| Modifications that You create or to which You contribute, must be under | ||||
| the terms of this License. You must inform recipients that the Source | ||||
| Code Form of the Covered Software is governed by the terms of this | ||||
| License, and how they can obtain a copy of this License. You may not | ||||
| attempt to alter or restrict the recipients' rights in the Source Code | ||||
| Form. | ||||
|  | ||||
| 3.2. Distribution of Executable Form | ||||
|  | ||||
| If You distribute Covered Software in Executable Form then: | ||||
|  | ||||
| (a) such Covered Software must also be made available in Source Code | ||||
|     Form, as described in Section 3.1, and You must inform recipients of | ||||
|     the Executable Form how they can obtain a copy of such Source Code | ||||
|     Form by reasonable means in a timely manner, at a charge no more | ||||
|     than the cost of distribution to the recipient; and | ||||
|  | ||||
| (b) You may distribute such Executable Form under the terms of this | ||||
|     License, or sublicense it under different terms, provided that the | ||||
|     license for the Executable Form does not attempt to limit or alter | ||||
|     the recipients' rights in the Source Code Form under this License. | ||||
|  | ||||
| 3.3. Distribution of a Larger Work | ||||
|  | ||||
| You may create and distribute a Larger Work under terms of Your choice, | ||||
| provided that You also comply with the requirements of this License for | ||||
| the Covered Software. If the Larger Work is a combination of Covered | ||||
| Software with a work governed by one or more Secondary Licenses, and the | ||||
| Covered Software is not Incompatible With Secondary Licenses, this | ||||
| License permits You to additionally distribute such Covered Software | ||||
| under the terms of such Secondary License(s), so that the recipient of | ||||
| the Larger Work may, at their option, further distribute the Covered | ||||
| Software under the terms of either this License or such Secondary | ||||
| License(s). | ||||
|  | ||||
| 3.4. Notices | ||||
|  | ||||
| You may not remove or alter the substance of any license notices | ||||
| (including copyright notices, patent notices, disclaimers of warranty, | ||||
| or limitations of liability) contained within the Source Code Form of | ||||
| the Covered Software, except that You may alter any license notices to | ||||
| the extent required to remedy known factual inaccuracies. | ||||
|  | ||||
| 3.5. Application of Additional Terms | ||||
|  | ||||
| You may choose to offer, and to charge a fee for, warranty, support, | ||||
| indemnity or liability obligations to one or more recipients of Covered | ||||
| Software. However, You may do so only on Your own behalf, and not on | ||||
| behalf of any Contributor. You must make it absolutely clear that any | ||||
| such warranty, support, indemnity, or liability obligation is offered by | ||||
| You alone, and You hereby agree to indemnify every Contributor for any | ||||
| liability incurred by such Contributor as a result of warranty, support, | ||||
| indemnity or liability terms You offer. You may include additional | ||||
| disclaimers of warranty and limitations of liability specific to any | ||||
| jurisdiction. | ||||
|  | ||||
| 4. Inability to Comply Due to Statute or Regulation | ||||
| --------------------------------------------------- | ||||
|  | ||||
| If it is impossible for You to comply with any of the terms of this | ||||
| License with respect to some or all of the Covered Software due to | ||||
| statute, judicial order, or regulation then You must: (a) comply with | ||||
| the terms of this License to the maximum extent possible; and (b) | ||||
| describe the limitations and the code they affect. Such description must | ||||
| be placed in a text file included with all distributions of the Covered | ||||
| Software under this License. Except to the extent prohibited by statute | ||||
| or regulation, such description must be sufficiently detailed for a | ||||
| recipient of ordinary skill to be able to understand it. | ||||
|  | ||||
| 5. Termination | ||||
| -------------- | ||||
|  | ||||
| 5.1. The rights granted under this License will terminate automatically | ||||
| if You fail to comply with any of its terms. However, if You become | ||||
| compliant, then the rights granted under this License from a particular | ||||
| Contributor are reinstated (a) provisionally, unless and until such | ||||
| Contributor explicitly and finally terminates Your grants, and (b) on an | ||||
| ongoing basis, if such Contributor fails to notify You of the | ||||
| non-compliance by some reasonable means prior to 60 days after You have | ||||
| come back into compliance. Moreover, Your grants from a particular | ||||
| Contributor are reinstated on an ongoing basis if such Contributor | ||||
| notifies You of the non-compliance by some reasonable means, this is the | ||||
| first time You have received notice of non-compliance with this License | ||||
| from such Contributor, and You become compliant prior to 30 days after | ||||
| Your receipt of the notice. | ||||
|  | ||||
| 5.2. If You initiate litigation against any entity by asserting a patent | ||||
| infringement claim (excluding declaratory judgment actions, | ||||
| counter-claims, and cross-claims) alleging that a Contributor Version | ||||
| directly or indirectly infringes any patent, then the rights granted to | ||||
| You by any and all Contributors for the Covered Software under Section | ||||
| 2.1 of this License shall terminate. | ||||
|  | ||||
| 5.3. In the event of termination under Sections 5.1 or 5.2 above, all | ||||
| end user license agreements (excluding distributors and resellers) which | ||||
| have been validly granted by You or Your distributors under this License | ||||
| prior to termination shall survive termination. | ||||
|  | ||||
| ************************************************************************ | ||||
| *                                                                      * | ||||
| *  6. Disclaimer of Warranty                                           * | ||||
| *  -------------------------                                           * | ||||
| *                                                                      * | ||||
| *  Covered Software is provided under this License on an "as is"       * | ||||
| *  basis, without warranty of any kind, either expressed, implied, or  * | ||||
| *  statutory, including, without limitation, warranties that the       * | ||||
| *  Covered Software is free of defects, merchantable, fit for a        * | ||||
| *  particular purpose or non-infringing. The entire risk as to the     * | ||||
| *  quality and performance of the Covered Software is with You.        * | ||||
| *  Should any Covered Software prove defective in any respect, You     * | ||||
| *  (not any Contributor) assume the cost of any necessary servicing,   * | ||||
| *  repair, or correction. This disclaimer of warranty constitutes an   * | ||||
| *  essential part of this License. No use of any Covered Software is   * | ||||
| *  authorized under this License except under this disclaimer.         * | ||||
| *                                                                      * | ||||
| ************************************************************************ | ||||
|  | ||||
| ************************************************************************ | ||||
| *                                                                      * | ||||
| *  7. Limitation of Liability                                          * | ||||
| *  --------------------------                                          * | ||||
| *                                                                      * | ||||
| *  Under no circumstances and under no legal theory, whether tort      * | ||||
| *  (including negligence), contract, or otherwise, shall any           * | ||||
| *  Contributor, or anyone who distributes Covered Software as          * | ||||
| *  permitted above, be liable to You for any direct, indirect,         * | ||||
| *  special, incidental, or consequential damages of any character      * | ||||
| *  including, without limitation, damages for lost profits, loss of    * | ||||
| *  goodwill, work stoppage, computer failure or malfunction, or any    * | ||||
| *  and all other commercial damages or losses, even if such party      * | ||||
| *  shall have been informed of the possibility of such damages. This   * | ||||
| *  limitation of liability shall not apply to liability for death or   * | ||||
| *  personal injury resulting from such party's negligence to the       * | ||||
| *  extent applicable law prohibits such limitation. Some               * | ||||
| *  jurisdictions do not allow the exclusion or limitation of           * | ||||
| *  incidental or consequential damages, so this exclusion and          * | ||||
| *  limitation may not apply to You.                                    * | ||||
| *                                                                      * | ||||
| ************************************************************************ | ||||
|  | ||||
| 8. Litigation | ||||
| ------------- | ||||
|  | ||||
| Any litigation relating to this License may be brought only in the | ||||
| courts of a jurisdiction where the defendant maintains its principal | ||||
| place of business and such litigation shall be governed by laws of that | ||||
| jurisdiction, without reference to its conflict-of-law provisions. | ||||
| Nothing in this Section shall prevent a party's ability to bring | ||||
| cross-claims or counter-claims. | ||||
|  | ||||
| 9. Miscellaneous | ||||
| ---------------- | ||||
|  | ||||
| This License represents the complete agreement concerning the subject | ||||
| matter hereof. If any provision of this License is held to be | ||||
| unenforceable, such provision shall be reformed only to the extent | ||||
| necessary to make it enforceable. Any law or regulation which provides | ||||
| that the language of a contract shall be construed against the drafter | ||||
| shall not be used to construe this License against a Contributor. | ||||
|  | ||||
| 10. Versions of the License | ||||
| --------------------------- | ||||
|  | ||||
| 10.1. New Versions | ||||
|  | ||||
| Mozilla Foundation is the license steward. Except as provided in Section | ||||
| 10.3, no one other than the license steward has the right to modify or | ||||
| publish new versions of this License. Each version will be given a | ||||
| distinguishing version number. | ||||
|  | ||||
| 10.2. Effect of New Versions | ||||
|  | ||||
| You may distribute the Covered Software under the terms of the version | ||||
| of the License under which You originally received the Covered Software, | ||||
| or under the terms of any subsequent version published by the license | ||||
| steward. | ||||
|  | ||||
| 10.3. Modified Versions | ||||
|  | ||||
| If you create software not governed by this License, and you want to | ||||
| create a new license for such software, you may create and use a | ||||
| modified version of this License if you rename the license and remove | ||||
| any references to the name of the license steward (except to note that | ||||
| such modified license differs from this License). | ||||
|  | ||||
| 10.4. Distributing Source Code Form that is Incompatible With Secondary | ||||
| Licenses | ||||
|  | ||||
| If You choose to distribute Source Code Form that is Incompatible With | ||||
| Secondary Licenses under the terms of this version of the License, the | ||||
| notice described in Exhibit B of this License must be attached. | ||||
|  | ||||
| Exhibit A - Source Code Form License Notice | ||||
| ------------------------------------------- | ||||
|  | ||||
|   This Source Code Form is subject to the terms of the Mozilla Public | ||||
|   License, v. 2.0. If a copy of the MPL was not distributed with this | ||||
|   file, You can obtain one at https://mozilla.org/MPL/2.0/. | ||||
|  | ||||
| If it is not possible or desirable to put the notice in a particular | ||||
| file, then You may include the notice in a location (such as a LICENSE | ||||
| file in a relevant directory) where a recipient would be likely to look | ||||
| for such a notice. | ||||
|  | ||||
| You may add additional accurate notices of copyright ownership. | ||||
|  | ||||
| Exhibit B - "Incompatible With Secondary Licenses" Notice | ||||
| --------------------------------------------------------- | ||||
|  | ||||
|   This Source Code Form is "Incompatible With Secondary Licenses", as | ||||
|   defined by the Mozilla Public License, v. 2.0. | ||||
							
								
								
									
										124
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										124
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,76 +1,74 @@ | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2017 The CC: Tweaked Developers | ||||
| <img src="logo.png" alt="CC: Restitched" width="100%"/> | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
| [](https://github.com/cc-tweaked/cc-restitched/actions "Current build status") | ||||
| [](https://www.curseforge.com/minecraft/mc-mods/cc-restitched "Download CC:  Restitched on CurseForge") | ||||
| [](https://gitpod.io/#https://github.com/cc-tweaked/cc-restitched/tree/1.17-alpha) | ||||
|  | ||||
| <picture> | ||||
|   <source media="(prefers-color-scheme: dark)" srcset="./doc/logo-darkmode.png"> | ||||
|   <source media="(prefers-color-scheme: light)" srcset="./doc/logo.png"> | ||||
|   <img alt="CC: Tweaked" src="./doc/logo.png"> | ||||
| </picture> | ||||
| # CC: R Version VS CC: T Version | ||||
| CC: R Strives to maintain perfect pairity with CC: T, however in some cases this is not possible, so CC: R might have a "newer" version than what CC: T has, these newer versions will be primarily bugfixes and the like because [Fabric](https://fabricmc.net/) is "weird" when porting a Forge mod. | ||||
|  | ||||
| [](https://github.com/cc-tweaked/CC-Tweaked/actions "Current build status") | ||||
| [][Modrinth] | ||||
| ## What is CC: Restiched | ||||
| This is an fork of [Zundrel/cc-tweaked-fabric](https://github.com/Zundrel/cc-tweaked-fabric) who's goal was to port [SquidDev-CC/CC-Tweaked](https://github.com/SquidDev-CC/CC-Tweaked) to the [Fabric](https://fabricmc.net/) modloader. I picked up maintaining the mod because the team working on Zundrel's fork can no longer mantain it so, I picked it up to make it as equal as possible with CC: T. | ||||
|  | ||||
| CC: Tweaked is a mod for Minecraft which adds programmable computers, turtles and more to the game. A fork of the | ||||
| much-beloved [ComputerCraft], it continues its legacy with improved performance and stability, along with a wealth of | ||||
| new features. | ||||
| ## Resource Packs | ||||
| This mod includes textures that are more in-line with the style of Mojang's new texture-artist, Jappa. If you prefer the original textures, enable the "Classic" resource pack. | ||||
|  | ||||
| CC: Tweaked can be installed from [Modrinth]. It runs on both [Minecraft Forge] and [Fabric]. | ||||
| <img src="https://raw.githubusercontent.com/3prm3/cc-pack/main/pack.png" alt="CC: Restitched" width="16"  height="16"/> [3prm3/cc-pack](https://github.com/3prm3/cc-pack/) | ||||
| We also have a second resourcepack made by [3prm3](https://github.com/3prm3), it features a complete overhaul and can be enabled by enabling the `overhaul` resource pack, go check out his resource pack over here! | ||||
|  | ||||
| # Does this work with shaders/[Sodium?](https://github.com/CaffeineMC/sodium-fabric) | ||||
| * [ YES ] Sodium | ||||
| * [ YES ] Optifine | ||||
| 	* Works with VBO Rendering (automatically set) | ||||
| 	* No issues | ||||
| * [ EHH ] Iris Shaders | ||||
| 	* "Works" with TBO Rendering (Default) | ||||
| 	* Crashes with VBO Rendering | ||||
| 	* <details> | ||||
| 		<summary>Shaders are broken</summary> | ||||
|  | ||||
| 		* Shaders will cause varrying results ranging from monitors being invisible, to straight up crashing. | ||||
| 		* Not using shaders will result in odd Z-Fighting of the monitor display and the transparent texture | ||||
| 			-  | ||||
| 			-  | ||||
| 		</details> | ||||
| * [ YES ] Canvas | ||||
| 	* Works with TBO Rendering (Default) | ||||
| 	* Scuffed with VBO Rendering | ||||
| 	* <details> | ||||
| 		<summary>VBO is broken</summary> | ||||
|  | ||||
|   		Monitors are just... scuffed beyond belief. | ||||
| 		-  | ||||
| 		-  | ||||
| 			* The content to the left is supposed to be on the monitors to the right, also the bottom one is supposed to `black/white` not colored. | ||||
| 		* Turtle Texture for some reason? | ||||
| 			-  | ||||
| 		</details> | ||||
|  | ||||
| ## Contributing | ||||
| Any contribution is welcome, be that using the mod, reporting bugs or contributing code. If you want to get started | ||||
| developing the mod, [check out the instructions here](CONTRIBUTING.md#developing). | ||||
| Any contribution is welcome, be that using the mod, reporting bugs or contributing code. In order to start helping develop CC: R there are a few rules; | ||||
| 1) Follow the [Fabric](https://fabricmc.net/) programming guidelines as close as possible. This means you have to use [`loom`](https://fabricmc.net/wiki/tutorial:mappings) mappings, if you use anything else, your code will be rejected. | ||||
| 2) You cannot intentionally implement bugs and security vulnerabilities. | ||||
| 3) Unless the code is taken directly from CC: Tweaked, `lua` code is offlimits from alteration. | ||||
|  | ||||
| ## Bleeding Edge Builds | ||||
| Bleeding edge builds can be found [here](https://github.com/cc-tweaked/cc-restitched/actions) at Github Actions. | ||||
|  | ||||
| ## Community | ||||
| If you need help getting started with CC: Tweaked, want to show off your latest project, or just want to chat about | ||||
| ComputerCraft, do check out our [GitHub discussions page][GitHub discussions]! There's also a fairly populated, | ||||
| albeit quiet IRC channel on [EsperNet], if that's more your cup of tea. You can join `#computercraft` through your | ||||
| desktop client, or online using [KiwiIRC]. | ||||
| If you need help getting started with CC: Restitched, want to show off your latest project, or just want to chat about ComputerCraft, here is the [Forum](https://forums.computercraft.cc/) and the [Discord](https://discord.gg/H2UyJXe). | ||||
|  | ||||
| We also host fairly comprehensive documentation at [tweaked.cc](https://tweaked.cc/ "The CC: Tweaked website"). | ||||
| ## Known Bugs/Issues | ||||
| Main Known issue | ||||
| * Mods that add blocks that can be used as peripherals for CC: T on Forge, don't work with CC: R. | ||||
| 	* This is because of the differences between Forge and [Fabric](https://fabricmc.net/), and that mod devs, to my knowledge have not agreed upon a standard method in which to implement cross compatibility between mods, | ||||
| * Occasionally a computer will say "File not found" when running a built in program, this is normal just hold `crtl+r` to reboot it. | ||||
| 	* We do not know what causes it directly, however it happens when the world is `/reload`ed, and currently running programs are not affected so nothing *should break* | ||||
|  | ||||
| ## Using | ||||
| CC: Tweaked is hosted on my maven repo, and so is relatively simple to depend on. You may wish to add a soft (or hard) | ||||
| dependency in your `mods.toml` file, with the appropriate version bounds, to ensure that API functionality you depend | ||||
| on is present. | ||||
| ## Perpherals | ||||
| Unfortunately, unlike the original CC: Tweaked project, CC: Restitched, does not have any actual peripheral mods, currently the only one we have is an example for mod devs to get started with making/adding the peripheral API to their mods! | ||||
|  | ||||
| ```groovy | ||||
| repositories { | ||||
|   maven { | ||||
|     url "https://maven.squiddev.cc" | ||||
|     content { | ||||
|       includeGroup("cc.tweaked") | ||||
|     } | ||||
|   } | ||||
| } | ||||
| If you're a mod dev, made a mod with CC: R peripheral support OR if you're a player who found a mod with CC: R support, please open an [issue here](https://github.com/cc-tweaked/cc-restitched/issues/new?assignees=&labels=peripheralShoutout&template=peripheral_shoutout.md) to let us know so we can add it to the list! | ||||
|  | ||||
| dependencies { | ||||
|   // Vanilla (i.e. for multi-loader systems) | ||||
|   compileOnly("cc.tweaked:cc-tweaked-$mcVersion-common-api:$cctVersion") | ||||
|  | ||||
|   // Forge Gradle | ||||
|   compileOnly("cc.tweaked:cc-tweaked-$mcVersion-forge-api:$cctVersion") | ||||
|   runtimeOnly("cc.tweaked:cc-tweaked-$mcVersion-forge:$cctVersion") | ||||
|  | ||||
|   // Fabric Loom | ||||
|   modCompileOnly("cc.tweaked:cc-tweaked-$mcVersion-fabric-api:$cctVersion") | ||||
|   modRuntimeOnly("cc.tweaked:cc-tweaked-$mcVersion-fabric:$cctVersion") | ||||
| } | ||||
| ``` | ||||
|  | ||||
| You should also be careful to only use classes within the `dan200.computercraft.api` package. Non-API classes are | ||||
| subject to change at any point. If you depend on functionality outside the API (or need to mixin to CC:T), please file | ||||
| an issue to let me know! | ||||
|  | ||||
| We bundle the API sources with the jar, so documentation should be easily viewable within your editor. Alternatively, | ||||
| the generated documentation [can be browsed online](https://tweaked.cc/javadoc/). | ||||
|  | ||||
| [computercraft]: https://github.com/dan200/ComputerCraft "ComputerCraft on GitHub" | ||||
| [modrinth]: https://modrinth.com/mod/gu7yAYhd "Download CC: Tweaked from Modrinth" | ||||
| [Minecraft Forge]: https://files.minecraftforge.net/ "Download Minecraft Forge." | ||||
| [Fabric]: https://fabricmc.net/use/installer/ "Download Fabric." | ||||
| [GitHub Discussions]: https://github.com/cc-tweaked/CC-Tweaked/discussions | ||||
| [EsperNet]: https://www.esper.net/ | ||||
| [KiwiIRC]: https://kiwiirc.com/nextclient/#irc://irc.esper.net:+6697/#computercraft "#computercraft on EsperNet" | ||||
| *  [CC Peripheral Test [1.16.5]](https://github.com/Toad-Dev/cc-peripheral-test) | ||||
| 	* This is an example mod for how to make peripherals that work as a block, or as a turtle upgrade! | ||||
|   | ||||
							
								
								
									
										100
									
								
								REUSE.toml
									
									
									
									
									
								
							
							
						
						
									
										100
									
								
								REUSE.toml
									
									
									
									
									
								
							| @@ -1,100 +0,0 @@ | ||||
| # SPDX-FileCopyrightText: 2017 The CC: Tweaked Developers | ||||
| # | ||||
| # SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| version = 1 | ||||
| SPDX-PackageName = "CC: Tweaked" | ||||
| SPDX-PackageSupplier = "Jonathan Coates <git@squiddev.cc>" | ||||
| SPDX-PackageDownloadLocation = "https://github.com/cc-tweaked/cc-tweaked" | ||||
|  | ||||
| [[annotations]] | ||||
| SPDX-FileCopyrightText = "The CC: Tweaked Developers" | ||||
| SPDX-License-Identifier = "CC0-1.0" | ||||
| path = [ | ||||
|     # Generated/data files are CC0. | ||||
|     "gradle/gradle-daemon-jvm.properties", | ||||
|     "projects/common/src/main/resources/assets/computercraft/sounds.json", | ||||
|     "projects/common/src/main/resources/assets/computercraft/sounds/empty.ogg", | ||||
|     "projects/common/src/testMod/resources/data/cctest/computercraft/turtle_upgrade/**", | ||||
|     "projects/common/src/testMod/resources/data/cctest/structures/**", | ||||
|     "projects/*/src/generated/**", | ||||
|     "projects/web/src/htmlTransform/export/index.json", | ||||
|     "projects/web/src/htmlTransform/export/items/minecraft/**", | ||||
|     # GitHub build scripts are CC0. While we could add a header to each file, | ||||
|     # it's unclear if it will break actions or issue templates in some way. | ||||
|     ".github/**", | ||||
|     # Example mod is CC0. | ||||
|     "projects/*/src/examples/**" | ||||
| ] | ||||
|  | ||||
| [[annotations]] | ||||
| # Several assets where it's inconvenient to create a .license file. | ||||
| SPDX-FileCopyrightText = "The CC: Tweaked Developers" | ||||
| SPDX-License-Identifier = "MPL-2.0" | ||||
| path = [ | ||||
|     "doc/images/**", | ||||
|     "package.json", | ||||
|     "package-lock.json", | ||||
|     "projects/*/src/*/resources/*.mixins.json", | ||||
|     "projects/fabric/src/*/resources/fabric.mod.json", | ||||
|     "projects/common/src/main/resources/assets/minecraft/shaders/core/computercraft/monitor_tbo.json", | ||||
|     "projects/common/src/testMod/resources/data/computercraft/loot_tables/treasure_disk.json", | ||||
|     "projects/common/src/testMod/resources/pack.mcmeta", | ||||
|     "projects/core/src/main/resources/data/computercraft/lua/rom/modules/command/.ignoreme", | ||||
|     "projects/core/src/main/resources/data/computercraft/lua/rom/modules/main/.ignoreme", | ||||
|     "projects/core/src/main/resources/data/computercraft/lua/rom/modules/turtle/.ignoreme", | ||||
|     "projects/core/src/main/resources/data/computercraft/lua/rom/motd.txt", | ||||
|     "projects/web/src/frontend/mount/.settings", | ||||
|     "projects/web/src/frontend/mount/example.nfp", | ||||
|     "projects/web/src/frontend/mount/example.nft", | ||||
|     "projects/web/src/frontend/mount/expr_template.lua", | ||||
|     "projects/web/tsconfig.json", | ||||
| ] | ||||
|  | ||||
| [[annotations]] | ||||
| # Bulk-license original assets as CCPL. | ||||
| SPDX-FileCopyrightText = "2011 Daniel Ratcliffe" | ||||
| SPDX-License-Identifier = "LicenseRef-CCPL" | ||||
| path = [ | ||||
|     "doc/logo.png", | ||||
|     "doc/logo-darkmode.png", | ||||
|     "projects/common/src/main/resources/assets/computercraft/models/**", | ||||
|     "projects/common/src/main/resources/assets/computercraft/textures/**", | ||||
|     "projects/common/src/main/resources/pack.mcmeta", | ||||
|     "projects/common/src/main/resources/pack.png", | ||||
|     "projects/core/src/main/resources/assets/computercraft/textures/gui/term_font.png", | ||||
|     "projects/core/src/main/resources/data/computercraft/lua/rom/autorun/.ignoreme", | ||||
|     "projects/core/src/main/resources/data/computercraft/lua/rom/help/**", | ||||
|     "projects/core/src/main/resources/data/computercraft/lua/rom/programs/fun/advanced/levels/**", | ||||
|     "projects/web/src/htmlTransform/export/items/computercraft/**", | ||||
| ] | ||||
|  | ||||
| [[annotations]] | ||||
| # Community-contributed language files | ||||
| SPDX-FileCopyrightText = "2017 The CC: Tweaked Developers" | ||||
| SPDX-License-Identifier = "LicenseRef-CCPL" | ||||
| path = [ | ||||
|     "projects/common/src/main/resources/assets/computercraft/lang/cs_cz.json", | ||||
|     "projects/common/src/main/resources/assets/computercraft/lang/ko_kr.json", | ||||
|     "projects/common/src/main/resources/assets/computercraft/lang/pl_pl.json", | ||||
|     "projects/common/src/main/resources/assets/computercraft/lang/pt_br.json", | ||||
|     "projects/common/src/main/resources/assets/computercraft/lang/ru_ru.json", | ||||
|     "projects/common/src/main/resources/assets/computercraft/lang/uk_ua.json", | ||||
|     "projects/common/src/main/resources/assets/computercraft/lang/zh_cn.json", | ||||
| ] | ||||
|  | ||||
| [[annotations]] | ||||
| # Community-contributed language files | ||||
| SPDX-FileCopyrightText = "2017 The CC: Tweaked Developers" | ||||
| SPDX-License-Identifier = "MPL-2.0" | ||||
| path = "projects/common/src/main/resources/assets/computercraft/lang/**" | ||||
|  | ||||
| [[annotations]] | ||||
| path = ["gradle/wrapper/**"] | ||||
| SPDX-FileCopyrightText = "Gradle Inc" | ||||
| SPDX-License-Identifier = "Apache-2.0" | ||||
|  | ||||
| [[annotations]] | ||||
| path = "projects/core/src/test/resources/test-rom/data/json-parsing/**" | ||||
| SPDX-FileCopyrightText = "2016 Nicolas Seriot" | ||||
| SPDX-License-Identifier = "MIT" | ||||
							
								
								
									
										159
									
								
								build.gradle
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										159
									
								
								build.gradle
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,159 @@ | ||||
| plugins { | ||||
|     id 'fabric-loom' version '0.9-SNAPSHOT' | ||||
|     id 'maven-publish' | ||||
|     id "checkstyle" | ||||
|     id "com.github.hierynomus.license" version "0.16.1" | ||||
| } | ||||
|  | ||||
| java { | ||||
|     toolchain { | ||||
|         languageVersion = JavaLanguageVersion.of(16) | ||||
|         vendor = JvmVendorSpec.ADOPTOPENJDK | ||||
|     } | ||||
| } | ||||
|  | ||||
| version = mod_version | ||||
|  | ||||
| group = "dan200.computercraft" | ||||
| archivesBaseName = "cc-restiched" | ||||
|  | ||||
| repositories { | ||||
|     mavenCentral() | ||||
|     maven { url 'https://jitpack.io' } | ||||
|     maven { url 'https://api.modrinth.com/maven'} | ||||
|     maven { url "https://maven.shedaniel.me/" } | ||||
|     maven { url "https://maven.terraformersmc.com/" } | ||||
|     maven { | ||||
|         name "SquidDev" | ||||
|         url "https://squiddev.cc/maven" | ||||
|     } | ||||
| } | ||||
|  | ||||
| loom { | ||||
|     accessWidenerPath = file("src/main/resources/cc.accesswidener") | ||||
| } | ||||
|  | ||||
| configurations { | ||||
|     implementation.extendsFrom shade | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     checkstyle 'com.puppycrawl.tools:checkstyle:8.45.1' | ||||
|  | ||||
|     minecraft "com.mojang:minecraft:${mc_version}" | ||||
|     mappings "net.fabricmc:yarn:${mc_version}+build.${mappings_version}:v2" | ||||
|     modImplementation "net.fabricmc:fabric-loader:${fabric_loader_version}" | ||||
|     modImplementation "net.fabricmc.fabric-api:fabric-api:${fabric_api_version}" | ||||
|  | ||||
|     modApi("me.shedaniel.cloth:cloth-config-fabric:${cloth_config_version}") { | ||||
|         exclude(group: "net.fabricmc.fabric-api") | ||||
|     } | ||||
|     modImplementation "com.terraformersmc:modmenu:${modmenu_version}" | ||||
|     modImplementation "me.shedaniel.cloth.api:cloth-utils-v1:${cloth_api_version}" | ||||
|  | ||||
|     implementation 'com.electronwill.night-config:toml:3.6.3' | ||||
|     implementation 'com.google.code.findbugs:jsr305:3.0.2' | ||||
|  | ||||
|     shade 'org.squiddev:Cobalt:0.5.2-SNAPSHOT' | ||||
|  | ||||
|     include "me.shedaniel.cloth.api:cloth-utils-v1:${cloth_api_version}" | ||||
|     include 'com.electronwill.night-config:core:3.6.3' | ||||
|     include 'com.electronwill.night-config:toml:3.6.3' | ||||
|     include "me.shedaniel.cloth:cloth-config-fabric:${cloth_config_version}" | ||||
|  | ||||
|     modRuntime "me.shedaniel:RoughlyEnoughItems-api-fabric:6.0.254-alpha" | ||||
|     modRuntime "me.shedaniel:RoughlyEnoughItems-fabric:6.0.254-alpha" | ||||
| } | ||||
|  | ||||
| processResources { | ||||
|     inputs.property "version", project.version | ||||
|  | ||||
| 	filesMatching("fabric.mod.json") { | ||||
| 		expand "version": project.version | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // ensure that the encoding is set to UTF-8, no matter what the system default is | ||||
| // this fixes some edge cases with special characters not displaying correctly | ||||
| // see http://yodaconditions.net/blog/fix-for-java-file-encoding-problems-with-gradle.html | ||||
| tasks.withType(JavaCompile) { | ||||
|     options.encoding = "UTF-8" | ||||
| } | ||||
|  | ||||
| // Loom will automatically attach sourcesJar to a RemapSourcesJar task and to the "build" task | ||||
| // if it is present. | ||||
| // If you remove this task, sources will not be generated. | ||||
| task sourcesJar(type: Jar, dependsOn: classes) { | ||||
|     classifier = "sources" | ||||
|     from sourceSets.main.allSource | ||||
| } | ||||
|  | ||||
| jar { | ||||
|     from "LICENSE" | ||||
|  | ||||
|     from configurations.shade.collect { it.isDirectory() ? it : zipTree(it) } | ||||
| } | ||||
|  | ||||
|  | ||||
| import com.hierynomus.gradle.license.tasks.LicenseCheck | ||||
| import com.hierynomus.gradle.license.tasks.LicenseFormat | ||||
|  | ||||
| license { | ||||
|     mapping("java", "SLASHSTAR_STYLE") | ||||
|     strictCheck true | ||||
|  | ||||
|     ext.year = Calendar.getInstance().get(Calendar.YEAR) | ||||
| } | ||||
|  | ||||
| [licenseMain, licenseFormatMain].forEach { | ||||
|     it.configure { | ||||
|         include("**/*.java") | ||||
|         exclude("dan200/computercraft/api/**") | ||||
|         header file('config/license/main.txt') | ||||
|     } | ||||
| } | ||||
|  | ||||
| [licenseTest, licenseFormatTest].forEach { | ||||
|     it.configure { | ||||
|         include("**/*.java") | ||||
|         header file('config/license/main.txt') | ||||
|     } | ||||
| } | ||||
|  | ||||
| gradle.projectsEvaluated { | ||||
|     tasks.withType(LicenseFormat) { | ||||
|         outputs.upToDateWhen { false } | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| task licenseAPI(type: LicenseCheck); | ||||
| task licenseFormatAPI(type: LicenseFormat); | ||||
| [licenseAPI, licenseFormatAPI].forEach { | ||||
|     it.configure { | ||||
|         source = sourceSets.main.java | ||||
|         include("dan200/computercraft/api/**") | ||||
|         header file('config/license/api.txt') | ||||
|     } | ||||
| } | ||||
|  | ||||
| // configure the maven publication | ||||
| publishing { | ||||
|     publications { | ||||
|         mavenJava(MavenPublication) { | ||||
|             // add all the jars that should be included when publishing to maven | ||||
|             artifact(remapJar) { | ||||
|                 builtBy remapJar | ||||
|             } | ||||
|             artifact(sourcesJar) { | ||||
|                 builtBy remapSourcesJar | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // select the repositories you want to publish to | ||||
|     repositories { | ||||
|         // uncomment to publish to the local maven | ||||
|         // mavenLocal() | ||||
|     } | ||||
| } | ||||
							
								
								
									
										122
									
								
								build.gradle.kts
									
									
									
									
									
								
							
							
						
						
									
										122
									
								
								build.gradle.kts
									
									
									
									
									
								
							| @@ -1,122 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| import cc.tweaked.gradle.JUnitExt | ||||
| import net.fabricmc.loom.api.LoomGradleExtensionAPI | ||||
| import net.fabricmc.loom.util.gradle.SourceSetHelper | ||||
| import org.jetbrains.gradle.ext.* | ||||
| import org.jetbrains.gradle.ext.Application | ||||
|  | ||||
| plugins { | ||||
|     publishing | ||||
|     alias(libs.plugins.taskTree) | ||||
|     alias(libs.plugins.githubRelease) | ||||
|     alias(libs.plugins.gradleVersions) | ||||
|     alias(libs.plugins.versionCatalogUpdate) | ||||
|     id("org.jetbrains.gradle.plugin.idea-ext") | ||||
|     id("cc-tweaked") | ||||
| } | ||||
|  | ||||
| val isUnstable = project.properties["isUnstable"] == "true" | ||||
| val modVersion: String by extra | ||||
| val mcVersion: String by extra | ||||
|  | ||||
| githubRelease { | ||||
|     token(findProperty("githubApiKey") as String? ?: "") | ||||
|     owner = "cc-tweaked" | ||||
|     repo = "CC-Tweaked" | ||||
|     targetCommitish = cct.gitBranch | ||||
|  | ||||
|     tagName = "v$mcVersion-$modVersion" | ||||
|     releaseName = "[$mcVersion] $modVersion" | ||||
|     body = provider { | ||||
|         "## " + project(":core").file("src/main/resources/data/computercraft/lua/rom/help/whatsnew.md") | ||||
|             .readLines() | ||||
|             .takeWhile { it != "Type \"help changelog\" to see the full version history." } | ||||
|             .joinToString("\n").trim() | ||||
|     } | ||||
|     prerelease = isUnstable | ||||
| } | ||||
|  | ||||
| tasks.publish { dependsOn(tasks.githubRelease) } | ||||
|  | ||||
| idea.project.settings.runConfigurations { | ||||
|     register<JUnitExt>("Core Tests") { | ||||
|         vmParameters = "-ea" | ||||
|         moduleName = "${idea.project.name}.core.test" | ||||
|         packageName = "" | ||||
|     } | ||||
|  | ||||
|     register<JUnitExt>("CraftOS Tests") { | ||||
|         vmParameters = "-ea" | ||||
|         moduleName = "${idea.project.name}.core.test" | ||||
|         className = "dan200.computercraft.core.ComputerTestDelegate" | ||||
|     } | ||||
|  | ||||
|     register<JUnitExt>("CraftOS Tests (Fast)") { | ||||
|         vmParameters = "-ea -Dcc.skip_keywords=slow" | ||||
|         moduleName = "${idea.project.name}.core.test" | ||||
|         className = "dan200.computercraft.core.ComputerTestDelegate" | ||||
|     } | ||||
|  | ||||
|     register<JUnitExt>("Common Tests") { | ||||
|         vmParameters = "-ea" | ||||
|         moduleName = "${idea.project.name}.common.test" | ||||
|         packageName = "" | ||||
|     } | ||||
|  | ||||
|     register<JUnitExt>("Fabric Tests") { | ||||
|         val fabricProject = evaluationDependsOn(":fabric") | ||||
|         val classPathGroup = fabricProject.extensions.getByType<LoomGradleExtensionAPI>().mods | ||||
|             .joinToString(File.pathSeparator + File.pathSeparator) { modSettings -> | ||||
|                 SourceSetHelper.getClasspath(modSettings, project).joinToString(File.pathSeparator) { it.absolutePath } | ||||
|             } | ||||
|  | ||||
|         vmParameters = "-ea -Dfabric.classPathGroups=$classPathGroup" | ||||
|         moduleName = "${idea.project.name}.fabric.test" | ||||
|         packageName = "" | ||||
|     } | ||||
|  | ||||
|     register<JUnitExt>("Forge Tests") { | ||||
|         vmParameters = "-ea" | ||||
|         moduleName = "${idea.project.name}.forge.test" | ||||
|         packageName = "" | ||||
|     } | ||||
|  | ||||
|     register<Application>("Standalone") { | ||||
|         moduleName = "${idea.project.name}.standalone.main" | ||||
|         mainClass = "cc.tweaked.standalone.Main" | ||||
|         programParameters = "--resources=projects/core/src/main/resources --term=80x30 --allow-local-domains" | ||||
|     } | ||||
| } | ||||
|  | ||||
| // Build with the IntelliJ, rather than through Gradle. This may require setting the "Compiler Output" option in | ||||
| // "Project Structure". | ||||
| idea.project.settings.delegateActions { | ||||
|     delegateBuildRunToGradle = false | ||||
|     testRunner = ActionDelegationConfig.TestRunner.PLATFORM | ||||
| } | ||||
|  | ||||
| idea.project.settings.compiler.javac { | ||||
|     // We want ErrorProne to be present when compiling via IntelliJ, as it offers some helpful warnings | ||||
|     // and errors. Loop through our source sets and find the appropriate flags. | ||||
|     moduleJavacAdditionalOptions = subprojects | ||||
|         .asSequence() | ||||
|         .map { evaluationDependsOn(it.path) } | ||||
|         .flatMap { project -> | ||||
|             val sourceSets = project.extensions.findByType(SourceSetContainer::class) ?: return@flatMap sequenceOf() | ||||
|             sourceSets.asSequence().map { sourceSet -> | ||||
|                 val name = "${idea.project.name}.${project.name}.${sourceSet.name}" | ||||
|                 val compile = project.tasks.named(sourceSet.compileJavaTaskName, JavaCompile::class).get() | ||||
|                 name to compile.options.allCompilerArgs.joinToString(" ") { if (it.contains(" ")) "\"$it\"" else it } | ||||
|             } | ||||
|         } | ||||
|         .toMap() | ||||
| } | ||||
|  | ||||
| versionCatalogUpdate { | ||||
|     sortByKey = false | ||||
|     pin { versions.addAll("fastutil", "guava", "netty", "slf4j") } | ||||
|     keep { keepUnusedLibraries = true } | ||||
| } | ||||
| @@ -1,74 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| plugins { | ||||
|     `java-gradle-plugin` | ||||
|     `kotlin-dsl` | ||||
|     alias(libs.plugins.gradleVersions) | ||||
|     alias(libs.plugins.versionCatalogUpdate) | ||||
| } | ||||
|  | ||||
| // Duplicated in settings.gradle.kts | ||||
| repositories { | ||||
|     mavenCentral() | ||||
|     gradlePluginPortal() | ||||
|  | ||||
|     maven("https://maven.neoforged.net") { | ||||
|         name = "NeoForge" | ||||
|         content { | ||||
|             includeGroup("net.neoforged") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     maven("https://maven.fabricmc.net/") { | ||||
|         name = "Fabric" | ||||
|         content { | ||||
|             includeGroup("net.fabricmc") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     maven("https://maven.squiddev.cc") { | ||||
|         name = "SquidDev" | ||||
|         content { | ||||
|             includeGroup("cc.tweaked.vanilla-extract") | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     implementation(libs.errorProne.plugin) | ||||
|     implementation(libs.kotlin.plugin) | ||||
|     implementation(libs.spotless) | ||||
|  | ||||
|     implementation(libs.fabric.loom) | ||||
|     implementation(libs.ideaExt) | ||||
|     implementation(libs.minotaur) | ||||
|     implementation(libs.modDevGradle) | ||||
|     implementation(libs.vanillaExtract) | ||||
| } | ||||
|  | ||||
| gradlePlugin { | ||||
|     plugins { | ||||
|         register("cc-tweaked") { | ||||
|             id = "cc-tweaked" | ||||
|             implementationClass = "cc.tweaked.gradle.CCTweakedPlugin" | ||||
|         } | ||||
|  | ||||
|         register("cc-tweaked.illuaminate") { | ||||
|             id = "cc-tweaked.illuaminate" | ||||
|             implementationClass = "cc.tweaked.gradle.IlluaminatePlugin" | ||||
|         } | ||||
|  | ||||
|         register("cc-tweaked.node") { | ||||
|             id = "cc-tweaked.node" | ||||
|             implementationClass = "cc.tweaked.gradle.NodePlugin" | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| versionCatalogUpdate { | ||||
|     sortByKey = false | ||||
|     keep { keepUnusedLibraries = true } | ||||
|     catalogFile = file("../gradle/libs.versions.toml") | ||||
| } | ||||
| @@ -1,11 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| dependencyResolutionManagement { | ||||
|     versionCatalogs { | ||||
|         create("libs") { | ||||
|             from(files("../gradle/libs.versions.toml")) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,69 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| /** Default configuration for Fabric projects. */ | ||||
|  | ||||
| import cc.tweaked.gradle.CCTweakedExtension | ||||
| import cc.tweaked.gradle.CCTweakedPlugin | ||||
| import cc.tweaked.gradle.IdeaRunConfigurations | ||||
| import cc.tweaked.gradle.MinecraftConfigurations | ||||
|  | ||||
| plugins { | ||||
|     `java-library` | ||||
|     id("fabric-loom") | ||||
|     id("cc-tweaked.java-convention") | ||||
| } | ||||
|  | ||||
| plugins.apply(CCTweakedPlugin::class.java) | ||||
|  | ||||
| val mcVersion: String by extra | ||||
|  | ||||
| repositories { | ||||
|     maven("https://maven.parchmentmc.org/") { | ||||
|         name = "Parchment" | ||||
|         content { | ||||
|             includeGroup("org.parchmentmc.data") | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| loom { | ||||
|     splitEnvironmentSourceSets() | ||||
|     splitModDependencies = true | ||||
| } | ||||
|  | ||||
| MinecraftConfigurations.setup(project) | ||||
|  | ||||
| extensions.configure(CCTweakedExtension::class.java) { | ||||
|     linters(minecraft = true, loader = "fabric") | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     val libs = project.extensions.getByType<VersionCatalogsExtension>().named("libs") | ||||
|  | ||||
|     minecraft("com.mojang:minecraft:$mcVersion") | ||||
|     mappings( | ||||
|         loom.layered { | ||||
|             officialMojangMappings() | ||||
|             parchment( | ||||
|                 project.dependencies.create( | ||||
|                     group = "org.parchmentmc.data", | ||||
|                     name = "parchment-${libs.findVersion("parchmentMc").get()}", | ||||
|                     version = libs.findVersion("parchment").get().toString(), | ||||
|                     ext = "zip", | ||||
|                 ), | ||||
|             ) | ||||
|         }, | ||||
|     ) | ||||
|  | ||||
|     modImplementation(libs.findLibrary("fabric-loader").get()) | ||||
|     modImplementation(libs.findLibrary("fabric-api").get()) | ||||
|  | ||||
|     // Depend on error prone annotations to silence a lot of compile warnings. | ||||
|     compileOnlyApi(libs.findLibrary("errorProne.annotations").get()) | ||||
| } | ||||
|  | ||||
| tasks.ideaSyncTask { | ||||
|     doLast { IdeaRunConfigurations(project).patch() } | ||||
| } | ||||
| @@ -1,35 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| /** Default configuration for Forge projects. */ | ||||
|  | ||||
| import cc.tweaked.gradle.CCTweakedExtension | ||||
| import cc.tweaked.gradle.CCTweakedPlugin | ||||
| import cc.tweaked.gradle.IdeaRunConfigurations | ||||
| import cc.tweaked.gradle.MinecraftConfigurations | ||||
|  | ||||
| plugins { | ||||
|     id("cc-tweaked.java-convention") | ||||
|     id("net.neoforged.moddev") | ||||
| } | ||||
|  | ||||
| plugins.apply(CCTweakedPlugin::class.java) | ||||
|  | ||||
| val mcVersion: String by extra | ||||
|  | ||||
| neoForge { | ||||
|     val libs = project.extensions.getByType<VersionCatalogsExtension>().named("libs") | ||||
|     version = libs.findVersion("neoForge").get().toString() | ||||
|  | ||||
|     parchment { | ||||
|         minecraftVersion = libs.findVersion("parchmentMc").get().toString() | ||||
|         mappingsVersion = libs.findVersion("parchment").get().toString() | ||||
|     } | ||||
| } | ||||
|  | ||||
| MinecraftConfigurations.setup(project) | ||||
|  | ||||
| extensions.configure(CCTweakedExtension::class.java) { | ||||
|     linters(minecraft = true, loader = "forge") | ||||
| } | ||||
| @@ -1,231 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| import cc.tweaked.gradle.CCTweakedExtension | ||||
| import cc.tweaked.gradle.CCTweakedPlugin | ||||
| import com.diffplug.gradle.spotless.FormatExtension | ||||
| import com.diffplug.spotless.LineEnding | ||||
| import net.ltgt.gradle.errorprone.CheckSeverity | ||||
| import net.ltgt.gradle.errorprone.errorprone | ||||
| import java.nio.charset.StandardCharsets | ||||
|  | ||||
| plugins { | ||||
|     `java-library` | ||||
|     idea | ||||
|     jacoco | ||||
|     checkstyle | ||||
|     id("com.diffplug.spotless") | ||||
|     id("net.ltgt.errorprone") | ||||
| } | ||||
|  | ||||
| val modVersion: String by extra | ||||
| val mcVersion: String by extra | ||||
|  | ||||
| group = "cc.tweaked" | ||||
| version = modVersion | ||||
|  | ||||
| base.archivesName.convention("cc-tweaked-$mcVersion-${project.name}") | ||||
|  | ||||
| java { | ||||
|     toolchain { | ||||
|         languageVersion= CCTweakedPlugin.JAVA_VERSION | ||||
|     } | ||||
|  | ||||
|     withSourcesJar() | ||||
| } | ||||
|  | ||||
| repositories { | ||||
|     mavenCentral() | ||||
|  | ||||
|     val mainMaven = maven("https://maven.squiddev.cc/mirror") { | ||||
|         name = "SquidDev" | ||||
|     } | ||||
|  | ||||
|     exclusiveContent { | ||||
|         forRepositories(mainMaven) | ||||
|         filter { | ||||
|             includeGroup("cc.tweaked") | ||||
|             // Things we mirror | ||||
|             includeGroup("com.simibubi.create") | ||||
|             includeGroup("net.commoble.morered") | ||||
|             includeGroup("dev.architectury") | ||||
|             includeGroup("dev.emi") | ||||
|             includeGroup("maven.modrinth") | ||||
|             includeGroup("me.shedaniel.cloth") | ||||
|             includeGroup("me.shedaniel") | ||||
|             includeGroup("mezz.jei") | ||||
|             includeGroup("org.teavm") | ||||
|             includeModule("com.terraformersmc", "modmenu") | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     val libs = project.extensions.getByType<VersionCatalogsExtension>().named("libs") | ||||
|     checkstyle(libs.findLibrary("checkstyle").get()) | ||||
|  | ||||
|     constraints { | ||||
|         checkstyle("org.codehaus.plexus:plexus-container-default:2.1.1") { | ||||
|             because("2.1.0 depends on deprecated Google collections module") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     errorprone(libs.findLibrary("errorProne-core").get()) | ||||
|     errorprone(libs.findLibrary("nullAway").get()) | ||||
| } | ||||
|  | ||||
| // Configure default JavaCompile tasks with our arguments. | ||||
| sourceSets.all { | ||||
|     tasks.named(compileJavaTaskName, JavaCompile::class.java) { | ||||
|  | ||||
|         options.compilerArgs.addAll( | ||||
|             listOf( | ||||
|                 "-Xlint", | ||||
|                 // Processing just gives us "No processor claimed any of these annotations", so skip that! | ||||
|                 "-Xlint:-processing", | ||||
|                 // We violate this pattern too often for it to be a helpful warning. Something to improve one day! | ||||
|                 "-Xlint:-this-escape", | ||||
|             ), | ||||
|         ) | ||||
|  | ||||
|         options.errorprone { | ||||
|             check("InvalidBlockTag", CheckSeverity.OFF) // Broken by @cc.xyz | ||||
|             check("InvalidParam", CheckSeverity.OFF) // Broken by records. | ||||
|             check("InlineMeSuggester", CheckSeverity.OFF) // Minecraft uses @Deprecated liberally | ||||
|             // Too many false positives right now. Maybe we need an indirection for it later on. | ||||
|             check("ReferenceEquality", CheckSeverity.OFF) | ||||
|             check("EnumOrdinal", CheckSeverity.OFF) // For now. We could replace most of these with EnumMap. | ||||
|             check("OperatorPrecedence", CheckSeverity.OFF) // For now. | ||||
|             check("NonOverridingEquals", CheckSeverity.OFF) // Peripheral.equals makes this hard to avoid | ||||
|             check("FutureReturnValueIgnored", CheckSeverity.OFF) // Too many false positives with Netty | ||||
|             check("InvalidInlineTag", CheckSeverity.OFF) // Triggered by @snippet. Can be removed on Java 21. | ||||
|  | ||||
|             check("NullAway", CheckSeverity.ERROR) | ||||
|             option( | ||||
|                 "NullAway:AnnotatedPackages", | ||||
|                 listOf("dan200.computercraft", "cc.tweaked", "net.fabricmc.fabric.api").joinToString(","), | ||||
|             ) | ||||
|             option("NullAway:ExcludedFieldAnnotations", listOf("org.spongepowered.asm.mixin.Shadow").joinToString(",")) | ||||
|             option("NullAway:CastToNonNullMethod", "dan200.computercraft.core.util.Nullability.assertNonNull") | ||||
|             option("NullAway:CheckOptionalEmptiness") | ||||
|             option("NullAway:AcknowledgeRestrictiveAnnotations") | ||||
|  | ||||
|             excludedPaths = ".*/jmh_generated/.*" | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| tasks.compileTestJava { | ||||
|     options.errorprone { | ||||
|         check("NullAway", CheckSeverity.OFF) | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| tasks.withType(JavaCompile::class.java).configureEach { | ||||
|     options.encoding = "UTF-8" | ||||
| } | ||||
|  | ||||
| tasks.processResources { | ||||
|     exclude("**/*.license") | ||||
|     exclude(".cache") | ||||
| } | ||||
|  | ||||
| tasks.withType(AbstractArchiveTask::class.java).configureEach { | ||||
|     isPreserveFileTimestamps = false | ||||
|     isReproducibleFileOrder = true | ||||
|     filePermissions {} | ||||
|     dirPermissions {} | ||||
| } | ||||
|  | ||||
| tasks.jar { | ||||
|     manifest { | ||||
|         attributes( | ||||
|             "Specification-Title" to "computercraft", | ||||
|             "Specification-Vendor" to "SquidDev", | ||||
|             "Specification-Version" to "1", | ||||
|             "Implementation-Title" to "cctweaked-${project.name}", | ||||
|             "Implementation-Version" to modVersion, | ||||
|             "Implementation-Vendor" to "SquidDev", | ||||
|         ) | ||||
|     } | ||||
| } | ||||
|  | ||||
| tasks.javadoc { | ||||
|     options { | ||||
|         val stdOptions = this as StandardJavadocDocletOptions | ||||
|         stdOptions.addBooleanOption("Xdoclint:all,-missing", true) | ||||
|         stdOptions.links("https://docs.oracle.com/en/java/javase/21/docs/api/") | ||||
|     } | ||||
| } | ||||
|  | ||||
| tasks.test { | ||||
|     finalizedBy("jacocoTestReport") | ||||
|  | ||||
|     useJUnitPlatform() | ||||
|     testLogging { | ||||
|         events("skipped", "failed") | ||||
|     } | ||||
| } | ||||
|  | ||||
| tasks.withType(JacocoReport::class.java).configureEach { | ||||
|     reports.xml.required = true | ||||
|     reports.html.required =true | ||||
| } | ||||
|  | ||||
| project.plugins.withType(CCTweakedPlugin::class.java) { | ||||
|     // Set up jacoco to read from /all/ our source directories. | ||||
|     val cct = project.extensions.getByType<CCTweakedExtension>() | ||||
|     project.tasks.named("jacocoTestReport", JacocoReport::class.java) { | ||||
|         for (ref in cct.sourceSets.get()) sourceDirectories.from(ref.allSource.sourceDirectories) | ||||
|     } | ||||
| } | ||||
|  | ||||
| tasks.register("checkstyle") { | ||||
|     description = "Run Checkstyle on all sources" | ||||
|     group = LifecycleBasePlugin.VERIFICATION_GROUP | ||||
|     dependsOn(tasks.withType(Checkstyle::class.java)) | ||||
| } | ||||
|  | ||||
| spotless { | ||||
|     encoding = StandardCharsets.UTF_8 | ||||
|     lineEndings = LineEnding.UNIX | ||||
|  | ||||
|     fun FormatExtension.defaults() { | ||||
|         endWithNewline() | ||||
|         trimTrailingWhitespace() | ||||
|         indentWithSpaces(4) | ||||
|     } | ||||
|  | ||||
|     java { | ||||
|         defaults() | ||||
|         removeUnusedImports() | ||||
|     } | ||||
|  | ||||
|     val ktlintConfig = mapOf( | ||||
|         "ktlint_standard_no-wildcard-imports" to "disabled", | ||||
|         "ktlint_standard_class-naming" to "disabled", | ||||
|         "ktlint_standard_function-naming" to "disabled", | ||||
|         "ij_kotlin_allow_trailing_comma" to "true", | ||||
|         "ij_kotlin_allow_trailing_comma_on_call_site" to "true", | ||||
|     ) | ||||
|  | ||||
|     kotlinGradle { | ||||
|         defaults() | ||||
|         ktlint().editorConfigOverride(ktlintConfig) | ||||
|     } | ||||
|  | ||||
|     kotlin { | ||||
|         defaults() | ||||
|         ktlint().editorConfigOverride(ktlintConfig) | ||||
|     } | ||||
| } | ||||
|  | ||||
| idea.module { | ||||
|     excludeDirs.addAll(project.files("run", "out", "logs").files) | ||||
|  | ||||
|     // Force Gradle to write to inherit the output directory from the parent, instead of writing to out/xxx/classes. | ||||
|     // This is required for Loom, and we patch Forge's run configurations to work there. | ||||
|     inheritOutputDirs = true | ||||
| } | ||||
| @@ -1,39 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2023 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| import cc.tweaked.gradle.setProvider | ||||
|  | ||||
| plugins { | ||||
|     id("com.modrinth.minotaur") | ||||
|     id("cc-tweaked.publishing") | ||||
| } | ||||
|  | ||||
| abstract class ModPublishingExtension { | ||||
|     abstract val output: Property<AbstractArchiveTask> | ||||
|  | ||||
|     init { | ||||
|         output.finalizeValueOnRead() | ||||
|     } | ||||
| } | ||||
|  | ||||
| val modPublishing = project.extensions.create("modPublishing", ModPublishingExtension::class.java) | ||||
|  | ||||
| val isUnstable = project.properties["isUnstable"] == "true" | ||||
| val modVersion: String by extra | ||||
| val mcVersion: String by extra | ||||
|  | ||||
| modrinth { | ||||
|     token = findProperty("modrinthApiKey") as String? ?: "" | ||||
|     projectId = "gu7yAYhd" | ||||
|     versionNumber = modVersion | ||||
|     versionName = modVersion | ||||
|     versionType = if (isUnstable) "alpha" else "release" | ||||
|     uploadFile.setProvider(modPublishing.output) | ||||
|     gameVersions.add(mcVersion) | ||||
|     changelog = "Release notes can be found on the [GitHub repository](https://github.com/cc-tweaked/CC-Tweaked/releases/tag/v$mcVersion-$modVersion)." | ||||
|  | ||||
|     syncBodyFrom = provider { rootProject.file("doc/mod-page.md").readText() } | ||||
| } | ||||
|  | ||||
| tasks.publish { dependsOn(tasks.modrinth) } | ||||
| @@ -1,54 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| /** | ||||
|  * Sets up the configurations for writing game tests. | ||||
|  * | ||||
|  * See notes in [cc.tweaked.gradle.MinecraftConfigurations] for the general design behind these cursed ideas. | ||||
|  */ | ||||
|  | ||||
| import cc.tweaked.gradle.MinecraftConfigurations | ||||
| import cc.tweaked.gradle.clientClasses | ||||
| import cc.tweaked.gradle.commonClasses | ||||
|  | ||||
| plugins { | ||||
|     kotlin("jvm") | ||||
|     id("cc-tweaked.java-convention") | ||||
| } | ||||
|  | ||||
| val main = sourceSets["main"] | ||||
| val client = sourceSets["client"] | ||||
|  | ||||
| MinecraftConfigurations.createDerivedConfiguration(project, MinecraftConfigurations.DATAGEN) | ||||
| MinecraftConfigurations.createDerivedConfiguration(project, MinecraftConfigurations.EXAMPLES) | ||||
| MinecraftConfigurations.createDerivedConfiguration(project, MinecraftConfigurations.TEST_MOD) | ||||
|  | ||||
| // Set up generated resources | ||||
| sourceSets.main { resources.srcDir("src/generated/resources") } | ||||
| sourceSets.named("examples") { resources.srcDir("src/examples/generatedResources") } | ||||
|  | ||||
| // Make sure our examples compile. | ||||
| tasks.check { dependsOn(tasks.named("compileExamplesJava")) } | ||||
|  | ||||
| // Similar to java-test-fixtures, but tries to avoid putting the obfuscated jar on the classpath. | ||||
|  | ||||
| val testFixtures by sourceSets.creating { | ||||
|     compileClasspath += main.compileClasspath + client.compileClasspath | ||||
| } | ||||
|  | ||||
| java.registerFeature("testFixtures") { | ||||
|     usingSourceSet(testFixtures) | ||||
|     disablePublication() | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     val libs = project.extensions.getByType<VersionCatalogsExtension>().named("libs") | ||||
|     add(testFixtures.apiConfigurationName, libs.findBundle("test").get()) | ||||
|     // Consumers of this project already have the common and client classes on the classpath, so it's fine for these | ||||
|     // to be compile-only. | ||||
|     add(testFixtures.compileOnlyApiConfigurationName, commonClasses(project)) | ||||
|     add(testFixtures.compileOnlyApiConfigurationName, clientClasses(project)) | ||||
|  | ||||
|     testImplementation(testFixtures(project)) | ||||
| } | ||||
| @@ -1,48 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| plugins { | ||||
|     `java-library` | ||||
|     `maven-publish` | ||||
| } | ||||
|  | ||||
| publishing { | ||||
|     publications { | ||||
|         register<MavenPublication>("maven") { | ||||
|             artifactId = base.archivesName.get() | ||||
|             from(components["java"]) | ||||
|             suppressAllPomMetadataWarnings() | ||||
|  | ||||
|             pom { | ||||
|                 name = "CC: Tweaked" | ||||
|                 description = "CC: Tweaked is a fork of ComputerCraft, adding programmable computers, turtles and more to Minecraft." | ||||
|                 url = "https://github.com/cc-tweaked/CC-Tweaked" | ||||
|  | ||||
|                 scm { | ||||
|                     url = "https://github.com/cc-tweaked/CC-Tweaked.git" | ||||
|                 } | ||||
|  | ||||
|                 issueManagement { | ||||
|                     system = "github" | ||||
|                     url = "https://github.com/cc-tweaked/CC-Tweaked/issues" | ||||
|                 } | ||||
|  | ||||
|                 licenses { | ||||
|                     license { | ||||
|                         name = "ComputerCraft Public License, Version 1.0" | ||||
|                         url = "https://github.com/cc-tweaked/CC-Tweaked/blob/HEAD/LICENSE" | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     repositories { | ||||
|         maven("https://maven.squiddev.cc") { | ||||
|             name = "SquidDev" | ||||
|  | ||||
|             credentials(PasswordCredentials::class) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,41 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| /** Default configuration for non-modloader-specific Minecraft projects. */ | ||||
|  | ||||
| import cc.tweaked.gradle.CCTweakedExtension | ||||
| import cc.tweaked.gradle.CCTweakedPlugin | ||||
| import cc.tweaked.gradle.MinecraftConfigurations | ||||
|  | ||||
| plugins { | ||||
|     id("cc-tweaked.java-convention") | ||||
|     id("cc.tweaked.vanilla-extract") | ||||
| } | ||||
|  | ||||
| plugins.apply(CCTweakedPlugin::class.java) | ||||
|  | ||||
| val mcVersion: String by extra | ||||
|  | ||||
| val libs = project.extensions.getByType<VersionCatalogsExtension>().named("libs") | ||||
|  | ||||
| minecraft { | ||||
|     version(mcVersion) | ||||
|  | ||||
|     mappings { | ||||
|         parchment(libs.findVersion("parchmentMc").get().toString(), libs.findVersion("parchment").get().toString()) | ||||
|     } | ||||
|  | ||||
|     unpick(libs.findLibrary("yarn").get()) | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     // Depend on error prone annotations to silence a lot of compile warnings. | ||||
|     compileOnly(libs.findLibrary("errorProne.annotations").get()) | ||||
| } | ||||
|  | ||||
| MinecraftConfigurations.setupBasic(project) | ||||
|  | ||||
| extensions.configure(CCTweakedExtension::class.java) { | ||||
|     linters(minecraft = true, loader = null) | ||||
| } | ||||
| @@ -1,254 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import net.ltgt.gradle.errorprone.CheckSeverity | ||||
| import net.ltgt.gradle.errorprone.errorprone | ||||
| import org.gradle.api.GradleException | ||||
| import org.gradle.api.NamedDomainObjectProvider | ||||
| import org.gradle.api.Project | ||||
| import org.gradle.api.Task | ||||
| import org.gradle.api.plugins.JavaPluginExtension | ||||
| import org.gradle.api.provider.Provider | ||||
| import org.gradle.api.provider.SetProperty | ||||
| import org.gradle.api.tasks.SourceSet | ||||
| import org.gradle.api.tasks.bundling.Jar | ||||
| import org.gradle.api.tasks.compile.JavaCompile | ||||
| import org.gradle.api.tasks.javadoc.Javadoc | ||||
| import org.gradle.language.base.plugins.LifecycleBasePlugin | ||||
| import org.gradle.language.jvm.tasks.ProcessResources | ||||
| import org.gradle.process.JavaForkOptions | ||||
| import org.gradle.testing.jacoco.plugins.JacocoPluginExtension | ||||
| import org.gradle.testing.jacoco.plugins.JacocoTaskExtension | ||||
| import org.gradle.testing.jacoco.tasks.JacocoReport | ||||
| import org.jetbrains.kotlin.gradle.dsl.KotlinProjectExtension | ||||
| import org.jetbrains.kotlin.gradle.tasks.KotlinCompile | ||||
| import java.io.File | ||||
| import java.io.IOException | ||||
| import java.net.URI | ||||
| import java.util.regex.Pattern | ||||
|  | ||||
| abstract class CCTweakedExtension(private val project: Project) { | ||||
|     /** Get the hash of the latest git commit. */ | ||||
|     val gitHash: Provider<String> = | ||||
|         gitProvider("<no git commit>", listOf("rev-parse", "HEAD")) { it.trim() } | ||||
|  | ||||
|     /** Get the current git branch. */ | ||||
|     val gitBranch: Provider<String> = | ||||
|         gitProvider("<no git branch>", listOf("rev-parse", "--abbrev-ref", "HEAD")) { it.trim() } | ||||
|  | ||||
|     /** Get a list of all contributors to the project. */ | ||||
|     val gitContributors: Provider<List<String>> = | ||||
|         gitProvider(listOf(), listOf("shortlog", "-ns", "--group=author", "--group=trailer:co-authored-by", "HEAD")) { input -> | ||||
|             input.lineSequence() | ||||
|                 .filter { it.isNotEmpty() } | ||||
|                 .map { | ||||
|                     val matcher = COMMIT_COUNTS.matcher(it) | ||||
|                     matcher.find() | ||||
|                     matcher.group(1) | ||||
|                 } | ||||
|                 .filter { !IGNORED_USERS.contains(it) } | ||||
|                 .toList() | ||||
|                 .sortedWith(String.CASE_INSENSITIVE_ORDER) | ||||
|         } | ||||
|  | ||||
|     /** | ||||
|      * References to other sources | ||||
|      */ | ||||
|     val sourceDirectories: SetProperty<SourceSetReference> = project.objects.setProperty(SourceSetReference::class.java) | ||||
|  | ||||
|     /** All source sets referenced by this project. */ | ||||
|     val sourceSets = sourceDirectories.map { x -> x.map { it.sourceSet } } | ||||
|  | ||||
|     init { | ||||
|         sourceDirectories.finalizeValueOnRead() | ||||
|         project.afterEvaluate { sourceDirectories.disallowChanges() } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Mark this project as consuming another project. Its [sourceDirectories] are added, allowing easier configuration | ||||
|      * of run configurations and other tasks which consume sources/classes. | ||||
|      */ | ||||
|     fun externalSources(project: Project) { | ||||
|         val otherCct = project.extensions.getByType(CCTweakedExtension::class.java) | ||||
|         for (sourceSet in otherCct.sourceDirectories.get()) { | ||||
|             sourceDirectories.add(SourceSetReference(sourceSet.sourceSet, classes = sourceSet.classes, external = true)) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Add a dependency on another project such that its sources and compiles are processed with this one. | ||||
|      * | ||||
|      * This is used when importing a common library into a loader-specific one, as we want to compile sources using | ||||
|      * the loader-specific sources. | ||||
|      */ | ||||
|     fun inlineProject(path: String) { | ||||
|         val otherProject = project.evaluationDependsOn(path) | ||||
|         val otherJava = otherProject.extensions.getByType(JavaPluginExtension::class.java) | ||||
|         val main = otherJava.sourceSets.getByName("main") | ||||
|         val client = otherJava.sourceSets.getByName("client") | ||||
|  | ||||
|         // Pull in sources from the other project. | ||||
|         extendSourceSet(otherProject, main) | ||||
|         extendSourceSet(otherProject, client) | ||||
|         for (sourceSet in listOf(MinecraftConfigurations.DATAGEN, MinecraftConfigurations.EXAMPLES, MinecraftConfigurations.TEST_MOD, "testFixtures")) { | ||||
|             otherJava.sourceSets.findByName(sourceSet)?.let { extendSourceSet(otherProject, it) } | ||||
|         } | ||||
|  | ||||
|         // The extra source-processing tasks should include these files too. | ||||
|         project.tasks.named(main.javadocTaskName, Javadoc::class.java) { source(main.allJava, client.allJava) } | ||||
|         project.tasks.named(main.sourcesJarTaskName, Jar::class.java) { from(main.allSource, client.allSource) } | ||||
|         sourceDirectories.addAll(SourceSetReference.inline(main), SourceSetReference.inline(client)) | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Extend a source set with files from another project. | ||||
|      * | ||||
|      * This actually extends the original compile tasks, as extending the source sets does not play well with IDEs. | ||||
|      */ | ||||
|     private fun extendSourceSet(otherProject: Project, sourceSet: SourceSet) { | ||||
|         project.tasks.named(sourceSet.compileJavaTaskName, JavaCompile::class.java) { | ||||
|             dependsOn(otherProject.tasks.named(sourceSet.compileJavaTaskName)) // Avoid duplicate compile errors | ||||
|             source(sourceSet.allJava) | ||||
|         } | ||||
|  | ||||
|         project.tasks.named(sourceSet.processResourcesTaskName, ProcessResources::class.java) { | ||||
|             from(sourceSet.resources) | ||||
|         } | ||||
|  | ||||
|         // Also try to depend on Kotlin if it exists | ||||
|         val kotlin = otherProject.extensions.findByType(KotlinProjectExtension::class.java) | ||||
|         if (kotlin != null) { | ||||
|             val compileKotlin = sourceSet.getCompileTaskName("kotlin") | ||||
|             project.tasks.named(compileKotlin, KotlinCompile::class.java) { | ||||
|                 dependsOn(otherProject.tasks.named(compileKotlin)) | ||||
|                 source(kotlin.sourceSets.getByName(sourceSet.name).kotlin) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // If we're doing an IDE sync, add a fake dependency to ensure it's on the classpath. | ||||
|         if (isIdeSync) project.dependencies.add(sourceSet.apiConfigurationName, sourceSet.output) | ||||
|     } | ||||
|  | ||||
|     fun linters(@Suppress("UNUSED_PARAMETER") vararg unused: UseNamedArgs, minecraft: Boolean, loader: String?) { | ||||
|         val java = project.extensions.getByType(JavaPluginExtension::class.java) | ||||
|         val sourceSets = java.sourceSets | ||||
|  | ||||
|         project.dependencies.run { add("errorprone", project(mapOf("path" to ":lints"))) } | ||||
|         sourceSets.all { | ||||
|             val name = name | ||||
|             project.tasks.named(compileJavaTaskName, JavaCompile::class.java) { | ||||
|                 options.errorprone { | ||||
|                     // Only the main source set should run the side checker | ||||
|                     check("SideChecker", if (minecraft && name == "main") CheckSeverity.DEFAULT else CheckSeverity.OFF) | ||||
|  | ||||
|                     // The MissingLoaderOverride check superseeds the MissingOverride one, so disable that. | ||||
|                     if (loader != null) { | ||||
|                         check("MissingOverride", CheckSeverity.OFF) | ||||
|                         option("ModLoader", loader) | ||||
|                     } else { | ||||
|                         check("LoaderOverride", CheckSeverity.OFF) | ||||
|                         check("MissingLoaderOverride", CheckSeverity.OFF) | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun <T> jacoco(task: NamedDomainObjectProvider<T>) where T : Task, T : JavaForkOptions { | ||||
|         val reportTaskName = "jacoco${task.name.capitalise()}Report" | ||||
|  | ||||
|         val jacoco = project.extensions.getByType(JacocoPluginExtension::class.java) | ||||
|         task.configure { | ||||
|             finalizedBy(reportTaskName) | ||||
|             jacoco.applyTo(this) | ||||
|  | ||||
|             extensions.configure(JacocoTaskExtension::class.java) { | ||||
|                 includes = listOf("dan200.computercraft.*") | ||||
|                 excludes = listOf( | ||||
|                     "dan200.computercraft.mixin.*", // Exclude mixins, as they're not executed at runtime. | ||||
|                     "dan200.computercraft.shared.Capabilities$*", // Exclude capability tokens, as Forge rewrites them. | ||||
|                 ) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         project.tasks.register(reportTaskName, JacocoReport::class.java) { | ||||
|             group = LifecycleBasePlugin.VERIFICATION_GROUP | ||||
|             description = "Generates code coverage report for the ${task.name} task." | ||||
|  | ||||
|             executionData(task.get()) | ||||
|  | ||||
|             // Don't want to use sourceSets(...) here as we don't use all class directories. | ||||
|             for (ref in this@CCTweakedExtension.sourceDirectories.get()) { | ||||
|                 sourceDirectories.from(ref.sourceSet.allSource.sourceDirectories) | ||||
|                 if (ref.classes) classDirectories.from(ref.sourceSet.output) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Download a file by creating a dummy Ivy repository. | ||||
|      * | ||||
|      * This should only be used for one-off downloads. Using a more conventional Ivy or Maven repository is preferred | ||||
|      * where possible. | ||||
|      */ | ||||
|     fun downloadFile(label: String, url: String): File { | ||||
|         val uri = URI(url) | ||||
|         val path = File(uri.path) | ||||
|  | ||||
|         project.repositories.ivy { | ||||
|             name = label | ||||
|             setUrl(URI(uri.scheme, uri.userInfo, uri.host, uri.port, path.parent, null, null)) | ||||
|             patternLayout { | ||||
|                 artifact("[artifact].[ext]") | ||||
|             } | ||||
|             metadataSources { | ||||
|                 artifact() | ||||
|             } | ||||
|             content { | ||||
|                 includeModule("cc.tweaked.internal", path.nameWithoutExtension) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return project.configurations.detachedConfiguration( | ||||
|             project.dependencies.create( | ||||
|                 mapOf( | ||||
|                     "group" to "cc.tweaked.internal", | ||||
|                     "name" to path.nameWithoutExtension, | ||||
|                     "ext" to path.extension, | ||||
|                 ), | ||||
|             ), | ||||
|         ).resolve().single() | ||||
|     } | ||||
|  | ||||
|     private fun <T> gitProvider(default: T, command: List<String>, process: (String) -> T): Provider<T> { | ||||
|         val baseResult = project.providers.exec { | ||||
|             commandLine = listOf("git", "-C", project.rootDir.absolutePath) + command | ||||
|         } | ||||
|  | ||||
|         return project.provider { | ||||
|             val res = try { | ||||
|                 baseResult.standardOutput.asText.get() | ||||
|             } catch (e: IOException) { | ||||
|                 project.logger.error("Cannot read Git repository: ${e.message}", e) | ||||
|                 return@provider default | ||||
|             } catch (e: GradleException) { | ||||
|                 project.logger.error("Cannot read Git repository: ${e.message}", e) | ||||
|                 return@provider default | ||||
|             } | ||||
|             process(res) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         private val COMMIT_COUNTS = Pattern.compile("""^\s*[0-9]+\s+(.*)$""") | ||||
|         private val IGNORED_USERS = setOf( | ||||
|             "GitHub", "Daniel Ratcliffe", "NotSquidDev", "Weblate", | ||||
|         ) | ||||
|  | ||||
|         private val isIdeSync: Boolean | ||||
|             get() = java.lang.Boolean.parseBoolean(System.getProperty("idea.sync.active", "false")) | ||||
|     } | ||||
| } | ||||
| @@ -1,47 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import org.gradle.api.Plugin | ||||
| import org.gradle.api.Project | ||||
| import org.gradle.api.plugins.JavaPlugin | ||||
| import org.gradle.api.plugins.JavaPluginExtension | ||||
| import org.gradle.jvm.toolchain.JavaLanguageVersion | ||||
| import org.gradle.plugins.ide.idea.model.IdeaModel | ||||
| import org.jetbrains.gradle.ext.IdeaExtPlugin | ||||
| import org.jetbrains.gradle.ext.runConfigurations | ||||
| import org.jetbrains.gradle.ext.settings | ||||
|  | ||||
| /** | ||||
|  * Configures projects to match a shared configuration. | ||||
|  */ | ||||
| class CCTweakedPlugin : Plugin<Project> { | ||||
|     override fun apply(project: Project) { | ||||
|         val cct = project.extensions.create("cct", CCTweakedExtension::class.java) | ||||
|  | ||||
|         project.plugins.withType(JavaPlugin::class.java) { | ||||
|             val sourceSets = project.extensions.getByType(JavaPluginExtension::class.java).sourceSets | ||||
|             cct.sourceDirectories.add(SourceSetReference.internal(sourceSets.getByName("main"))) | ||||
|         } | ||||
|  | ||||
|         project.plugins.withType(IdeaExtPlugin::class.java) { extendIdea(project) } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Extend the [IdeaExtPlugin] plugin's `runConfiguration` container to also support [JUnitExt]. | ||||
|      */ | ||||
|     private fun extendIdea(project: Project) { | ||||
|         val ideaModel = project.extensions.findByName("idea") as IdeaModel? ?: return | ||||
|         val ideaProject = ideaModel.project ?: return | ||||
|  | ||||
|         ideaProject.settings.runConfigurations { | ||||
|             registerFactory(JUnitExt::class.java) { name -> project.objects.newInstance(JUnitExt::class.java, name) } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         val JAVA_VERSION = JavaLanguageVersion.of(21) | ||||
|     } | ||||
| } | ||||
| @@ -1,68 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import org.gradle.api.DefaultTask | ||||
| import org.gradle.api.GradleException | ||||
| import org.gradle.api.file.RegularFileProperty | ||||
| import org.gradle.api.provider.Property | ||||
| import org.gradle.api.tasks.* | ||||
| import org.gradle.language.base.plugins.LifecycleBasePlugin | ||||
|  | ||||
| /** | ||||
|  * Checks the `changelog.md` and `whatsnew.md` files are well-formed. | ||||
|  */ | ||||
| @CacheableTask | ||||
| abstract class CheckChangelog : DefaultTask() { | ||||
|     init { | ||||
|         group = LifecycleBasePlugin.VERIFICATION_GROUP | ||||
|         description = "Verifies the changelog and whatsnew file are consistent." | ||||
|     } | ||||
|  | ||||
|     @get:Input | ||||
|     abstract val version: Property<String> | ||||
|  | ||||
|     @get:InputFile | ||||
|     @get:PathSensitive(PathSensitivity.NONE) | ||||
|     abstract val changelog: RegularFileProperty | ||||
|  | ||||
|     @get:InputFile | ||||
|     @get:PathSensitive(PathSensitivity.NONE) | ||||
|     abstract val whatsNew: RegularFileProperty | ||||
|  | ||||
|     @TaskAction | ||||
|     fun check() { | ||||
|         val version = version.get() | ||||
|  | ||||
|         var ok = true | ||||
|  | ||||
|         // Check we're targeting the current version | ||||
|         var whatsNew = whatsNew.get().asFile.readLines() | ||||
|         if (whatsNew[0] != "New features in CC: Tweaked $version") { | ||||
|             ok = false | ||||
|             logger.error("Expected `whatsnew.md' to target $version.") | ||||
|         } | ||||
|  | ||||
|         // Check "read more" exists and trim it | ||||
|         val idx = whatsNew.indexOfFirst { it == "Type \"help changelog\" to see the full version history." } | ||||
|         if (idx == -1) { | ||||
|             ok = false | ||||
|             logger.error("Must mention the changelog in whatsnew.md") | ||||
|         } else { | ||||
|             whatsNew = whatsNew.slice(0 until idx) | ||||
|         } | ||||
|  | ||||
|         // Check whatsnew and changelog match. | ||||
|         val expectedChangelog = sequenceOf("# ${whatsNew[0]}") + whatsNew.slice(1 until whatsNew.size).asSequence() | ||||
|         val changelog = changelog.get().asFile.readLines() | ||||
|         val mismatch = expectedChangelog.zip(changelog.asSequence()).filter { (a, b) -> a != b }.firstOrNull() | ||||
|         if (mismatch != null) { | ||||
|             ok = false | ||||
|             logger.error("whatsnew and changelog are not in sync") | ||||
|         } | ||||
|  | ||||
|         if (!ok) throw GradleException("Could not check release") | ||||
|     } | ||||
| } | ||||
| @@ -1,98 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2023 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import org.gradle.api.DefaultTask | ||||
| import org.gradle.api.GradleException | ||||
| import org.gradle.api.artifacts.Configuration | ||||
| import org.gradle.api.artifacts.MinimalExternalModuleDependency | ||||
| import org.gradle.api.artifacts.component.ModuleComponentIdentifier | ||||
| import org.gradle.api.artifacts.component.ModuleComponentSelector | ||||
| import org.gradle.api.artifacts.component.ProjectComponentIdentifier | ||||
| import org.gradle.api.artifacts.result.DependencyResult | ||||
| import org.gradle.api.artifacts.result.ResolvedDependencyResult | ||||
| import org.gradle.api.provider.ListProperty | ||||
| import org.gradle.api.provider.MapProperty | ||||
| import org.gradle.api.provider.Provider | ||||
| import org.gradle.api.tasks.Input | ||||
| import org.gradle.api.tasks.TaskAction | ||||
| import org.gradle.language.base.plugins.LifecycleBasePlugin | ||||
|  | ||||
| abstract class DependencyCheck : DefaultTask() { | ||||
|     @get:Input | ||||
|     protected abstract val dependencies: ListProperty<DependencyResult> | ||||
|  | ||||
|     /** | ||||
|      * A mapping of module coordinates (`group:module`) to versions, overriding the requested version. | ||||
|      */ | ||||
|     @get:Input | ||||
|     protected abstract val overrides: MapProperty<String, String> | ||||
|  | ||||
|     init { | ||||
|         description = "Check :core's dependencies are consistent with Minecraft's." | ||||
|         group = LifecycleBasePlugin.VERIFICATION_GROUP | ||||
|  | ||||
|         dependencies.finalizeValueOnRead() | ||||
|         overrides.finalizeValueOnRead() | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Override a module with a different version. | ||||
|      */ | ||||
|     fun override(module: Provider<MinimalExternalModuleDependency>, version: String) { | ||||
|         overrides.putAll(project.provider { mutableMapOf(module.get().module.toString() to version) }) | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Add a configuration to check. | ||||
|      */ | ||||
|     fun configuration(configuration: Provider<Configuration>) { | ||||
|         // We can't store the Configuration in the cache, so store the resolved dependencies instead. | ||||
|         dependencies.addAll(configuration.map { it.incoming.resolutionResult.allDependencies }) | ||||
|     } | ||||
|  | ||||
|     @TaskAction | ||||
|     fun run() { | ||||
|         var ok = true | ||||
|         for (configuration in dependencies.get()) { | ||||
|             if (!check(configuration)) ok = false | ||||
|         } | ||||
|  | ||||
|         if (!ok) { | ||||
|             throw GradleException("Mismatched versions in Minecraft dependencies. gradle/libs.versions.toml may need updating.") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun check(dependency: DependencyResult): Boolean { | ||||
|         if (dependency !is ResolvedDependencyResult) { | ||||
|             logger.warn("Found unexpected dependency result {}", dependency) | ||||
|             return false | ||||
|         } | ||||
|  | ||||
|         // Skip dependencies on non-modules. | ||||
|         val requested = dependency.requested | ||||
|         if (requested !is ModuleComponentSelector) return true | ||||
|  | ||||
|         // If this dependency is specified within some project (so is non-transitive), or is pulled in via Minecraft, | ||||
|         // then check for consistency. | ||||
|         // It would be nice to be smarter about transitive dependencies, but avoiding false positives is hard. | ||||
|         val from = dependency.from.id | ||||
|         if ( | ||||
|             from is ProjectComponentIdentifier || | ||||
|             from is ModuleComponentIdentifier && (from.group == "net.minecraft" || from.group == "io.netty") | ||||
|         ) { | ||||
|             // If the version is different between the requested and selected version, report an error. | ||||
|             val selected = dependency.selected.moduleVersion!!.version | ||||
|             val requestedVersion = overrides.get()["${requested.group}:${requested.module}"] ?: requested.version | ||||
|             if (requestedVersion != selected) { | ||||
|                 logger.error("Requested dependency {} (via {}) but got version {}", requested, from, selected) | ||||
|                 return false | ||||
|             } | ||||
|  | ||||
|             return true | ||||
|         } | ||||
|         return true | ||||
|     } | ||||
| } | ||||
| @@ -1,16 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import org.gradle.api.file.DirectoryProperty | ||||
| import org.gradle.api.provider.Property | ||||
| import org.gradle.api.tasks.AbstractExecTask | ||||
| import org.gradle.api.tasks.OutputDirectory | ||||
| import java.io.File | ||||
|  | ||||
| abstract class ExecToDir : AbstractExecTask<ExecToDir>(ExecToDir::class.java) { | ||||
|     @get:OutputDirectory | ||||
|     abstract val output: DirectoryProperty | ||||
| } | ||||
| @@ -1,185 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import org.gradle.api.artifacts.dsl.DependencyHandler | ||||
| import org.gradle.api.file.FileSystemLocation | ||||
| import org.gradle.api.provider.Property | ||||
| import org.gradle.api.provider.Provider | ||||
| import org.gradle.api.tasks.JavaExec | ||||
| import org.gradle.kotlin.dsl.getByName | ||||
| import org.gradle.process.BaseExecSpec | ||||
| import org.gradle.process.JavaExecSpec | ||||
| import org.gradle.process.ProcessForkOptions | ||||
|  | ||||
| /** | ||||
|  * Add an annotation processor to all source sets. | ||||
|  */ | ||||
| fun DependencyHandler.annotationProcessorEverywhere(dep: Any) { | ||||
|     add("compileOnly", dep) | ||||
|     add("annotationProcessor", dep) | ||||
|  | ||||
|     add("clientCompileOnly", dep) | ||||
|     add("clientAnnotationProcessor", dep) | ||||
|  | ||||
|     add("testCompileOnly", dep) | ||||
|     add("testAnnotationProcessor", dep) | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * A version of [JavaExecSpec.copyTo] which copies *all* properties. | ||||
|  */ | ||||
| fun JavaExec.copyToFull(spec: JavaExec) { | ||||
|     copyTo(spec) | ||||
|  | ||||
|     // Additional Java options | ||||
|     spec.jvmArgs = jvmArgs // Fabric overrides getJvmArgs so copyTo doesn't do the right thing. | ||||
|     spec.args = args | ||||
|     spec.argumentProviders.addAll(argumentProviders) | ||||
|     spec.mainClass.set(mainClass) | ||||
|     spec.classpath = classpath | ||||
|     spec.javaLauncher.set(javaLauncher) | ||||
|     if (executable != null) spec.setExecutable(executable!!) | ||||
|  | ||||
|     // Additional ExecSpec options | ||||
|     copyToExec(spec) | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Base this [JavaExec] task on an existing task. | ||||
|  */ | ||||
| fun JavaExec.copyFromTask(task: JavaExec) { | ||||
|     for (dep in task.dependsOn) dependsOn(dep) | ||||
|     task.copyToFull(this) | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Base this [JavaExec] task on an existing task. | ||||
|  */ | ||||
| fun JavaExec.copyFromTask(task: String) { | ||||
|     copyFromTask(project.tasks.getByName<JavaExec>(task)) | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Copy additional [BaseExecSpec] options which aren't handled by [ProcessForkOptions.copyTo]. | ||||
|  */ | ||||
| fun BaseExecSpec.copyToExec(spec: BaseExecSpec) { | ||||
|     spec.isIgnoreExitValue = isIgnoreExitValue | ||||
|     if (standardInput != null) spec.standardInput = standardInput | ||||
|     if (standardOutput != null) spec.standardOutput = standardOutput | ||||
|     if (errorOutput != null) spec.errorOutput = errorOutput | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * An alternative to [Nothing] with a more descriptive name. Use to enforce calling a function with named arguments: | ||||
|  * | ||||
|  * ```kotlin | ||||
|  * fun f(vararg unused: UseNamedArgs, arg1: Int, arg2: Int) { | ||||
|  *   // ... | ||||
|  * } | ||||
|  * ``` | ||||
|  */ | ||||
| class UseNamedArgs private constructor() | ||||
|  | ||||
| /** | ||||
|  * An [AutoCloseable] implementation which can be used to combine other [AutoCloseable] instances. | ||||
|  * | ||||
|  * Values which implement [AutoCloseable] can be dynamically registered with [CloseScope.add]. When the scope is closed, | ||||
|  * each value is closed in the opposite order. | ||||
|  * | ||||
|  * This is largely intended for cases where it's not appropriate to nest [AutoCloseable.use], for instance when nested | ||||
|  * would be too deep. | ||||
|  */ | ||||
| class CloseScope : AutoCloseable { | ||||
|     private val toClose = ArrayDeque<AutoCloseable>() | ||||
|  | ||||
|     /** | ||||
|      * Add a value to be closed when this scope is closed. | ||||
|      */ | ||||
|     public fun add(value: AutoCloseable) { | ||||
|         toClose.addLast(value) | ||||
|     } | ||||
|  | ||||
|     override fun close() { | ||||
|         close(null) | ||||
|     } | ||||
|  | ||||
|     @PublishedApi | ||||
|     internal fun close(baseException: Throwable?) { | ||||
|         var exception = baseException | ||||
|  | ||||
|         while (true) { | ||||
|             var toClose = toClose.removeLastOrNull() ?: break | ||||
|             try { | ||||
|                 toClose.close() | ||||
|             } catch (e: Throwable) { | ||||
|                 if (exception == null) { | ||||
|                     exception = e | ||||
|                 } else { | ||||
|                     exception.addSuppressed(e) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (exception != null) throw exception | ||||
|     } | ||||
|  | ||||
|     inline fun <R> use(block: (CloseScope) -> R): R { | ||||
|         var exception: Throwable? = null | ||||
|         try { | ||||
|             return block(this) | ||||
|         } catch (e: Throwable) { | ||||
|             exception = e | ||||
|             throw e | ||||
|         } finally { | ||||
|             close(exception) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** Proxy method to avoid overload ambiguity. */ | ||||
| fun <T> Property<T>.setProvider(provider: Provider<out T>) = set(provider) | ||||
|  | ||||
| /** Short-cut method to get the absolute path of a [FileSystemLocation] provider. */ | ||||
| fun Provider<out FileSystemLocation>.getAbsolutePath(): String = get().asFile.absolutePath | ||||
|  | ||||
| /** | ||||
|  * Get the version immediately after the provided version. | ||||
|  * | ||||
|  * For example, given "1.2.3", this will return "1.2.4". | ||||
|  */ | ||||
| fun getNextVersion(version: String): String { | ||||
|     // Split a version like x.y.z-SNAPSHOT into x.y.z and -SNAPSHOT | ||||
|     val dashIndex = version.indexOf('-') | ||||
|     val mainVersion = if (dashIndex < 0) version else version.substring(0, dashIndex) | ||||
|  | ||||
|     // Find the last component in x.y.z and increment it. | ||||
|     val lastIndex = mainVersion.lastIndexOf('.') | ||||
|     if (lastIndex < 0) throw IllegalArgumentException("Cannot parse version format \"$version\"") | ||||
|     val lastVersion = try { | ||||
|         mainVersion.substring(lastIndex + 1).toInt() | ||||
|     } catch (e: NumberFormatException) { | ||||
|         throw IllegalArgumentException("Cannot parse version format \"$version\"", e) | ||||
|     } | ||||
|  | ||||
|     // Then append all components together. | ||||
|     val out = StringBuilder() | ||||
|     out.append(version, 0, lastIndex + 1) | ||||
|     out.append(lastVersion + 1) | ||||
|     if (dashIndex >= 0) out.append(version, dashIndex, version.length) | ||||
|     return out.toString() | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Capitalise the first letter of the string. | ||||
|  * | ||||
|  * This is a replacement for the now deprecated [String.capitalize]. | ||||
|  */ | ||||
| fun String.capitalise(): String { | ||||
|     if (isEmpty()) return this | ||||
|     val first = this[0] | ||||
|     val firstTitle = first.titlecaseChar() | ||||
|     return if (first == firstTitle) this else firstTitle + substring(1) | ||||
| } | ||||
| @@ -1,23 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2023 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import org.jetbrains.gradle.ext.JUnit | ||||
| import javax.inject.Inject | ||||
|  | ||||
| /** | ||||
|  * A version of [JUnit] with a functional [className]. | ||||
|  * | ||||
|  * See [#92](https://github.com/JetBrains/gradle-idea-ext-plugin/issues/92). | ||||
|  */ | ||||
| open class JUnitExt @Inject constructor(nameParam: String) : JUnit(nameParam) { | ||||
|     override fun toMap(): MutableMap<String, *> { | ||||
|         val map = HashMap(super.toMap()) | ||||
|         // Should be "class" instead of "className". | ||||
|         // See https://github.com/JetBrains/intellij-community/blob/9ba394021dc73a3926f13d6d6cdf434f9ee7046d/plugins/junit/src/com/intellij/execution/junit/JUnitRunConfigurationImporter.kt#L39 | ||||
|         map["class"] = className | ||||
|         return map | ||||
|     } | ||||
| } | ||||
| @@ -1,174 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import org.gradle.api.Project | ||||
| import org.gradle.api.logging.Logging | ||||
| import org.w3c.dom.Attr | ||||
| import org.w3c.dom.Document | ||||
| import org.w3c.dom.Node | ||||
| import org.xml.sax.InputSource | ||||
| import java.nio.file.Files | ||||
| import java.nio.file.Path | ||||
| import javax.xml.parsers.DocumentBuilderFactory | ||||
| import javax.xml.transform.TransformerFactory | ||||
| import javax.xml.transform.dom.DOMSource | ||||
| import javax.xml.transform.stream.StreamResult | ||||
| import javax.xml.xpath.XPathConstants | ||||
| import javax.xml.xpath.XPathFactory | ||||
|  | ||||
| /** | ||||
|  * Patches up run configurations from ForgeGradle and Loom. | ||||
|  * | ||||
|  * Would be good to PR some (or all) of these changes upstream at some point. | ||||
|  * | ||||
|  * @see net.fabricmc.loom.configuration.ide.idea.IdeaSyncTask | ||||
|  * @see net.minecraftforge.gradle.common.util.runs.IntellijRunGenerator | ||||
|  */ | ||||
| internal class IdeaRunConfigurations(project: Project) { | ||||
|     private val rootProject = project.rootProject | ||||
|  | ||||
|     private val documentBuilder = DocumentBuilderFactory.newInstance().newDocumentBuilder() | ||||
|     private val xpath = XPathFactory.newInstance().newXPath() | ||||
|     private val writer = TransformerFactory.newInstance().newTransformer() | ||||
|  | ||||
|     private val ideaDir = rootProject.file(".idea/") | ||||
|     private val buildDir: Lazy<String?> = lazy { | ||||
|         val ideaMisc = ideaDir.resolve("misc.xml") | ||||
|  | ||||
|         try { | ||||
|             val doc = Files.newBufferedReader(ideaMisc.toPath()).use { | ||||
|                 documentBuilder.parse(InputSource(it)) | ||||
|             } | ||||
|             val node = | ||||
|                 xpath.evaluate("//component[@name=\"ProjectRootManager\"]/output", doc, XPathConstants.NODE) as Node | ||||
|             val attr = node.attributes.getNamedItem("url") as Attr | ||||
|             attr.value.removePrefix("file://") | ||||
|         } catch (e: Exception) { | ||||
|             LOGGER.error("Failed to find root directory", e) | ||||
|             null | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun patch() = synchronized(LOCK) { | ||||
|         val runConfigDir = ideaDir.resolve("runConfigurations") | ||||
|         if (!runConfigDir.isDirectory) return | ||||
|  | ||||
|         Files.list(runConfigDir.toPath()).use { | ||||
|             for (configuration in it) { | ||||
|                 val filename = configuration.fileName.toString(); | ||||
|                 when { | ||||
|                     filename.endsWith("_fabric.xml") -> patchFabric(configuration) | ||||
|                     filename.startsWith("forge_") && filename.endsWith(".xml") -> patchForge(configuration) | ||||
|                     else -> {} | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun patchFabric(path: Path) = withXml(path) { | ||||
|         setXml("//configuration", "folderName") { "Fabric" } | ||||
|     } | ||||
|  | ||||
|     private fun patchForge(path: Path) = withXml(path) { | ||||
|         val configId = path.fileName.toString().removePrefix("forge_").removeSuffix(".xml") | ||||
|         val sourceSet = forgeConfigs[configId] | ||||
|         if (sourceSet == null) { | ||||
|             LOGGER.error("[{}] Cannot map run configuration to a known source set", path) | ||||
|             return@withXml | ||||
|         } | ||||
|  | ||||
|         setXml("//configuration", "folderName") { "Forge" } | ||||
|         setXml("//configuration/module", "name") { "${rootProject.name}.forge.$sourceSet" } | ||||
|  | ||||
|         if (buildDir.value == null) return@withXml | ||||
|         setXml("//configuration/envs/env[@name=\"MOD_CLASSES\"]", "value") { classpath -> | ||||
|             val classes = classpath!!.split(':') | ||||
|             val newClasses = mutableListOf<String>() | ||||
|             fun appendUnique(x: String) { | ||||
|                 if (!newClasses.contains(x)) newClasses.add(x) | ||||
|             } | ||||
|  | ||||
|             for (entry in classes) { | ||||
|                 if (!entry.contains("/out/")) { | ||||
|                     appendUnique(entry) | ||||
|                     continue | ||||
|                 } | ||||
|  | ||||
|                 val match = CLASSPATH_ENTRY.matchEntire(entry) | ||||
|                 if (match != null) { | ||||
|                     val modId = match.groups["modId"]!!.value | ||||
|                     val proj = match.groups["proj"]!!.value | ||||
|                     var component = match.groups["component"]!!.value | ||||
|                     if (component == "production") component = "main" | ||||
|  | ||||
|                     appendUnique(forgeModEntry(modId, proj, component)) | ||||
|                 } else { | ||||
|                     LOGGER.warn("[{}] Unknown classpath entry {}", path, entry) | ||||
|                     appendUnique(entry) | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             // Ensure common code is on the classpath | ||||
|             for (proj in listOf("common", "common-api")) { | ||||
|                 for (component in listOf("main", "client")) { | ||||
|                     appendUnique(forgeModEntry("computercraft", proj, component)) | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (newClasses.any { it.startsWith("cctest%%") }) { | ||||
|                 appendUnique(forgeModEntry("cctest", "core", "testFixtures")) | ||||
|                 appendUnique(forgeModEntry("cctest", "common", "testFixtures")) | ||||
|                 appendUnique(forgeModEntry("cctest", "common", "testMod")) | ||||
|             } | ||||
|  | ||||
|             newClasses.joinToString(":") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun forgeModEntry(mod: String, project: String, component: String) = | ||||
|         "$mod%%${buildDir.value}/production/${rootProject.name}.$project.$component" | ||||
|  | ||||
|     private fun LocatedDocument.setXml(xpath: String, attribute: String, value: (String?) -> String) { | ||||
|         val node = this@IdeaRunConfigurations.xpath.evaluate(xpath, document, XPathConstants.NODE) as Node? | ||||
|         if (node == null) { | ||||
|             LOGGER.error("[{}] Cannot find {}", path.fileName, xpath) | ||||
|             return | ||||
|         } | ||||
|  | ||||
|         val attr = node.attributes.getNamedItem(attribute) as Attr? ?: document.createAttribute(attribute) | ||||
|         val oldValue = attr.value | ||||
|         attr.value = value(attr.value) | ||||
|         node.attributes.setNamedItem(attr) | ||||
|  | ||||
|         if (oldValue != attr.value) { | ||||
|             LOGGER.info("[{}] Setting {}@{}:\n  Old: {}\n  New: {}", path.fileName, xpath, attribute, oldValue, attr.value) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun withXml(path: Path, run: LocatedDocument.() -> Unit) { | ||||
|         val doc = Files.newBufferedReader(path).use { documentBuilder.parse(InputSource(it)) } | ||||
|         run(LocatedDocument(path, doc)) | ||||
|         Files.newBufferedWriter(path).use { writer.transform(DOMSource(doc), StreamResult(it)) } | ||||
|     } | ||||
|  | ||||
|     private class LocatedDocument(val path: Path, val document: Document) | ||||
|  | ||||
|     companion object { | ||||
|         private val LOGGER = Logging.getLogger(IdeaRunConfigurations::class.java) | ||||
|         private val LOCK = Any() | ||||
|  | ||||
|         private val CLASSPATH_ENTRY = | ||||
|             Regex("(?<modId>[a-z]+)%%\\\$PROJECT_DIR\\\$/projects/(?<proj>[a-z-]+)/out/(?<component>\\w+)/(?<type>[a-z]+)\$") | ||||
|  | ||||
|         private val forgeConfigs = mapOf( | ||||
|             "runClient" to "client", | ||||
|             "runData" to "main", | ||||
|             "runGameTestServer" to "testMod", | ||||
|             "runServer" to "main", | ||||
|             "runTestClient" to "testMod", | ||||
|         ) | ||||
|     } | ||||
| } | ||||
| @@ -1,127 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import org.gradle.api.DefaultTask | ||||
| import org.gradle.api.Plugin | ||||
| import org.gradle.api.Project | ||||
| import org.gradle.api.Task | ||||
| import org.gradle.api.artifacts.Dependency | ||||
| import org.gradle.api.provider.Property | ||||
| import org.gradle.api.provider.Provider | ||||
| import org.gradle.api.tasks.AbstractExecTask | ||||
| import org.gradle.api.tasks.Input | ||||
| import org.gradle.api.tasks.TaskAction | ||||
| import java.io.File | ||||
|  | ||||
| abstract class IlluaminateExtension { | ||||
|     /** The version of illuaminate to use. */ | ||||
|     abstract val version: Property<String> | ||||
|  | ||||
|     /** The path to illuaminate. If not given, illuaminate will be downloaded automatically. */ | ||||
|     abstract val file: Property<File> | ||||
| } | ||||
|  | ||||
| class IlluaminatePlugin : Plugin<Project> { | ||||
|     override fun apply(project: Project) { | ||||
|         val extension = project.extensions.create("illuaminate", IlluaminateExtension::class.java) | ||||
|         extension.file.convention(setupDependency(project, extension.version)) | ||||
|  | ||||
|         project.tasks.register(SetupIlluaminate.NAME, SetupIlluaminate::class.java) { | ||||
|             file.set(extension.file.map { it.absolutePath }) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** Set up a repository for illuaminate and download our binary from it. */ | ||||
|     private fun setupDependency(project: Project, version: Provider<String>): Provider<File> { | ||||
|         project.repositories.ivy { | ||||
|             name = "Illuaminate" | ||||
|             setUrl("https://squiddev.cc/illuaminate/bin/") | ||||
|             patternLayout { | ||||
|                 artifact("[revision]/[artifact]-[ext]") | ||||
|             } | ||||
|             metadataSources { | ||||
|                 artifact() | ||||
|             } | ||||
|             content { | ||||
|                 includeModule("cc.squiddev", "illuaminate") | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return version.map { | ||||
|             val dep = illuaminateArtifact(project, it) | ||||
|             val configuration = project.configurations.detachedConfiguration(dep) | ||||
|             configuration.isTransitive = false | ||||
|             configuration.resolve().single() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** Define a dependency for illuaminate from a version number and the current operating system. */ | ||||
|     private fun illuaminateArtifact(project: Project, version: String): Dependency { | ||||
|         val osName = System.getProperty("os.name").lowercase() | ||||
|         val (os, suffix) = when { | ||||
|             osName.contains("windows") -> Pair("windows", ".exe") | ||||
|             osName.contains("mac os") || osName.contains("darwin") -> Pair("macos", "") | ||||
|             osName.contains("linux") -> Pair("linux", "") | ||||
|             else -> error("Unsupported OS $osName for illuaminate") | ||||
|         } | ||||
|  | ||||
|         val osArch = System.getProperty("os.arch").lowercase() | ||||
|         val arch = when { | ||||
|             // On macOS the x86_64 binary will work for both ARM and Intel Macs through Rosetta. | ||||
|             os == "macos" -> "x86_64" | ||||
|             osArch == "arm" || osArch.startsWith("aarch") -> error("Unsupported architecture '$osArch' for illuaminate") | ||||
|             osArch.contains("64") -> "x86_64" | ||||
|             else -> error("Unsupported architecture '$osArch' for illuaminate") | ||||
|         } | ||||
|  | ||||
|         return project.dependencies.create( | ||||
|             mapOf( | ||||
|                 "group" to "cc.squiddev", | ||||
|                 "name" to "illuaminate", | ||||
|                 "version" to version, | ||||
|                 "ext" to "$os-$arch$suffix", | ||||
|             ), | ||||
|         ) | ||||
|     } | ||||
| } | ||||
|  | ||||
| private val Task.illuaminatePath: String? // "?" needed to avoid overload ambiguity in setExecutable below. | ||||
|     get() = project.extensions.getByType(IlluaminateExtension::class.java).file.get().absolutePath | ||||
|  | ||||
| /** Prepares illuaminate for being run. This simply requests the dependency and then marks it as executable. */ | ||||
| abstract class SetupIlluaminate : DefaultTask() { | ||||
|     @get:Input | ||||
|     abstract val file: Property<String> | ||||
|  | ||||
|     @TaskAction | ||||
|     fun setExecutable() { | ||||
|         val file = File(this.file.get()) | ||||
|         if (file.canExecute()) { | ||||
|             didWork = false | ||||
|             return | ||||
|         } | ||||
|  | ||||
|         file.setExecutable(true) | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         const val NAME: String = "setupIlluaminate" | ||||
|     } | ||||
| } | ||||
|  | ||||
| abstract class IlluaminateExec : AbstractExecTask<IlluaminateExec>(IlluaminateExec::class.java) { | ||||
|     init { | ||||
|         dependsOn(SetupIlluaminate.NAME) | ||||
|         executable = illuaminatePath | ||||
|     } | ||||
| } | ||||
|  | ||||
| abstract class IlluaminateExecToDir : ExecToDir() { | ||||
|     init { | ||||
|         dependsOn(SetupIlluaminate.NAME) | ||||
|         executable = illuaminatePath | ||||
|     } | ||||
| } | ||||
| @@ -1,120 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2024 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import cc.tweaked.vanillaextract.core.util.MoreFiles | ||||
| import org.gradle.api.Action | ||||
| import org.gradle.api.DefaultTask | ||||
| import org.gradle.api.GradleException | ||||
| import org.gradle.api.file.* | ||||
| import org.gradle.api.model.ObjectFactory | ||||
| import org.gradle.api.provider.ListProperty | ||||
| import org.gradle.api.tasks.* | ||||
| import javax.inject.Inject | ||||
|  | ||||
| /** | ||||
|  * Merge common files across multiple directories into one destination directory. | ||||
|  * | ||||
|  * This is intended for merging the generated resources from the Forge and Fabric projects. Files common between the two | ||||
|  * are written to the global [output] directory, while distinct files are written to the per-source | ||||
|  * [MergeTrees.Source.output] directory. | ||||
|  */ | ||||
| abstract class MergeTrees : DefaultTask() { | ||||
|     /** | ||||
|      * A source directory to read from. | ||||
|      */ | ||||
|     interface Source { | ||||
|         /** | ||||
|          * The folder contianing all input files. | ||||
|          */ | ||||
|         @get:InputFiles | ||||
|         @get:PathSensitive(PathSensitivity.RELATIVE) | ||||
|         val input: ConfigurableFileTree | ||||
|  | ||||
|         fun input(configure: Action<ConfigurableFileTree>) { | ||||
|             configure.execute(input) | ||||
|         } | ||||
|  | ||||
|         /** | ||||
|          * The folder to write files unique to this folder to. | ||||
|          */ | ||||
|         @get:OutputDirectory | ||||
|         val output: DirectoryProperty | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * The list of sources. | ||||
|      */ | ||||
|     @get:Nested | ||||
|     abstract val sources: ListProperty<Source> | ||||
|  | ||||
|     /** | ||||
|      * Add and configure a new source. | ||||
|      */ | ||||
|     fun source(configure: Action<Source>) { | ||||
|         val instance = objectFactory.newInstance(Source::class.java) | ||||
|         configure.execute(instance) | ||||
|         instance.output.disallowChanges() | ||||
|         sources.add(instance) | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * The directory to write common files to. | ||||
|      */ | ||||
|     @get:OutputDirectory | ||||
|     abstract val output: DirectoryProperty | ||||
|  | ||||
|     @get:Inject | ||||
|     protected abstract val objectFactory: ObjectFactory | ||||
|  | ||||
|     @get:Inject | ||||
|     protected abstract val fsOperations: FileSystemOperations | ||||
|  | ||||
|     @TaskAction | ||||
|     fun run() { | ||||
|         val sources = this.sources.get() | ||||
|         if (sources.isEmpty()) throw GradleException("Cannot have an empty list of sources") | ||||
|  | ||||
|         val files = mutableMapOf<String, SharedFile>() | ||||
|         for (source in sources) { | ||||
|             source.input.visit( | ||||
|                 object : FileVisitor { | ||||
|                     override fun visitDir(dirDetails: FileVisitDetails) = Unit | ||||
|                     override fun visitFile(fileDetails: FileVisitDetails) { | ||||
|                         val path = fileDetails.file.toRelativeString(source.input.dir) | ||||
|                         val hash = MoreFiles.computeSha1(fileDetails.file.toPath()) | ||||
|  | ||||
|                         val existing = files[path] | ||||
|                         if (existing == null) { | ||||
|                             files[path] = SharedFile(hash, 1) | ||||
|                         } else if (existing.hash == hash) { | ||||
|                             existing.found++ | ||||
|                         } | ||||
|                     } | ||||
|                 }, | ||||
|             ) | ||||
|         } | ||||
|  | ||||
|         val sharedFiles = files.entries.asSequence().filter { (_, v) -> v.found == sources.size }.map { (k, _) -> k }.toList() | ||||
|  | ||||
|         // Copy shared files to the common directory | ||||
|         fsOperations.sync { | ||||
|             from(sources[0].input) | ||||
|             into(output) | ||||
|             include(sharedFiles) | ||||
|         } | ||||
|  | ||||
|         // And all other files to their per-source directory | ||||
|         for (source in sources) { | ||||
|             fsOperations.sync { | ||||
|                 from(source.input) | ||||
|                 into(source.output) | ||||
|                 exclude(sharedFiles) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     class SharedFile(val hash: String, var found: Int) | ||||
| } | ||||
| @@ -1,144 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import cc.tweaked.vanillaextract.configurations.Capabilities | ||||
| import cc.tweaked.vanillaextract.configurations.MinecraftSetup | ||||
| import org.gradle.api.Project | ||||
| import org.gradle.api.artifacts.ModuleDependency | ||||
| import org.gradle.api.artifacts.dsl.DependencyHandler | ||||
| import org.gradle.api.plugins.BasePlugin | ||||
| import org.gradle.api.plugins.JavaPluginExtension | ||||
| import org.gradle.api.tasks.SourceSet | ||||
| import org.gradle.api.tasks.bundling.Jar | ||||
| import org.gradle.api.tasks.javadoc.Javadoc | ||||
| import org.gradle.kotlin.dsl.get | ||||
|  | ||||
| /** | ||||
|  * This sets up a separate client-only source set, and extends that and the main/common source set with additional | ||||
|  * metadata, to make it easier to consume jars downstream. | ||||
|  */ | ||||
| class MinecraftConfigurations private constructor(private val project: Project) { | ||||
|     private val java = project.extensions.getByType(JavaPluginExtension::class.java) | ||||
|     private val sourceSets = java.sourceSets | ||||
|     private val configurations = project.configurations | ||||
|  | ||||
|     private val main = sourceSets[SourceSet.MAIN_SOURCE_SET_NAME] | ||||
|     private val test = sourceSets[SourceSet.TEST_SOURCE_SET_NAME] | ||||
|  | ||||
|     /** | ||||
|      * Performs the initial setup of our configurations. | ||||
|      */ | ||||
|     private fun setup() { | ||||
|         // Define a client source set. | ||||
|         val client = sourceSets.maybeCreate("client") | ||||
|  | ||||
|         // Ensure the client classpaths behave the same as the main ones. | ||||
|         consistentWithMain(client) | ||||
|  | ||||
|         // Set up an API configuration for clients (to ensure it's consistent with the main source set). | ||||
|         val clientApi = configurations.maybeCreate(client.apiConfigurationName).apply { | ||||
|             isVisible = false | ||||
|             isCanBeConsumed = false | ||||
|             isCanBeResolved = false | ||||
|         } | ||||
|         configurations.named(client.implementationConfigurationName) { extendsFrom(clientApi) } | ||||
|  | ||||
|         project.tasks.register(client.jarTaskName, Jar::class.java) { | ||||
|             description = "An empty jar standing in for the client classes." | ||||
|             group = BasePlugin.BUILD_GROUP | ||||
|             archiveClassifier.set("client") | ||||
|         } | ||||
|  | ||||
|         MinecraftSetup(project).setupOutgoingConfigurations() | ||||
|  | ||||
|         // Reset the client classpath (Loom configures it slightly differently to this) and add a main -> client | ||||
|         // dependency. Here we /can/ use source set outputs as we add transitive deps by patching the classpath. Nasty, | ||||
|         // but avoids accidentally pulling in Forge's obfuscated jar. | ||||
|         client.compileClasspath = client.compileClasspath + main.compileClasspath | ||||
|         client.runtimeClasspath = client.runtimeClasspath + main.runtimeClasspath | ||||
|         project.dependencies.add(client.apiConfigurationName, main.output) | ||||
|  | ||||
|         // Also add client classes to the test classpath. We do the same nasty tricks as needed for main -> client. | ||||
|         test.compileClasspath += client.compileClasspath | ||||
|         test.runtimeClasspath += client.runtimeClasspath | ||||
|         project.dependencies.add(test.implementationConfigurationName, client.output) | ||||
|  | ||||
|         // Configure some tasks to include our additional files. | ||||
|         project.tasks.named("javadoc", Javadoc::class.java) { | ||||
|             source(client.allJava) | ||||
|             classpath = main.compileClasspath + main.output + client.compileClasspath + client.output | ||||
|         } | ||||
|         // This are already done by Fabric, but we need it for Forge and vanilla. It shouldn't conflict at all. | ||||
|         project.tasks.named("jar", Jar::class.java) { from(client.output) } | ||||
|         project.tasks.named("sourcesJar", Jar::class.java) { from(client.allSource) } | ||||
|  | ||||
|         setupBasic() | ||||
|     } | ||||
|  | ||||
|     private fun consistentWithMain(sourceSet: SourceSet) { | ||||
|         configurations.named(sourceSet.compileClasspathConfigurationName) { | ||||
|             shouldResolveConsistentlyWith(configurations[main.compileClasspathConfigurationName]) | ||||
|         } | ||||
|  | ||||
|         configurations.named(sourceSet.runtimeClasspathConfigurationName) { | ||||
|             shouldResolveConsistentlyWith(configurations[main.runtimeClasspathConfigurationName]) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun setupBasic() { | ||||
|         val client = sourceSets["client"] | ||||
|  | ||||
|         project.extensions.configure(CCTweakedExtension::class.java) { | ||||
|             sourceDirectories.add(SourceSetReference.internal(client)) | ||||
|         } | ||||
|  | ||||
|         // Register a task to check there are no conflicts with the core project. | ||||
|         val checkDependencyConsistency = | ||||
|             project.tasks.register("checkDependencyConsistency", DependencyCheck::class.java) { | ||||
|                 // We need to check both the main and client classpath *configurations*, as the actual configuration | ||||
|                 configuration(configurations.named(main.runtimeClasspathConfigurationName)) | ||||
|                 configuration(configurations.named(client.runtimeClasspathConfigurationName)) | ||||
|             } | ||||
|         project.tasks.named("check") { dependsOn(checkDependencyConsistency) } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Create a new configuration that pulls in the main and client classes from the mod. | ||||
|      */ | ||||
|     private fun createDerivedConfiguration(name: String) { | ||||
|         val client = sourceSets["client"] | ||||
|         val sourceSet = sourceSets.create(name) | ||||
|         sourceSet.compileClasspath += main.compileClasspath + client.compileClasspath | ||||
|         sourceSet.runtimeClasspath += main.runtimeClasspath + client.runtimeClasspath | ||||
|         consistentWithMain(sourceSet) | ||||
|         project.dependencies.add(sourceSet.implementationConfigurationName, main.output) | ||||
|         project.dependencies.add(sourceSet.implementationConfigurationName, client.output) | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         const val DATAGEN = "datagen" | ||||
|         const val EXAMPLES = "examples" | ||||
|         const val TEST_MOD = "testMod" | ||||
|  | ||||
|         fun setupBasic(project: Project) { | ||||
|             MinecraftConfigurations(project).setupBasic() | ||||
|         } | ||||
|  | ||||
|         fun setup(project: Project) { | ||||
|             MinecraftConfigurations(project).setup() | ||||
|         } | ||||
|  | ||||
|         fun createDerivedConfiguration(project: Project, name: String) { | ||||
|             MinecraftConfigurations(project).createDerivedConfiguration(name) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun DependencyHandler.clientClasses(notation: Any): ModuleDependency = | ||||
|     Capabilities.clientClasses(create(notation) as ModuleDependency) | ||||
|  | ||||
| fun DependencyHandler.commonClasses(notation: Any): ModuleDependency = | ||||
|     Capabilities.commonClasses(create(notation) as ModuleDependency) | ||||
| @@ -1,241 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import net.neoforged.moddevgradle.internal.RunGameTask | ||||
| import org.gradle.api.GradleException | ||||
| import org.gradle.api.file.FileSystemOperations | ||||
| import org.gradle.api.invocation.Gradle | ||||
| import org.gradle.api.provider.Provider | ||||
| import org.gradle.api.services.BuildService | ||||
| import org.gradle.api.services.BuildServiceParameters | ||||
| import org.gradle.api.tasks.* | ||||
| import org.gradle.kotlin.dsl.getByName | ||||
| import org.gradle.language.base.plugins.LifecycleBasePlugin | ||||
| import java.io.File | ||||
| import java.nio.file.Files | ||||
| import java.util.concurrent.TimeUnit | ||||
| import java.util.function.Supplier | ||||
| import javax.inject.Inject | ||||
| import kotlin.collections.set | ||||
| import kotlin.random.Random | ||||
|  | ||||
| /** | ||||
|  * A [JavaExec] task for client-tests. This sets some common setup, and uses [MinecraftRunnerService] to ensure only one | ||||
|  * test runs at once. | ||||
|  */ | ||||
| abstract class ClientJavaExec : JavaExec() { | ||||
|     private val clientRunner: Provider<MinecraftRunnerService> = MinecraftRunnerService.get(project.gradle) | ||||
|  | ||||
|     init { | ||||
|         group = LifecycleBasePlugin.VERIFICATION_GROUP | ||||
|         usesService(clientRunner) | ||||
|     } | ||||
|  | ||||
|     @get:Input | ||||
|     val renderdoc get() = project.hasProperty("renderdoc") | ||||
|  | ||||
|     /** | ||||
|      * When [false], tests will not be run automatically, allowing the user to debug rendering. | ||||
|      */ | ||||
|     @get:Input | ||||
|     val clientDebug get() = renderdoc || project.hasProperty("clientDebug") | ||||
|  | ||||
|     /** | ||||
|      * When [false], tests will not run under a framebuffer. | ||||
|      */ | ||||
|     @get:Input | ||||
|     val useFramebuffer get() = !clientDebug && !project.hasProperty("clientNoFramebuffer") | ||||
|  | ||||
|     /** | ||||
|      * The path test results are written to. | ||||
|      */ | ||||
|     @get:OutputFile | ||||
|     val testResults = project.layout.buildDirectory.file("test-results/$name.xml") | ||||
|  | ||||
|     private fun setTestProperties() { | ||||
|         if (!clientDebug) systemProperty("cctest.client", "") | ||||
|         if (renderdoc) environment("LD_PRELOAD", "/usr/lib/librenderdoc.so") | ||||
|         systemProperty("cctest.gametest-report", testResults.get().asFile.absoluteFile) | ||||
|         workingDir(project.layout.buildDirectory.dir("gametest/$name")) | ||||
|     } | ||||
|  | ||||
|     init { | ||||
|         setTestProperties() | ||||
|     } | ||||
|  | ||||
|     fun copyFromForge(path: String) = copyFromForge(project.tasks.getByName(path, RunGameTask::class)) | ||||
|  | ||||
|     /** | ||||
|      * Set this task to run a given [RunGameTask]. | ||||
|      */ | ||||
|     fun copyFromForge(task: RunGameTask) { | ||||
|         copyFrom(task) | ||||
|  | ||||
|         // Eagerly evaluate the behaviour of RunGameTask.exec | ||||
|         environment.putAll(task.environmentProperty.get()) | ||||
|         classpath(task.classpathProvider) | ||||
|         workingDir = task.gameDirectory.get().asFile | ||||
|  | ||||
|         setTestProperties() // setRunConfig may clobber some properties, ensure everything is set. | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Copy configuration from a task with the given name. | ||||
|      */ | ||||
|     fun copyFrom(path: String) = copyFrom(project.tasks.getByName(path, JavaExec::class)) | ||||
|  | ||||
|     /** | ||||
|      * Copy configuration from an existing [JavaExec] task. | ||||
|      */ | ||||
|     fun copyFrom(task: JavaExec) { | ||||
|         for (dep in task.dependsOn) dependsOn(dep) | ||||
|         task.copyToFull(this) | ||||
|         setTestProperties() // copyToFull may clobber some properties, ensure everything is set. | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Only run tests with the given tags. | ||||
|      */ | ||||
|     fun tags(vararg tags: String) { | ||||
|         systemProperty("cctest.tags", tags.joinToString(",")) | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Write a file with the given contents before starting Minecraft. This may be useful for writing config files. | ||||
|      */ | ||||
|     fun withFileContents(path: Any, contents: Supplier<String>) { | ||||
|         val file = project.file(path).toPath() | ||||
|         doFirst { | ||||
|             Files.createDirectories(file.parent) | ||||
|             Files.writeString(file, contents.get()) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Copy a file to the provided path before starting Minecraft. This copy only occurs if the file does not already | ||||
|      * exist. | ||||
|      */ | ||||
|     fun withFileFrom(path: Any, source: Supplier<File>) { | ||||
|         val file = project.file(path).toPath() | ||||
|         doFirst { | ||||
|             Files.createDirectories(file.parent) | ||||
|             if (!Files.exists(file)) Files.copy(source.get().toPath(), file) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Configure Iris to use Complementary Shaders. | ||||
|      */ | ||||
|     fun withComplementaryShaders() { | ||||
|         val cct = project.extensions.getByType(CCTweakedExtension::class.java) | ||||
|  | ||||
|         withFileFrom(workingDir.resolve("shaderpacks/ComplementaryShaders_v4.6.zip")) { | ||||
|             cct.downloadFile("Complementary Shaders", "https://edge.forgecdn.net/files/3951/170/ComplementaryShaders_v4.6.zip") | ||||
|         } | ||||
|         withFileContents(workingDir.resolve("config/iris.properties")) { | ||||
|             """ | ||||
|             enableShaders=true | ||||
|             shaderPack=ComplementaryShaders_v4.6.zip | ||||
|             """.trimIndent() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @TaskAction | ||||
|     override fun exec() { | ||||
|         Files.createDirectories(workingDir.toPath()) | ||||
|         fsOperations.delete { delete(workingDir.resolve("screenshots")) } | ||||
|  | ||||
|         if (useFramebuffer) { | ||||
|             clientRunner.get().wrapClient(this) { super.exec() } | ||||
|         } else { | ||||
|             super.exec() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @get:Inject | ||||
|     protected abstract val fsOperations: FileSystemOperations | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * A service for [JavaExec] tasks which start Minecraft. | ||||
|  * | ||||
|  * Tasks may run `usesService(MinecraftRunnerService.get(gradle))` to ensure that only one Minecraft-related task runs | ||||
|  * at once. | ||||
|  */ | ||||
| abstract class MinecraftRunnerService : BuildService<BuildServiceParameters.None> { | ||||
|     private val hasXvfb = lazy { | ||||
|         System.getProperty("os.name", "").equals("linux", ignoreCase = true) && ProcessHelpers.onPath("xvfb-run") | ||||
|     } | ||||
|  | ||||
|     internal fun wrapClient(exec: JavaExec, run: () -> Unit) = when { | ||||
|         hasXvfb.value -> runXvfb(exec, run) | ||||
|         else -> run() | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Run a program under Xvfb, preventing it spawning a window. | ||||
|      */ | ||||
|     private fun runXvfb(exec: JavaExec, run: () -> Unit) { | ||||
|         fun ProcessBuilder.startVerbose(): Process { | ||||
|             exec.logger.info("Running ${this.command()}") | ||||
|             return start() | ||||
|         } | ||||
|  | ||||
|         CloseScope().use { scope -> | ||||
|             val dir = Files.createTempDirectory("cctweaked").toAbsolutePath() | ||||
|             scope.add { fsOperations.delete { delete(dir) } } | ||||
|  | ||||
|             val authFile = Files.createTempFile(dir, "Xauthority", "").toAbsolutePath() | ||||
|  | ||||
|             val cookie = StringBuilder().also { | ||||
|                 for (i in 0..31) it.append("0123456789abcdef"[Random.nextInt(16)]) | ||||
|             }.toString() | ||||
|  | ||||
|             val xvfb = | ||||
|                 ProcessBuilder("Xvfb", "-displayfd", "1", "-screen", "0", "640x480x24", "-nolisten", "tcp").also { | ||||
|                     it.inheritIO() | ||||
|                     it.environment()["XAUTHORITY"] = authFile.toString() | ||||
|                     it.redirectOutput(ProcessBuilder.Redirect.PIPE) | ||||
|                 }.startVerbose() | ||||
|             scope.add { xvfb.destroyForcibly().waitFor() } | ||||
|  | ||||
|             val server = xvfb.inputReader().use { it.readLine().trim() } | ||||
|             exec.logger.info("Running at :$server (XAUTHORITY=$authFile.toA") | ||||
|  | ||||
|             ProcessBuilder("xauth", "add", ":$server", ".", cookie).also { | ||||
|                 it.inheritIO() | ||||
|                 it.environment()["XAUTHORITY"] = authFile.toString() | ||||
|             }.startVerbose().waitForOrThrow("Failed to setup XAuthority file") | ||||
|  | ||||
|             scope.add { | ||||
|                 ProcessBuilder("xauth", "remove", ":$server").also { | ||||
|                     it.inheritIO() | ||||
|                     it.environment()["XAUTHORITY"] = authFile.toString() | ||||
|                 }.startVerbose().waitFor() | ||||
|             } | ||||
|  | ||||
|             // Wait a few seconds for Xvfb to start. Ugly, but identical to xvfb-run. | ||||
|             if (xvfb.waitFor(3, TimeUnit.SECONDS)) { | ||||
|                 throw GradleException("Xvfb unexpectedly exited (with status code ${xvfb.exitValue()})") | ||||
|             } | ||||
|  | ||||
|             exec.environment("XAUTHORITY", authFile.toString()) | ||||
|             exec.environment("DISPLAY", ":$server") | ||||
|  | ||||
|             run() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @get:Inject | ||||
|     protected abstract val fsOperations: FileSystemOperations | ||||
|  | ||||
|     companion object { | ||||
|         fun get(gradle: Gradle): Provider<MinecraftRunnerService> = | ||||
|             gradle.sharedServices.registerIfAbsent("cc.tweaked.gradle.ClientJavaExec", MinecraftRunnerService::class.java) { | ||||
|                 maxParallelUsages.set(1) | ||||
|             } | ||||
|     } | ||||
| } | ||||
| @@ -1,69 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import org.gradle.api.DefaultTask | ||||
| import org.gradle.api.Plugin | ||||
| import org.gradle.api.Project | ||||
| import org.gradle.api.file.Directory | ||||
| import org.gradle.api.file.DirectoryProperty | ||||
| import org.gradle.api.provider.Provider | ||||
| import org.gradle.api.tasks.* | ||||
| import org.gradle.process.ExecOperations | ||||
| import java.io.File | ||||
| import javax.inject.Inject | ||||
|  | ||||
| class NodePlugin : Plugin<Project> { | ||||
|     override fun apply(project: Project) { | ||||
|         val extension = project.extensions.create("node", NodeExtension::class.java) | ||||
|         project.tasks.register(NpmInstall.TASK_NAME, NpmInstall::class.java) { | ||||
|             projectRoot.convention(extension.projectRoot) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| abstract class NodeExtension(project: Project) { | ||||
|     /** The directory containing `package-lock.json` and `node_modules/`. */ | ||||
|     abstract val projectRoot: DirectoryProperty | ||||
|  | ||||
|     init { | ||||
|         projectRoot.convention(project.layout.projectDirectory) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** Installs node modules as dependencies. */ | ||||
| abstract class NpmInstall : DefaultTask() { | ||||
|     @get:Internal | ||||
|     abstract val projectRoot: DirectoryProperty | ||||
|  | ||||
|     @get:InputFile | ||||
|     @get:PathSensitive(PathSensitivity.NONE) | ||||
|     val packageLock: Provider<File> = projectRoot.file("package-lock.json").map { it.asFile } | ||||
|  | ||||
|     @get:OutputDirectory | ||||
|     val nodeModules: Provider<Directory> = projectRoot.dir("node_modules") | ||||
|  | ||||
|     @get:Inject | ||||
|     protected abstract val execOperations: ExecOperations | ||||
|  | ||||
|     @TaskAction | ||||
|     fun install() { | ||||
|         execOperations.exec { | ||||
|             commandLine(ProcessHelpers.getExecutable("npm"), "ci") | ||||
|             workingDir = projectRoot.get().asFile | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         internal const val TASK_NAME: String = "npmInstall" | ||||
|     } | ||||
| } | ||||
|  | ||||
| abstract class NpxExecToDir : ExecToDir() { | ||||
|     init { | ||||
|         dependsOn(NpmInstall.TASK_NAME) | ||||
|         executable = ProcessHelpers.getExecutable("npx") | ||||
|     } | ||||
| } | ||||
| @@ -1,42 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import org.gradle.api.GradleException | ||||
| import java.io.File | ||||
|  | ||||
| internal object ProcessHelpers { | ||||
|     fun onPath(name: String): Boolean { | ||||
|         val path = System.getenv("PATH") ?: return false | ||||
|         return path.splitToSequence(File.pathSeparator).any { File(it, name).exists() } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Search for an executable on the `PATH` if required. | ||||
|      * | ||||
|      * [Process]/[ProcessBuilder] does not handle all executable file extensions on Windows (such as `.com). When on | ||||
|      * Windows, this function searches `PATH` and `PATHEXT` for an executable matching [name]. | ||||
|      */ | ||||
|     fun getExecutable(name: String): String { | ||||
|         if (!System.getProperty("os.name").lowercase().contains("windows")) return name | ||||
|  | ||||
|         val path = (System.getenv("PATH") ?: return name).split(File.pathSeparator) | ||||
|         val pathExt = (System.getenv("PATHEXT") ?: return name).split(File.pathSeparator) | ||||
|  | ||||
|         for (pathEntry in path) { | ||||
|             for (ext in pathExt) { | ||||
|                 val resolved = File(pathEntry, name + ext) | ||||
|                 if (resolved.exists()) return resolved.getAbsolutePath() | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return name | ||||
|     } | ||||
| } | ||||
|  | ||||
| internal fun Process.waitForOrThrow(message: String) { | ||||
|     val ret = waitFor() | ||||
|     if (ret != 0) throw GradleException("$message (exited with $ret)") | ||||
| } | ||||
| @@ -1,24 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import org.gradle.api.tasks.SourceSet | ||||
|  | ||||
| data class SourceSetReference( | ||||
|     val sourceSet: SourceSet, | ||||
|     val classes: Boolean, | ||||
|     val external: Boolean, | ||||
| ) { | ||||
|     companion object { | ||||
|         /** A source set in the current project. */ | ||||
|         fun internal(sourceSet: SourceSet) = SourceSetReference(sourceSet, classes = true, external = false) | ||||
|  | ||||
|         /** A source set from another project. */ | ||||
|         fun external(sourceSet: SourceSet) = SourceSetReference(sourceSet, classes = true, external = true) | ||||
|  | ||||
|         /** A source set which is inlined into the current project. */ | ||||
|         fun inline(sourceSet: SourceSet) = SourceSetReference(sourceSet, classes = false, external = false) | ||||
|     } | ||||
| } | ||||
| @@ -1,16 +0,0 @@ | ||||
| // SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
| // | ||||
| // SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| package cc.tweaked.gradle | ||||
|  | ||||
| import groovy.util.Node | ||||
| import groovy.util.NodeList | ||||
|  | ||||
| object XmlUtil { | ||||
|     fun findChild(node: Node, name: String): Node? = when (val child = node.get(name)) { | ||||
|         is Node -> child | ||||
|         is NodeList -> child.singleOrNull() as Node? | ||||
|         else -> null | ||||
|     } | ||||
| } | ||||
							
								
								
									
										2491
									
								
								codeInspectionSettings.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2491
									
								
								codeInspectionSettings.xml
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										61
									
								
								codeStyleSettings.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								codeStyleSettings.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| <code_scheme name="Project" version="173"> | ||||
|   <JSON> | ||||
|     <option name="OBJECT_WRAPPING" value="1" /> | ||||
|     <option name="ARRAY_WRAPPING" value="1" /> | ||||
|   </JSON> | ||||
|   <JavaCodeStyleSettings> | ||||
|     <option name="PACKAGES_TO_USE_IMPORT_ON_DEMAND"> | ||||
|       <value /> | ||||
|     </option> | ||||
|     <option name="JD_P_AT_EMPTY_LINES" value="false" /> | ||||
|     <option name="JD_PRESERVE_LINE_FEEDS" value="true" /> | ||||
|   </JavaCodeStyleSettings> | ||||
|   <codeStyleSettings language="JAVA"> | ||||
|     <option name="KEEP_FIRST_COLUMN_COMMENT" value="false" /> | ||||
|     <option name="BRACE_STYLE" value="2" /> | ||||
|     <option name="CLASS_BRACE_STYLE" value="2" /> | ||||
|     <option name="METHOD_BRACE_STYLE" value="2" /> | ||||
|     <option name="LAMBDA_BRACE_STYLE" value="5" /> | ||||
|     <option name="ELSE_ON_NEW_LINE" value="true" /> | ||||
|     <option name="CATCH_ON_NEW_LINE" value="true" /> | ||||
|     <option name="FINALLY_ON_NEW_LINE" value="true" /> | ||||
|     <option name="SPACE_WITHIN_METHOD_CALL_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_METHOD_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_IF_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_WHILE_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_FOR_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_TRY_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_CATCH_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_SWITCH_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_SYNCHRONIZED_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_ARRAY_INITIALIZER_BRACES" value="true" /> | ||||
|     <option name="SPACE_BEFORE_IF_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_WHILE_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_FOR_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_TRY_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_CATCH_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_SWITCH_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_SYNCHRONIZED_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_ARRAY_INITIALIZER_LBRACE" value="true" /> | ||||
|     <option name="KEEP_SIMPLE_METHODS_IN_ONE_LINE" value="true" /> | ||||
|     <option name="KEEP_SIMPLE_LAMBDAS_IN_ONE_LINE" value="true" /> | ||||
|     <option name="KEEP_SIMPLE_CLASSES_IN_ONE_LINE" value="true" /> | ||||
|     <option name="IF_BRACE_FORCE" value="1" /> | ||||
|     <option name="DOWHILE_BRACE_FORCE" value="1" /> | ||||
|     <option name="WHILE_BRACE_FORCE" value="1" /> | ||||
|     <option name="FOR_BRACE_FORCE" value="1" /> | ||||
|     <option name="SPACE_WITHIN_ANNOTATION_PARENTHESES" value="true" /> | ||||
|     <indentOptions> | ||||
|       <option name="CONTINUATION_INDENT_SIZE" value="4" /> | ||||
|     </indentOptions> | ||||
|   </codeStyleSettings> | ||||
|   <codeStyleSettings language="JSON"> | ||||
|     <option name="KEEP_BLANK_LINES_IN_CODE" value="1" /> | ||||
|     <option name="SPACE_WITHIN_BRACKETS" value="true" /> | ||||
|     <option name="SPACE_WITHIN_BRACES" value="true" /> | ||||
|     <indentOptions> | ||||
|       <option name="INDENT_SIZE" value="4" /> | ||||
|       <option name="CONTINUATION_INDENT_SIZE" value="4" /> | ||||
|     </indentOptions> | ||||
|   </codeStyleSettings> | ||||
| </code_scheme> | ||||
| @@ -1,11 +1,4 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2019 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| <!DOCTYPE module PUBLIC | ||||
|     "-//Checkstyle//DTD Checkstyle Configuration 1.3//EN" | ||||
|     "https://checkstyle.org/dtds/configuration_1_3.dtd"> | ||||
| @@ -13,21 +6,8 @@ SPDX-License-Identifier: MPL-2.0 | ||||
|     <property name="tabWidth" value="4"/> | ||||
|     <property name="charset" value="UTF-8" /> | ||||
|  | ||||
|     <module name="BeforeExecutionExclusionFileFilter"> | ||||
|         <property name="fileNamePattern" value="module\-info\.java$"/> | ||||
|     </module> | ||||
|  | ||||
|     <module name="SuppressionFilter"> | ||||
|         <property name="file" value="${config_loc}/suppressions.xml" /> | ||||
|     </module> | ||||
|  | ||||
|     <!-- | ||||
|         Checkstyle doesn't support @snippet (https://github.com/checkstyle/checkstyle/issues/11455), | ||||
|         so suppress warnings nearby | ||||
|     --> | ||||
|     <module name="SuppressWithNearbyTextFilter"> | ||||
|         <property name="nearbyTextPattern" value="@snippet" /> | ||||
|         <property name="lineRange" value="20" /> | ||||
| 	<property name="file" value="${config_loc}/suppressions.xml" /> | ||||
|     </module> | ||||
|  | ||||
|     <module name="BeforeExecutionExclusionFileFilter"> | ||||
| @@ -37,10 +17,7 @@ SPDX-License-Identifier: MPL-2.0 | ||||
|     <module name="TreeWalker"> | ||||
|         <!-- Annotations --> | ||||
|         <module name="AnnotationLocation" /> | ||||
|         <module name="AnnotationUseStyle"> | ||||
|             <!-- We want trailing commas on multiline arrays. --> | ||||
|             <property name="trailingArrayComma" value="ignore" /> | ||||
|         </module> | ||||
|         <module name="AnnotationUseStyle" /> | ||||
|         <module name="MissingDeprecated" /> | ||||
|         <module name="MissingOverride" /> | ||||
|  | ||||
| @@ -49,11 +26,17 @@ SPDX-License-Identifier: MPL-2.0 | ||||
|         <module name="EmptyCatchBlock"> | ||||
|             <property name="exceptionVariableName" value="ignored" /> | ||||
|         </module> | ||||
|         <module name="LeftCurly" /> | ||||
|         <module name="LeftCurly"> | ||||
|             <property name="option" value="nl" /> | ||||
|             <!-- The defaults, minus lambdas. --> | ||||
|             <property name="tokens" value="ANNOTATION_DEF,CLASS_DEF,CTOR_DEF,ENUM_CONSTANT_DEF,ENUM_DEF,INTERFACE_DEF,LITERAL_CASE,LITERAL_CATCH,LITERAL_DEFAULT,LITERAL_DO,LITERAL_ELSE,LITERAL_FINALLY,LITERAL_FOR,LITERAL_IF,LITERAL_SWITCH,LITERAL_SYNCHRONIZED,LITERAL_TRY,LITERAL_WHILE,METHOD_DEF,OBJBLOCK,STATIC_INIT" /> | ||||
|         </module> | ||||
|         <module name="NeedBraces"> | ||||
|             <property name="allowSingleLineStatement" value="true"/> | ||||
|         </module> | ||||
|         <module name="RightCurly" /> | ||||
|         <module name="RightCurly"> | ||||
|             <property name="option" value="alone" /> | ||||
|         </module> | ||||
|  | ||||
|         <!-- Class design. As if we've ever followed good practice here. --> | ||||
|         <module name="FinalClass" /> | ||||
| @@ -75,36 +58,25 @@ SPDX-License-Identifier: MPL-2.0 | ||||
|         <module name="SimplifyBooleanExpression" /> | ||||
|         <module name="SimplifyBooleanReturn" /> | ||||
|         <module name="StringLiteralEquality" /> | ||||
|         <module name="UnnecessaryParentheses"> | ||||
|             <!-- Default minus LAND. --> | ||||
|             <property name="tokens" value="EXPR,IDENT,NUM_DOUBLE,NUM_FLOAT,NUM_INT,NUM_LONG,STRING_LITERAL,LITERAL_NULL,LITERAL_FALSE,LITERAL_TRUE,ASSIGN,BAND_ASSIGN,BOR_ASSIGN,BSR_ASSIGN,BXOR_ASSIGN,DIV_ASSIGN,MINUS_ASSIGN,MOD_ASSIGN,PLUS_ASSIGN,SL_ASSIGN,SR_ASSIGN,STAR_ASSIGN,LAMBDA,TEXT_BLOCK_LITERAL_BEGIN,LITERAL_INSTANCEOF,GT,LT,GE,LE,EQUAL,NOT_EQUAL,UNARY_MINUS,UNARY_PLUS,INC,DEC,LNOT,BNOT,POST_INC,POST_DEC" /> | ||||
|         </module> | ||||
|         <!-- <module name="UnnecessaryParentheses" /> --> | ||||
|         <module name="UnnecessarySemicolonAfterTypeMemberDeclaration" /> | ||||
|         <module name="UnnecessarySemicolonInTryWithResources" /> | ||||
|         <module name="UnnecessarySemicolonInEnumeration" /> | ||||
|  | ||||
|         <!-- Imports --> | ||||
|         <module name="CustomImportOrder"> | ||||
|             <property name="customImportOrderRules" | ||||
|                 value="THIRD_PARTY_PACKAGE###STANDARD_JAVA_PACKAGE###STATIC" | ||||
|             /> | ||||
|         </module> | ||||
|         <!--<module name="CustomImportOrder" />--> | ||||
|         <module name="IllegalImport" /> | ||||
|         <module name="RedundantImport" /> | ||||
|         <module name="UnusedImports" /> | ||||
|  | ||||
|         <!-- Javadoc --> | ||||
|         <!-- TODO: Missing* checks for the dan200.computercraft.api package? --> | ||||
|         <module name="AtclauseOrder"> | ||||
|             <property name="tagOrder" value="@param, @return, @throws, @deprecated"/> | ||||
|         </module> | ||||
|         <module name="AtclauseOrder" /> | ||||
|         <module name="InvalidJavadocPosition" /> | ||||
|         <module name="JavadocBlockTagLocation" /> | ||||
|         <module name="JavadocMethod"/> | ||||
|         <module name="JavadocType"/> | ||||
|         <module name="JavadocStyle"> | ||||
|             <property name="checkHtml" value="false" /> | ||||
|         </module> | ||||
|         <module name="JavadocStyle" /> | ||||
|         <module name="NonEmptyAtclauseDescription" /> | ||||
|         <module name="SingleLineJavadoc" /> | ||||
|         <module name="SummaryJavadocCheck"/> | ||||
| @@ -125,19 +97,20 @@ SPDX-License-Identifier: MPL-2.0 | ||||
|         <module name="LambdaParameterName" /> | ||||
|         <module name="LocalFinalVariableName" /> | ||||
|         <module name="LocalVariableName" /> | ||||
|         <module name="MemberName"> | ||||
|             <property name="format" value="^\$?[a-z][a-zA-Z0-9]*$" /> | ||||
|         </module> | ||||
|         <module name="MethodName"> | ||||
|             <property name="format" value="^(computercraft\$)?[a-z][a-zA-Z0-9]*$" /> | ||||
|         </module> | ||||
|         <module name="MemberName" /> | ||||
|         <module name="MethodName" /> | ||||
|         <module name="MethodTypeParameterName" /> | ||||
|         <module name="PackageName"> | ||||
|             <property name="format" value="^(dan200\.computercraft|cc\.tweaked|com\.example\.examplemod)(\.[a-z][a-z0-9]*)*" /> | ||||
|             <property name="format" value="^dan200\.computercraft(\.[a-z][a-z0-9]*)*" /> | ||||
|         </module> | ||||
|         <module name="ParameterName" /> | ||||
|         <module name="StaticVariableName"> | ||||
|             <property name="format" value="^[a-z][a-zA-Z0-9]*$" /> | ||||
|             <property name="format" value="^[a-z][a-zA-Z0-9]*|CAPABILITY(_[A-Z_]+)?$" /> | ||||
|             <property name="applyToPrivate" value="false" /> | ||||
|         </module> | ||||
|         <module name="StaticVariableName"> | ||||
|             <property name="format" value="^(s_)?[a-z][a-zA-Z0-9]*|CAPABILITY(_[A-Z_]+)?$" /> | ||||
|             <property name="applyToPrivate" value="true" /> | ||||
|         </module> | ||||
|         <module name="TypeName" /> | ||||
|  | ||||
| @@ -150,11 +123,18 @@ SPDX-License-Identifier: MPL-2.0 | ||||
|         <module name="MethodParamPad" /> | ||||
|         <module name="NoLineWrap" /> | ||||
|         <module name="NoWhitespaceAfter"> | ||||
|             <property name="tokens" value="AT,INC,DEC,UNARY_MINUS,UNARY_PLUS,BNOT,LNOT,DOT,ARRAY_DECLARATOR,INDEX_OP,METHOD_REF" /> | ||||
|             <property name="tokens" value="AT,INC,DEC,UNARY_MINUS,UNARY_PLUS,BNOT,LNOT,DOT,ARRAY_DECLARATOR,INDEX_OP" /> | ||||
|         </module> | ||||
|         <module name="NoWhitespaceBefore" /> | ||||
|         <!-- TODO: Decide on an OperatorWrap style. --> | ||||
|         <module name="ParenPad" /> | ||||
|         <module name="ParenPad"> | ||||
|             <property name="option" value="space" /> | ||||
|             <property name="tokens" value="ANNOTATION,ANNOTATION_FIELD_DEF,CTOR_CALL,CTOR_DEF,ENUM_CONSTANT_DEF,LITERAL_CATCH,LITERAL_DO,LITERAL_FOR,LITERAL_IF,LITERAL_NEW,LITERAL_SWITCH,LITERAL_SYNCHRONIZED,LITERAL_WHILE,METHOD_CALL,METHOD_DEF,RESOURCE_SPECIFICATION,SUPER_CTOR_CALL,LAMBDA" /> | ||||
|         </module> | ||||
|         <module name="ParenPad"> | ||||
|             <property name="option" value="nospace" /> | ||||
|             <property name="tokens" value="DOT,EXPR,QUESTION" /> | ||||
|         </module> | ||||
|         <module name="SeparatorWrap"> | ||||
|             <property name="option" value="eol" /> | ||||
|             <property name="tokens" value="COMMA,SEMI,ELLIPSIS,ARRAY_DECLARATOR,RBRACK,METHOD_REF" /> | ||||
| @@ -169,13 +149,8 @@ SPDX-License-Identifier: MPL-2.0 | ||||
|             <property name="tokens" value="COMMA" /> | ||||
|         </module> | ||||
|         <module name="WhitespaceAround"> | ||||
|             <property name="ignoreEnhancedForColon" value="false" /> | ||||
|             <!-- Allow empty functions --> | ||||
|             <property name="allowEmptyLambdas" value="true" /> | ||||
|             <property name="allowEmptyMethods" value="true" /> | ||||
|             <property name="allowEmptyConstructors" value="true" /> | ||||
|             <property name="allowEmptyTypes" value="true" /> | ||||
|  | ||||
|             <property name="ignoreEnhancedForColon" value="false" /> | ||||
|             <property name="tokens" value="ASSIGN,BAND,BAND_ASSIGN,BOR,BOR_ASSIGN,BSR,BSR_ASSIGN,BXOR,BXOR_ASSIGN,COLON,DIV,DIV_ASSIGN,EQUAL,GE,GT,LAMBDA,LAND,LCURLY,LE,LITERAL_RETURN,LOR,LT,MINUS,MINUS_ASSIGN,MOD,MOD_ASSIGN,NOT_EQUAL,PLUS,PLUS_ASSIGN,QUESTION,RCURLY,SL,SLIST,SL_ASSIGN,SR,SR_ASSIGN,STAR,STAR_ASSIGN,LITERAL_ASSERT,TYPE_EXTENSION_AND" /> | ||||
|         </module> | ||||
|     </module> | ||||
|   | ||||
| @@ -1,11 +1,4 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2019 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| <!DOCTYPE suppressions PUBLIC | ||||
|     "-//Checkstyle//DTD SuppressionFilter Configuration 1.2//EN" | ||||
|     "https://checkstyle.org/dtds/suppressions_1_2.dtd"> | ||||
| @@ -14,15 +7,9 @@ SPDX-License-Identifier: MPL-2.0 | ||||
|     <suppress checks="StaticVariableName" files=".*[\\/]ComputerCraft.java" /> | ||||
|     <suppress checks="StaticVariableName" files=".*[\\/]ComputerCraftAPI.java" /> | ||||
|  | ||||
|     <!-- Some shadowed fields in mixins can't be renamed --> | ||||
|     <suppress checks="StaticVariableName" files=".*[\\/]mixin[\\/].*" /> | ||||
|  | ||||
|     <!-- The commands API is documented in Lua. --> | ||||
|     <suppress checks="SummaryJavadocCheck" files=".*[\\/]CommandAPI.java" /> | ||||
|  | ||||
|     <!-- Allow putting files in other packages if they look like our TeaVM stubs. --> | ||||
|     <suppress checks="PackageName" files=".*[\\/]T[A-Za-z]+.java" /> | ||||
|  | ||||
|     <!-- Allow underscores in our test classes. --> | ||||
|     <suppress checks="MethodName" files=".*(Contract|Test).java" /> | ||||
|  | ||||
|     <!-- Allow underscores in Mixin classes --> | ||||
|     <suppress checks="TypeName" files=".*[\\/]mixin[\\/].*.java" /> | ||||
| </suppressions> | ||||
|   | ||||
							
								
								
									
										2491
									
								
								config/idea/codeInspectionSettings.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2491
									
								
								config/idea/codeInspectionSettings.xml
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										61
									
								
								config/idea/codeStyleSettings.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								config/idea/codeStyleSettings.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| <code_scheme name="Project" version="173"> | ||||
|   <JSON> | ||||
|     <option name="OBJECT_WRAPPING" value="1" /> | ||||
|     <option name="ARRAY_WRAPPING" value="1" /> | ||||
|   </JSON> | ||||
|   <JavaCodeStyleSettings> | ||||
|     <option name="PACKAGES_TO_USE_IMPORT_ON_DEMAND"> | ||||
|       <value /> | ||||
|     </option> | ||||
|     <option name="JD_P_AT_EMPTY_LINES" value="false" /> | ||||
|     <option name="JD_PRESERVE_LINE_FEEDS" value="true" /> | ||||
|   </JavaCodeStyleSettings> | ||||
|   <codeStyleSettings language="JAVA"> | ||||
|     <option name="KEEP_FIRST_COLUMN_COMMENT" value="false" /> | ||||
|     <option name="BRACE_STYLE" value="2" /> | ||||
|     <option name="CLASS_BRACE_STYLE" value="2" /> | ||||
|     <option name="METHOD_BRACE_STYLE" value="2" /> | ||||
|     <option name="LAMBDA_BRACE_STYLE" value="5" /> | ||||
|     <option name="ELSE_ON_NEW_LINE" value="true" /> | ||||
|     <option name="CATCH_ON_NEW_LINE" value="true" /> | ||||
|     <option name="FINALLY_ON_NEW_LINE" value="true" /> | ||||
|     <option name="SPACE_WITHIN_METHOD_CALL_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_METHOD_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_IF_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_WHILE_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_FOR_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_TRY_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_CATCH_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_SWITCH_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_SYNCHRONIZED_PARENTHESES" value="true" /> | ||||
|     <option name="SPACE_WITHIN_ARRAY_INITIALIZER_BRACES" value="true" /> | ||||
|     <option name="SPACE_BEFORE_IF_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_WHILE_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_FOR_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_TRY_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_CATCH_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_SWITCH_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_SYNCHRONIZED_PARENTHESES" value="false" /> | ||||
|     <option name="SPACE_BEFORE_ARRAY_INITIALIZER_LBRACE" value="true" /> | ||||
|     <option name="KEEP_SIMPLE_METHODS_IN_ONE_LINE" value="true" /> | ||||
|     <option name="KEEP_SIMPLE_LAMBDAS_IN_ONE_LINE" value="true" /> | ||||
|     <option name="KEEP_SIMPLE_CLASSES_IN_ONE_LINE" value="true" /> | ||||
|     <option name="IF_BRACE_FORCE" value="1" /> | ||||
|     <option name="DOWHILE_BRACE_FORCE" value="1" /> | ||||
|     <option name="WHILE_BRACE_FORCE" value="1" /> | ||||
|     <option name="FOR_BRACE_FORCE" value="1" /> | ||||
|     <option name="SPACE_WITHIN_ANNOTATION_PARENTHESES" value="true" /> | ||||
|     <indentOptions> | ||||
|       <option name="CONTINUATION_INDENT_SIZE" value="4" /> | ||||
|     </indentOptions> | ||||
|   </codeStyleSettings> | ||||
|   <codeStyleSettings language="JSON"> | ||||
|     <option name="KEEP_BLANK_LINES_IN_CODE" value="1" /> | ||||
|     <option name="SPACE_WITHIN_BRACKETS" value="true" /> | ||||
|     <option name="SPACE_WITHIN_BRACES" value="true" /> | ||||
|     <indentOptions> | ||||
|       <option name="INDENT_SIZE" value="4" /> | ||||
|       <option name="CONTINUATION_INDENT_SIZE" value="4" /> | ||||
|     </indentOptions> | ||||
|   </codeStyleSettings> | ||||
| </code_scheme> | ||||
							
								
								
									
										3
									
								
								config/license/api.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								config/license/api.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| This file is part of the public ComputerCraft API - http://www.computercraft.info | ||||
| Copyright Daniel Ratcliffe, 2011-${year}. This API may be redistributed unmodified and in full only. | ||||
| For help using the API, and posting your mods, visit the forums at computercraft.info. | ||||
							
								
								
									
										3
									
								
								config/license/main.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								config/license/main.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| This file is part of ComputerCraft - http://www.computercraft.info | ||||
| Copyright Daniel Ratcliffe, 2011-${year}. Do not distribute without permission. | ||||
| Send enquiries to dratcliffe@gmail.com | ||||
							
								
								
									
										49
									
								
								config/pre-commit/config.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								config/pre-commit/config.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | ||||
| # See https://pre-commit.com for more information | ||||
| # See https://pre-commit.com/hooks.html for more hooks | ||||
| repos: | ||||
| - repo: https://github.com/pre-commit/pre-commit-hooks | ||||
|   rev: v3.2.0 | ||||
|   hooks: | ||||
|   - id: trailing-whitespace | ||||
|   - id: end-of-file-fixer | ||||
|   - id: check-merge-conflict | ||||
|  | ||||
|   # Quick syntax checkers | ||||
|   - id: check-xml | ||||
|   - id: check-yaml | ||||
|   - id: check-toml | ||||
|   - id: check-json | ||||
|     exclude: "tsconfig\\.json$" | ||||
|  | ||||
| - repo: https://github.com/editorconfig-checker/editorconfig-checker.python | ||||
|   rev: 2.3.5 | ||||
|   hooks: | ||||
|   - id: editorconfig-checker | ||||
|     args: ['-disable-indentation'] | ||||
|     exclude: "^(.*\\.(bat)|LICENSE)$" | ||||
|  | ||||
| - repo: local | ||||
|   hooks: | ||||
|   - id: checkstyle | ||||
|     name: Check Java codestyle | ||||
|     files: ".*\\.java$" | ||||
|     language: system | ||||
|     entry: ./gradlew checkstyleMain checkstyleTest | ||||
|     pass_filenames: false | ||||
|     require_serial: true | ||||
|   - id: license | ||||
|     name: Check Java license headers | ||||
|     files: ".*\\.java$" | ||||
|     language: system | ||||
|     entry: ./gradlew licenseFormat | ||||
|     pass_filenames: false | ||||
|     require_serial: true | ||||
|  | ||||
| exclude: | | ||||
|   (?x)^( | ||||
|     src/generated| | ||||
|     src/test/resources/test-rom/data/json-parsing/| | ||||
|     src/test/server-files/| | ||||
|     config/idea/| | ||||
|     .vscode/ | ||||
|   ) | ||||
							
								
								
									
										16
									
								
								config/pre-commit/illuaminate-lint.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										16
									
								
								config/pre-commit/illuaminate-lint.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| #!/usr/bin/env sh | ||||
| set -e | ||||
|  | ||||
| test -d bin || mkdir bin | ||||
| test -f bin/illuaminate || curl -s -obin/illuaminate https://squiddev.cc/illuaminate/linux-x86-64/illuaminate | ||||
| chmod +x bin/illuaminate | ||||
|  | ||||
| if [ -n ${GITHUB_ACTIONS+x} ]; then | ||||
|     # Register a problem matcher (see https://github.com/actions/toolkit/blob/master/docs/problem-matchers.md) | ||||
|     # for illuaminate. | ||||
|     echo "::add-matcher::.github/matchers/illuaminate.json" | ||||
|     trap 'echo "::remove-matcher owner=illuaminate::"' EXIT | ||||
| fi | ||||
|  | ||||
| ./gradlew luaJavadoc | ||||
| bin/illuaminate lint | ||||
							
								
								
									
										28
									
								
								crowdin.yml
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								crowdin.yml
									
									
									
									
									
								
							| @@ -1,28 +0,0 @@ | ||||
| # SPDX-FileCopyrightText: 2017 The CC: Tweaked Developers | ||||
| # | ||||
| # SPDX-License-Identifier: MPL-2.0 | ||||
|  | ||||
| files: | ||||
|   - source: projects/common/src/generated/resources/assets/computercraft/lang/en_us.json | ||||
|     translation: /projects/common/src/main/resources/assets/computercraft/lang/%locale_with_underscore%.json | ||||
|     languages_mapping: | ||||
|       locale_with_underscore: | ||||
|         cs: cs_cz    # Czech | ||||
|         da: da_dk    # Danish | ||||
|         de: de_de    # German | ||||
|         es-ES: es_es # Spanish | ||||
|         fr: fr_fr    # French | ||||
|         it: it_it    # Italian | ||||
|         ja: ja_jp    # Japanese | ||||
|         ko: ko_kr    # Korean | ||||
|         nb: nb_no    # Norwegian Bokmal | ||||
|         nl: nl_nl    # Dutch | ||||
|         pl: pl_pl    # Polish | ||||
|         pt-BR: pt_br # Portuguese, Brazilian | ||||
|         ru: ru_ru    # Russian | ||||
|         sv-SE: sv_se # Sweedish | ||||
|         tok: tok     # Toki Pona | ||||
|         tr: tr_tr    # Turkish | ||||
|         uk: uk_ua    # Ukraine | ||||
|         vi: vi_vn    # Vietnamese | ||||
|         zh-CN: zh_cn # Chinese Simplified | ||||
| @@ -1,28 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] alarm | ||||
| see: os.setAlarm To start an alarm. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`alarm`] event is fired when an alarm started with [`os.setAlarm`] completes. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`number`]: The ID of the alarm that finished. | ||||
|  | ||||
| ## Example | ||||
| Starts a timer and then waits for it to complete. | ||||
|  | ||||
| ```lua | ||||
| local alarm_id = os.setAlarm(os.time() + 0.05) | ||||
| local event, id | ||||
| repeat | ||||
|     event, id = os.pullEvent("alarm") | ||||
| until id == alarm_id | ||||
| print("Alarm with ID " .. id .. " was fired") | ||||
| ``` | ||||
| @@ -1,31 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] char | ||||
| see: key To listen to any key press. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`char`] event is fired when a character is typed on the keyboard. | ||||
|  | ||||
| The [`char`] event is different to a key press. Sometimes multiple key presses may result in one character being | ||||
| typed (for instance, on some European keyboards). Similarly, some keys (e.g. <kbd>Ctrl</kbd>) do not have any | ||||
| corresponding character. The [`key`] should be used if you want to listen to key presses themselves. | ||||
|  | ||||
| ## Return values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The string representing the character that was pressed. | ||||
|  | ||||
|  | ||||
| ## Example | ||||
| Prints each character the user presses: | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|     local event, character = os.pullEvent("char") | ||||
|     print(character .. " was pressed.") | ||||
| end | ||||
| ``` | ||||
| @@ -1,24 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] computer_command | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`computer_command`] event is fired when the `/computercraft queue` command is run for the current computer. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]<abbr title="Variable number of arguments">…</abbr>: The arguments passed to the command. | ||||
|  | ||||
| ## Example | ||||
| Prints the contents of messages sent: | ||||
| ```lua | ||||
| while true do | ||||
|   local event = {os.pullEvent("computer_command")} | ||||
|   print("Received message:", table.unpack(event, 2)) | ||||
| end | ||||
| ``` | ||||
| @@ -1,25 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] disk | ||||
| see: disk_eject For the event sent when a disk is removed. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`disk`] event is fired when a disk is inserted into an adjacent or networked disk drive. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The side of the disk drive that had a disk inserted. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a disk is inserted: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, side = os.pullEvent("disk") | ||||
|   print("Inserted a disk on side " .. side) | ||||
| end | ||||
| ``` | ||||
| @@ -1,25 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] disk_eject | ||||
| see: disk For the event sent when a disk is inserted. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`disk_eject`] event is fired when a disk is removed from an adjacent or networked disk drive. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The side of the disk drive that had a disk removed. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a disk is removed: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, side = os.pullEvent("disk_eject") | ||||
|   print("Removed a disk on side " .. side) | ||||
| end | ||||
| ``` | ||||
| @@ -1,48 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] file_transfer | ||||
| since: 1.101.0 | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2022 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`file_transfer`] event is queued when a user drags-and-drops a file on an open computer. | ||||
|  | ||||
| This event contains a single argument of type [`TransferredFiles`], which can be used to [get the files to be | ||||
| transferred][`TransferredFiles.getFiles`]. Each file returned is a [binary file handle][`fs.ReadHandle`] with an | ||||
| additional [getName][`TransferredFile.getName`] method. | ||||
|  | ||||
| ## Return values | ||||
| 1. [`string`]: The event name | ||||
| 2. [`TransferredFiles`]: The list of transferred files. | ||||
|  | ||||
| ## Example | ||||
| Waits for a user to drop files on top of the computer, then prints the list of files and the size of each file. | ||||
|  | ||||
| ```lua | ||||
| local _, files = os.pullEvent("file_transfer") | ||||
| for _, file in ipairs(files.getFiles()) do | ||||
|   -- Seek to the end of the file to get its size, then go back to the beginning. | ||||
|   local size = file.seek("end") | ||||
|   file.seek("set", 0) | ||||
|  | ||||
|   print(file.getName() .. " " .. size) | ||||
| end | ||||
| ``` | ||||
|  | ||||
| ## Example | ||||
| Save each transferred file to the computer's storage. | ||||
|  | ||||
| ```lua | ||||
| local _, files = os.pullEvent("file_transfer") | ||||
| for _, file in ipairs(files.getFiles()) do | ||||
|   local handle = fs.open(file.getName(), "wb") | ||||
|   handle.write(file.readAll()) | ||||
|  | ||||
|   handle.close() | ||||
|   file.close() | ||||
| end | ||||
| ``` | ||||
| @@ -1,20 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] http_check | ||||
| see: http.checkURLAsync To check a URL asynchronously. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`http_check`] event is fired when a URL check finishes. | ||||
|  | ||||
| This event is normally handled inside [`http.checkURL`], but it can still be seen when using [`http.checkURLAsync`]. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The URL requested to be checked. | ||||
| 3. [`boolean`]: Whether the check succeeded. | ||||
| 4. <span class="type">[`string`]|[`nil`]</span>: If the check failed, a reason explaining why the check failed. | ||||
| @@ -1,46 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] http_failure | ||||
| see: http.request To send an HTTP request. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`http_failure`] event is fired when an HTTP request fails. | ||||
|  | ||||
| This event is normally handled inside [`http.get`] and [`http.post`], but it can still be seen when using [`http.request`]. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The URL of the site requested. | ||||
| 3. [`string`]: An error describing the failure. | ||||
| 4. <span class="type">[`http.Response`]|[`nil`]</span>: A response handle if the connection succeeded, but the server's | ||||
|    response indicated failure. | ||||
|  | ||||
| ## Example | ||||
| Prints an error why the website cannot be contacted: | ||||
| ```lua | ||||
| local myURL = "https://does.not.exist.tweaked.cc" | ||||
| http.request(myURL) | ||||
| local event, url, err | ||||
| repeat | ||||
|     event, url, err = os.pullEvent("http_failure") | ||||
| until url == myURL | ||||
| print("The URL " .. url .. " could not be reached: " .. err) | ||||
| ``` | ||||
|  | ||||
| Prints the contents of a webpage that does not exist: | ||||
| ```lua | ||||
| local myURL = "https://tweaked.cc/this/does/not/exist" | ||||
| http.request(myURL) | ||||
| local event, url, err, handle | ||||
| repeat | ||||
|     event, url, err, handle = os.pullEvent("http_failure") | ||||
| until url == myURL | ||||
| print("The URL " .. url .. " could not be reached: " .. err) | ||||
| print(handle.getResponseCode()) | ||||
| handle.close() | ||||
| ``` | ||||
| @@ -1,33 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] http_success | ||||
| see: http.request To make an HTTP request. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`http_success`] event is fired when an HTTP request returns successfully. | ||||
|  | ||||
| This event is normally handled inside [`http.get`] and [`http.post`], but it can still be seen when using [`http.request`]. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The URL of the site requested. | ||||
| 3. [`http.Response`]: The successful HTTP response. | ||||
|  | ||||
| ## Example | ||||
| Prints the content of a website (this may fail if the request fails): | ||||
| ```lua | ||||
| local myURL = "https://tweaked.cc/" | ||||
| http.request(myURL) | ||||
| local event, url, handle | ||||
| repeat | ||||
|     event, url, handle = os.pullEvent("http_success") | ||||
| until url == myURL | ||||
| print("Contents of " .. url .. ":") | ||||
| print(handle.readAll()) | ||||
| handle.close() | ||||
| ``` | ||||
| @@ -1,32 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] key | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| This event is fired when any key is pressed while the terminal is focused. | ||||
|  | ||||
| This event returns a numerical "key code" (for instance, <kbd>F1</kbd> is 290). This value may vary between versions and | ||||
| so it is recommended to use the constants in the [`keys`] API rather than hard coding numeric values. | ||||
|  | ||||
| If the button pressed represented a printable character, then the [`key`] event will be followed immediately by a [`char`] | ||||
| event. If you are consuming text input, use a [`char`] event instead! | ||||
|  | ||||
| ## Return values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`number`]: The numerical key value of the key pressed. | ||||
| 3. [`boolean`]: Whether the key event was generated while holding the key ([`true`]), rather than pressing it the first time ([`false`]). | ||||
|  | ||||
| ## Example | ||||
| Prints each key when the user presses it, and if the key is being held. | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   local event, key, is_held = os.pullEvent("key") | ||||
|   print(("%s held=%s"):format(keys.getName(key), is_held)) | ||||
| end | ||||
| ``` | ||||
| @@ -1,30 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] key_up | ||||
| see: keys For a lookup table of the given keys. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| Fired whenever a key is released (or the terminal is closed while a key was being pressed). | ||||
|  | ||||
| This event returns a numerical "key code" (for instance, <kbd>F1</kbd> is 290). This value may vary between versions and | ||||
| so it is recommended to use the constants in the [`keys`] API rather than hard coding numeric values. | ||||
|  | ||||
| ## Return values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`number`]: The numerical key value of the key pressed. | ||||
|  | ||||
| ## Example | ||||
| Prints each key released on the keyboard whenever a [`key_up`] event is fired. | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   local event, key = os.pullEvent("key_up") | ||||
|   local name = keys.getName(key) or "unknown key" | ||||
|   print(name .. " was released.") | ||||
| end | ||||
| ``` | ||||
| @@ -1,34 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] modem_message | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`modem_message`] event is fired when a message is received on an open channel on any [`modem`]. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The side of the modem that received the message. | ||||
| 3. [`number`]: The channel that the message was sent on. | ||||
| 4. [`number`]: The reply channel set by the sender. | ||||
| 5. [`any`]: The message as sent by the sender. | ||||
| 6. <span class="type">[`number`]|[`nil`]</span>: The distance between the sender and the receiver in blocks, or [`nil`] if the message was sent between dimensions. | ||||
|  | ||||
| ## Example | ||||
| Wraps a [`modem`] peripheral, opens channel 0 for listening, and prints all received messages. | ||||
|  | ||||
| ```lua | ||||
| local modem = peripheral.find("modem") or error("No modem attached", 0) | ||||
| modem.open(0) | ||||
|  | ||||
| while true do | ||||
|     local event, side, channel, replyChannel, message, distance = os.pullEvent("modem_message") | ||||
|     print(("Message received on side %s on channel %d (reply to %d) from %f blocks away with message %s"):format( | ||||
|         side, channel, replyChannel, distance, tostring(message) | ||||
|     )) | ||||
| end | ||||
| ``` | ||||
| @@ -1,25 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] monitor_resize | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`monitor_resize`] event is fired when an adjacent or networked [monitor's][`monitor`] size is changed. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The side or network ID of the monitor that was resized. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a monitor is resized: | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   local event, side = os.pullEvent("monitor_resize") | ||||
|   print("The monitor on side " .. side .. " was resized.") | ||||
| end | ||||
| ``` | ||||
| @@ -1,26 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] monitor_touch | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`monitor_touch`] event is fired when an adjacent or networked [Advanced Monitor][`monitor`] is right-clicked. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The side or network ID of the monitor that was touched. | ||||
| 3. [`number`]: The X coordinate of the touch, in characters. | ||||
| 4. [`number`]: The Y coordinate of the touch, in characters. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a monitor is touched: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, side, x, y = os.pullEvent("monitor_touch") | ||||
|   print("The monitor on side " .. side .. " was touched at (" .. x .. ", " .. y .. ")") | ||||
| end | ||||
| ``` | ||||
| @@ -1,38 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] mouse_click | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| This event is fired when the terminal is clicked with a mouse. This event is only fired on advanced computers (including | ||||
| advanced turtles and pocket computers). | ||||
|  | ||||
| ## Return values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`number`]: The mouse button that was clicked. | ||||
| 3. [`number`]: The X-coordinate of the click. | ||||
| 4. [`number`]: The Y-coordinate of the click. | ||||
|  | ||||
| ## Mouse buttons | ||||
| Several mouse events ([`mouse_click`], [`mouse_up`], [`mouse_scroll`]) contain a "mouse button" code. This takes a | ||||
| numerical value depending on which button on your mouse was last pressed when this event occurred. | ||||
|  | ||||
| | Button Code | Mouse Button  | | ||||
| |------------:|---------------| | ||||
| |           1 | Left button   | | ||||
| |           2 | Right button  | | ||||
| |           3 | Middle button | | ||||
|  | ||||
| ## Example | ||||
| Print the button and the coordinates whenever the mouse is clicked. | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   local event, button, x, y = os.pullEvent("mouse_click") | ||||
|   print(("The mouse button %s was pressed at %d, %d"):format(button, x, y)) | ||||
| end | ||||
| ``` | ||||
| @@ -1,28 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] mouse_drag | ||||
| see: mouse_click For when a mouse button is initially pressed. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| This event is fired every time the mouse is moved while a mouse button is being held. | ||||
|  | ||||
| ## Return values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`number`]: The [mouse button](mouse_click.html#Mouse_buttons) that is being pressed. | ||||
| 3. [`number`]: The X-coordinate of the mouse. | ||||
| 4. [`number`]: The Y-coordinate of the mouse. | ||||
|  | ||||
| ## Example | ||||
| Print the button and the coordinates whenever the mouse is dragged. | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   local event, button, x, y = os.pullEvent("mouse_drag") | ||||
|   print(("The mouse button %s was dragged at %d, %d"):format(button, x, y)) | ||||
| end | ||||
| ``` | ||||
| @@ -1,27 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] mouse_scroll | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| This event is fired when a mouse wheel is scrolled in the terminal. | ||||
|  | ||||
| ## Return values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`number`]: The direction of the scroll. (-1 = up, 1 = down) | ||||
| 3. [`number`]: The X-coordinate of the mouse when scrolling. | ||||
| 4. [`number`]: The Y-coordinate of the mouse when scrolling. | ||||
|  | ||||
| ## Example | ||||
| Prints the direction of each scroll, and the position of the mouse at the time. | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   local event, dir, x, y = os.pullEvent("mouse_scroll") | ||||
|   print(("The mouse was scrolled in direction %s at %d, %d"):format(dir, x, y)) | ||||
| end | ||||
| ``` | ||||
| @@ -1,27 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] mouse_up | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| This event is fired when a mouse button is released or a held mouse leaves the computer's terminal. | ||||
|  | ||||
| ## Return values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`number`]: The [mouse button](mouse_click.html#Mouse_buttons) that was released. | ||||
| 3. [`number`]: The X-coordinate of the mouse. | ||||
| 4. [`number`]: The Y-coordinate of the mouse. | ||||
|  | ||||
| ## Example | ||||
| Prints the coordinates and button number whenever the mouse is released. | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   local event, button, x, y = os.pullEvent("mouse_up") | ||||
|   print(("The mouse button %s was released at %d, %d"):format(button, x, y)) | ||||
| end | ||||
| ``` | ||||
| @@ -1,24 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] paste | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`paste`] event is fired when text is pasted into the computer through Ctrl-V (or ⌘V on Mac). | ||||
|  | ||||
| ## Return values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`] The text that was pasted. | ||||
|  | ||||
| ## Example | ||||
| Prints pasted text: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, text = os.pullEvent("paste") | ||||
|   print('"' .. text .. '" was pasted') | ||||
| end | ||||
| ``` | ||||
| @@ -1,25 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] peripheral | ||||
| see: peripheral_detach For the event fired when a peripheral is detached. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`peripheral`] event is fired when a peripheral is attached on a side or to a modem. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The side the peripheral was attached to. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a peripheral is attached: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, side = os.pullEvent("peripheral") | ||||
|   print("A peripheral was attached on side " .. side) | ||||
| end | ||||
| ``` | ||||
| @@ -1,25 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] peripheral_detach | ||||
| see: peripheral For the event fired when a peripheral is attached. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`peripheral_detach`] event is fired when a peripheral is detached from a side or from a modem. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The side the peripheral was detached from. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a peripheral is detached: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, side = os.pullEvent("peripheral_detach") | ||||
|   print("A peripheral was detached on side " .. side) | ||||
| end | ||||
| ``` | ||||
| @@ -1,36 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] rednet_message | ||||
| see: modem_message For raw modem messages sent outside of Rednet. | ||||
| see: rednet.receive To wait for a Rednet message with an optional timeout and protocol filter. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`rednet_message`] event is fired when a message is sent over Rednet. | ||||
|  | ||||
| This event is usually handled by [`rednet.receive`], but it can also be pulled manually. | ||||
|  | ||||
| [`rednet_message`] events are sent by [`rednet.run`] in the top-level coroutine in response to [`modem_message`] events. A [`rednet_message`] event is always preceded by a [`modem_message`] event. They are generated inside CraftOS rather than being sent by the ComputerCraft machine. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`number`]: The ID of the sending computer. | ||||
| 3. [`any`]: The message sent. | ||||
| 4. <span class="type">[`string`]|[`nil`]</span>: The protocol of the message, if provided. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when one is sent: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, sender, message, protocol = os.pullEvent("rednet_message") | ||||
|   if protocol ~= nil then | ||||
|     print("Received message from " .. sender .. " with protocol " .. protocol .. " and message " .. tostring(message)) | ||||
|   else | ||||
|     print("Received message from " .. sender .. " with message " .. tostring(message)) | ||||
|   end | ||||
| end | ||||
| ``` | ||||
| @@ -1,27 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] redstone | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`event!redstone`] event is fired whenever any redstone inputs on the computer or [relay][`redstone_relay`] change. | ||||
|  | ||||
| ## Return values | ||||
| 1. [`string`]: The event name. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a redstone input changes: | ||||
| ```lua | ||||
| while true do | ||||
|   os.pullEvent("redstone") | ||||
|   print("A redstone input has changed!") | ||||
| end | ||||
| ``` | ||||
|  | ||||
| ## See also | ||||
|  - [The `redstone` API on computers][`module!redstone`] | ||||
|  - [The `redstone_relay` peripheral][`redstone_relay`] | ||||
| @@ -1,33 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] speaker_audio_empty | ||||
| see: speaker.playAudio To play audio using the speaker | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The name of the speaker which is available to play more audio. | ||||
|  | ||||
|  | ||||
| ## Example | ||||
| This uses [`io.lines`] to read audio data in blocks of 16KiB from "example_song.dfpwm", and then attempts to play it | ||||
| using [`speaker.playAudio`]. If the speaker's buffer is full, it waits for an event and tries again. | ||||
|  | ||||
| ```lua {data-peripheral=speaker} | ||||
| local dfpwm = require("cc.audio.dfpwm") | ||||
| local speaker = peripheral.find("speaker") | ||||
|  | ||||
| local decoder = dfpwm.make_decoder() | ||||
| for chunk in io.lines("data/example.dfpwm", 16 * 1024) do | ||||
|     local buffer = decoder(chunk) | ||||
|  | ||||
|     while not speaker.playAudio(buffer) do | ||||
|         os.pullEvent("speaker_audio_empty") | ||||
|     end | ||||
| end | ||||
| ``` | ||||
| @@ -1,34 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] task_complete | ||||
| see: commands.execAsync To run a command which fires a task_complete event. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`task_complete`] event is fired when an asynchronous task completes. This is usually handled inside the function call that queued the task; however, functions such as [`commands.execAsync`] return immediately so the user can wait for completion. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`number`]: The ID of the task that completed. | ||||
| 3. [`boolean`]: Whether the command succeeded. | ||||
| 4. [`string`]: If the command failed, an error message explaining the failure. (This is not present if the command succeeded.) | ||||
| 5. <abbr title="Variable number of arguments">…</abbr>: Any parameters returned from the command. | ||||
|  | ||||
| ## Example | ||||
| Prints the results of an asynchronous command: | ||||
| ```lua | ||||
| local taskID = commands.execAsync("say Hello") | ||||
| local event | ||||
| repeat | ||||
|     event = {os.pullEvent("task_complete")} | ||||
| until event[2] == taskID | ||||
| if event[3] == true then | ||||
|   print("Task " .. event[2] .. " succeeded:", table.unpack(event, 4)) | ||||
| else | ||||
|   print("Task " .. event[2] .. " failed: " .. event[4]) | ||||
| end | ||||
| ``` | ||||
| @@ -1,30 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] term_resize | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`term_resize`] event is fired when the main terminal is resized. For instance: | ||||
|  - When a the tab bar is shown or hidden in [`multishell`]. | ||||
|  - When the terminal is redirected to a monitor via the "monitor" program and the monitor is resized. | ||||
|  | ||||
| When this event fires, some parts of the terminal may have been moved or deleted. Simple terminal programs (those | ||||
| not using [`term.setCursorPos`]) can ignore this event, but more complex GUI programs should redraw the entire screen. | ||||
|  | ||||
| ## Return values | ||||
| 1. [`string`]: The event name. | ||||
|  | ||||
| ## Example | ||||
| Print a message each time the terminal is resized. | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   os.pullEvent("term_resize") | ||||
|   local w, h = term.getSize() | ||||
|   print("The term was resized to (" .. w .. ", " .. h .. ")") | ||||
| end | ||||
| ``` | ||||
| @@ -1,34 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] terminate | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`terminate`] event is fired when <kbd>Ctrl-T</kbd> is held down. | ||||
|  | ||||
| This event is normally handled by [`os.pullEvent`], and will not be returned. However, [`os.pullEventRaw`] will return this event when fired. | ||||
|  | ||||
| [`terminate`] will be sent even when a filter is provided to [`os.pullEventRaw`]. When using [`os.pullEventRaw`] with a filter, make sure to check that the event is not [`terminate`]. | ||||
|  | ||||
| ## Return values | ||||
| 1. [`string`]: The event name. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when Ctrl-T is held: | ||||
| ```lua | ||||
| while true do | ||||
|   local event = os.pullEventRaw("terminate") | ||||
|   if event == "terminate" then print("Terminate requested!") end | ||||
| end | ||||
| ``` | ||||
|  | ||||
| Exits when Ctrl-T is held: | ||||
| ```lua | ||||
| while true do | ||||
|   os.pullEvent() | ||||
| end | ||||
| ``` | ||||
| @@ -1,27 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] timer | ||||
| see: os.startTimer To start a timer. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`timer`] event is fired when a timer started with [`os.startTimer`] completes. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`number`]: The ID of the timer that finished. | ||||
|  | ||||
| ## Example | ||||
| Start and wait for a timer to finish. | ||||
| ```lua | ||||
| local timer_id = os.startTimer(2) | ||||
| local event, id | ||||
| repeat | ||||
|     event, id = os.pullEvent("timer") | ||||
| until id == timer_id | ||||
| print("Timer with ID " .. id .. " was fired") | ||||
| ``` | ||||
| @@ -1,23 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] turtle_inventory | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`turtle_inventory`] event is fired when a turtle's inventory is changed. | ||||
|  | ||||
| ## Return values | ||||
| 1. [`string`]: The event name. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when the inventory is changed: | ||||
| ```lua | ||||
| while true do | ||||
|   os.pullEvent("turtle_inventory") | ||||
|   print("The inventory was changed.") | ||||
| end | ||||
| ``` | ||||
| @@ -1,36 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] websocket_closed | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`websocket_closed`] event is fired when an open WebSocket connection is closed. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The URL of the WebSocket that was closed. | ||||
| 3. <span class="type">[`string`]|[`nil`]</span>: The [server-provided reason][close_reason] | ||||
|    the websocket was closed. This will be [`nil`] if the connection was closed | ||||
|    abnormally. | ||||
| 4. <span class="type">[`number`]|[`nil`]</span>: The [connection close code][close_code], | ||||
|    indicating why the socket was closed. This will be [`nil`] if the connection | ||||
|    was closed abnormally. | ||||
|  | ||||
| [close_reason]: https://www.rfc-editor.org/rfc/rfc6455.html#section-7.1.6 "The WebSocket Connection Close Reason, RFC 6455" | ||||
| [close_code]: https://www.rfc-editor.org/rfc/rfc6455.html#section-7.1.5 "The WebSocket Connection Close Code, RFC 6455" | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a WebSocket is closed (this may take a minute): | ||||
| ```lua | ||||
| local myURL = "wss://example.tweaked.cc/echo" | ||||
| local ws = http.websocket(myURL) | ||||
| local event, url | ||||
| repeat | ||||
|     event, url = os.pullEvent("websocket_closed") | ||||
| until url == myURL | ||||
| print("The WebSocket at " .. url .. " was closed.") | ||||
| ``` | ||||
| @@ -1,31 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] websocket_failure | ||||
| see: http.websocketAsync To send an HTTP request. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`websocket_failure`] event is fired when a WebSocket connection request fails. | ||||
|  | ||||
| This event is normally handled inside [`http.websocket`], but it can still be seen when using [`http.websocketAsync`]. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The URL of the site requested. | ||||
| 3. [`string`]: An error describing the failure. | ||||
|  | ||||
| ## Example | ||||
| Prints an error why the website cannot be contacted: | ||||
| ```lua | ||||
| local myURL = "wss://example.tweaked.cc/not-a-websocket" | ||||
| http.websocketAsync(myURL) | ||||
| local event, url, err | ||||
| repeat | ||||
|     event, url, err = os.pullEvent("websocket_failure") | ||||
| until url == myURL | ||||
| print("The URL " .. url .. " could not be reached: " .. err) | ||||
| ``` | ||||
| @@ -1,33 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] websocket_message | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`websocket_message`] event is fired when a message is received on an open WebSocket connection. | ||||
|  | ||||
| This event is normally handled by [`http.Websocket.receive`], but it can also be pulled manually. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The URL of the WebSocket. | ||||
| 3. [`string`]: The contents of the message. | ||||
| 4. [`boolean`]: Whether this is a binary message. | ||||
|  | ||||
| ## Example | ||||
| Prints a message sent by a WebSocket: | ||||
| ```lua | ||||
| local myURL = "wss://example.tweaked.cc/echo" | ||||
| local ws = http.websocket(myURL) | ||||
| ws.send("Hello!") | ||||
| local event, url, message | ||||
| repeat | ||||
|     event, url, message = os.pullEvent("websocket_message") | ||||
| until url == myURL | ||||
| print("Received message from " .. url .. " with contents " .. message) | ||||
| ws.close() | ||||
| ``` | ||||
| @@ -1,34 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] websocket_success | ||||
| see: http.websocketAsync To open a WebSocket asynchronously. | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| SPDX-FileCopyrightText: 2021 The CC: Tweaked Developers | ||||
|  | ||||
| SPDX-License-Identifier: MPL-2.0 | ||||
| --> | ||||
|  | ||||
| The [`websocket_success`] event is fired when a WebSocket connection request returns successfully. | ||||
|  | ||||
| This event is normally handled inside [`http.websocket`], but it can still be seen when using [`http.websocketAsync`]. | ||||
|  | ||||
| ## Return Values | ||||
| 1. [`string`]: The event name. | ||||
| 2. [`string`]: The URL of the site. | ||||
| 3. [`http.Websocket`]: The handle for the WebSocket. | ||||
|  | ||||
| ## Example | ||||
| Prints the content of a website (this may fail if the request fails): | ||||
| ```lua | ||||
| local myURL = "wss://example.tweaked.cc/echo" | ||||
| http.websocketAsync(myURL) | ||||
| local event, url, handle | ||||
| repeat | ||||
|     event, url, handle = os.pullEvent("websocket_success") | ||||
| until url == myURL | ||||
| print("Connected to " .. url) | ||||
| handle.send("Hello!") | ||||
| print(handle.receive()) | ||||
| handle.close() | ||||
| ``` | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user