mirror of
				https://github.com/IntellectualSites/PlotSquared.git
				synced 2025-10-22 14:13:44 +02:00 
			
		
		
		
	Compare commits
	
		
			344 Commits
		
	
	
		
			7.0.0-rc.4
			...
			feat/range
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 5a100ef988 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 711fba0b2a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 671a27fa6f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 03de685dc4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f616885206 | ||
|   | 7da4eb1ab5 | ||
|   | 629646ab06 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 74a1a1f954 | ||
|   | aa44078018 | ||
|   | bfbf406418 | ||
|   | 2accedf264 | ||
|   | 6ef0d58480 | ||
|   | fbf4a638b4 | ||
|   | 9abfa21078 | ||
|   | b84599b4b3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 058983cdd5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6ba3694121 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8e8e31b80e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | da2e66c1f8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d5d6fcb859 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 96f73331f9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 020947d90c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7dbd0bcff8 | ||
|   | 9626302f04 | ||
|   | 1b4a347e8b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 19e6ed4b9b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6b1b0f2d6a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e499bc02ec | ||
|   | 62084fffdd | ||
|   | d012f79349 | ||
|   | 139d6efc70 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6f5cb917f2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9bf66f1b7c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 92875ebe7f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 25a4545f14 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3fdeed019b | ||
|   | f3400df811 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | beb7cb40f4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2c0bb03e5c | ||
|   | e8c170686c | ||
|   | ff8676cde3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 817effb735 | ||
|   | a0a3d8828a | ||
|   | 8741bfcf88 | ||
|   | 6a6c113e5b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3f573b4d46 | ||
|   | 2f6db9c3db | ||
|   | 2f050b7b47 | ||
|   | eb0d854870 | ||
|   | d4f10422e3 | ||
|   | 4e2ea67992 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f533e194f4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 98bc2e7407 | ||
|   | 1b9d0d5317 | ||
|   | 8bb15d5c65 | ||
|   | e9baa802ec | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 63e2d325cf | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 94abd69e22 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 661e4ae8d3 | ||
|   | 974c639a51 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 11b806bd4d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0275372051 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c18cf3acfe | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fa52149394 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f5108ec253 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a7058eeff2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | eabae8db44 | ||
|   | a836e8e763 | ||
|   | 6930a9cecb | ||
|   | effbacb823 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 47d1f1e0cb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6bedd9b25f | ||
|   | 960b7b2a8b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 198052b7a8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a5af3a9d16 | ||
|   | bc4e2c51da | ||
|   | c46b4ddeaa | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c8e7367987 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1a0450eefb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0cb8075184 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5d979b0a4f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fe1554c03c | ||
|   | f0fd9986b4 | ||
|   | bab6f20a5d | ||
|   | 32d36b28fa | ||
|   | a11c560d4e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 66bb0a6214 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7218e9829f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2e17c941fc | ||
|   | 5e628cc758 | ||
|   | a42e08dc0e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 10bf45c128 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9fe61e5053 | ||
|   | 8f5bdf5dbb | ||
|   | 9df1387f81 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | dcf5a7d940 | ||
|   | 641e3840cb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5642061d6f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 003898f6a7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fecf8104e9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8a9dab4f8e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0832656a12 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5525085e6d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7a429fd05c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2c0ad36939 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5e0957c14a | ||
|   | c83306a2ea | ||
|   | 6f4c156585 | ||
|   | afb36d98c7 | ||
|   | 001ae78fb2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5fc8e06c22 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f9401dda94 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d1a48dba4d | ||
|   | db9b51a535 | ||
|   | 511db0af37 | ||
|   | e1ccda3e6d | ||
|   | a69cd609b9 | ||
|   | 5d4e6c5819 | ||
|   | db05f1481a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ee3dd00225 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 346a48225d | ||
|   | dfd80c4723 | ||
|   | fad038ef78 | ||
|   | 84b1af8856 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 13c5a67cb1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8ae894d51d | ||
|   | bbb3736846 | ||
|   | 1ebcbf60a6 | ||
|   | 494144dc4f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 895cf0da66 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 05811d80ce | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4de9967ef4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 034bd866eb | ||
|   | 98aab56616 | ||
|   | 8f980c726b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7f59c03f06 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 480a5925b6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cee4493723 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b8ac1a22c1 | ||
|   | 67e69e3fc1 | ||
|   | 670f5a802e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c4bd6b6500 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 161e3ffdc7 | ||
|   | be8f07c556 | ||
|   | 215053e364 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4839a83279 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0649ef33f0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1590a4b6eb | ||
|   | c793b4454a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7ab1a3eafb | ||
|   | c65c9e7827 | ||
|   | f88ea94bfe | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c8e8eb919f | ||
|   | 83fe761fe4 | ||
|   | a7447c9d75 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5867cc51a7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 86e21f3e1a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d7d884ad6d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1c45e01a14 | ||
|   | 6ef1163325 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c57d784df7 | ||
|   | c239908aa3 | ||
|   | a6412581a6 | ||
|   | f20c5f46e3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4db5954490 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9f68654614 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2e4c6199e5 | ||
|   | 7edca600fd | ||
|   | bc1cc074b8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d383187c6e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 125a3f6772 | ||
|   | faca8c2da0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0ad5ef4f94 | ||
|   | 5e8d8629c2 | ||
|   | 9f4f213a8c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ce14036949 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2dbb6ee025 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0da1d9f17a | ||
|   | f1f41b0523 | ||
|   | fe324d3ea9 | ||
|   | ff83868cbc | ||
|   | 111ea7029e | ||
|   | 9be2eedf7f | ||
|   | 82f868ae7d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e46dbd826c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 809ddce2b3 | ||
|   | 1b40cea51f | ||
|   | 022e0fc224 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b32137a650 | ||
|   | 17c41c0494 | ||
|   | 1ee76bf2d9 | ||
|   | 2321831044 | ||
|   | 25e98618b9 | ||
|   | 5344efd1b7 | ||
|   | 68701b6201 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 64c610ef37 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 931bb90600 | ||
|   | 1dfa3b4e66 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8f236a56a6 | ||
|   | e51121960d | ||
|   | cc011de032 | ||
|   | 28298ffdd6 | ||
|   | 499d3c39bc | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3d56937f14 | ||
|   | 0f1c2cb4e4 | ||
|   | 7b233c944a | ||
|   | d9537ee9df | ||
|   | 0de6887526 | ||
|   | a2e3274215 | ||
|   | b369683b9c | ||
|   | 7f1f1e025e | ||
|   | 59787fe7f3 | ||
|   | 6783d5ece6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d9aa2a496c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 809ed6778c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2fe44053a2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c36b87ca14 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5aec7653b6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cc5d01e225 | ||
|   | 448577774a | ||
|   | 966c878a72 | ||
|   | 5021f5b379 | ||
|   | 76ea9e0d3c | ||
|   | 951f08bc8b | ||
|   | ae941e67a4 | ||
|   | 9566af5fda | ||
|   | fccc146053 | ||
|   | a1d94af242 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6371cd4c5a | ||
|   | e4613cfc62 | ||
|   | 8c44b2d2d2 | ||
|   | 449af2f3a4 | ||
|   | ead7acdd76 | ||
|   | 1991142d48 | ||
|   | 63ae11b3d3 | ||
|   | 86fe3c6846 | ||
|   | a90e179338 | ||
|   | a6ae287908 | ||
|   | 1a33997099 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6edd4b8220 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9b0d1e484c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6971fa4c10 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3c818f3e33 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 31be2e5eb3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 945a8ad306 | ||
|   | c6b0b99cd6 | ||
|   | dbfc43e3cd | ||
|   | c8b4a2fa39 | ||
|   | d851e27aed | ||
|   | 4a45729c9e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7931c0864e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1456b29d93 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 761477b76d | ||
|   | e61bcf905f | ||
|   | 85bec710df | ||
|   | d130794453 | ||
|   | f5f875eb11 | ||
|   | 89511f07f9 | ||
|   | 1a18adcd95 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 65858c5f3e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5c7520b5f5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f3b9cd5ded | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8a3eb25805 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 48bbd3c018 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | bf85013f70 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d36a2d236b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 79f111ec0a | ||
|   | 31ae62b62c | ||
|   | cdb44d4884 | ||
|   | eb63e4351d | ||
|   | ba7880241b | ||
|   | be6838f29e | ||
|   | dc73116401 | ||
|   | b6a87df072 | ||
|   | 8195afaa2f | ||
|   | 561eac2fbd | ||
|   | fdc887850c | ||
|   | e3bfd9b8bf | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e689337188 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ee6ae6cba0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | dc8d7809bd | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | dcd63ed4d9 | ||
|   | 3cc770970f | ||
|   | 1c3776b605 | ||
|   | 95c7f621fb | ||
|   | 15b4cbdb0f | ||
|   | 812eac18d3 | ||
|   | 16a4ee835c | ||
|   | c013b92e62 | ||
|   | b00a46b286 | ||
|   | 44b1127181 | ||
|   | c7bfd48a21 | ||
|   | dc13783db8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0a390ab342 | ||
|   | d111740f64 | ||
|   | 28e97e8441 | ||
|   | a30cdb37d6 | ||
|   | f848162066 | ||
|   | 40c70aa98d | ||
|   | 0d2b36bac8 | ||
|   | d7e5bcdaa5 | ||
|   | fc783574a3 | ||
|   | 5f7bb784f0 | ||
|   | 26c55a318f | ||
|   | ee68bc3d9e | ||
|   | a3bc3968a5 | ||
|   | 79454da1a6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 12a4c92ad9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 167692d464 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ae26e8155c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 286ea62a21 | ||
|   | d95c74d8c9 | ||
|   | c1555ddbc7 | ||
|   | 4fe0c586d9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | aae6ea4fee | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 385d018504 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f4def082c1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 69c9f1df83 | ||
|   | e138dc0267 | ||
|   | ca50b53f94 | ||
|   | f705487055 | ||
|   | b7c9453a1a | ||
|   | 1aa370d562 | ||
|   | d3dab0d736 | ||
|   | 764156b267 | ||
|   | 665f5251bf | ||
|   | 7c328095d7 | ||
|   | 7884c91d52 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e9a19e0821 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 022847fc4b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1ee673be58 | ||
|   | 3c2aa99e86 | ||
|   | 11fac3f060 | ||
|   | 3e57e524b9 | ||
|   | f582ec03c5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 893be136f0 | ||
|   | b74ba30281 | ||
|   | ba9dab1f73 | ||
|   | 8e60fdb477 | ||
|   | 443fe8dd47 | ||
|   | e56e52ba4f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cd008bed9b | ||
|   | d4c90283d6 | ||
|   | dc04ec955a | 
							
								
								
									
										16
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							| @@ -24,20 +24,16 @@ body: | |||||||
|   - type: dropdown |   - type: dropdown | ||||||
|     attributes: |     attributes: | ||||||
|       label: Server Version |       label: Server Version | ||||||
|       description: Which server version version you using? If your server version is not listed, it is not supported. Update to a supported version first. |       description: Which server version are you using? If your server version is not listed, it is not supported. Update to a supported version first. | ||||||
|       multiple: false |       multiple: false | ||||||
|       options: |       options: | ||||||
|         - '1.20.1' |         - '1.21.4' | ||||||
|  |         - '1.21.3' | ||||||
|  |         - '1.21.1' | ||||||
|  |         - '1.20.6' | ||||||
|  |         - '1.20.4' | ||||||
|         - '1.20' |         - '1.20' | ||||||
|         - '1.19.4' |         - '1.19.4' | ||||||
|         - '1.19.3' |  | ||||||
|         - '1.19.2' |  | ||||||
|         - '1.19.1' |  | ||||||
|         - '1.19' |  | ||||||
|         - '1.18.2' |  | ||||||
|         - '1.18.1' |  | ||||||
|         - '1.17.1' |  | ||||||
|         - '1.16.5' |  | ||||||
|     validations: |     validations: | ||||||
|       required: true |       required: true | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								.github/renovate.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.github/renovate.json
									
									
									
									
										vendored
									
									
								
							| @@ -1,7 +1,7 @@ | |||||||
| { | { | ||||||
|   "$schema": "https://docs.renovatebot.com/renovate-schema.json", |   "$schema": "https://docs.renovatebot.com/renovate-schema.json", | ||||||
|   "extends": [ |   "extends": [ | ||||||
|     "config:base", |     "config:recommended", | ||||||
|     ":semanticCommitsDisabled" |     ":semanticCommitsDisabled" | ||||||
|   ], |   ], | ||||||
|   "automerge": true, |   "automerge": true, | ||||||
| @@ -9,5 +9,10 @@ | |||||||
|     "dependencies" |     "dependencies" | ||||||
|   ], |   ], | ||||||
|   "rebaseWhen": "conflicted", |   "rebaseWhen": "conflicted", | ||||||
|   "schedule": ["on the first day of the month"] |   "ignoreDeps": [ | ||||||
|  |     "com.google.code.gson:gson", | ||||||
|  |     "com.google.guava:guava", | ||||||
|  |     "org.yaml:snakeyaml", | ||||||
|  |     "org.apache.logging.log4j:log4j-api" | ||||||
|  |   ] | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								.github/workflows/build-pr.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/build-pr.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,13 +9,13 @@ jobs: | |||||||
|         os: [ ubuntu-latest, windows-latest, macos-latest ] |         os: [ ubuntu-latest, windows-latest, macos-latest ] | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout Repository |       - name: Checkout Repository | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v4 | ||||||
|       - name: Validate Gradle Wrapper |       - name: Validate Gradle Wrapper | ||||||
|         uses: gradle/wrapper-validation-action@v1 |         uses: gradle/actions/wrapper-validation@v4 | ||||||
|       - name: Setup Java |       - name: Setup Java | ||||||
|         uses: actions/setup-java@v3 |         uses: actions/setup-java@v4 | ||||||
|         with: |         with: | ||||||
|           distribution: temurin |           distribution: temurin | ||||||
|           java-version: 17 |           java-version: 21 | ||||||
|       - name: Clean Build |       - name: Clean Build | ||||||
|         run: ./gradlew clean build |         run: ./gradlew clean build | ||||||
|   | |||||||
							
								
								
									
										20
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										20
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,14 +9,14 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout Repository |       - name: Checkout Repository | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v4 | ||||||
|       - name: Validate Gradle Wrapper |       - name: Validate Gradle Wrapper | ||||||
|         uses: gradle/wrapper-validation-action@v1 |         uses: gradle/actions/wrapper-validation@v4 | ||||||
|       - name: Setup Java |       - name: Setup Java | ||||||
|         uses: actions/setup-java@v3 |         uses: actions/setup-java@v4 | ||||||
|         with: |         with: | ||||||
|           distribution: temurin |           distribution: temurin | ||||||
|           java-version: 17 |           java-version: 21 | ||||||
|       - name: Clean Build |       - name: Clean Build | ||||||
|         run: ./gradlew clean build |         run: ./gradlew clean build | ||||||
|       - name: Determine release status |       - name: Determine release status | ||||||
| @@ -29,18 +29,18 @@ jobs: | |||||||
|           fi |           fi | ||||||
|       - name: Publish Release |       - name: Publish Release | ||||||
|         if: ${{ runner.os == 'Linux' && env.STATUS == 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main'}} |         if: ${{ runner.os == 'Linux' && env.STATUS == 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main'}} | ||||||
|         run: ./gradlew publishToSonatype closeAndReleaseSonatypeStagingRepository |         run: ./gradlew publishAndReleaseToMavenCentral --no-configuration-cache | ||||||
|         env: |         env: | ||||||
|           ORG_GRADLE_PROJECT_sonatypeUsername: ${{ secrets.SONATYPE_USERNAME }} |           ORG_GRADLE_PROJECT_mavenCentralUsername: ${{ secrets.CENTRAL_USERNAME }} | ||||||
|           ORG_GRADLE_PROJECT_sonatypePassword: ${{ secrets.SONATYPE_PASSWORD }} |           ORG_GRADLE_PROJECT_mavenCentralPassword: ${{ secrets.CENTRAL_PASSWORD }} | ||||||
|           ORG_GRADLE_PROJECT_signingKey: ${{ secrets.SIGNING_KEY }} |           ORG_GRADLE_PROJECT_signingKey: ${{ secrets.SIGNING_KEY }} | ||||||
|           ORG_GRADLE_PROJECT_signingPassword: ${{ secrets.SIGNING_PASSWORD }} |           ORG_GRADLE_PROJECT_signingPassword: ${{ secrets.SIGNING_PASSWORD }} | ||||||
|       - name: Publish Snapshot |       - name: Publish Snapshot | ||||||
|         if: ${{ runner.os == 'Linux' && env.STATUS != 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main' }} |         if: ${{ runner.os == 'Linux' && env.STATUS != 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main' }} | ||||||
|         run: ./gradlew publishToSonatype |         run: ./gradlew publishAllPublicationsToMavenCentralRepository | ||||||
|         env: |         env: | ||||||
|           ORG_GRADLE_PROJECT_sonatypeUsername: ${{ secrets.SONATYPE_USERNAME }} |           ORG_GRADLE_PROJECT_mavenCentralUsername: ${{ secrets.CENTRAL_USERNAME }} | ||||||
|           ORG_GRADLE_PROJECT_sonatypePassword: ${{ secrets.SONATYPE_PASSWORD }} |           ORG_GRADLE_PROJECT_mavenCentralPassword: ${{ secrets.CENTRAL_PASSWORD }} | ||||||
|       - name: Publish core javadoc |       - name: Publish core javadoc | ||||||
|         if: ${{ runner.os == 'Linux' && env.STATUS == 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main'}} |         if: ${{ runner.os == 'Linux' && env.STATUS == 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main'}} | ||||||
|         uses: cpina/github-action-push-to-another-repository@main |         uses: cpina/github-action-push-to-another-repository@main | ||||||
|   | |||||||
							
								
								
									
										12
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
								
							| @@ -20,17 +20,17 @@ jobs: | |||||||
|         language: [ 'java' ] |         language: [ 'java' ] | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout repository |       - name: Checkout repository | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v4 | ||||||
|       - name: Setup Java |       - name: Setup Java | ||||||
|         uses: actions/setup-java@v3 |         uses: actions/setup-java@v4 | ||||||
|         with: |         with: | ||||||
|           distribution: temurin |           distribution: temurin | ||||||
|           java-version: 17 |           java-version: 21 | ||||||
|       - name: Initialize CodeQL |       - name: Initialize CodeQL | ||||||
|         uses: github/codeql-action/init@v2 |         uses: github/codeql-action/init@v3 | ||||||
|         with: |         with: | ||||||
|           languages: ${{ matrix.language }} |           languages: ${{ matrix.language }} | ||||||
|       - name: Autobuild |       - name: Autobuild | ||||||
|         uses: github/codeql-action/autobuild@v2 |         uses: github/codeql-action/autobuild@v3 | ||||||
|       - name: Perform CodeQL Analysis |       - name: Perform CodeQL Analysis | ||||||
|         uses: github/codeql-action/analyze@v2 |         uses: github/codeql-action/analyze@v3 | ||||||
|   | |||||||
							
								
								
									
										23
									
								
								.github/workflows/label-merge-conflicts.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								.github/workflows/label-merge-conflicts.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | name: "Label conflicting PRs" | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |   pull_request_target: | ||||||
|  |     types: [ synchronize ] | ||||||
|  |   pull_request: | ||||||
|  |     types: [ synchronize ] | ||||||
|  |  | ||||||
|  | permissions: | ||||||
|  |   pull-requests: write | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   main: | ||||||
|  |     if: github.event.pull_request.user.login != 'dependabot[bot]' | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - name: Label conflicting PRs | ||||||
|  |         uses: eps1lon/actions-label-merge-conflict@v3.0.3 | ||||||
|  |         with: | ||||||
|  |           dirtyLabel: "unresolved-merge-conflict" | ||||||
|  |           repoToken: "${{ secrets.GITHUB_TOKEN }}" | ||||||
|  |           commentOnDirty: "Please take a moment and address the merge conflicts of your pull request. Thanks!" | ||||||
|  |           continueOnMissingPermissions: true | ||||||
							
								
								
									
										2
									
								
								.github/workflows/release-drafter.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/release-drafter.yml
									
									
									
									
										vendored
									
									
								
							| @@ -12,6 +12,6 @@ jobs: | |||||||
|     if: ${{ github.event_name != 'pull_request' || github.repository != github.event.pull_request.head.repo.full_name }} |     if: ${{ github.event_name != 'pull_request' || github.repository != github.event.pull_request.head.repo.full_name }} | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: release-drafter/release-drafter@v5 |       - uses: release-drafter/release-drafter@v6 | ||||||
|         env: |         env: | ||||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -138,6 +138,5 @@ build/ | |||||||
|  |  | ||||||
| .DS_Store | .DS_Store | ||||||
| # Ignore run folders | # Ignore run folders | ||||||
| run-[0-0].[0-9]/ | run-[0-9].[0-9][0-9]/ | ||||||
| run-[0-0].[0-9].[0-9]/ | run-[0-9].[0-9][0-9].[0-9]/ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ import com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar | |||||||
| repositories { | repositories { | ||||||
|     maven { |     maven { | ||||||
|         name = "PlaceholderAPI" |         name = "PlaceholderAPI" | ||||||
|         url = uri("https://repo.extendedclip.com/content/repositories/placeholderapi/") |         url = uri("https://repo.extendedclip.com/releases/") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     maven { |     maven { | ||||||
| @@ -21,38 +21,40 @@ dependencies { | |||||||
|     api(projects.plotsquaredCore) |     api(projects.plotsquaredCore) | ||||||
|  |  | ||||||
|     // Metrics |     // Metrics | ||||||
|     implementation("org.bstats:bstats-bukkit") |     implementation(libs.bstatsBukkit) | ||||||
|  |  | ||||||
|     // Paper |     // Paper | ||||||
|     compileOnly("io.papermc.paper:paper-api") |     compileOnly(libs.paper) | ||||||
|     implementation("io.papermc:paperlib") |     implementation(libs.paperlib) | ||||||
|  |  | ||||||
|     // Plugins |     // Plugins | ||||||
|     compileOnly(libs.worldeditBukkit) { |     compileOnly(libs.worldeditBukkit) { | ||||||
|         exclude(group = "org.bukkit") |         exclude(group = "org.bukkit") | ||||||
|         exclude(group = "org.spigotmc") |         exclude(group = "org.spigotmc") | ||||||
|     } |     } | ||||||
|     compileOnly("com.fastasyncworldedit:FastAsyncWorldEdit-Bukkit") { isTransitive = false } |     compileOnly(libs.faweBukkit) { isTransitive = false } | ||||||
|     testImplementation("com.fastasyncworldedit:FastAsyncWorldEdit-Bukkit") { isTransitive = false } |     testImplementation(libs.faweBukkit) { isTransitive = false } | ||||||
|     compileOnly("com.github.MilkBowl:VaultAPI") { |     compileOnly(libs.vault) { | ||||||
|         exclude(group = "org.bukkit") |         exclude(group = "org.bukkit") | ||||||
|     } |     } | ||||||
|     compileOnly(libs.placeholderapi) |     compileOnly(libs.placeholderapi) | ||||||
|     compileOnly(libs.luckperms) |     compileOnly(libs.luckperms) | ||||||
|     compileOnly(libs.essentialsx) |     compileOnly(libs.essentialsx) { | ||||||
|  |         exclude(group = "org.spigotmc") | ||||||
|  |     } | ||||||
|     compileOnly(libs.mvdwapi) { isTransitive = false } |     compileOnly(libs.mvdwapi) { isTransitive = false } | ||||||
|  |  | ||||||
|     // Other libraries |     // Other libraries | ||||||
|     implementation(libs.squirrelid) { isTransitive = false } |     implementation(libs.squirrelid) { isTransitive = false } | ||||||
|     implementation("dev.notmyfault.serverlib:ServerLib") |     implementation(libs.serverlib) | ||||||
|  |  | ||||||
|     // Our libraries |     // Our libraries | ||||||
|     implementation(libs.arkitektonika) |     implementation(libs.arkitektonika) | ||||||
|     implementation("com.intellectualsites.paster:Paster") |     implementation(libs.paster) | ||||||
|     implementation("com.intellectualsites.informative-annotations:informative-annotations") |     implementation(libs.informativeAnnotations) | ||||||
|  |  | ||||||
|     // Adventure |     // Adventure | ||||||
|     implementation("net.kyori:adventure-platform-bukkit") |     implementation(libs.adventureBukkit) | ||||||
| } | } | ||||||
|  |  | ||||||
| tasks.processResources { | tasks.processResources { | ||||||
| @@ -67,6 +69,7 @@ tasks.named<ShadowJar>("shadowJar") { | |||||||
|         exclude(dependency("org.checkerframework:")) |         exclude(dependency("org.checkerframework:")) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     relocate("net.kyori.option", "com.plotsquared.core.configuration.option") | ||||||
|     relocate("net.kyori.adventure", "com.plotsquared.core.configuration.adventure") |     relocate("net.kyori.adventure", "com.plotsquared.core.configuration.adventure") | ||||||
|     relocate("net.kyori.examination", "com.plotsquared.core.configuration.examination") |     relocate("net.kyori.examination", "com.plotsquared.core.configuration.examination") | ||||||
|     relocate("io.papermc.lib", "com.plotsquared.bukkit.paperlib") |     relocate("io.papermc.lib", "com.plotsquared.bukkit.paperlib") | ||||||
| @@ -100,10 +103,10 @@ tasks { | |||||||
|     withType<Javadoc> { |     withType<Javadoc> { | ||||||
|         val isRelease = if (rootProject.version.toString().endsWith("-SNAPSHOT")) "TODO" else rootProject.version.toString() |         val isRelease = if (rootProject.version.toString().endsWith("-SNAPSHOT")) "TODO" else rootProject.version.toString() | ||||||
|         val opt = options as StandardJavadocDocletOptions |         val opt = options as StandardJavadocDocletOptions | ||||||
|         opt.links("https://jd.papermc.io/paper/1.19/") |         opt.links("https://jd.papermc.io/paper/1.20.4/") | ||||||
|         opt.links("https://docs.enginehub.org/javadoc/com.sk89q.worldedit/worldedit-bukkit/" + libs.worldeditBukkit.get().versionConstraint.toString()) |         opt.links("https://docs.enginehub.org/javadoc/com.sk89q.worldedit/worldedit-bukkit/" + libs.worldeditBukkit.get().versionConstraint.toString()) | ||||||
|         opt.links("https://intellectualsites.github.io/plotsquared-javadocs/core/") |         opt.links("https://intellectualsites.github.io/plotsquared-javadocs/core/") | ||||||
|         opt.links("https://jd.advntr.dev/api/4.14.0/") |         opt.links("https://jd.advntr.dev/api/" + libs.adventureApi.get().versionConstraint.toString()) | ||||||
|         opt.links("https://google.github.io/guice/api-docs/" + libs.guice.get().versionConstraint.toString() + "/javadoc/") |         opt.links("https://google.github.io/guice/api-docs/" + libs.guice.get().versionConstraint.toString() + "/javadoc/") | ||||||
|         opt.links("https://checkerframework.org/api/") |         opt.links("https://checkerframework.org/api/") | ||||||
|         opt.isLinkSource = true |         opt.isLinkSource = true | ||||||
| @@ -112,5 +115,6 @@ tasks { | |||||||
|         opt.encoding("UTF-8") |         opt.encoding("UTF-8") | ||||||
|         opt.keyWords() |         opt.keyWords() | ||||||
|         opt.addStringOption("-since", isRelease) |         opt.addStringOption("-since", isRelease) | ||||||
|  |         opt.noTimestamp() | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -24,7 +24,6 @@ import com.google.inject.Injector; | |||||||
| import com.google.inject.Key; | import com.google.inject.Key; | ||||||
| import com.google.inject.Singleton; | import com.google.inject.Singleton; | ||||||
| import com.google.inject.Stage; | import com.google.inject.Stage; | ||||||
| import com.google.inject.TypeLiteral; |  | ||||||
| import com.plotsquared.bukkit.generator.BukkitPlotGenerator; | import com.plotsquared.bukkit.generator.BukkitPlotGenerator; | ||||||
| import com.plotsquared.bukkit.inject.BackupModule; | import com.plotsquared.bukkit.inject.BackupModule; | ||||||
| import com.plotsquared.bukkit.inject.BukkitModule; | import com.plotsquared.bukkit.inject.BukkitModule; | ||||||
| @@ -35,8 +34,10 @@ import com.plotsquared.bukkit.listener.BlockEventListener117; | |||||||
| import com.plotsquared.bukkit.listener.ChunkListener; | import com.plotsquared.bukkit.listener.ChunkListener; | ||||||
| import com.plotsquared.bukkit.listener.EntityEventListener; | import com.plotsquared.bukkit.listener.EntityEventListener; | ||||||
| import com.plotsquared.bukkit.listener.EntitySpawnListener; | import com.plotsquared.bukkit.listener.EntitySpawnListener; | ||||||
|  | import com.plotsquared.bukkit.listener.HighFreqBlockEventListener; | ||||||
| import com.plotsquared.bukkit.listener.PaperListener; | import com.plotsquared.bukkit.listener.PaperListener; | ||||||
| import com.plotsquared.bukkit.listener.PlayerEventListener; | import com.plotsquared.bukkit.listener.PlayerEventListener; | ||||||
|  | import com.plotsquared.bukkit.listener.PlayerEventListener1201; | ||||||
| import com.plotsquared.bukkit.listener.ProjectileEventListener; | import com.plotsquared.bukkit.listener.ProjectileEventListener; | ||||||
| import com.plotsquared.bukkit.listener.ServerListener; | import com.plotsquared.bukkit.listener.ServerListener; | ||||||
| import com.plotsquared.bukkit.listener.SingleWorldListener; | import com.plotsquared.bukkit.listener.SingleWorldListener; | ||||||
| @@ -44,12 +45,12 @@ import com.plotsquared.bukkit.listener.SpigotListener; | |||||||
| import com.plotsquared.bukkit.listener.WorldEvents; | import com.plotsquared.bukkit.listener.WorldEvents; | ||||||
| import com.plotsquared.bukkit.placeholder.PAPIPlaceholders; | import com.plotsquared.bukkit.placeholder.PAPIPlaceholders; | ||||||
| import com.plotsquared.bukkit.placeholder.PlaceholderFormatter; | import com.plotsquared.bukkit.placeholder.PlaceholderFormatter; | ||||||
| import com.plotsquared.bukkit.player.BukkitPlayer; |  | ||||||
| import com.plotsquared.bukkit.player.BukkitPlayerManager; | import com.plotsquared.bukkit.player.BukkitPlayerManager; | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
| import com.plotsquared.bukkit.util.BukkitWorld; | import com.plotsquared.bukkit.util.BukkitWorld; | ||||||
| import com.plotsquared.bukkit.util.SetGenCB; | import com.plotsquared.bukkit.util.SetGenCB; | ||||||
| import com.plotsquared.bukkit.util.TranslationUpdateManager; | import com.plotsquared.bukkit.util.TranslationUpdateManager; | ||||||
|  | import com.plotsquared.bukkit.util.UpdateUtility; | ||||||
| import com.plotsquared.bukkit.util.task.BukkitTaskManager; | import com.plotsquared.bukkit.util.task.BukkitTaskManager; | ||||||
| import com.plotsquared.bukkit.util.task.PaperTimeConverter; | import com.plotsquared.bukkit.util.task.PaperTimeConverter; | ||||||
| import com.plotsquared.bukkit.util.task.SpigotTimeConverter; | import com.plotsquared.bukkit.util.task.SpigotTimeConverter; | ||||||
| @@ -134,6 +135,7 @@ import org.bukkit.generator.ChunkGenerator; | |||||||
| import org.bukkit.metadata.FixedMetadataValue; | import org.bukkit.metadata.FixedMetadataValue; | ||||||
| import org.bukkit.metadata.MetadataValue; | import org.bukkit.metadata.MetadataValue; | ||||||
| import org.bukkit.plugin.Plugin; | import org.bukkit.plugin.Plugin; | ||||||
|  | import org.bukkit.plugin.RegisteredServiceProvider; | ||||||
| import org.bukkit.plugin.java.JavaPlugin; | import org.bukkit.plugin.java.JavaPlugin; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
| import org.checkerframework.checker.nullness.qual.Nullable; | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
| @@ -251,6 +253,12 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|  |     public @NonNull String serverBrand() { | ||||||
|  |         return Bukkit.getName(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     @SuppressWarnings("deprecation") // Paper deprecation | ||||||
|     public void onEnable() { |     public void onEnable() { | ||||||
|         this.pluginName = getDescription().getName(); |         this.pluginName = getDescription().getName(); | ||||||
|  |  | ||||||
| @@ -302,11 +310,9 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|  |  | ||||||
|         this.serverLocale = Locale.forLanguageTag(Settings.Enabled_Components.DEFAULT_LOCALE); |         this.serverLocale = Locale.forLanguageTag(Settings.Enabled_Components.DEFAULT_LOCALE); | ||||||
|  |  | ||||||
|         /* TODO Enable update checker before v7 is released to GA |  | ||||||
|         if (PremiumVerification.isPremium() && Settings.Enabled_Components.UPDATE_NOTIFICATIONS) { |         if (PremiumVerification.isPremium() && Settings.Enabled_Components.UPDATE_NOTIFICATIONS) { | ||||||
|             injector.getInstance(UpdateUtility.class).updateChecker(); |             injector.getInstance(UpdateUtility.class).updateChecker(); | ||||||
|         } |         } | ||||||
|          */ |  | ||||||
|  |  | ||||||
|         if (PremiumVerification.isPremium()) { |         if (PremiumVerification.isPremium()) { | ||||||
|             LOGGER.info("PlotSquared version licensed to Spigot user {}", getUserID()); |             LOGGER.info("PlotSquared version licensed to Spigot user {}", getUserID()); | ||||||
| @@ -358,7 +364,13 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|  |  | ||||||
|         if (Settings.Enabled_Components.EVENTS) { |         if (Settings.Enabled_Components.EVENTS) { | ||||||
|             getServer().getPluginManager().registerEvents(injector().getInstance(PlayerEventListener.class), this); |             getServer().getPluginManager().registerEvents(injector().getInstance(PlayerEventListener.class), this); | ||||||
|  |             if ((serverVersion()[1] == 20 && serverVersion()[2] >= 1) || serverVersion()[1] > 20) { | ||||||
|  |                 getServer().getPluginManager().registerEvents(injector().getInstance(PlayerEventListener1201.class), this); | ||||||
|  |             } | ||||||
|             getServer().getPluginManager().registerEvents(injector().getInstance(BlockEventListener.class), this); |             getServer().getPluginManager().registerEvents(injector().getInstance(BlockEventListener.class), this); | ||||||
|  |             if (Settings.HIGH_FREQUENCY_LISTENER) { | ||||||
|  |                 getServer().getPluginManager().registerEvents(injector().getInstance(HighFreqBlockEventListener.class), this); | ||||||
|  |             } | ||||||
|             if (serverVersion()[1] >= 17) { |             if (serverVersion()[1] >= 17) { | ||||||
|                 getServer().getPluginManager().registerEvents(injector().getInstance(BlockEventListener117.class), this); |                 getServer().getPluginManager().registerEvents(injector().getInstance(BlockEventListener117.class), this); | ||||||
|             } |             } | ||||||
| @@ -551,7 +563,7 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|         this.startMetrics(); |         this.startMetrics(); | ||||||
|  |  | ||||||
|         if (Settings.Enabled_Components.WORLDS) { |         if (Settings.Enabled_Components.WORLDS) { | ||||||
|             TaskManager.getPlatformImplementation().taskRepeat(this::unload, TaskTime.seconds(1L)); |             TaskManager.getPlatformImplementation().taskRepeat(this::unload, TaskTime.seconds(10L)); | ||||||
|             try { |             try { | ||||||
|                 singleWorldListener = injector().getInstance(SingleWorldListener.class); |                 singleWorldListener = injector().getInstance(SingleWorldListener.class); | ||||||
|                 Bukkit.getPluginManager().registerEvents(singleWorldListener, this); |                 Bukkit.getPluginManager().registerEvents(singleWorldListener, this); | ||||||
| @@ -774,22 +786,31 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|                 Iterator<Entity> iterator = entities.iterator(); |                 Iterator<Entity> iterator = entities.iterator(); | ||||||
|                 while (iterator.hasNext()) { |                 while (iterator.hasNext()) { | ||||||
|                     Entity entity = iterator.next(); |                     Entity entity = iterator.next(); | ||||||
|  |                     //noinspection ConstantValue - getEntitySpawnReason annotated as NotNull, but is not NotNull. lol. | ||||||
|  |                     if (PaperLib.isPaper() && entity.getEntitySpawnReason() != null && "CUSTOM".equals(entity.getEntitySpawnReason().name())) { | ||||||
|  |                         continue; | ||||||
|  |                     } | ||||||
|  |                     // Fallback for Spigot not having Entity#getEntitySpawnReason | ||||||
|  |                     if (entity.getMetadata("ps_custom_spawned").stream().anyMatch(MetadataValue::asBoolean)) { | ||||||
|  |                         continue; | ||||||
|  |                     } | ||||||
|  |                     // TODO: use (type) pattern matching when targeting java 21 | ||||||
|                     switch (entity.getType().toString()) { |                     switch (entity.getType().toString()) { | ||||||
|                         case "EGG": |                         case "EGG": | ||||||
|                         case "FISHING_HOOK": |                         case "FISHING_HOOK", "FISHING_BOBBER": | ||||||
|                         case "ENDER_SIGNAL": |                         case "ENDER_SIGNAL", "EYE_OF_ENDER": | ||||||
|                         case "AREA_EFFECT_CLOUD": |                         case "AREA_EFFECT_CLOUD": | ||||||
|                         case "EXPERIENCE_ORB": |                         case "EXPERIENCE_ORB": | ||||||
|                         case "LEASH_HITCH": |                         case "LEASH_HITCH", "LEASH_KNOT": | ||||||
|                         case "FIREWORK": |                         case "FIREWORK", "FIREWORK_ROCKET": | ||||||
|                         case "LIGHTNING": |                         case "LIGHTNING", "LIGHTNING_BOLT": | ||||||
|                         case "WITHER_SKULL": |                         case "WITHER_SKULL": | ||||||
|                         case "UNKNOWN": |                         case "UNKNOWN": | ||||||
|                         case "PLAYER": |                         case "PLAYER": | ||||||
|                             // non moving / unmovable |                             // non moving / unmovable | ||||||
|                             continue; |                             continue; | ||||||
|                         case "THROWN_EXP_BOTTLE": |                         case "THROWN_EXP_BOTTLE", "EXPERIENCE_BOTTLE": | ||||||
|                         case "SPLASH_POTION": |                         case "SPLASH_POTION", "POTION": | ||||||
|                         case "SNOWBALL": |                         case "SNOWBALL": | ||||||
|                         case "SHULKER_BULLET": |                         case "SHULKER_BULLET": | ||||||
|                         case "SPECTRAL_ARROW": |                         case "SPECTRAL_ARROW": | ||||||
| @@ -807,13 +828,25 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|                             // Temporarily classify as vehicle |                             // Temporarily classify as vehicle | ||||||
|                         case "MINECART": |                         case "MINECART": | ||||||
|                         case "MINECART_CHEST": |                         case "MINECART_CHEST": | ||||||
|  |                         case "CHEST_MINECART": | ||||||
|                         case "MINECART_COMMAND": |                         case "MINECART_COMMAND": | ||||||
|  |                         case "COMMAND_BLOCK_MINECART": | ||||||
|                         case "MINECART_FURNACE": |                         case "MINECART_FURNACE": | ||||||
|  |                         case "FURNACE_MINECART": | ||||||
|                         case "MINECART_HOPPER": |                         case "MINECART_HOPPER": | ||||||
|  |                         case "HOPPER_MINECART": | ||||||
|                         case "MINECART_MOB_SPAWNER": |                         case "MINECART_MOB_SPAWNER": | ||||||
|  |                         case "SPAWNER_MINECART": | ||||||
|                         case "ENDER_CRYSTAL": |                         case "ENDER_CRYSTAL": | ||||||
|                         case "MINECART_TNT": |                         case "MINECART_TNT": | ||||||
|  |                         case "TNT_MINECART": | ||||||
|  |                         case "CHEST_BOAT": | ||||||
|                         case "BOAT": |                         case "BOAT": | ||||||
|  |                         case "ACACIA_BOAT", "BIRCH_BOAT", "CHERRY_BOAT", "DARK_OAK_BOAT", "JUNGLE_BOAT", "MANGROVE_BOAT", | ||||||
|  |                              "OAK_BOAT", "PALE_OAK_BOAT", "SPRUCE_BOAT", "BAMBOO_RAFT": | ||||||
|  |                         case "ACACIA_CHEST_BOAT", "BIRCH_CHEST_BOAT", "CHERRY_CHEST_BOAT", "DARK_OAK_CHEST_BOAT", | ||||||
|  |                              "JUNGLE_CHEST_BOAT", "MANGROVE_CHEST_BOAT", "OAK_CHEST_BOAT", "PALE_OAK_CHEST_BOAT", | ||||||
|  |                              "SPRUCE_CHEST_BOAT", "BAMBOO_CHEST_RAFT": | ||||||
|                             if (Settings.Enabled_Components.KILL_ROAD_VEHICLES) { |                             if (Settings.Enabled_Components.KILL_ROAD_VEHICLES) { | ||||||
|                                 com.plotsquared.core.location.Location location = BukkitUtil.adapt(entity.getLocation()); |                                 com.plotsquared.core.location.Location location = BukkitUtil.adapt(entity.getLocation()); | ||||||
|                                 Plot plot = location.getPlot(); |                                 Plot plot = location.getPlot(); | ||||||
| @@ -842,14 +875,14 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|                         case "SMALL_FIREBALL": |                         case "SMALL_FIREBALL": | ||||||
|                         case "FIREBALL": |                         case "FIREBALL": | ||||||
|                         case "DRAGON_FIREBALL": |                         case "DRAGON_FIREBALL": | ||||||
|                         case "DROPPED_ITEM": |                         case "DROPPED_ITEM", "ITEM": | ||||||
|                             if (Settings.Enabled_Components.KILL_ROAD_ITEMS |                             if (Settings.Enabled_Components.KILL_ROAD_ITEMS | ||||||
|                                     && plotArea.getOwnedPlotAbs(BukkitUtil.adapt(entity.getLocation())) == null) { |                                     && plotArea.getOwnedPlotAbs(BukkitUtil.adapt(entity.getLocation())) == null) { | ||||||
|                                 this.removeRoadEntity(entity, iterator); |                                 this.removeRoadEntity(entity, iterator); | ||||||
|                             } |                             } | ||||||
|                             // dropped item |                             // dropped item | ||||||
|                             continue; |                             continue; | ||||||
|                         case "PRIMED_TNT": |                         case "PRIMED_TNT", "TNT": | ||||||
|                         case "FALLING_BLOCK": |                         case "FALLING_BLOCK": | ||||||
|                             // managed elsewhere |                             // managed elsewhere | ||||||
|                             continue; |                             continue; | ||||||
| @@ -861,8 +894,7 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|                                     if (livingEntity.isLeashed() && !Settings.Enabled_Components.KILL_OWNED_ROAD_MOBS) { |                                     if (livingEntity.isLeashed() && !Settings.Enabled_Components.KILL_OWNED_ROAD_MOBS) { | ||||||
|                                         continue; |                                         continue; | ||||||
|                                     } |                                     } | ||||||
|                                     List<MetadataValue> keep = entity.getMetadata("keep"); |                                     if (entity.hasMetadata("keep")) { | ||||||
|                                     if (!keep.isEmpty()) { |  | ||||||
|                                         continue; |                                         continue; | ||||||
|                                     } |                                     } | ||||||
|  |  | ||||||
| @@ -928,7 +960,7 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|                         case "HORSE": |                         case "HORSE": | ||||||
|                         case "IRON_GOLEM": |                         case "IRON_GOLEM": | ||||||
|                         case "MAGMA_CUBE": |                         case "MAGMA_CUBE": | ||||||
|                         case "MUSHROOM_COW": |                         case "MUSHROOM_COW", "MOOSHROOM": | ||||||
|                         case "OCELOT": |                         case "OCELOT": | ||||||
|                         case "PIG": |                         case "PIG": | ||||||
|                         case "PIG_ZOMBIE": |                         case "PIG_ZOMBIE": | ||||||
| @@ -937,7 +969,7 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|                         case "SILVERFISH": |                         case "SILVERFISH": | ||||||
|                         case "SKELETON": |                         case "SKELETON": | ||||||
|                         case "SLIME": |                         case "SLIME": | ||||||
|                         case "SNOWMAN": |                         case "SNOWMAN", "SNOW_GOLEM": | ||||||
|                         case "SPIDER": |                         case "SPIDER": | ||||||
|                         case "SQUID": |                         case "SQUID": | ||||||
|                         case "VILLAGER": |                         case "VILLAGER": | ||||||
| @@ -1150,7 +1182,9 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|     @Override |     @Override | ||||||
|     public @NonNull String serverNativePackage() { |     public @NonNull String serverNativePackage() { | ||||||
|         final String name = Bukkit.getServer().getClass().getPackage().getName(); |         final String name = Bukkit.getServer().getClass().getPackage().getName(); | ||||||
|         return name.substring(name.lastIndexOf('.') + 1); |         String ver = name.substring(name.lastIndexOf('.') + 1); | ||||||
|  |         // org.bukkit.craftbukkit is no longer suffixed by a version | ||||||
|  |         return ver.equals("craftbukkit") ? "" : ver; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -1161,6 +1195,7 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|         return new BukkitPlotGenerator(world, generator, this.plotAreaManager); |         return new BukkitPlotGenerator(world, generator, this.plotAreaManager); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @SuppressWarnings("deprecation") // Paper deprecation | ||||||
|     @Override |     @Override | ||||||
|     public @NonNull String pluginsFormatted() { |     public @NonNull String pluginsFormatted() { | ||||||
|         StringBuilder msg = new StringBuilder(); |         StringBuilder msg = new StringBuilder(); | ||||||
| @@ -1177,12 +1212,23 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|                     .append("  • Load Before: ").append(p.getDescription().getLoadBefore()).append("\n") |                     .append("  • Load Before: ").append(p.getDescription().getLoadBefore()).append("\n") | ||||||
|                     .append("  • Dependencies: ").append(p.getDescription().getDepend()).append("\n") |                     .append("  • Dependencies: ").append(p.getDescription().getDepend()).append("\n") | ||||||
|                     .append("  • Soft Dependencies: ").append(p.getDescription().getSoftDepend()).append("\n"); |                     .append("  • Soft Dependencies: ").append(p.getDescription().getSoftDepend()).append("\n"); | ||||||
|  |             List<RegisteredServiceProvider<?>> providers = Bukkit.getServicesManager().getRegistrations(p); | ||||||
|  |             if (!providers.isEmpty()) { | ||||||
|  |                 msg.append("  • Provided Services: \n"); | ||||||
|  |                 for (RegisteredServiceProvider<?> provider : providers) { | ||||||
|  |                     msg.append("    • ") | ||||||
|  |                             .append(provider.getService().getName()).append(" = ") | ||||||
|  |                             .append(provider.getProvider().getClass().getName()) | ||||||
|  |                             .append(" (priority: ").append(provider.getPriority()).append(")") | ||||||
|  |                             .append("\n"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|         return msg.toString(); |         return msg.toString(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     @SuppressWarnings("ConstantConditions") |     @SuppressWarnings({"ConstantConditions", "deprecation"}) // Paper deprecation | ||||||
|     public @NonNull String worldEditImplementations() { |     public @NonNull String worldEditImplementations() { | ||||||
|         StringBuilder msg = new StringBuilder(); |         StringBuilder msg = new StringBuilder(); | ||||||
|         if (Bukkit.getPluginManager().getPlugin("FastAsyncWorldEdit") != null) { |         if (Bukkit.getPluginManager().getPlugin("FastAsyncWorldEdit") != null) { | ||||||
| @@ -1246,15 +1292,13 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public @NonNull PlatformWorldManager<?> worldManager() { |     public @NonNull PlatformWorldManager<?> worldManager() { | ||||||
|         return injector().getInstance(Key.get(new TypeLiteral<PlatformWorldManager<World>>() { |         return this.worldManager; | ||||||
|         })); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     @NonNull |     @NonNull | ||||||
|     @SuppressWarnings("unchecked") |  | ||||||
|     public PlayerManager<? extends PlotPlayer<Player>, ? extends Player> playerManager() { |     public PlayerManager<? extends PlotPlayer<Player>, ? extends Player> playerManager() { | ||||||
|         return (PlayerManager<BukkitPlayer, Player>) injector().getInstance(PlayerManager.class); |         return this.playerManager; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|   | |||||||
| @@ -25,7 +25,6 @@ import org.bukkit.Art; | |||||||
| import org.bukkit.DyeColor; | import org.bukkit.DyeColor; | ||||||
| import org.bukkit.Location; | import org.bukkit.Location; | ||||||
| import org.bukkit.Rotation; | import org.bukkit.Rotation; | ||||||
| import org.bukkit.TreeSpecies; |  | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.block.BlockFace; | import org.bukkit.block.BlockFace; | ||||||
| import org.bukkit.entity.AbstractHorse; | import org.bukkit.entity.AbstractHorse; | ||||||
| @@ -33,6 +32,8 @@ import org.bukkit.entity.Ageable; | |||||||
| import org.bukkit.entity.ArmorStand; | import org.bukkit.entity.ArmorStand; | ||||||
| import org.bukkit.entity.Bat; | import org.bukkit.entity.Bat; | ||||||
| import org.bukkit.entity.Boat; | import org.bukkit.entity.Boat; | ||||||
|  | import org.bukkit.entity.Breedable; | ||||||
|  | import org.bukkit.entity.ChestBoat; | ||||||
| import org.bukkit.entity.ChestedHorse; | import org.bukkit.entity.ChestedHorse; | ||||||
| import org.bukkit.entity.EnderDragon; | import org.bukkit.entity.EnderDragon; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
| @@ -43,7 +44,6 @@ import org.bukkit.entity.LivingEntity; | |||||||
| import org.bukkit.entity.Painting; | import org.bukkit.entity.Painting; | ||||||
| import org.bukkit.entity.Rabbit; | import org.bukkit.entity.Rabbit; | ||||||
| import org.bukkit.entity.Sheep; | import org.bukkit.entity.Sheep; | ||||||
| import org.bukkit.entity.Slime; |  | ||||||
| import org.bukkit.entity.Tameable; | import org.bukkit.entity.Tameable; | ||||||
| import org.bukkit.inventory.EntityEquipment; | import org.bukkit.inventory.EntityEquipment; | ||||||
| import org.bukkit.inventory.InventoryHolder; | import org.bukkit.inventory.InventoryHolder; | ||||||
| @@ -74,6 +74,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|     private HorseStats horse; |     private HorseStats horse; | ||||||
|     private boolean noGravity; |     private boolean noGravity; | ||||||
|  |  | ||||||
|  |     @SuppressWarnings("deprecation") // Deprecation exists since 1.20, while we support 1.16 onwards | ||||||
|     public ReplicatingEntityWrapper(Entity entity, short depth) { |     public ReplicatingEntityWrapper(Entity entity, short depth) { | ||||||
|         super(entity); |         super(entity); | ||||||
|  |  | ||||||
| @@ -101,11 +102,19 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|             this.noGravity = true; |             this.noGravity = true; | ||||||
|         } |         } | ||||||
|         switch (entity.getType().toString()) { |         switch (entity.getType().toString()) { | ||||||
|             case "BOAT" -> { |             case "BOAT", "ACACIA_BOAT", "BIRCH_BOAT", "CHERRY_BOAT", "DARK_OAK_BOAT", "JUNGLE_BOAT", "MANGROVE_BOAT", | ||||||
|  |                  "OAK_BOAT", "PALE_OAK_BOAT", "SPRUCE_BOAT", "BAMBOO_RAFT" -> { | ||||||
|                 Boat boat = (Boat) entity; |                 Boat boat = (Boat) entity; | ||||||
|                 this.dataByte = getOrdinal(TreeSpecies.values(), boat.getWoodType()); |                 this.dataByte = getOrdinal(Boat.Type.values(), boat.getBoatType()); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|  |             case "ACACIA_CHEST_BOAT", "BIRCH_CHEST_BOAT", "CHERRY_CHEST_BOAT", "DARK_OAK_CHEST_BOAT", | ||||||
|  |                  "JUNGLE_CHEST_BOAT", "MANGROVE_CHEST_BOAT", "OAK_CHEST_BOAT", "PALE_OAK_CHEST_BOAT", | ||||||
|  |                  "SPRUCE_CHEST_BOAT", "BAMBOO_CHEST_RAFT" -> { | ||||||
|  |                 ChestBoat boat = (ChestBoat) entity; | ||||||
|  |                 this.dataByte = getOrdinal(Boat.Type.values(), boat.getBoatType()); | ||||||
|  |                 storeInventory(boat); | ||||||
|  |             } | ||||||
|             case "ARROW", "EGG", "ENDER_CRYSTAL", "ENDER_PEARL", "ENDER_SIGNAL", "EXPERIENCE_ORB", "FALLING_BLOCK", "FIREBALL", |             case "ARROW", "EGG", "ENDER_CRYSTAL", "ENDER_PEARL", "ENDER_SIGNAL", "EXPERIENCE_ORB", "FALLING_BLOCK", "FIREBALL", | ||||||
|                     "FIREWORK", "FISHING_HOOK", "LEASH_HITCH", "LIGHTNING", "MINECART", "MINECART_COMMAND", "MINECART_MOB_SPAWNER", |                     "FIREWORK", "FISHING_HOOK", "LEASH_HITCH", "LIGHTNING", "MINECART", "MINECART_COMMAND", "MINECART_MOB_SPAWNER", | ||||||
|                     "MINECART_TNT", "PLAYER", "PRIMED_TNT", "SLIME", "SMALL_FIREBALL", "SNOWBALL", "MINECART_FURNACE", "SPLASH_POTION", |                     "MINECART_TNT", "PLAYER", "PRIMED_TNT", "SLIME", "SMALL_FIREBALL", "SNOWBALL", "MINECART_FURNACE", "SPLASH_POTION", | ||||||
| @@ -115,7 +124,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             // MISC // |             // MISC // | ||||||
|             case "DROPPED_ITEM" -> { |             case "DROPPED_ITEM", "ITEM" -> { | ||||||
|                 Item item = (Item) entity; |                 Item item = (Item) entity; | ||||||
|                 this.stack = item.getItemStack(); |                 this.stack = item.getItemStack(); | ||||||
|                 return; |                 return; | ||||||
| @@ -145,14 +154,14 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|             } |             } | ||||||
|             // END MISC // |             // END MISC // | ||||||
|             // INVENTORY HOLDER // |             // INVENTORY HOLDER // | ||||||
|             case "MINECART_CHEST", "MINECART_HOPPER" -> { |             case "MINECART_CHEST", "CHEST_MINECART", "MINECART_HOPPER", "HOPPER_MINECART" -> { | ||||||
|                 storeInventory((InventoryHolder) entity); |                 storeInventory((InventoryHolder) entity); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             // START LIVING ENTITY // |             // START LIVING ENTITY // | ||||||
|             // START AGEABLE // |             // START AGEABLE // | ||||||
|             // START TAMEABLE // |             // START TAMEABLE // | ||||||
|             case "HORSE", "DONKEY", "LLAMA", "MULE", "SKELETON_HORSE" -> { |             case "CAMEL", "HORSE", "DONKEY", "LLAMA", "TRADER_LLAMA", "MULE", "SKELETON_HORSE", "ZOMBIE_HORSE" -> { | ||||||
|                 AbstractHorse horse = (AbstractHorse) entity; |                 AbstractHorse horse = (AbstractHorse) entity; | ||||||
|                 this.horse = new HorseStats(); |                 this.horse = new HorseStats(); | ||||||
|                 this.horse.jump = horse.getJumpStrength(); |                 this.horse.jump = horse.getJumpStrength(); | ||||||
| @@ -164,20 +173,19 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                 //this.horse.style = horse.getStyle(); |                 //this.horse.style = horse.getStyle(); | ||||||
|                 //this.horse.color = horse.getColor(); |                 //this.horse.color = horse.getColor(); | ||||||
|                 storeTameable(horse); |                 storeTameable(horse); | ||||||
|                 storeAgeable(horse); |                 storeBreedable(horse); | ||||||
|                 storeLiving(horse); |                 storeLiving(horse); | ||||||
|                 storeInventory(horse); |                 storeInventory(horse); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             // END INVENTORY HOLDER // |             // END INVENTORY HOLDER // | ||||||
|             case "WOLF", "OCELOT" -> { |             case "WOLF", "OCELOT", "CAT", "PARROT" -> { | ||||||
|                 storeTameable((Tameable) entity); |                 storeTameable((Tameable) entity); | ||||||
|                 storeAgeable((Ageable) entity); |                 storeBreedable((Breedable) entity); | ||||||
|                 storeLiving((LivingEntity) entity); |                 storeLiving((LivingEntity) entity); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             // END TAMEABLE // |             // END TAMEABLE // | ||||||
|             //todo fix sheep |  | ||||||
|             case "SHEEP" -> { |             case "SHEEP" -> { | ||||||
|                 Sheep sheep = (Sheep) entity; |                 Sheep sheep = (Sheep) entity; | ||||||
|                 if (sheep.isSheared()) { |                 if (sheep.isSheared()) { | ||||||
| @@ -185,19 +193,19 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                 } else { |                 } else { | ||||||
|                     this.dataByte = (byte) 0; |                     this.dataByte = (byte) 0; | ||||||
|                 } |                 } | ||||||
|                 this.dataByte2 = sheep.getColor().getDyeData(); |                 this.dataByte2 = getOrdinal(DyeColor.values(), sheep.getColor()); | ||||||
|                 storeAgeable(sheep); |                 storeBreedable(sheep); | ||||||
|                 storeLiving(sheep); |                 storeLiving(sheep); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             case "VILLAGER", "CHICKEN", "COW", "MUSHROOM_COW", "PIG", "TURTLE", "POLAR_BEAR" -> { |             case "VILLAGER", "CHICKEN", "COW", "MUSHROOM_COW", "PIG", "TURTLE", "POLAR_BEAR" -> { | ||||||
|                 storeAgeable((Ageable) entity); |                 storeBreedable((Breedable) entity); | ||||||
|                 storeLiving((LivingEntity) entity); |                 storeLiving((LivingEntity) entity); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             case "RABBIT" -> { |             case "RABBIT" -> { | ||||||
|                 this.dataByte = getOrdinal(Rabbit.Type.values(), ((Rabbit) entity).getRabbitType()); |                 this.dataByte = getOrdinal(Rabbit.Type.values(), ((Rabbit) entity).getRabbitType()); | ||||||
|                 storeAgeable((Ageable) entity); |                 storeBreedable((Breedable) entity); | ||||||
|                 storeLiving((LivingEntity) entity); |                 storeLiving((LivingEntity) entity); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
| @@ -266,7 +274,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|             } |             } | ||||||
|             case "SKELETON", "WITHER_SKELETON", "GUARDIAN", "ELDER_GUARDIAN", "GHAST", "MAGMA_CUBE", "SQUID", "PIG_ZOMBIE", "HOGLIN", |             case "SKELETON", "WITHER_SKELETON", "GUARDIAN", "ELDER_GUARDIAN", "GHAST", "MAGMA_CUBE", "SQUID", "PIG_ZOMBIE", "HOGLIN", | ||||||
|                     "ZOMBIFIED_PIGLIN", "PIGLIN", "PIGLIN_BRUTE", "ZOMBIE", "WITHER", "WITCH", "SPIDER", "CAVE_SPIDER", "SILVERFISH", |                     "ZOMBIFIED_PIGLIN", "PIGLIN", "PIGLIN_BRUTE", "ZOMBIE", "WITHER", "WITCH", "SPIDER", "CAVE_SPIDER", "SILVERFISH", | ||||||
|                     "GIANT", "ENDERMAN", "CREEPER", "BLAZE", "SHULKER", "SNOWMAN" -> { |                     "GIANT", "ENDERMAN", "CREEPER", "BLAZE", "SHULKER", "SNOWMAN", "SNOW_GOLEM" -> { | ||||||
|                 storeLiving((LivingEntity) entity); |                 storeLiving((LivingEntity) entity); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
| @@ -381,6 +389,11 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @deprecated Use {@link #restoreBreedable(Breedable)} instead | ||||||
|  |      * @since 7.1.0 | ||||||
|  |      */ | ||||||
|  |     @Deprecated(forRemoval = true, since = "7.1.0") | ||||||
|     private void restoreAgeable(Ageable entity) { |     private void restoreAgeable(Ageable entity) { | ||||||
|         if (!this.aged.adult) { |         if (!this.aged.adult) { | ||||||
|             entity.setBaby(); |             entity.setBaby(); | ||||||
| @@ -391,6 +404,11 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @deprecated Use {@link #storeBreedable(Breedable)} instead | ||||||
|  |      * @since 7.1.0 | ||||||
|  |      */ | ||||||
|  |     @Deprecated(forRemoval = true, since = "7.1.0") | ||||||
|     public void storeAgeable(Ageable aged) { |     public void storeAgeable(Ageable aged) { | ||||||
|         this.aged = new AgeableStats(); |         this.aged = new AgeableStats(); | ||||||
|         this.aged.age = aged.getAge(); |         this.aged.age = aged.getAge(); | ||||||
| @@ -398,6 +416,29 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|         this.aged.adult = aged.isAdult(); |         this.aged.adult = aged.isAdult(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @since 7.1.0 | ||||||
|  |      */ | ||||||
|  |     private void restoreBreedable(Breedable entity) { | ||||||
|  |         if (!this.aged.adult) { | ||||||
|  |             entity.setBaby(); | ||||||
|  |         } | ||||||
|  |         entity.setAgeLock(this.aged.locked); | ||||||
|  |         if (this.aged.age > 0) { | ||||||
|  |             entity.setAge(this.aged.age); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @since 7.1.0 | ||||||
|  |      */ | ||||||
|  |     private void storeBreedable(Breedable breedable) { | ||||||
|  |         this.aged = new AgeableStats(); | ||||||
|  |         this.aged.age = breedable.getAge(); | ||||||
|  |         this.aged.locked = breedable.getAgeLock(); | ||||||
|  |         this.aged.adult = breedable.isAdult(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public void storeTameable(Tameable tamed) { |     public void storeTameable(Tameable tamed) { | ||||||
|         this.tamed = new TameableStats(); |         this.tamed = new TameableStats(); | ||||||
|         this.tamed.owner = tamed.getOwner(); |         this.tamed.owner = tamed.getOwner(); | ||||||
| @@ -415,7 +456,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|         } |         } | ||||||
|         Entity entity; |         Entity entity; | ||||||
|         switch (this.getType().toString()) { |         switch (this.getType().toString()) { | ||||||
|             case "DROPPED_ITEM" -> { |             case "DROPPED_ITEM", "ITEM" -> { | ||||||
|                 return world.dropItem(location, this.stack); |                 return world.dropItem(location, this.stack); | ||||||
|             } |             } | ||||||
|             case "PLAYER", "LEASH_HITCH" -> { |             case "PLAYER", "LEASH_HITCH" -> { | ||||||
| @@ -451,15 +492,25 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|             entity.setGravity(false); |             entity.setGravity(false); | ||||||
|         } |         } | ||||||
|         switch (entity.getType().toString()) { |         switch (entity.getType().toString()) { | ||||||
|             case "BOAT" -> { |             case "BOAT", "ACACIA_BOAT", "BIRCH_BOAT", "CHERRY_BOAT", "DARK_OAK_BOAT", "JUNGLE_BOAT", "MANGROVE_BOAT", | ||||||
|  |                  "OAK_BOAT", "PALE_OAK_BOAT", "SPRUCE_BOAT", "BAMBOO_RAFT" -> { | ||||||
|                 Boat boat = (Boat) entity; |                 Boat boat = (Boat) entity; | ||||||
|                 boat.setWoodType(TreeSpecies.values()[dataByte]); |                 boat.setBoatType(Boat.Type.values()[dataByte]); | ||||||
|                 return entity; |                 return entity; | ||||||
|             } |             } | ||||||
|             case "SLIME" -> { |             case "ACACIA_CHEST_BOAT", "BIRCH_CHEST_BOAT", "CHERRY_CHEST_BOAT", "DARK_OAK_CHEST_BOAT", | ||||||
|  |                  "JUNGLE_CHEST_BOAT", "MANGROVE_CHEST_BOAT", "OAK_CHEST_BOAT", "PALE_OAK_CHEST_BOAT", | ||||||
|  |                  "SPRUCE_CHEST_BOAT", "BAMBOO_CHEST_RAFT" -> { | ||||||
|  |                 ChestBoat boat = (ChestBoat) entity; | ||||||
|  |                 boat.setBoatType(Boat.Type.values()[dataByte]); | ||||||
|  |                 restoreInventory(boat); | ||||||
|  |                 return entity; | ||||||
|  |             } | ||||||
|  |             // SLIME is not even stored | ||||||
|  |             /* case "SLIME" -> { | ||||||
|                 ((Slime) entity).setSize(this.dataByte); |                 ((Slime) entity).setSize(this.dataByte); | ||||||
|                 return entity; |                 return entity; | ||||||
|             } |             } */ | ||||||
|             case "ARROW", "EGG", "ENDER_CRYSTAL", "ENDER_PEARL", "ENDER_SIGNAL", "DROPPED_ITEM", "EXPERIENCE_ORB", "FALLING_BLOCK", |             case "ARROW", "EGG", "ENDER_CRYSTAL", "ENDER_PEARL", "ENDER_SIGNAL", "DROPPED_ITEM", "EXPERIENCE_ORB", "FALLING_BLOCK", | ||||||
|                     "FIREBALL", "FIREWORK", "FISHING_HOOK", "LEASH_HITCH", "LIGHTNING", "MINECART", "MINECART_COMMAND", |                     "FIREBALL", "FIREWORK", "FISHING_HOOK", "LEASH_HITCH", "LIGHTNING", "MINECART", "MINECART_COMMAND", | ||||||
|                     "MINECART_MOB_SPAWNER", "MINECART_TNT", "PLAYER", "PRIMED_TNT", "SMALL_FIREBALL", "SNOWBALL", |                     "MINECART_MOB_SPAWNER", "MINECART_TNT", "PLAYER", "PRIMED_TNT", "SMALL_FIREBALL", "SNOWBALL", | ||||||
| @@ -483,14 +534,14 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|             } |             } | ||||||
|             // END MISC // |             // END MISC // | ||||||
|             // INVENTORY HOLDER // |             // INVENTORY HOLDER // | ||||||
|             case "MINECART_CHEST", "MINECART_HOPPER" -> { |             case "MINECART_CHEST", "CHEST_MINECART", "MINECART_HOPPER", "HOPPER_MINECART" -> { | ||||||
|                 restoreInventory((InventoryHolder) entity); |                 restoreInventory((InventoryHolder) entity); | ||||||
|                 return entity; |                 return entity; | ||||||
|             } |             } | ||||||
|             // START LIVING ENTITY // |             // START LIVING ENTITY // | ||||||
|             // START AGEABLE // |             // START AGEABLE // | ||||||
|             // START TAMEABLE // |             // START TAMEABLE // | ||||||
|             case "HORSE", "LLAMA", "SKELETON_HORSE", "DONKEY", "MULE" -> { |             case "CAMEL", "HORSE", "DONKEY", "LLAMA", "TRADER_LLAMA", "MULE", "SKELETON_HORSE", "ZOMBIE_HORSE" -> { | ||||||
|                 AbstractHorse horse = (AbstractHorse) entity; |                 AbstractHorse horse = (AbstractHorse) entity; | ||||||
|                 horse.setJumpStrength(this.horse.jump); |                 horse.setJumpStrength(this.horse.jump); | ||||||
|                 if (horse instanceof ChestedHorse) { |                 if (horse instanceof ChestedHorse) { | ||||||
| @@ -501,15 +552,15 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                 //horse.setStyle(this.horse.style); |                 //horse.setStyle(this.horse.style); | ||||||
|                 //horse.setColor(this.horse.color); |                 //horse.setColor(this.horse.color); | ||||||
|                 restoreTameable(horse); |                 restoreTameable(horse); | ||||||
|                 restoreAgeable(horse); |                 restoreBreedable(horse); | ||||||
|                 restoreLiving(horse); |                 restoreLiving(horse); | ||||||
|                 restoreInventory(horse); |                 restoreInventory(horse); | ||||||
|                 return entity; |                 return entity; | ||||||
|             } |             } | ||||||
|             // END INVENTORY HOLDER // |             // END INVENTORY HOLDER // | ||||||
|             case "WOLF", "OCELOT" -> { |             case "WOLF", "OCELOT", "CAT", "PARROT" -> { | ||||||
|                 restoreTameable((Tameable) entity); |                 restoreTameable((Tameable) entity); | ||||||
|                 restoreAgeable((Ageable) entity); |                 restoreBreedable((Breedable) entity); | ||||||
|                 restoreLiving((LivingEntity) entity); |                 restoreLiving((LivingEntity) entity); | ||||||
|                 return entity; |                 return entity; | ||||||
|             } |             } | ||||||
| @@ -520,14 +571,14 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                     sheep.setSheared(true); |                     sheep.setSheared(true); | ||||||
|                 } |                 } | ||||||
|                 if (this.dataByte2 != 0) { |                 if (this.dataByte2 != 0) { | ||||||
|                     sheep.setColor(DyeColor.getByDyeData(this.dataByte2)); |                     sheep.setColor(DyeColor.values()[this.dataByte2]); | ||||||
|                 } |                 } | ||||||
|                 restoreAgeable(sheep); |                 restoreBreedable(sheep); | ||||||
|                 restoreLiving(sheep); |                 restoreLiving(sheep); | ||||||
|                 return sheep; |                 return sheep; | ||||||
|             } |             } | ||||||
|             case "VILLAGER", "CHICKEN", "COW", "TURTLE", "POLAR_BEAR", "MUSHROOM_COW", "PIG" -> { |             case "VILLAGER", "CHICKEN", "COW", "TURTLE", "POLAR_BEAR", "MUSHROOM_COW", "PIG" -> { | ||||||
|                 restoreAgeable((Ageable) entity); |                 restoreBreedable((Breedable) entity); | ||||||
|                 restoreLiving((LivingEntity) entity); |                 restoreLiving((LivingEntity) entity); | ||||||
|                 return entity; |                 return entity; | ||||||
|             } |             } | ||||||
| @@ -536,7 +587,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                 if (this.dataByte != 0) { |                 if (this.dataByte != 0) { | ||||||
|                     ((Rabbit) entity).setRabbitType(Rabbit.Type.values()[this.dataByte]); |                     ((Rabbit) entity).setRabbitType(Rabbit.Type.values()[this.dataByte]); | ||||||
|                 } |                 } | ||||||
|                 restoreAgeable((Ageable) entity); |                 restoreBreedable((Breedable) entity); | ||||||
|                 restoreLiving((LivingEntity) entity); |                 restoreLiving((LivingEntity) entity); | ||||||
|                 return entity; |                 return entity; | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -52,7 +52,7 @@ import org.jetbrains.annotations.Nullable; | |||||||
|  |  | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.Arrays; | import java.util.Arrays; | ||||||
| import java.util.EnumSet; | import java.util.HashSet; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Random; | import java.util.Random; | ||||||
| import java.util.Set; | import java.util.Set; | ||||||
| @@ -445,7 +445,7 @@ public class BukkitPlotGenerator extends ChunkGenerator implements GeneratorWrap | |||||||
|         private static final List<Biome> BIOMES; |         private static final List<Biome> BIOMES; | ||||||
|  |  | ||||||
|         static { |         static { | ||||||
|             Set<Biome> disabledBiomes = EnumSet.of(Biome.CUSTOM); |             Set<Biome> disabledBiomes = new HashSet<>(List.of(Biome.CUSTOM)); | ||||||
|             if (PlotSquared.platform().serverVersion()[1] <= 19) { |             if (PlotSquared.platform().serverVersion()[1] <= 19) { | ||||||
|                 final Biome cherryGrove = Registry.BIOME.get(NamespacedKey.minecraft("cherry_grove")); |                 final Biome cherryGrove = Registry.BIOME.get(NamespacedKey.minecraft("cherry_grove")); | ||||||
|                 if (cherryGrove != null) { |                 if (cherryGrove != null) { | ||||||
|   | |||||||
| @@ -23,13 +23,13 @@ import com.google.inject.Provides; | |||||||
| import com.google.inject.Singleton; | import com.google.inject.Singleton; | ||||||
| import com.google.inject.assistedinject.FactoryModuleBuilder; | import com.google.inject.assistedinject.FactoryModuleBuilder; | ||||||
| import com.plotsquared.bukkit.BukkitPlatform; | import com.plotsquared.bukkit.BukkitPlatform; | ||||||
|  | import com.plotsquared.bukkit.listener.ServerListener; | ||||||
| import com.plotsquared.bukkit.listener.SingleWorldListener; | import com.plotsquared.bukkit.listener.SingleWorldListener; | ||||||
| import com.plotsquared.bukkit.player.BukkitPlayerManager; | import com.plotsquared.bukkit.player.BukkitPlayerManager; | ||||||
| import com.plotsquared.bukkit.queue.BukkitChunkCoordinator; | import com.plotsquared.bukkit.queue.BukkitChunkCoordinator; | ||||||
| import com.plotsquared.bukkit.queue.BukkitQueueCoordinator; | import com.plotsquared.bukkit.queue.BukkitQueueCoordinator; | ||||||
| import com.plotsquared.bukkit.schematic.BukkitSchematicHandler; | import com.plotsquared.bukkit.schematic.BukkitSchematicHandler; | ||||||
| import com.plotsquared.bukkit.util.BukkitChunkManager; | import com.plotsquared.bukkit.util.BukkitChunkManager; | ||||||
| import com.plotsquared.bukkit.util.BukkitEconHandler; |  | ||||||
| import com.plotsquared.bukkit.util.BukkitInventoryUtil; | import com.plotsquared.bukkit.util.BukkitInventoryUtil; | ||||||
| import com.plotsquared.bukkit.util.BukkitRegionManager; | import com.plotsquared.bukkit.util.BukkitRegionManager; | ||||||
| import com.plotsquared.bukkit.util.BukkitSetupUtils; | import com.plotsquared.bukkit.util.BukkitSetupUtils; | ||||||
| @@ -47,6 +47,9 @@ import com.plotsquared.core.inject.factory.ChunkCoordinatorBuilderFactory; | |||||||
| import com.plotsquared.core.inject.factory.ChunkCoordinatorFactory; | import com.plotsquared.core.inject.factory.ChunkCoordinatorFactory; | ||||||
| import com.plotsquared.core.inject.factory.HybridPlotWorldFactory; | import com.plotsquared.core.inject.factory.HybridPlotWorldFactory; | ||||||
| import com.plotsquared.core.inject.factory.ProgressSubscriberFactory; | import com.plotsquared.core.inject.factory.ProgressSubscriberFactory; | ||||||
|  | import com.plotsquared.core.player.OfflinePlotPlayer; | ||||||
|  | import com.plotsquared.core.player.PlotPlayer; | ||||||
|  | import com.plotsquared.core.plot.PlotArea; | ||||||
| import com.plotsquared.core.plot.world.DefaultPlotAreaManager; | import com.plotsquared.core.plot.world.DefaultPlotAreaManager; | ||||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | import com.plotsquared.core.plot.world.PlotAreaManager; | ||||||
| import com.plotsquared.core.plot.world.SinglePlotAreaManager; | import com.plotsquared.core.plot.world.SinglePlotAreaManager; | ||||||
| @@ -72,6 +75,8 @@ import org.bukkit.command.ConsoleCommandSender; | |||||||
| import org.bukkit.plugin.java.JavaPlugin; | import org.bukkit.plugin.java.JavaPlugin; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
|  | import java.util.Objects; | ||||||
|  |  | ||||||
| public class BukkitModule extends AbstractModule { | public class BukkitModule extends AbstractModule { | ||||||
|  |  | ||||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + BukkitModule.class.getSimpleName()); |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + BukkitModule.class.getSimpleName()); | ||||||
| @@ -128,21 +133,64 @@ public class BukkitModule extends AbstractModule { | |||||||
|     @Provides |     @Provides | ||||||
|     @Singleton |     @Singleton | ||||||
|     @NonNull EconHandler provideEconHandler() { |     @NonNull EconHandler provideEconHandler() { | ||||||
|         if (!Settings.Enabled_Components.ECONOMY) { |         if (!Settings.Enabled_Components.ECONOMY || !Bukkit.getPluginManager().isPluginEnabled("Vault")) { | ||||||
|             return EconHandler.nullEconHandler(); |             return EconHandler.nullEconHandler(); | ||||||
|         } |         } | ||||||
|         if (Bukkit.getPluginManager().isPluginEnabled("Vault")) { |         // Guice eagerly initializes singletons, so we need to bring the laziness ourselves | ||||||
|             try { |         return new LazyEconHandler(); | ||||||
|                 BukkitEconHandler econHandler = new BukkitEconHandler(); |     } | ||||||
|                 if (!econHandler.init()) { |  | ||||||
|                     LOGGER.warn("Economy is enabled but no plugin is providing an economy service. Falling back..."); |     private static final class LazyEconHandler extends EconHandler implements ServerListener.MutableEconHandler { | ||||||
|                     return EconHandler.nullEconHandler(); |         private volatile EconHandler implementation; | ||||||
|                 } |  | ||||||
|                 return econHandler; |         public void setImplementation(EconHandler econHandler) { | ||||||
|             } catch (final Exception ignored) { |             this.implementation = econHandler; | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|         return EconHandler.nullEconHandler(); |  | ||||||
|  |         @Override | ||||||
|  |         public boolean init() { | ||||||
|  |             return get().init(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public double getBalance(final PlotPlayer<?> player) { | ||||||
|  |             return get().getBalance(player); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public void withdrawMoney(final PlotPlayer<?> player, final double amount) { | ||||||
|  |             get().withdrawMoney(player, amount); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public void depositMoney(final PlotPlayer<?> player, final double amount) { | ||||||
|  |             get().depositMoney(player, amount); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public void depositMoney(final OfflinePlotPlayer player, final double amount) { | ||||||
|  |             get().depositMoney(player, amount); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public boolean isEnabled(final PlotArea plotArea) { | ||||||
|  |             return get().isEnabled(plotArea); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public @NonNull String format(final double balance) { | ||||||
|  |             return get().format(balance); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public boolean isSupported() { | ||||||
|  |             return get().isSupported(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private EconHandler get() { | ||||||
|  |             return Objects.requireNonNull(this.implementation, "EconHandler not ready yet."); | ||||||
|  |         } | ||||||
|  |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -22,12 +22,20 @@ import com.google.inject.AbstractModule; | |||||||
| import com.google.inject.Provides; | import com.google.inject.Provides; | ||||||
| import com.google.inject.Singleton; | import com.google.inject.Singleton; | ||||||
| import com.plotsquared.bukkit.permissions.BukkitPermissionHandler; | import com.plotsquared.bukkit.permissions.BukkitPermissionHandler; | ||||||
|  | import com.plotsquared.bukkit.permissions.BukkitRangedPermissionResolver; | ||||||
|  | import com.plotsquared.bukkit.permissions.LuckPermsRangedPermissionResolver; | ||||||
| import com.plotsquared.bukkit.permissions.VaultPermissionHandler; | import com.plotsquared.bukkit.permissions.VaultPermissionHandler; | ||||||
|  | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.permissions.PermissionHandler; | import com.plotsquared.core.permissions.PermissionHandler; | ||||||
|  | import com.plotsquared.core.permissions.RangedPermissionResolver; | ||||||
|  | import org.apache.logging.log4j.LogManager; | ||||||
|  | import org.apache.logging.log4j.Logger; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
|  |  | ||||||
| public class PermissionModule extends AbstractModule { | public class PermissionModule extends AbstractModule { | ||||||
|  |  | ||||||
|  |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + PermissionModule.class.getSimpleName()); | ||||||
|  |  | ||||||
|     @Provides |     @Provides | ||||||
|     @Singleton |     @Singleton | ||||||
|     PermissionHandler providePermissionHandler() { |     PermissionHandler providePermissionHandler() { | ||||||
| @@ -40,4 +48,18 @@ public class PermissionModule extends AbstractModule { | |||||||
|         return new BukkitPermissionHandler(); |         return new BukkitPermissionHandler(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Provides | ||||||
|  |     @Singleton | ||||||
|  |     RangedPermissionResolver provideRangedPermissionResolver() { | ||||||
|  |         if (Settings.Permissions.USE_LUCKPERMS_RANGE_RESOLVER) { | ||||||
|  |             if (Bukkit.getPluginManager().isPluginEnabled("LuckPerms")) { | ||||||
|  |                 LOGGER.info("Using experimental LuckPerms ranged permission resolver"); | ||||||
|  |                 return new LuckPermsRangedPermissionResolver(); | ||||||
|  |             } | ||||||
|  |             LOGGER.warn("Enabled LuckPerms ranged permission resolver, but LuckPerms is not installed. " + | ||||||
|  |                     "Falling back to default Bukkit ranged permission resolver"); | ||||||
|  |         } | ||||||
|  |         return new BukkitRangedPermissionResolver(); | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -24,7 +24,6 @@ import com.plotsquared.bukkit.util.BukkitUtil; | |||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.database.DBFunc; |  | ||||||
| import com.plotsquared.core.location.Location; | import com.plotsquared.core.location.Location; | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| @@ -33,6 +32,7 @@ import com.plotsquared.core.plot.PlotArea; | |||||||
| import com.plotsquared.core.plot.flag.implementations.BlockBurnFlag; | import com.plotsquared.core.plot.flag.implementations.BlockBurnFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.BlockIgnitionFlag; | import com.plotsquared.core.plot.flag.implementations.BlockIgnitionFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.BreakFlag; | import com.plotsquared.core.plot.flag.implementations.BreakFlag; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.ConcreteHardenFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.CoralDryFlag; | import com.plotsquared.core.plot.flag.implementations.CoralDryFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.CropGrowFlag; | import com.plotsquared.core.plot.flag.implementations.CropGrowFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.DisablePhysicsFlag; | import com.plotsquared.core.plot.flag.implementations.DisablePhysicsFlag; | ||||||
| @@ -47,7 +47,6 @@ import com.plotsquared.core.plot.flag.implementations.LeafDecayFlag; | |||||||
| import com.plotsquared.core.plot.flag.implementations.LiquidFlowFlag; | import com.plotsquared.core.plot.flag.implementations.LiquidFlowFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.MycelGrowFlag; | import com.plotsquared.core.plot.flag.implementations.MycelGrowFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.PlaceFlag; | import com.plotsquared.core.plot.flag.implementations.PlaceFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.RedstoneFlag; |  | ||||||
| import com.plotsquared.core.plot.flag.implementations.SnowFormFlag; | import com.plotsquared.core.plot.flag.implementations.SnowFormFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.SnowMeltFlag; | import com.plotsquared.core.plot.flag.implementations.SnowMeltFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.SoilDryFlag; | import com.plotsquared.core.plot.flag.implementations.SoilDryFlag; | ||||||
| @@ -61,7 +60,6 @@ import com.plotsquared.core.util.task.TaskTime; | |||||||
| import com.sk89q.worldedit.WorldEdit; | import com.sk89q.worldedit.WorldEdit; | ||||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||||
| import com.sk89q.worldedit.world.block.BlockType; | import com.sk89q.worldedit.world.block.BlockType; | ||||||
| import net.kyori.adventure.text.Component; |  | ||||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | import net.kyori.adventure.text.minimessage.tag.Tag; | ||||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
| @@ -91,11 +89,9 @@ import org.bukkit.event.block.BlockFromToEvent; | |||||||
| import org.bukkit.event.block.BlockGrowEvent; | import org.bukkit.event.block.BlockGrowEvent; | ||||||
| import org.bukkit.event.block.BlockIgniteEvent; | import org.bukkit.event.block.BlockIgniteEvent; | ||||||
| import org.bukkit.event.block.BlockMultiPlaceEvent; | import org.bukkit.event.block.BlockMultiPlaceEvent; | ||||||
| import org.bukkit.event.block.BlockPhysicsEvent; |  | ||||||
| import org.bukkit.event.block.BlockPistonExtendEvent; | import org.bukkit.event.block.BlockPistonExtendEvent; | ||||||
| import org.bukkit.event.block.BlockPistonRetractEvent; | import org.bukkit.event.block.BlockPistonRetractEvent; | ||||||
| import org.bukkit.event.block.BlockPlaceEvent; | import org.bukkit.event.block.BlockPlaceEvent; | ||||||
| import org.bukkit.event.block.BlockRedstoneEvent; |  | ||||||
| import org.bukkit.event.block.BlockSpreadEvent; | import org.bukkit.event.block.BlockSpreadEvent; | ||||||
| import org.bukkit.event.block.CauldronLevelChangeEvent; | import org.bukkit.event.block.CauldronLevelChangeEvent; | ||||||
| import org.bukkit.event.block.EntityBlockFormEvent; | import org.bukkit.event.block.EntityBlockFormEvent; | ||||||
| @@ -110,10 +106,6 @@ import org.checkerframework.checker.nullness.qual.NonNull; | |||||||
| import java.util.Iterator; | import java.util.Iterator; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Objects; | import java.util.Objects; | ||||||
| import java.util.Set; |  | ||||||
| import java.util.UUID; |  | ||||||
| import java.util.stream.Collectors; |  | ||||||
| import java.util.stream.Stream; |  | ||||||
|  |  | ||||||
| import static org.bukkit.Tag.CORALS; | import static org.bukkit.Tag.CORALS; | ||||||
| import static org.bukkit.Tag.CORAL_BLOCKS; | import static org.bukkit.Tag.CORAL_BLOCKS; | ||||||
| @@ -121,20 +113,6 @@ import static org.bukkit.Tag.WALL_CORALS; | |||||||
|  |  | ||||||
| @SuppressWarnings("unused") | @SuppressWarnings("unused") | ||||||
| public class BlockEventListener implements Listener { | public class BlockEventListener implements Listener { | ||||||
|  |  | ||||||
|     private static final Set<Material> PISTONS = Set.of( |  | ||||||
|             Material.PISTON, |  | ||||||
|             Material.STICKY_PISTON |  | ||||||
|     ); |  | ||||||
|     private static final Set<Material> PHYSICS_BLOCKS = Set.of( |  | ||||||
|             Material.TURTLE_EGG, |  | ||||||
|             Material.TURTLE_SPAWN_EGG |  | ||||||
|     ); |  | ||||||
|     private static final Set<Material> SNOW = Stream.of(Material.values()) // needed as Tag.SNOW isn't present in 1.16.5 |  | ||||||
|             .filter(material -> material.name().contains("SNOW")) |  | ||||||
|             .filter(Material::isBlock) |  | ||||||
|             .collect(Collectors.toUnmodifiableSet()); |  | ||||||
|  |  | ||||||
|     private final PlotAreaManager plotAreaManager; |     private final PlotAreaManager plotAreaManager; | ||||||
|     private final WorldEdit worldEdit; |     private final WorldEdit worldEdit; | ||||||
|  |  | ||||||
| @@ -163,111 +141,6 @@ public class BlockEventListener implements Listener { | |||||||
|         }, TaskTime.ticks(3L)); |         }, TaskTime.ticks(3L)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler |  | ||||||
|     public void onRedstoneEvent(BlockRedstoneEvent event) { |  | ||||||
|         Block block = event.getBlock(); |  | ||||||
|         Location location = BukkitUtil.adapt(block.getLocation()); |  | ||||||
|         PlotArea area = location.getPlotArea(); |  | ||||||
|         if (area == null) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         Plot plot = location.getOwnedPlot(); |  | ||||||
|         if (plot == null) { |  | ||||||
|             if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, RedstoneFlag.class, false)) { |  | ||||||
|                 event.setNewCurrent(0); |  | ||||||
|             } |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         if (!plot.getFlag(RedstoneFlag.class)) { |  | ||||||
|             event.setNewCurrent(0); |  | ||||||
|             plot.debug("Redstone event was cancelled because redstone = false"); |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         if (Settings.Redstone.DISABLE_OFFLINE) { |  | ||||||
|             boolean disable = false; |  | ||||||
|             if (!DBFunc.SERVER.equals(plot.getOwner())) { |  | ||||||
|                 if (plot.isMerged()) { |  | ||||||
|                     disable = true; |  | ||||||
|                     for (UUID owner : plot.getOwners()) { |  | ||||||
|                         if (PlotSquared.platform().playerManager().getPlayerIfExists(owner) != null) { |  | ||||||
|                             disable = false; |  | ||||||
|                             break; |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } else { |  | ||||||
|                     disable = PlotSquared.platform().playerManager().getPlayerIfExists(plot.getOwnerAbs()) == null; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             if (disable) { |  | ||||||
|                 for (UUID trusted : plot.getTrusted()) { |  | ||||||
|                     if (PlotSquared.platform().playerManager().getPlayerIfExists(trusted) != null) { |  | ||||||
|                         disable = false; |  | ||||||
|                         break; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 if (disable) { |  | ||||||
|                     event.setNewCurrent(0); |  | ||||||
|                     plot.debug("Redstone event was cancelled because no trusted player was in the plot"); |  | ||||||
|                     return; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         if (Settings.Redstone.DISABLE_UNOCCUPIED) { |  | ||||||
|             for (final PlotPlayer<?> player : PlotSquared.platform().playerManager().getPlayers()) { |  | ||||||
|                 if (plot.equals(player.getCurrentPlot())) { |  | ||||||
|                     return; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             event.setNewCurrent(0); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @EventHandler(ignoreCancelled = true, priority = EventPriority.HIGHEST) |  | ||||||
|     public void onPhysicsEvent(BlockPhysicsEvent event) { |  | ||||||
|         Block block = event.getBlock(); |  | ||||||
|         Location location = BukkitUtil.adapt(block.getLocation()); |  | ||||||
|         PlotArea area = location.getPlotArea(); |  | ||||||
|         if (area == null) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         Plot plot = area.getOwnedPlotAbs(location); |  | ||||||
|         if (plot == null) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         if (event.getChangedType().hasGravity() && plot.getFlag(DisablePhysicsFlag.class)) { |  | ||||||
|             event.setCancelled(true); |  | ||||||
|             sendBlockChange(event.getBlock().getLocation(), event.getBlock().getBlockData()); |  | ||||||
|             plot.debug("Prevented block physics and resent block change because disable-physics = true"); |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         if (event.getChangedType() == Material.COMPARATOR) { |  | ||||||
|             if (!plot.getFlag(RedstoneFlag.class)) { |  | ||||||
|                 event.setCancelled(true); |  | ||||||
|                 plot.debug("Prevented comparator update because redstone = false"); |  | ||||||
|             } |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         if (PHYSICS_BLOCKS.contains(event.getChangedType())) { |  | ||||||
|             if (plot.getFlag(DisablePhysicsFlag.class)) { |  | ||||||
|                 event.setCancelled(true); |  | ||||||
|                 plot.debug("Prevented block physics because disable-physics = true"); |  | ||||||
|             } |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         if (Settings.Redstone.DETECT_INVALID_EDGE_PISTONS) { |  | ||||||
|             if (PISTONS.contains(block.getType())) { |  | ||||||
|                 org.bukkit.block.data.Directional piston = (org.bukkit.block.data.Directional) block.getBlockData(); |  | ||||||
|                 final BlockFace facing = piston.getFacing(); |  | ||||||
|                 location = location.add(facing.getModX(), facing.getModY(), facing.getModZ()); |  | ||||||
|                 Plot newPlot = area.getOwnedPlotAbs(location); |  | ||||||
|                 if (!plot.equals(newPlot)) { |  | ||||||
|                     event.setCancelled(true); |  | ||||||
|                     plot.debug("Prevented piston update because of invalid edge piston detection"); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|     public void blockCreate(BlockPlaceEvent event) { |     public void blockCreate(BlockPlaceEvent event) { | ||||||
|         Location location = BukkitUtil.adapt(event.getBlock().getLocation()); |         Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||||
| @@ -281,13 +154,6 @@ public class BlockEventListener implements Listener { | |||||||
|         if (plot != null) { |         if (plot != null) { | ||||||
|             if (area.notifyIfOutsideBuildArea(pp, location.getY())) { |             if (area.notifyIfOutsideBuildArea(pp, location.getY())) { | ||||||
|                 event.setCancelled(true); |                 event.setCancelled(true); | ||||||
|                 pp.sendMessage( |  | ||||||
|                         TranslatableCaption.of("height.height_limit"), |  | ||||||
|                         TagResolver.builder() |  | ||||||
|                                 .tag("minheight", Tag.inserting(Component.text(area.getMinBuildHeight()))) |  | ||||||
|                                 .tag("maxheight", Tag.inserting(Component.text(area.getMaxBuildHeight()))) |  | ||||||
|                                 .build() |  | ||||||
|                 ); |  | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             if (!plot.hasOwner()) { |             if (!plot.hasOwner()) { | ||||||
| @@ -379,13 +245,6 @@ public class BlockEventListener implements Listener { | |||||||
|                 } |                 } | ||||||
|             } else if (area.notifyIfOutsideBuildArea(plotPlayer, location.getY())) { |             } else if (area.notifyIfOutsideBuildArea(plotPlayer, location.getY())) { | ||||||
|                 event.setCancelled(true); |                 event.setCancelled(true); | ||||||
|                 plotPlayer.sendMessage( |  | ||||||
|                         TranslatableCaption.of("height.height_limit"), |  | ||||||
|                         TagResolver.builder() |  | ||||||
|                                 .tag("minheight", Tag.inserting(Component.text(area.getMinBuildHeight()))) |  | ||||||
|                                 .tag("maxheight", Tag.inserting(Component.text(area.getMaxBuildHeight()))) |  | ||||||
|                                 .build() |  | ||||||
|                 ); |  | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             if (!plot.hasOwner()) { |             if (!plot.hasOwner()) { | ||||||
| @@ -586,6 +445,12 @@ public class BlockEventListener implements Listener { | |||||||
|                 event.setCancelled(true); |                 event.setCancelled(true); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |         if (event.getNewState().getType().toString().endsWith("CONCRETE")) { | ||||||
|  |             if (!plot.getFlag(ConcreteHardenFlag.class)) { | ||||||
|  |                 plot.debug("Concrete powder could not harden because concrete-harden = false"); | ||||||
|  |                 event.setCancelled(true); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
| @@ -660,7 +525,11 @@ public class BlockEventListener implements Listener { | |||||||
|                 BlockBreakEvent call = new BlockBreakEvent(block, player); |                 BlockBreakEvent call = new BlockBreakEvent(block, player); | ||||||
|                 Bukkit.getServer().getPluginManager().callEvent(call); |                 Bukkit.getServer().getPluginManager().callEvent(call); | ||||||
|                 if (!call.isCancelled()) { |                 if (!call.isCancelled()) { | ||||||
|                     event.getBlock().breakNaturally(); |                     if (Settings.Flags.INSTABREAK_CONSIDER_TOOL) { | ||||||
|  |                         block.breakNaturally(event.getItemInHand()); | ||||||
|  |                     } else { | ||||||
|  |                         block.breakNaturally(); | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             // == rather than <= as we only care about the "ground level" not being destroyed |             // == rather than <= as we only care about the "ground level" not being destroyed | ||||||
| @@ -1331,18 +1200,9 @@ public class BlockEventListener implements Listener { | |||||||
|             if (pp.hasPermission(Permission.PERMISSION_ADMIN_BUILD_HEIGHT_LIMIT)) { |             if (pp.hasPermission(Permission.PERMISSION_ADMIN_BUILD_HEIGHT_LIMIT)) { | ||||||
|                 continue; |                 continue; | ||||||
|             } |             } | ||||||
|             if (currentLocation.getY() >= area.getMaxBuildHeight() || currentLocation.getY() < area.getMinBuildHeight()) { |             if (area.notifyIfOutsideBuildArea(pp, currentLocation.getY())) { | ||||||
|                 pp.sendMessage( |                 event.setCancelled(true); | ||||||
|                         TranslatableCaption.of("height.height_limit"), |                 break; | ||||||
|                         TagResolver.builder() |  | ||||||
|                                 .tag("minheight", Tag.inserting(Component.text(area.getMinBuildHeight()))) |  | ||||||
|                                 .tag("maxheight", Tag.inserting(Component.text(area.getMaxBuildHeight()))) |  | ||||||
|                                 .build() |  | ||||||
|                 ); |  | ||||||
|                 if (area.notifyIfOutsideBuildArea(pp, currentLocation.getY())) { |  | ||||||
|                     event.setCancelled(true); |  | ||||||
|                     break; |  | ||||||
|                 } |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -28,7 +28,6 @@ import com.plotsquared.core.plot.flag.implementations.CopperOxideFlag; | |||||||
| import com.plotsquared.core.plot.flag.implementations.MiscInteractFlag; | import com.plotsquared.core.plot.flag.implementations.MiscInteractFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.SculkSensorInteractFlag; | import com.plotsquared.core.plot.flag.implementations.SculkSensorInteractFlag; | ||||||
| import com.plotsquared.core.util.PlotFlagUtil; | import com.plotsquared.core.util.PlotFlagUtil; | ||||||
| import org.bukkit.Material; |  | ||||||
| import org.bukkit.block.Block; | import org.bukkit.block.Block; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
| import org.bukkit.entity.Item; | import org.bukkit.entity.Item; | ||||||
| @@ -42,31 +41,11 @@ import org.bukkit.event.block.BlockReceiveGameEvent; | |||||||
|  |  | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Objects; | import java.util.Objects; | ||||||
| import java.util.Set; |  | ||||||
| import java.util.UUID; | import java.util.UUID; | ||||||
|  |  | ||||||
| @SuppressWarnings("unused") | @SuppressWarnings("unused") | ||||||
| public class BlockEventListener117 implements Listener { | public class BlockEventListener117 implements Listener { | ||||||
|  |  | ||||||
|     private static final Set<Material> COPPER_OXIDIZING = Set.of( |  | ||||||
|             Material.COPPER_BLOCK, |  | ||||||
|             Material.EXPOSED_COPPER, |  | ||||||
|             Material.WEATHERED_COPPER, |  | ||||||
|             Material.OXIDIZED_COPPER, |  | ||||||
|             Material.CUT_COPPER, |  | ||||||
|             Material.EXPOSED_CUT_COPPER, |  | ||||||
|             Material.WEATHERED_CUT_COPPER, |  | ||||||
|             Material.OXIDIZED_CUT_COPPER, |  | ||||||
|             Material.CUT_COPPER_STAIRS, |  | ||||||
|             Material.EXPOSED_CUT_COPPER_STAIRS, |  | ||||||
|             Material.WEATHERED_CUT_COPPER_STAIRS, |  | ||||||
|             Material.OXIDIZED_CUT_COPPER_STAIRS, |  | ||||||
|             Material.CUT_COPPER_SLAB, |  | ||||||
|             Material.EXPOSED_CUT_COPPER_SLAB, |  | ||||||
|             Material.WEATHERED_CUT_COPPER_SLAB, |  | ||||||
|             Material.OXIDIZED_CUT_COPPER_SLAB |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     @Inject |     @Inject | ||||||
|     public BlockEventListener117() { |     public BlockEventListener117() { | ||||||
|     } |     } | ||||||
| @@ -134,7 +113,7 @@ public class BlockEventListener117 implements Listener { | |||||||
|     public void onBlockFertilize(BlockFertilizeEvent event) { |     public void onBlockFertilize(BlockFertilizeEvent event) { | ||||||
|         Block block = event.getBlock(); |         Block block = event.getBlock(); | ||||||
|         List<org.bukkit.block.BlockState> blocks = event.getBlocks(); |         List<org.bukkit.block.BlockState> blocks = event.getBlocks(); | ||||||
|         Location location = BukkitUtil.adapt(blocks.get(0).getLocation()); |         Location location = BukkitUtil.adapt(block.getLocation()); | ||||||
|  |  | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
| @@ -184,7 +163,7 @@ public class BlockEventListener117 implements Listener { | |||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         if (COPPER_OXIDIZING.contains(event.getNewState().getType())) { |         if (event.getNewState().getType().name().contains("COPPER")) { | ||||||
|             if (!plot.getFlag(CopperOxideFlag.class)) { |             if (!plot.getFlag(CopperOxideFlag.class)) { | ||||||
|                 plot.debug("Copper could not oxide because copper-oxide = false"); |                 plot.debug("Copper could not oxide because copper-oxide = false"); | ||||||
|                 event.setCancelled(true); |                 event.setCancelled(true); | ||||||
|   | |||||||
| @@ -19,6 +19,7 @@ | |||||||
| package com.plotsquared.bukkit.listener; | package com.plotsquared.bukkit.listener; | ||||||
|  |  | ||||||
| import com.google.inject.Inject; | import com.google.inject.Inject; | ||||||
|  | import com.plotsquared.bukkit.BukkitPlatform; | ||||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | import com.plotsquared.bukkit.player.BukkitPlayer; | ||||||
| import com.plotsquared.bukkit.util.BukkitEntityUtil; | import com.plotsquared.bukkit.util.BukkitEntityUtil; | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
| @@ -35,11 +36,15 @@ import com.plotsquared.core.plot.flag.implementations.DisablePhysicsFlag; | |||||||
| import com.plotsquared.core.plot.flag.implementations.EntityChangeBlockFlag; | import com.plotsquared.core.plot.flag.implementations.EntityChangeBlockFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.ExplosionFlag; | import com.plotsquared.core.plot.flag.implementations.ExplosionFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.InvincibleFlag; | import com.plotsquared.core.plot.flag.implementations.InvincibleFlag; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.ProjectileChangeBlockFlag; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.WeavingDeathPlace; | ||||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | import com.plotsquared.core.plot.world.PlotAreaManager; | ||||||
| import com.plotsquared.core.util.EventDispatcher; | import com.plotsquared.core.util.EventDispatcher; | ||||||
| import com.plotsquared.core.util.PlotFlagUtil; | import com.plotsquared.core.util.PlotFlagUtil; | ||||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||||
|  | import com.sk89q.worldedit.util.Enums; | ||||||
| import com.sk89q.worldedit.world.block.BlockType; | import com.sk89q.worldedit.world.block.BlockType; | ||||||
|  | import io.papermc.lib.PaperLib; | ||||||
| import org.bukkit.Material; | import org.bukkit.Material; | ||||||
| import org.bukkit.Particle; | import org.bukkit.Particle; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| @@ -53,6 +58,8 @@ import org.bukkit.entity.Player; | |||||||
| import org.bukkit.entity.Projectile; | import org.bukkit.entity.Projectile; | ||||||
| import org.bukkit.entity.TNTPrimed; | import org.bukkit.entity.TNTPrimed; | ||||||
| import org.bukkit.entity.Vehicle; | import org.bukkit.entity.Vehicle; | ||||||
|  | import org.bukkit.entity.minecart.ExplosiveMinecart; | ||||||
|  | import org.bukkit.event.Cancellable; | ||||||
| import org.bukkit.event.EventHandler; | import org.bukkit.event.EventHandler; | ||||||
| import org.bukkit.event.EventPriority; | import org.bukkit.event.EventPriority; | ||||||
| import org.bukkit.event.Listener; | import org.bukkit.event.Listener; | ||||||
| @@ -73,56 +80,54 @@ import org.checkerframework.checker.nullness.qual.NonNull; | |||||||
|  |  | ||||||
| import java.util.Iterator; | import java.util.Iterator; | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  | import java.util.Objects; | ||||||
|  |  | ||||||
| @SuppressWarnings("unused") | @SuppressWarnings("unused") | ||||||
| public class EntityEventListener implements Listener { | public class EntityEventListener implements Listener { | ||||||
|  |  | ||||||
|  |     private static final Particle EXPLOSION_HUGE = Objects.requireNonNull(Enums.findByValue( | ||||||
|  |             Particle.class, | ||||||
|  |             "EXPLOSION_EMITTER", | ||||||
|  |             "EXPLOSION_HUGE" | ||||||
|  |     )); | ||||||
|  |  | ||||||
|  |     private final BukkitPlatform platform; | ||||||
|     private final PlotAreaManager plotAreaManager; |     private final PlotAreaManager plotAreaManager; | ||||||
|     private final EventDispatcher eventDispatcher; |     private final EventDispatcher eventDispatcher; | ||||||
|     private float lastRadius; |     private float lastRadius; | ||||||
|  |  | ||||||
|     @Inject |     @Inject | ||||||
|     public EntityEventListener( |     public EntityEventListener( | ||||||
|  |             final @NonNull BukkitPlatform platform, | ||||||
|             final @NonNull PlotAreaManager plotAreaManager, |             final @NonNull PlotAreaManager plotAreaManager, | ||||||
|             final @NonNull EventDispatcher eventDispatcher |             final @NonNull EventDispatcher eventDispatcher | ||||||
|     ) { |     ) { | ||||||
|  |         this.platform = platform; | ||||||
|         this.plotAreaManager = plotAreaManager; |         this.plotAreaManager = plotAreaManager; | ||||||
|         this.eventDispatcher = eventDispatcher; |         this.eventDispatcher = eventDispatcher; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST) |     @EventHandler(priority = EventPriority.HIGHEST) | ||||||
|     public void onEntityCombustByEntity(EntityCombustByEntityEvent event) { |     public void onEntityCombustByEntity(EntityCombustByEntityEvent event) { | ||||||
|         EntityDamageByEntityEvent eventChange = |         onEntityDamageByEntityCommon(event.getCombuster(), event.getEntity(), EntityDamageEvent.DamageCause.FIRE_TICK, event); | ||||||
|                 new EntityDamageByEntityEvent( |  | ||||||
|                         event.getCombuster(), |  | ||||||
|                         event.getEntity(), |  | ||||||
|                         EntityDamageEvent.DamageCause.FIRE_TICK, |  | ||||||
|                         event.getDuration() |  | ||||||
|                 ); |  | ||||||
|         onEntityDamageByEntityEvent(eventChange); |  | ||||||
|         if (eventChange.isCancelled()) { |  | ||||||
|             event.setCancelled(true); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST) |     @EventHandler(priority = EventPriority.HIGHEST) | ||||||
|     public void onEntityDamageByEntityEvent(EntityDamageByEntityEvent event) { |     public void onEntityDamageByEntityEvent(EntityDamageByEntityEvent event) { | ||||||
|         Entity damager = event.getDamager(); |         onEntityDamageByEntityCommon(event.getDamager(), event.getEntity(), event.getCause(), event); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void onEntityDamageByEntityCommon( | ||||||
|  |             final Entity damager, | ||||||
|  |             final Entity victim, | ||||||
|  |             final EntityDamageEvent.DamageCause cause, | ||||||
|  |             final Cancellable event | ||||||
|  |     ) { | ||||||
|         Location location = BukkitUtil.adapt(damager.getLocation()); |         Location location = BukkitUtil.adapt(damager.getLocation()); | ||||||
|         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { |         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Entity victim = event.getEntity(); |         if (!BukkitEntityUtil.entityDamage(damager, victim, cause)) { | ||||||
| /* |  | ||||||
|         if (victim.getType().equals(EntityType.ITEM_FRAME)) { |  | ||||||
|             Plot plot = BukkitUtil.getLocation(victim).getPlot(); |  | ||||||
|             if (plot != null && !plot.isAdded(damager.getUniqueId())) { |  | ||||||
|                 event.setCancelled(true); |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| */ |  | ||||||
|         if (!BukkitEntityUtil.entityDamage(damager, victim, event.getCause())) { |  | ||||||
|             if (event.isCancelled()) { |             if (event.isCancelled()) { | ||||||
|                 if (victim instanceof Ageable ageable) { |                 if (victim instanceof Ageable ageable) { | ||||||
|                     if (ageable.getAge() == -24000) { |                     if (ageable.getAge() == -24000) { | ||||||
| @@ -143,6 +148,10 @@ public class EntityEventListener implements Listener { | |||||||
|         if (area == null) { |         if (area == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |         // Armour-stands are handled elsewhere and should not be handled by area-wide entity-spawn options | ||||||
|  |         if (entity.getType() == EntityType.ARMOR_STAND) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|         CreatureSpawnEvent.SpawnReason reason = event.getSpawnReason(); |         CreatureSpawnEvent.SpawnReason reason = event.getSpawnReason(); | ||||||
|         switch (reason.toString()) { |         switch (reason.toString()) { | ||||||
|             case "DISPENSE_EGG", "EGG", "OCELOT_BABY", "SPAWNER_EGG" -> { |             case "DISPENSE_EGG", "EGG", "OCELOT_BABY", "SPAWNER_EGG" -> { | ||||||
| @@ -152,21 +161,32 @@ public class EntityEventListener implements Listener { | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             case "REINFORCEMENTS", "NATURAL", "MOUNT", "PATROL", "RAID", "SHEARED", "SILVERFISH_BLOCK", "ENDER_PEARL", |             case "REINFORCEMENTS", "NATURAL", "MOUNT", "PATROL", "RAID", "SHEARED", "SILVERFISH_BLOCK", "ENDER_PEARL", | ||||||
|                     "TRAP", "VILLAGE_DEFENSE", "VILLAGE_INVASION", "BEEHIVE", "CHUNK_GEN", "NETHER_PORTAL", |                  "TRAP", "VILLAGE_DEFENSE", "VILLAGE_INVASION", "BEEHIVE", "CHUNK_GEN", "NETHER_PORTAL", | ||||||
|                     "DUPLICATION", "FROZEN", "SPELL" -> { |                  "FROZEN", "SPELL", "DEFAULT" -> { | ||||||
|                 if (!area.isMobSpawning()) { |                 if (!area.isMobSpawning()) { | ||||||
|                     event.setCancelled(true); |                     event.setCancelled(true); | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             case "BREEDING" -> { |             case "BREEDING", "DUPLICATION" -> { | ||||||
|                 if (!area.isSpawnBreeding()) { |                 if (!area.isSpawnBreeding()) { | ||||||
|                     event.setCancelled(true); |                     event.setCancelled(true); | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             case "BUILD_IRONGOLEM", "BUILD_SNOWMAN", "BUILD_WITHER", "CUSTOM" -> { |             case "CUSTOM" -> { | ||||||
|                 if (!area.isSpawnCustom() && entity.getType() != EntityType.ARMOR_STAND) { |                 if (!area.isSpawnCustom()) { | ||||||
|  |                     event.setCancelled(true); | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |                 // No need to clutter metadata if running paper | ||||||
|  |                 if (!PaperLib.isPaper()) { | ||||||
|  |                     entity.setMetadata("ps_custom_spawned", new FixedMetadataValue(this.platform, true)); | ||||||
|  |                 } | ||||||
|  |                 return; // Don't cancel if mob spawning is disabled | ||||||
|  |             } | ||||||
|  |             case "BUILD_IRONGOLEM", "BUILD_SNOWMAN", "BUILD_WITHER" -> { | ||||||
|  |                 if (!area.isSpawnCustom()) { | ||||||
|                     event.setCancelled(true); |                     event.setCancelled(true); | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
| @@ -233,6 +253,29 @@ public class EntityEventListener implements Listener { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @EventHandler(ignoreCancelled = true, priority = EventPriority.HIGHEST) | ||||||
|  |     public void onWeavingEffect(EntityChangeBlockEvent event) { | ||||||
|  |         if (event.getTo() != Material.COBWEB) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||||
|  |         PlotArea area = location.getPlotArea(); | ||||||
|  |         if (area == null) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         Plot plot = location.getOwnedPlot(); | ||||||
|  |         if (plot == null) { | ||||||
|  |             if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, WeavingDeathPlace.class, false)) { | ||||||
|  |                 event.setCancelled(true); | ||||||
|  |             } | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (!plot.getFlag(WeavingDeathPlace.class)) { | ||||||
|  |             plot.debug(event.getTo() + " could not spawn because weaving-death-place = false"); | ||||||
|  |             event.setCancelled(true); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGH) |     @EventHandler(priority = EventPriority.HIGH) | ||||||
|     public void onDamage(EntityDamageEvent event) { |     public void onDamage(EntityDamageEvent event) { | ||||||
|         if (event.getEntityType() != EntityType.PLAYER) { |         if (event.getEntityType() != EntityType.PLAYER) { | ||||||
| @@ -280,7 +323,7 @@ public class EntityEventListener implements Listener { | |||||||
|                 if (this.lastRadius != 0) { |                 if (this.lastRadius != 0) { | ||||||
|                     List<Entity> nearby = event.getEntity().getNearbyEntities(this.lastRadius, this.lastRadius, this.lastRadius); |                     List<Entity> nearby = event.getEntity().getNearbyEntities(this.lastRadius, this.lastRadius, this.lastRadius); | ||||||
|                     for (Entity near : nearby) { |                     for (Entity near : nearby) { | ||||||
|                         if (near instanceof TNTPrimed || near.getType().equals(EntityType.MINECART_TNT)) { |                         if (near instanceof TNTPrimed || near instanceof ExplosiveMinecart) { | ||||||
|                             if (!near.hasMetadata("plot")) { |                             if (!near.hasMetadata("plot")) { | ||||||
|                                 near.setMetadata("plot", new FixedMetadataValue((Plugin) PlotSquared.platform(), plot)); |                                 near.setMetadata("plot", new FixedMetadataValue((Plugin) PlotSquared.platform(), plot)); | ||||||
|                             } |                             } | ||||||
| @@ -304,7 +347,7 @@ public class EntityEventListener implements Listener { | |||||||
|         event.setCancelled(true); |         event.setCancelled(true); | ||||||
|         //Spawn Explosion Particles when enabled in settings |         //Spawn Explosion Particles when enabled in settings | ||||||
|         if (Settings.General.ALWAYS_SHOW_EXPLOSIONS) { |         if (Settings.General.ALWAYS_SHOW_EXPLOSIONS) { | ||||||
|             event.getLocation().getWorld().spawnParticle(Particle.EXPLOSION_HUGE, event.getLocation(), 0); |             event.getLocation().getWorld().spawnParticle(EXPLOSION_HUGE, event.getLocation(), 0); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -354,13 +397,13 @@ public class EntityEventListener implements Listener { | |||||||
|             if (shooter instanceof Player) { |             if (shooter instanceof Player) { | ||||||
|                 PlotPlayer<?> pp = BukkitUtil.adapt((Player) shooter); |                 PlotPlayer<?> pp = BukkitUtil.adapt((Player) shooter); | ||||||
|                 if (plot == null) { |                 if (plot == null) { | ||||||
|                     if (!pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { |                     if (area.isRoadFlags() && !area.getRoadFlag(ProjectileChangeBlockFlag.class) && !pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { | ||||||
|                         entity.remove(); |                         entity.remove(); | ||||||
|                         event.setCancelled(true); |                         event.setCancelled(true); | ||||||
|                     } |                     } | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|                 if (plot.isAdded(pp.getUUID()) || pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { |                 if (plot.isAdded(pp.getUUID()) || plot.getFlag(ProjectileChangeBlockFlag.class) || pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|                 entity.remove(); |                 entity.remove(); | ||||||
| @@ -391,7 +434,13 @@ public class EntityEventListener implements Listener { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         Plot plot = area.getOwnedPlot(location); |         Plot plot = area.getOwnedPlot(location); | ||||||
|         if (plot != null && !plot.getFlag(EntityChangeBlockFlag.class)) { |         if (plot == null) { | ||||||
|  |             if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, EntityChangeBlockFlag.class, false)) { | ||||||
|  |                 event.setCancelled(true); | ||||||
|  |             } | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (!plot.getFlag(EntityChangeBlockFlag.class)) { | ||||||
|             plot.debug(e.getType() + " could not change block because entity-change-block = false"); |             plot.debug(e.getType() + " could not change block because entity-change-block = false"); | ||||||
|             event.setCancelled(true); |             event.setCancelled(true); | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -31,8 +31,10 @@ import org.bukkit.Chunk; | |||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.block.Block; | import org.bukkit.block.Block; | ||||||
| import org.bukkit.entity.ArmorStand; | import org.bukkit.entity.ArmorStand; | ||||||
|  | import org.bukkit.entity.EnderCrystal; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
| import org.bukkit.entity.EntityType; | import org.bukkit.entity.EntityType; | ||||||
|  | import org.bukkit.entity.Item; | ||||||
| import org.bukkit.entity.Vehicle; | import org.bukkit.entity.Vehicle; | ||||||
| import org.bukkit.event.EventHandler; | import org.bukkit.event.EventHandler; | ||||||
| import org.bukkit.event.EventPriority; | import org.bukkit.event.EventPriority; | ||||||
| @@ -120,13 +122,19 @@ public class EntitySpawnListener implements Listener { | |||||||
|         Entity entity = event.getEntity(); |         Entity entity = event.getEntity(); | ||||||
|         Location location = BukkitUtil.adapt(entity.getLocation()); |         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (!location.isPlotArea()) { |         if (!location.isPlotArea() || area == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |         if (PaperLib.isPaper()) { | ||||||
|  |             //noinspection ConstantValue - getEntitySpawnReason annotated as NotNull, but is not NotNull. lol. | ||||||
|  |             if (area.isSpawnCustom() && entity.getEntitySpawnReason() != null && "CUSTOM".equals(entity.getEntitySpawnReason().name())) { | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|         Plot plot = location.getOwnedPlotAbs(); |         Plot plot = location.getOwnedPlotAbs(); | ||||||
|         EntityType type = entity.getType(); |         EntityType type = entity.getType(); | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             if (type == EntityType.DROPPED_ITEM) { |             if (entity instanceof Item) { | ||||||
|                 if (Settings.Enabled_Components.KILL_ROAD_ITEMS) { |                 if (Settings.Enabled_Components.KILL_ROAD_ITEMS) { | ||||||
|                     event.setCancelled(true); |                     event.setCancelled(true); | ||||||
|                 } |                 } | ||||||
| @@ -148,7 +156,7 @@ public class EntitySpawnListener implements Listener { | |||||||
|         if (Settings.Done.RESTRICT_BUILDING && DoneFlag.isDone(plot)) { |         if (Settings.Done.RESTRICT_BUILDING && DoneFlag.isDone(plot)) { | ||||||
|             event.setCancelled(true); |             event.setCancelled(true); | ||||||
|         } |         } | ||||||
|         if (type == EntityType.ENDER_CRYSTAL) { |         if (entity instanceof EnderCrystal || type == EntityType.ARMOR_STAND) { | ||||||
|             if (BukkitEntityUtil.checkEntity(entity, plot)) { |             if (BukkitEntityUtil.checkEntity(entity, plot)) { | ||||||
|                 event.setCancelled(true); |                 event.setCancelled(true); | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -0,0 +1,201 @@ | |||||||
|  | /* | ||||||
|  |  * PlotSquared, a land and world management plugin for Minecraft. | ||||||
|  |  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||||
|  |  * Copyright (C) IntellectualSites team and contributors | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 3 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  | package com.plotsquared.bukkit.listener; | ||||||
|  |  | ||||||
|  | import com.google.inject.Inject; | ||||||
|  | import com.plotsquared.bukkit.player.BukkitPlayer; | ||||||
|  | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
|  | import com.plotsquared.core.PlotSquared; | ||||||
|  | import com.plotsquared.core.configuration.Settings; | ||||||
|  | import com.plotsquared.core.database.DBFunc; | ||||||
|  | import com.plotsquared.core.location.Location; | ||||||
|  | import com.plotsquared.core.player.PlotPlayer; | ||||||
|  | import com.plotsquared.core.plot.Plot; | ||||||
|  | import com.plotsquared.core.plot.PlotArea; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.DisablePhysicsFlag; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.RedstoneFlag; | ||||||
|  | import com.plotsquared.core.plot.world.PlotAreaManager; | ||||||
|  | import com.plotsquared.core.util.PlotFlagUtil; | ||||||
|  | import com.plotsquared.core.util.task.TaskManager; | ||||||
|  | import com.plotsquared.core.util.task.TaskTime; | ||||||
|  | import com.sk89q.worldedit.WorldEdit; | ||||||
|  | import org.bukkit.Bukkit; | ||||||
|  | import org.bukkit.Material; | ||||||
|  | import org.bukkit.block.Block; | ||||||
|  | import org.bukkit.block.BlockFace; | ||||||
|  | import org.bukkit.block.data.BlockData; | ||||||
|  | import org.bukkit.event.EventHandler; | ||||||
|  | import org.bukkit.event.EventPriority; | ||||||
|  | import org.bukkit.event.Listener; | ||||||
|  | import org.bukkit.event.block.BlockPhysicsEvent; | ||||||
|  | import org.bukkit.event.block.BlockRedstoneEvent; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
|  | import java.util.Set; | ||||||
|  | import java.util.UUID; | ||||||
|  |  | ||||||
|  | @SuppressWarnings("unused") | ||||||
|  | public class HighFreqBlockEventListener implements Listener { | ||||||
|  |  | ||||||
|  |     private static final Set<Material> PISTONS = Set.of( | ||||||
|  |             Material.PISTON, | ||||||
|  |             Material.STICKY_PISTON | ||||||
|  |     ); | ||||||
|  |     private static final Set<Material> PHYSICS_BLOCKS = Set.of( | ||||||
|  |             Material.TURTLE_EGG, | ||||||
|  |             Material.TURTLE_SPAWN_EGG | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     private final PlotAreaManager plotAreaManager; | ||||||
|  |     private final WorldEdit worldEdit; | ||||||
|  |  | ||||||
|  |     @Inject | ||||||
|  |     public HighFreqBlockEventListener(final @NonNull PlotAreaManager plotAreaManager, final @NonNull WorldEdit worldEdit) { | ||||||
|  |         this.plotAreaManager = plotAreaManager; | ||||||
|  |         this.worldEdit = worldEdit; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static void sendBlockChange(final org.bukkit.Location bloc, final BlockData data) { | ||||||
|  |         TaskManager.runTaskLater(() -> { | ||||||
|  |             String world = bloc.getWorld().getName(); | ||||||
|  |             int x = bloc.getBlockX(); | ||||||
|  |             int z = bloc.getBlockZ(); | ||||||
|  |             int distance = Bukkit.getViewDistance() * 16; | ||||||
|  |  | ||||||
|  |             for (final PlotPlayer<?> player : PlotSquared.platform().playerManager().getPlayers()) { | ||||||
|  |                 Location location = player.getLocation(); | ||||||
|  |                 if (location.getWorldName().equals(world)) { | ||||||
|  |                     if (16 * Math.abs(location.getX() - x) / 16 > distance || 16 * Math.abs(location.getZ() - z) / 16 > distance) { | ||||||
|  |                         continue; | ||||||
|  |                     } | ||||||
|  |                     ((BukkitPlayer) player).player.sendBlockChange(bloc, data); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }, TaskTime.ticks(3L)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @EventHandler | ||||||
|  |     public void onRedstoneEvent(BlockRedstoneEvent event) { | ||||||
|  |         Block block = event.getBlock(); | ||||||
|  |         Location location = BukkitUtil.adapt(block.getLocation()); | ||||||
|  |         PlotArea area = location.getPlotArea(); | ||||||
|  |         if (area == null) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         Plot plot = location.getOwnedPlot(); | ||||||
|  |         if (plot == null) { | ||||||
|  |             if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, RedstoneFlag.class, false)) { | ||||||
|  |                 event.setNewCurrent(0); | ||||||
|  |             } | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (!plot.getFlag(RedstoneFlag.class)) { | ||||||
|  |             event.setNewCurrent(0); | ||||||
|  |             plot.debug("Redstone event was cancelled because redstone = false"); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (Settings.Redstone.DISABLE_OFFLINE) { | ||||||
|  |             boolean disable = false; | ||||||
|  |             if (!DBFunc.SERVER.equals(plot.getOwner())) { | ||||||
|  |                 if (plot.isMerged()) { | ||||||
|  |                     disable = true; | ||||||
|  |                     for (UUID owner : plot.getOwners()) { | ||||||
|  |                         if (PlotSquared.platform().playerManager().getPlayerIfExists(owner) != null) { | ||||||
|  |                             disable = false; | ||||||
|  |                             break; | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } else { | ||||||
|  |                     disable = PlotSquared.platform().playerManager().getPlayerIfExists(plot.getOwnerAbs()) == null; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             if (disable) { | ||||||
|  |                 for (UUID trusted : plot.getTrusted()) { | ||||||
|  |                     if (PlotSquared.platform().playerManager().getPlayerIfExists(trusted) != null) { | ||||||
|  |                         disable = false; | ||||||
|  |                         break; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 if (disable) { | ||||||
|  |                     event.setNewCurrent(0); | ||||||
|  |                     plot.debug("Redstone event was cancelled because no trusted player was in the plot"); | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         if (Settings.Redstone.DISABLE_UNOCCUPIED) { | ||||||
|  |             for (final PlotPlayer<?> player : PlotSquared.platform().playerManager().getPlayers()) { | ||||||
|  |                 if (plot.equals(player.getCurrentPlot())) { | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             event.setNewCurrent(0); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @EventHandler(ignoreCancelled = true, priority = EventPriority.HIGHEST) | ||||||
|  |     public void onPhysicsEvent(BlockPhysicsEvent event) { | ||||||
|  |         Block block = event.getBlock(); | ||||||
|  |         Location location = BukkitUtil.adapt(block.getLocation()); | ||||||
|  |         PlotArea area = location.getPlotArea(); | ||||||
|  |         if (area == null) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         Plot plot = area.getOwnedPlotAbs(location); | ||||||
|  |         if (plot == null) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (event.getChangedType().hasGravity() && plot.getFlag(DisablePhysicsFlag.class)) { | ||||||
|  |             event.setCancelled(true); | ||||||
|  |             sendBlockChange(event.getBlock().getLocation(), event.getBlock().getBlockData()); | ||||||
|  |             plot.debug("Prevented block physics and resent block change because disable-physics = true"); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (event.getChangedType() == Material.COMPARATOR) { | ||||||
|  |             if (!plot.getFlag(RedstoneFlag.class)) { | ||||||
|  |                 event.setCancelled(true); | ||||||
|  |                 plot.debug("Prevented comparator update because redstone = false"); | ||||||
|  |             } | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (PHYSICS_BLOCKS.contains(event.getChangedType())) { | ||||||
|  |             if (plot.getFlag(DisablePhysicsFlag.class)) { | ||||||
|  |                 event.setCancelled(true); | ||||||
|  |                 plot.debug("Prevented block physics because disable-physics = true"); | ||||||
|  |             } | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (Settings.Redstone.DETECT_INVALID_EDGE_PISTONS) { | ||||||
|  |             if (PISTONS.contains(block.getType())) { | ||||||
|  |                 org.bukkit.block.data.Directional piston = (org.bukkit.block.data.Directional) block.getBlockData(); | ||||||
|  |                 final BlockFace facing = piston.getFacing(); | ||||||
|  |                 location = location.add(facing.getModX(), facing.getModY(), facing.getModZ()); | ||||||
|  |                 Plot newPlot = area.getOwnedPlotAbs(location); | ||||||
|  |                 if (plot.equals(newPlot)) { | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |                 if (!plot.isMerged() || !plot.getConnectedPlots().contains(newPlot)) { | ||||||
|  |                     event.setCancelled(true); | ||||||
|  |                     plot.debug("Prevented piston update because of invalid edge piston detection"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -19,6 +19,7 @@ | |||||||
| package com.plotsquared.bukkit.listener; | package com.plotsquared.bukkit.listener; | ||||||
|  |  | ||||||
| import com.destroystokyo.paper.event.block.BeaconEffectEvent; | import com.destroystokyo.paper.event.block.BeaconEffectEvent; | ||||||
|  | import com.destroystokyo.paper.event.block.BlockDestroyEvent; | ||||||
| import com.destroystokyo.paper.event.entity.EntityPathfindEvent; | import com.destroystokyo.paper.event.entity.EntityPathfindEvent; | ||||||
| import com.destroystokyo.paper.event.entity.PlayerNaturallySpawnCreaturesEvent; | import com.destroystokyo.paper.event.entity.PlayerNaturallySpawnCreaturesEvent; | ||||||
| import com.destroystokyo.paper.event.entity.PreCreatureSpawnEvent; | import com.destroystokyo.paper.event.entity.PreCreatureSpawnEvent; | ||||||
| @@ -28,6 +29,7 @@ import com.destroystokyo.paper.event.player.PlayerLaunchProjectileEvent; | |||||||
| import com.destroystokyo.paper.event.server.AsyncTabCompleteEvent; | import com.destroystokyo.paper.event.server.AsyncTabCompleteEvent; | ||||||
| import com.google.inject.Inject; | import com.google.inject.Inject; | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
|  | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.command.Command; | import com.plotsquared.core.command.Command; | ||||||
| import com.plotsquared.core.command.MainCommand; | import com.plotsquared.core.command.MainCommand; | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
| @@ -37,17 +39,23 @@ import com.plotsquared.core.permissions.Permission; | |||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| import com.plotsquared.core.plot.PlotArea; | import com.plotsquared.core.plot.PlotArea; | ||||||
|  | import com.plotsquared.core.plot.PlotAreaType; | ||||||
| import com.plotsquared.core.plot.flag.FlagContainer; | import com.plotsquared.core.plot.flag.FlagContainer; | ||||||
| import com.plotsquared.core.plot.flag.implementations.BeaconEffectsFlag; | import com.plotsquared.core.plot.flag.implementations.BeaconEffectsFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.DoneFlag; | import com.plotsquared.core.plot.flag.implementations.DoneFlag; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.FishingFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.ProjectilesFlag; | import com.plotsquared.core.plot.flag.implementations.ProjectilesFlag; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.TileDropFlag; | ||||||
| import com.plotsquared.core.plot.flag.types.BooleanFlag; | import com.plotsquared.core.plot.flag.types.BooleanFlag; | ||||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | import com.plotsquared.core.plot.world.PlotAreaManager; | ||||||
| import com.plotsquared.core.util.PlotFlagUtil; | import com.plotsquared.core.util.PlotFlagUtil; | ||||||
|  | import io.papermc.paper.event.entity.EntityMoveEvent; | ||||||
|  | import io.papermc.paper.event.world.StructuresLocateEvent; | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | import net.kyori.adventure.text.minimessage.tag.Tag; | ||||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| import org.bukkit.Chunk; | import org.bukkit.Chunk; | ||||||
|  | import org.bukkit.NamespacedKey; | ||||||
| import org.bukkit.block.Block; | import org.bukkit.block.Block; | ||||||
| import org.bukkit.block.TileState; | import org.bukkit.block.TileState; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
| @@ -55,6 +63,7 @@ import org.bukkit.entity.EntityType; | |||||||
| import org.bukkit.entity.Player; | import org.bukkit.entity.Player; | ||||||
| import org.bukkit.entity.Projectile; | import org.bukkit.entity.Projectile; | ||||||
| import org.bukkit.entity.Slime; | import org.bukkit.entity.Slime; | ||||||
|  | import org.bukkit.event.Cancellable; | ||||||
| import org.bukkit.event.EventHandler; | import org.bukkit.event.EventHandler; | ||||||
| import org.bukkit.event.EventPriority; | import org.bukkit.event.EventPriority; | ||||||
| import org.bukkit.event.Listener; | import org.bukkit.event.Listener; | ||||||
| @@ -75,6 +84,9 @@ import java.util.regex.Pattern; | |||||||
| @SuppressWarnings("unused") | @SuppressWarnings("unused") | ||||||
| public class PaperListener implements Listener { | public class PaperListener implements Listener { | ||||||
|  |  | ||||||
|  |     private static final NamespacedKey ITEM = NamespacedKey.minecraft("item"); | ||||||
|  |     private static final NamespacedKey FISHING_BOBBER = NamespacedKey.minecraft("fishing_bobber"); | ||||||
|  |  | ||||||
|     private final PlotAreaManager plotAreaManager; |     private final PlotAreaManager plotAreaManager; | ||||||
|     private Chunk lastChunk; |     private Chunk lastChunk; | ||||||
|  |  | ||||||
| @@ -83,38 +95,25 @@ public class PaperListener implements Listener { | |||||||
|         this.plotAreaManager = plotAreaManager; |         this.plotAreaManager = plotAreaManager; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @EventHandler(ignoreCancelled = true, priority = EventPriority.LOWEST) | ||||||
|  |     public void onBlockDestroy(final BlockDestroyEvent event) { | ||||||
|  |         Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||||
|  |         PlotArea area = location.getPlotArea(); | ||||||
|  |         if (area == null) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         Plot plot = area.getPlot(location); | ||||||
|  |         if (plot != null) { | ||||||
|  |             event.setWillDrop(plot.getFlag(TileDropFlag.class)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @EventHandler |     @EventHandler | ||||||
|     public void onEntityPathfind(EntityPathfindEvent event) { |     public void onEntityPathfind(EntityPathfindEvent event) { | ||||||
|         if (!Settings.Paper_Components.ENTITY_PATHING) { |         if (!Settings.Paper_Components.ENTITY_PATHING) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Location toLoc = BukkitUtil.adapt(event.getLoc()); |         handleEntityMovement(event, event.getEntity().getLocation(), event.getLoc()); | ||||||
|         Location fromLoc = BukkitUtil.adapt(event.getEntity().getLocation()); |  | ||||||
|         PlotArea tarea = toLoc.getPlotArea(); |  | ||||||
|         if (tarea == null) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         PlotArea farea = fromLoc.getPlotArea(); |  | ||||||
|         if (farea == null) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         if (tarea != farea) { |  | ||||||
|             event.setCancelled(true); |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         Plot tplot = toLoc.getPlot(); |  | ||||||
|         Plot fplot = fromLoc.getPlot(); |  | ||||||
|         if (tplot == null ^ fplot == null) { |  | ||||||
|             event.setCancelled(true); |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         if (tplot == null || tplot.getId().hashCode() == fplot.getId().hashCode()) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         if (fplot.isMerged() && fplot.getConnectedPlots().contains(fplot)) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         event.setCancelled(true); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler |     @EventHandler | ||||||
| @@ -124,13 +123,28 @@ public class PaperListener implements Listener { | |||||||
|         } |         } | ||||||
|         Slime slime = event.getEntity(); |         Slime slime = event.getEntity(); | ||||||
|  |  | ||||||
|         Block b = slime.getTargetBlock(4); |         Block b = slime.getTargetBlockExact(4); | ||||||
|         if (b == null) { |         if (b == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Location toLoc = BukkitUtil.adapt(b.getLocation()); |         handleEntityMovement(event, event.getEntity().getLocation(),  b.getLocation()); | ||||||
|         Location fromLoc = BukkitUtil.adapt(event.getEntity().getLocation()); |     } | ||||||
|  |  | ||||||
|  |     @EventHandler | ||||||
|  |     public void onEntityMove(EntityMoveEvent event) { | ||||||
|  |         if (!Settings.Paper_Components.ENTITY_MOVEMENT) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (!event.hasExplicitlyChangedBlock()) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         handleEntityMovement(event, event.getFrom(), event.getTo()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static void handleEntityMovement(Cancellable event, org.bukkit.Location from, org.bukkit.Location target) { | ||||||
|  |         Location toLoc = BukkitUtil.adapt(target); | ||||||
|  |         Location fromLoc = BukkitUtil.adapt(from); | ||||||
|         PlotArea tarea = toLoc.getPlotArea(); |         PlotArea tarea = toLoc.getPlotArea(); | ||||||
|         if (tarea == null) { |         if (tarea == null) { | ||||||
|             return; |             return; | ||||||
| @@ -139,7 +153,6 @@ public class PaperListener implements Listener { | |||||||
|         if (farea == null) { |         if (farea == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (tarea != farea) { |         if (tarea != farea) { | ||||||
|             event.setCancelled(true); |             event.setCancelled(true); | ||||||
|             return; |             return; | ||||||
| @@ -150,10 +163,10 @@ public class PaperListener implements Listener { | |||||||
|             event.setCancelled(true); |             event.setCancelled(true); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         if (tplot == null || tplot.getId().hashCode() == fplot.getId().hashCode()) { |         if (tplot == null || tplot.getId().equals(fplot.getId())) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         if (fplot.isMerged() && fplot.getConnectedPlots().contains(fplot)) { |         if (fplot.isMerged() && fplot.getConnectedPlots().contains(tplot)) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         event.setCancelled(true); |         event.setCancelled(true); | ||||||
| @@ -166,12 +179,16 @@ public class PaperListener implements Listener { | |||||||
|         } |         } | ||||||
|         Location location = BukkitUtil.adapt(event.getSpawnLocation()); |         Location location = BukkitUtil.adapt(event.getSpawnLocation()); | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (!location.isPlotArea()) { |         if (area == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         //If entities are spawning... the chunk should be loaded? |         // Armour-stands are handled elsewhere and should not be handled by area-wide entity-spawn options | ||||||
|  |         if (event.getType() == EntityType.ARMOR_STAND) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         // If entities are spawning... the chunk should be loaded? | ||||||
|         Entity[] entities = event.getSpawnLocation().getChunk().getEntities(); |         Entity[] entities = event.getSpawnLocation().getChunk().getEntities(); | ||||||
|         if (entities.length > Settings.Chunk_Processor.MAX_ENTITIES) { |         if (entities.length >= Settings.Chunk_Processor.MAX_ENTITIES) { | ||||||
|             event.setShouldAbortSpawn(true); |             event.setShouldAbortSpawn(true); | ||||||
|             event.setCancelled(true); |             event.setCancelled(true); | ||||||
|             return; |             return; | ||||||
| @@ -200,7 +217,7 @@ public class PaperListener implements Listener { | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             case "BUILD_IRONGOLEM", "BUILD_SNOWMAN", "BUILD_WITHER", "CUSTOM" -> { |             case "BUILD_IRONGOLEM", "BUILD_SNOWMAN", "BUILD_WITHER", "CUSTOM" -> { | ||||||
|                 if (!area.isSpawnCustom() && event.getType() != EntityType.ARMOR_STAND) { |                 if (!area.isSpawnCustom()) { | ||||||
|                     event.setShouldAbortSpawn(true); |                     event.setShouldAbortSpawn(true); | ||||||
|                     event.setCancelled(true); |                     event.setCancelled(true); | ||||||
|                     return; |                     return; | ||||||
| @@ -218,7 +235,7 @@ public class PaperListener implements Listener { | |||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             EntityType type = event.getType(); |             EntityType type = event.getType(); | ||||||
|             // PreCreatureSpawnEvent **should** not be called for DROPPED_ITEM, just for the sake of consistency |             // PreCreatureSpawnEvent **should** not be called for DROPPED_ITEM, just for the sake of consistency | ||||||
|             if (type == EntityType.DROPPED_ITEM) { |             if (type.getKey().equals(ITEM)) { | ||||||
|                 if (Settings.Enabled_Components.KILL_ROAD_ITEMS) { |                 if (Settings.Enabled_Components.KILL_ROAD_ITEMS) { | ||||||
|                     event.setCancelled(true); |                     event.setCancelled(true); | ||||||
|                 } |                 } | ||||||
| @@ -344,6 +361,11 @@ public class PaperListener implements Listener { | |||||||
|                 event.setCancelled(true); |                 event.setCancelled(true); | ||||||
|             } |             } | ||||||
|         } else if (!plot.isAdded(pp.getUUID())) { |         } else if (!plot.isAdded(pp.getUUID())) { | ||||||
|  |             if (entity.getType().getKey().equals(FISHING_BOBBER)) { | ||||||
|  |                 if (plot.getFlag(FishingFlag.class)) { | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|             if (!plot.getFlag(ProjectilesFlag.class)) { |             if (!plot.getFlag(ProjectilesFlag.class)) { | ||||||
|                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { |                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||||
|                     pp.sendMessage( |                     pp.sendMessage( | ||||||
| @@ -439,6 +461,21 @@ public class PaperListener implements Listener { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Don't let the server die when populating cartographers (villager offering maps) in classic plot worlds | ||||||
|  |      * (as those don't generate POIs) | ||||||
|  |      */ | ||||||
|  |     @EventHandler(ignoreCancelled = true, priority = EventPriority.HIGHEST) | ||||||
|  |     public void onStructuresLocate(StructuresLocateEvent event) { | ||||||
|  |         if (!PlotSquared.get().getPlotAreaManager().hasPlotArea(event.getWorld().getName())) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         final PlotArea area = PlotSquared.get().getPlotAreaManager().getPlotAreaByString(event.getWorld().getName()); | ||||||
|  |         if (area != null && area.getType() == PlotAreaType.NORMAL) { | ||||||
|  |             event.setCancelled(true); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     private boolean getBooleanFlagValue( |     private boolean getBooleanFlagValue( | ||||||
|             @NonNull FlagContainer container, |             @NonNull FlagContainer container, | ||||||
|             @NonNull Class<? extends BooleanFlag<?>> flagClass, |             @NonNull Class<? extends BooleanFlag<?>> flagClass, | ||||||
|   | |||||||
| @@ -50,9 +50,11 @@ import com.plotsquared.core.plot.flag.implementations.DenyPortalsFlag; | |||||||
| import com.plotsquared.core.plot.flag.implementations.DenyTeleportFlag; | import com.plotsquared.core.plot.flag.implementations.DenyTeleportFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.DoneFlag; | import com.plotsquared.core.plot.flag.implementations.DoneFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.DropProtectionFlag; | import com.plotsquared.core.plot.flag.implementations.DropProtectionFlag; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.EditSignFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.HangingBreakFlag; | import com.plotsquared.core.plot.flag.implementations.HangingBreakFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.HangingPlaceFlag; | import com.plotsquared.core.plot.flag.implementations.HangingPlaceFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.HostileInteractFlag; | import com.plotsquared.core.plot.flag.implementations.HostileInteractFlag; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.InteractionInteractFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.ItemDropFlag; | import com.plotsquared.core.plot.flag.implementations.ItemDropFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.KeepInventoryFlag; | import com.plotsquared.core.plot.flag.implementations.KeepInventoryFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.LecternReadBookFlag; | import com.plotsquared.core.plot.flag.implementations.LecternReadBookFlag; | ||||||
| @@ -60,6 +62,7 @@ import com.plotsquared.core.plot.flag.implementations.MiscInteractFlag; | |||||||
| import com.plotsquared.core.plot.flag.implementations.PlayerInteractFlag; | import com.plotsquared.core.plot.flag.implementations.PlayerInteractFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.PreventCreativeCopyFlag; | import com.plotsquared.core.plot.flag.implementations.PreventCreativeCopyFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.TamedInteractFlag; | import com.plotsquared.core.plot.flag.implementations.TamedInteractFlag; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.TileDropFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.UntrustedVisitFlag; | import com.plotsquared.core.plot.flag.implementations.UntrustedVisitFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.VehicleBreakFlag; | import com.plotsquared.core.plot.flag.implementations.VehicleBreakFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.VehicleUseFlag; | import com.plotsquared.core.plot.flag.implementations.VehicleUseFlag; | ||||||
| @@ -85,9 +88,8 @@ import org.bukkit.Bukkit; | |||||||
| import org.bukkit.FluidCollisionMode; | import org.bukkit.FluidCollisionMode; | ||||||
| import org.bukkit.Material; | import org.bukkit.Material; | ||||||
| import org.bukkit.block.Block; | import org.bukkit.block.Block; | ||||||
| import org.bukkit.block.BlockFace; |  | ||||||
| import org.bukkit.block.BlockState; | import org.bukkit.block.BlockState; | ||||||
| import org.bukkit.block.data.Waterlogged; | import org.bukkit.block.Sign; | ||||||
| import org.bukkit.command.PluginCommand; | import org.bukkit.command.PluginCommand; | ||||||
| import org.bukkit.entity.ArmorStand; | import org.bukkit.entity.ArmorStand; | ||||||
| import org.bukkit.entity.Boat; | import org.bukkit.entity.Boat; | ||||||
| @@ -105,6 +107,7 @@ import org.bukkit.event.EventHandler; | |||||||
| import org.bukkit.event.EventPriority; | import org.bukkit.event.EventPriority; | ||||||
| import org.bukkit.event.Listener; | import org.bukkit.event.Listener; | ||||||
| import org.bukkit.event.block.Action; | import org.bukkit.event.block.Action; | ||||||
|  | import org.bukkit.event.block.BlockBreakEvent; | ||||||
| import org.bukkit.event.entity.EntityPickupItemEvent; | import org.bukkit.event.entity.EntityPickupItemEvent; | ||||||
| import org.bukkit.event.entity.EntityPlaceEvent; | import org.bukkit.event.entity.EntityPlaceEvent; | ||||||
| import org.bukkit.event.entity.EntityPotionEffectEvent; | import org.bukkit.event.entity.EntityPotionEffectEvent; | ||||||
| @@ -175,6 +178,98 @@ public class PlayerEventListener implements Listener { | |||||||
|             Material.WRITABLE_BOOK, |             Material.WRITABLE_BOOK, | ||||||
|             Material.WRITTEN_BOOK |             Material.WRITTEN_BOOK | ||||||
|     ); |     ); | ||||||
|  |     private static final Set<String> DYES; | ||||||
|  |     static { | ||||||
|  |         Set<String> mutableDyes = new HashSet<>(Set.of( | ||||||
|  |                 "WHITE_DYE", | ||||||
|  |                 "LIGHT_GRAY_DYE", | ||||||
|  |                 "GRAY_DYE", | ||||||
|  |                 "BLACK_DYE", | ||||||
|  |                 "BROWN_DYE", | ||||||
|  |                 "RED_DYE", | ||||||
|  |                 "ORANGE_DYE", | ||||||
|  |                 "YELLOW_DYE", | ||||||
|  |                 "LIME_DYE", | ||||||
|  |                 "GREEN_DYE", | ||||||
|  |                 "CYAN_DYE", | ||||||
|  |                 "LIGHT_BLUE_DYE", | ||||||
|  |                 "BLUE_DYE", | ||||||
|  |                 "PURPLE_DYE", | ||||||
|  |                 "MAGENTA_DYE", | ||||||
|  |                 "PINK_DYE", | ||||||
|  |                 "GLOW_INK_SAC" | ||||||
|  |         )); | ||||||
|  |         int[] version = PlotSquared.platform().serverVersion(); | ||||||
|  |         if (version[1] >= 20) { | ||||||
|  |             mutableDyes.add("HONEYCOMB"); | ||||||
|  |         } | ||||||
|  |         DYES = Set.copyOf(mutableDyes); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static final Set<String> INTERACTABLE_MATERIALS; | ||||||
|  |  | ||||||
|  |     static { | ||||||
|  |         // @formatter:off | ||||||
|  |         // "temporary" fix for https://hub.spigotmc.org/jira/browse/SPIGOT-7813 | ||||||
|  |         // can (and should) be removed when 1.21 support is dropped | ||||||
|  |         // List of all interactable 1.21 materials | ||||||
|  |         INTERACTABLE_MATERIALS = Material.CHEST.isInteractable() ? null :  Set.of( | ||||||
|  |                 "REDSTONE_ORE", "DEEPSLATE_REDSTONE_ORE", "CHISELED_BOOKSHELF", "DECORATED_POT", "CHEST", "CRAFTING_TABLE", | ||||||
|  |                 "FURNACE", "JUKEBOX", "OAK_FENCE", "SPRUCE_FENCE", "BIRCH_FENCE", "JUNGLE_FENCE", "ACACIA_FENCE", "CHERRY_FENCE", | ||||||
|  |                 "DARK_OAK_FENCE", "MANGROVE_FENCE", "BAMBOO_FENCE", "CRIMSON_FENCE", "WARPED_FENCE", "PUMPKIN", | ||||||
|  |                 "NETHER_BRICK_FENCE", "ENCHANTING_TABLE", "DRAGON_EGG", "ENDER_CHEST", "COMMAND_BLOCK", "BEACON", "ANVIL", | ||||||
|  |                 "CHIPPED_ANVIL", "DAMAGED_ANVIL", "LIGHT", "REPEATING_COMMAND_BLOCK", "CHAIN_COMMAND_BLOCK", "SHULKER_BOX", | ||||||
|  |                 "WHITE_SHULKER_BOX", "ORANGE_SHULKER_BOX", "MAGENTA_SHULKER_BOX", "LIGHT_BLUE_SHULKER_BOX", "YELLOW_SHULKER_BOX", | ||||||
|  |                 "LIME_SHULKER_BOX", "PINK_SHULKER_BOX", "GRAY_SHULKER_BOX", "LIGHT_GRAY_SHULKER_BOX", "CYAN_SHULKER_BOX", | ||||||
|  |                 "PURPLE_SHULKER_BOX", "BLUE_SHULKER_BOX", "BROWN_SHULKER_BOX", "GREEN_SHULKER_BOX", "RED_SHULKER_BOX", | ||||||
|  |                 "BLACK_SHULKER_BOX", "REPEATER", "COMPARATOR", "HOPPER", "DISPENSER", "DROPPER", "LECTERN", "LEVER", | ||||||
|  |                 "DAYLIGHT_DETECTOR", "TRAPPED_CHEST", "TNT", "NOTE_BLOCK", "STONE_BUTTON", "POLISHED_BLACKSTONE_BUTTON", | ||||||
|  |                 "OAK_BUTTON", "SPRUCE_BUTTON", "BIRCH_BUTTON", "JUNGLE_BUTTON", "ACACIA_BUTTON", "CHERRY_BUTTON", | ||||||
|  |                 "DARK_OAK_BUTTON", "MANGROVE_BUTTON", "BAMBOO_BUTTON", "CRIMSON_BUTTON", "WARPED_BUTTON", "IRON_DOOR", "OAK_DOOR", | ||||||
|  |                 "SPRUCE_DOOR", "BIRCH_DOOR", "JUNGLE_DOOR", "ACACIA_DOOR", "CHERRY_DOOR", "DARK_OAK_DOOR", "MANGROVE_DOOR", | ||||||
|  |                 "BAMBOO_DOOR", "CRIMSON_DOOR", "WARPED_DOOR", "COPPER_DOOR", "EXPOSED_COPPER_DOOR", "WEATHERED_COPPER_DOOR", | ||||||
|  |                 "OXIDIZED_COPPER_DOOR", "WAXED_COPPER_DOOR", "WAXED_EXPOSED_COPPER_DOOR", "WAXED_WEATHERED_COPPER_DOOR", | ||||||
|  |                 "WAXED_OXIDIZED_COPPER_DOOR", "IRON_TRAPDOOR", "OAK_TRAPDOOR", "SPRUCE_TRAPDOOR", "BIRCH_TRAPDOOR", | ||||||
|  |                 "JUNGLE_TRAPDOOR", "ACACIA_TRAPDOOR", "CHERRY_TRAPDOOR", "DARK_OAK_TRAPDOOR", "MANGROVE_TRAPDOOR", | ||||||
|  |                 "BAMBOO_TRAPDOOR", "CRIMSON_TRAPDOOR", "WARPED_TRAPDOOR", "COPPER_TRAPDOOR", "EXPOSED_COPPER_TRAPDOOR", | ||||||
|  |                 "WEATHERED_COPPER_TRAPDOOR", "OXIDIZED_COPPER_TRAPDOOR", "WAXED_COPPER_TRAPDOOR", "WAXED_EXPOSED_COPPER_TRAPDOOR", | ||||||
|  |                 "WAXED_WEATHERED_COPPER_TRAPDOOR", "WAXED_OXIDIZED_COPPER_TRAPDOOR", "OAK_FENCE_GATE", "SPRUCE_FENCE_GATE", | ||||||
|  |                 "BIRCH_FENCE_GATE", "JUNGLE_FENCE_GATE", "ACACIA_FENCE_GATE", "CHERRY_FENCE_GATE", "DARK_OAK_FENCE_GATE", | ||||||
|  |                 "MANGROVE_FENCE_GATE", "BAMBOO_FENCE_GATE", "CRIMSON_FENCE_GATE", "WARPED_FENCE_GATE", "STRUCTURE_BLOCK", | ||||||
|  |                 "JIGSAW", "OAK_SIGN", "SPRUCE_SIGN", "BIRCH_SIGN", "JUNGLE_SIGN", "ACACIA_SIGN", "CHERRY_SIGN", "DARK_OAK_SIGN", | ||||||
|  |                 "MANGROVE_SIGN", "BAMBOO_SIGN", "CRIMSON_SIGN", "WARPED_SIGN", "OAK_HANGING_SIGN", "SPRUCE_HANGING_SIGN", | ||||||
|  |                 "BIRCH_HANGING_SIGN", "JUNGLE_HANGING_SIGN", "ACACIA_HANGING_SIGN", "CHERRY_HANGING_SIGN", | ||||||
|  |                 "DARK_OAK_HANGING_SIGN", "MANGROVE_HANGING_SIGN", "BAMBOO_HANGING_SIGN", "CRIMSON_HANGING_SIGN", | ||||||
|  |                 "WARPED_HANGING_SIGN", "CAKE", "WHITE_BED", "ORANGE_BED", "MAGENTA_BED", "LIGHT_BLUE_BED", "YELLOW_BED", | ||||||
|  |                 "LIME_BED", "PINK_BED", "GRAY_BED", "LIGHT_GRAY_BED", "CYAN_BED", "PURPLE_BED", "BLUE_BED", "BROWN_BED", | ||||||
|  |                 "GREEN_BED", "RED_BED", "BLACK_BED", "CRAFTER", "BREWING_STAND", "CAULDRON", "FLOWER_POT", "LOOM", "COMPOSTER", | ||||||
|  |                 "BARREL", "SMOKER", "BLAST_FURNACE", "CARTOGRAPHY_TABLE", "FLETCHING_TABLE", "GRINDSTONE", "SMITHING_TABLE", | ||||||
|  |                 "STONECUTTER", "BELL", "CAMPFIRE", "SOUL_CAMPFIRE", "BEE_NEST", "BEEHIVE", "RESPAWN_ANCHOR", "CANDLE", | ||||||
|  |                 "WHITE_CANDLE", "ORANGE_CANDLE", "MAGENTA_CANDLE", "LIGHT_BLUE_CANDLE", "YELLOW_CANDLE", "LIME_CANDLE", | ||||||
|  |                 "PINK_CANDLE", "GRAY_CANDLE", "LIGHT_GRAY_CANDLE", "CYAN_CANDLE", "PURPLE_CANDLE", "BLUE_CANDLE", "BROWN_CANDLE", | ||||||
|  |                 "GREEN_CANDLE", "RED_CANDLE", "BLACK_CANDLE", "VAULT", "MOVING_PISTON", "REDSTONE_WIRE", "OAK_WALL_SIGN", | ||||||
|  |                 "SPRUCE_WALL_SIGN", "BIRCH_WALL_SIGN", "ACACIA_WALL_SIGN", "CHERRY_WALL_SIGN", "JUNGLE_WALL_SIGN", | ||||||
|  |                 "DARK_OAK_WALL_SIGN", "MANGROVE_WALL_SIGN", "BAMBOO_WALL_SIGN", "OAK_WALL_HANGING_SIGN", | ||||||
|  |                 "SPRUCE_WALL_HANGING_SIGN", "BIRCH_WALL_HANGING_SIGN", "ACACIA_WALL_HANGING_SIGN", | ||||||
|  |                 "CHERRY_WALL_HANGING_SIGN", "JUNGLE_WALL_HANGING_SIGN", "DARK_OAK_WALL_HANGING_SIGN", | ||||||
|  |                 "MANGROVE_WALL_HANGING_SIGN", "CRIMSON_WALL_HANGING_SIGN", "WARPED_WALL_HANGING_SIGN", "BAMBOO_WALL_HANGING_SIGN", | ||||||
|  |                 "WATER_CAULDRON", "LAVA_CAULDRON", "POWDER_SNOW_CAULDRON", "POTTED_TORCHFLOWER", "POTTED_OAK_SAPLING", | ||||||
|  |                 "POTTED_SPRUCE_SAPLING", "POTTED_BIRCH_SAPLING", "POTTED_JUNGLE_SAPLING", "POTTED_ACACIA_SAPLING", | ||||||
|  |                 "POTTED_CHERRY_SAPLING", "POTTED_DARK_OAK_SAPLING", "POTTED_MANGROVE_PROPAGULE", "POTTED_FERN", | ||||||
|  |                 "POTTED_DANDELION", "POTTED_POPPY", "POTTED_BLUE_ORCHID", "POTTED_ALLIUM", "POTTED_AZURE_BLUET", | ||||||
|  |                 "POTTED_RED_TULIP", "POTTED_ORANGE_TULIP", "POTTED_WHITE_TULIP", "POTTED_PINK_TULIP", "POTTED_OXEYE_DAISY", | ||||||
|  |                 "POTTED_CORNFLOWER", "POTTED_LILY_OF_THE_VALLEY", "POTTED_WITHER_ROSE", "POTTED_RED_MUSHROOM", | ||||||
|  |                 "POTTED_BROWN_MUSHROOM", "POTTED_DEAD_BUSH", "POTTED_CACTUS", "POTTED_BAMBOO", "SWEET_BERRY_BUSH", | ||||||
|  |                 "CRIMSON_WALL_SIGN", "WARPED_WALL_SIGN", "POTTED_CRIMSON_FUNGUS", "POTTED_WARPED_FUNGUS", "POTTED_CRIMSON_ROOTS", | ||||||
|  |                 "POTTED_WARPED_ROOTS", "CANDLE_CAKE", "WHITE_CANDLE_CAKE", "ORANGE_CANDLE_CAKE", "MAGENTA_CANDLE_CAKE", | ||||||
|  |                 "LIGHT_BLUE_CANDLE_CAKE", "YELLOW_CANDLE_CAKE", "LIME_CANDLE_CAKE", "PINK_CANDLE_CAKE", "GRAY_CANDLE_CAKE", | ||||||
|  |                 "LIGHT_GRAY_CANDLE_CAKE", "CYAN_CANDLE_CAKE", "PURPLE_CANDLE_CAKE", "BLUE_CANDLE_CAKE", "BROWN_CANDLE_CAKE", | ||||||
|  |                 "GREEN_CANDLE_CAKE", "RED_CANDLE_CAKE", "BLACK_CANDLE_CAKE", "CAVE_VINES", "CAVE_VINES_PLANT", | ||||||
|  |                 "POTTED_AZALEA_BUSH", "POTTED_FLOWERING_AZALEA_BUSH" | ||||||
|  |         ); | ||||||
|  |         // @formatter:on | ||||||
|  |     } | ||||||
|  |  | ||||||
|     private final EventDispatcher eventDispatcher; |     private final EventDispatcher eventDispatcher; | ||||||
|     private final WorldEdit worldEdit; |     private final WorldEdit worldEdit; | ||||||
|     private final PlotAreaManager plotAreaManager; |     private final PlotAreaManager plotAreaManager; | ||||||
| @@ -207,6 +302,53 @@ public class PlayerEventListener implements Listener { | |||||||
|         this.plotListener = plotListener; |         this.plotListener = plotListener; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @EventHandler(ignoreCancelled = true, priority = EventPriority.LOWEST) | ||||||
|  |     public void onBlockBreak(final BlockBreakEvent event) { | ||||||
|  |         Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||||
|  |         PlotArea area = location.getPlotArea(); | ||||||
|  |         if (area == null) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         Plot plot = area.getPlot(location); | ||||||
|  |         if (plot != null) { | ||||||
|  |             event.setDropItems(plot.getFlag(TileDropFlag.class)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|  |     public void onPlayerDyeSign(PlayerInteractEvent event) { | ||||||
|  |         ItemStack itemStack = event.getItem(); | ||||||
|  |         if (itemStack == null) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         Block block = event.getClickedBlock(); | ||||||
|  |         if (block != null && block.getState() instanceof Sign) { | ||||||
|  |             if (DYES.contains(itemStack.getType().toString())) { | ||||||
|  |                 Location location = BukkitUtil.adapt(block.getLocation()); | ||||||
|  |                 PlotArea area = location.getPlotArea(); | ||||||
|  |                 if (area == null) { | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |                 Plot plot = location.getOwnedPlot(); | ||||||
|  |                 if (plot == null) { | ||||||
|  |                     if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, EditSignFlag.class, false) | ||||||
|  |                             && !event.getPlayer().hasPermission(Permission.PERMISSION_ADMIN_INTERACT_ROAD.toString())) { | ||||||
|  |                         event.setCancelled(true); | ||||||
|  |                     } | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |                 if (plot.isAdded(event.getPlayer().getUniqueId())) { | ||||||
|  |                     return; // allow for added players | ||||||
|  |                 } | ||||||
|  |                 if (!plot.getFlag(EditSignFlag.class) | ||||||
|  |                         && !event.getPlayer().hasPermission(Permission.PERMISSION_ADMIN_INTERACT_OTHER.toString())) { | ||||||
|  |                     plot.debug(event.getPlayer().getName() + " could not color the sign because of edit-sign = false"); | ||||||
|  |                     event.setCancelled(true); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @EventHandler(ignoreCancelled = true) |     @EventHandler(ignoreCancelled = true) | ||||||
|     public void onEffect(@NonNull EntityPotionEffectEvent event) { |     public void onEffect(@NonNull EntityPotionEffectEvent event) { | ||||||
|         if (Settings.Enabled_Components.DISABLE_BEACON_EFFECT_OVERFLOW || |         if (Settings.Enabled_Components.DISABLE_BEACON_EFFECT_OVERFLOW || | ||||||
| @@ -232,7 +374,7 @@ public class PlayerEventListener implements Listener { | |||||||
|  |  | ||||||
|     @EventHandler |     @EventHandler | ||||||
|     public void onVehicleEntityCollision(VehicleEntityCollisionEvent e) { |     public void onVehicleEntityCollision(VehicleEntityCollisionEvent e) { | ||||||
|         if (e.getVehicle().getType() == EntityType.BOAT) { |         if (e.getVehicle() instanceof Boat) { | ||||||
|             Location location = BukkitUtil.adapt(e.getEntity().getLocation()); |             Location location = BukkitUtil.adapt(e.getEntity().getLocation()); | ||||||
|             if (location.isPlotArea()) { |             if (location.isPlotArea()) { | ||||||
|                 if (e.getEntity() instanceof Player) { |                 if (e.getEntity() instanceof Player) { | ||||||
| @@ -369,6 +511,7 @@ public class PlayerEventListener implements Listener { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|  |     @SuppressWarnings("deprecation") // Paper deprecation | ||||||
|     public void onConnect(PlayerJoinEvent event) { |     public void onConnect(PlayerJoinEvent event) { | ||||||
|         final Player player = event.getPlayer(); |         final Player player = event.getPlayer(); | ||||||
|         PlotSquared.platform().playerManager().removePlayer(player.getUniqueId()); |         PlotSquared.platform().playerManager().removePlayer(player.getUniqueId()); | ||||||
| @@ -448,15 +591,19 @@ public class PlayerEventListener implements Listener { | |||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|                 Plot plot = area.getPlot(location); |                 Plot plot = area.getPlot(location); | ||||||
|                 if (plot != null) { |                 if (plot != null && !plot.equals(lastPlot)) { | ||||||
|                     final boolean result = DenyTeleportFlag.allowsTeleport(pp, plot); |                     final boolean result = DenyTeleportFlag.allowsTeleport(pp, plot); | ||||||
|                     // there is one possibility to still allow teleportation: |                     // there is one possibility to still allow teleportation: | ||||||
|                     // to is identical to the plot's home location, and untrusted-visit is true |                     // to is identical to the plot's home location, and untrusted-visit is true | ||||||
|                     // i.e. untrusted-visit can override deny-teleport |                     // i.e. untrusted-visit can override deny-teleport | ||||||
|                     // this is acceptable, because otherwise it wouldn't make sense to have both flags set |                     // this is acceptable, because otherwise it wouldn't make sense to have both flags set | ||||||
|                     if (!result && !(plot.getFlag(UntrustedVisitFlag.class) && plot |                     if (result || (plot.getFlag(UntrustedVisitFlag.class) && plot.getHomeSynchronous().equals(BukkitUtil.adaptComplete(to)))) { | ||||||
|                             .getHomeSynchronous() |                         // returns false if the player is not allowed to enter the plot (if they are denied, for example) | ||||||
|                             .equals(BukkitUtil.adaptComplete(to)))) { |                         // don't let the move event cancel the entry after teleport, but rather catch and cancel early (#4647) | ||||||
|  |                         if (!plotListener.plotEntry(pp, plot)) { | ||||||
|  |                             event.setCancelled(true); | ||||||
|  |                         } | ||||||
|  |                     } else { | ||||||
|                         pp.sendMessage( |                         pp.sendMessage( | ||||||
|                                 TranslatableCaption.of("deny.no_enter"), |                                 TranslatableCaption.of("deny.no_enter"), | ||||||
|                                 TagResolver.resolver("plot", Tag.inserting(Component.text(plot.toString()))) |                                 TagResolver.resolver("plot", Tag.inserting(Component.text(plot.toString()))) | ||||||
| @@ -469,6 +616,19 @@ public class PlayerEventListener implements Listener { | |||||||
|         playerMove(event); |         playerMove(event); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|  |     public void onWorldChanged(PlayerChangedWorldEvent event) { | ||||||
|  |         Player player = event.getPlayer(); | ||||||
|  |         BukkitPlayer pp = BukkitUtil.adapt(player); | ||||||
|  |         if (this.worldEdit != null) { | ||||||
|  |             if (!pp.hasPermission(Permission.PERMISSION_WORLDEDIT_BYPASS)) { | ||||||
|  |                 if (pp.getAttribute("worldedit")) { | ||||||
|  |                     pp.removeAttribute("worldedit"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|     public void vehicleMove(VehicleMoveEvent event) |     public void vehicleMove(VehicleMoveEvent event) | ||||||
|             throws IllegalAccessException { |             throws IllegalAccessException { | ||||||
| @@ -607,7 +767,7 @@ public class PlayerEventListener implements Listener { | |||||||
|                 this.tmpTeleport = true; |                 this.tmpTeleport = true; | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             int border = area.getBorder(); |             int border = area.getBorder(true); | ||||||
|             int x1; |             int x1; | ||||||
|             if (x2 > border && this.tmpTeleport) { |             if (x2 > border && this.tmpTeleport) { | ||||||
|                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_BYPASS_BORDER)) { |                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_BYPASS_BORDER)) { | ||||||
| @@ -702,7 +862,7 @@ public class PlayerEventListener implements Listener { | |||||||
|                 this.tmpTeleport = true; |                 this.tmpTeleport = true; | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             int border = area.getBorder(); |             int border = area.getBorder(true); | ||||||
|             int z1; |             int z1; | ||||||
|             if (z2 > border && this.tmpTeleport) { |             if (z2 > border && this.tmpTeleport) { | ||||||
|                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_BYPASS_BORDER)) { |                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_BYPASS_BORDER)) { | ||||||
| @@ -733,6 +893,7 @@ public class PlayerEventListener implements Listener { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.LOW) |     @EventHandler(priority = EventPriority.LOW) | ||||||
|  |     @SuppressWarnings("deprecation") // Paper deprecation | ||||||
|     public void onChat(AsyncPlayerChatEvent event) { |     public void onChat(AsyncPlayerChatEvent event) { | ||||||
|         if (event.isCancelled()) { |         if (event.isCancelled()) { | ||||||
|             return; |             return; | ||||||
| @@ -807,40 +968,6 @@ public class PlayerEventListener implements Listener { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) |  | ||||||
|     public void onWorldChanged(PlayerChangedWorldEvent event) { |  | ||||||
|         Player player = event.getPlayer(); |  | ||||||
|         BukkitPlayer pp = BukkitUtil.adapt(player); |  | ||||||
|         // Delete last location |  | ||||||
|         Plot plot; |  | ||||||
|         try (final MetaDataAccess<Plot> lastPlotAccess = |  | ||||||
|                      pp.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { |  | ||||||
|             plot = lastPlotAccess.remove(); |  | ||||||
|         } |  | ||||||
|         try (final MetaDataAccess<Location> lastLocationAccess = |  | ||||||
|                      pp.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LOCATION)) { |  | ||||||
|             lastLocationAccess.remove(); |  | ||||||
|         } |  | ||||||
|         if (plot != null) { |  | ||||||
|             plotListener.plotExit(pp, plot); |  | ||||||
|         } |  | ||||||
|         if (this.worldEdit != null) { |  | ||||||
|             if (!pp.hasPermission(Permission.PERMISSION_WORLDEDIT_BYPASS)) { |  | ||||||
|                 if (pp.getAttribute("worldedit")) { |  | ||||||
|                     pp.removeAttribute("worldedit"); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         Location location = pp.getLocation(); |  | ||||||
|         PlotArea area = location.getPlotArea(); |  | ||||||
|         if (location.isPlotArea()) { |  | ||||||
|             plot = location.getPlot(); |  | ||||||
|             if (plot != null) { |  | ||||||
|                 plotListener.plotEntry(pp, plot); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @SuppressWarnings("deprecation") |     @SuppressWarnings("deprecation") | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|     public void onInventoryClick(InventoryClickEvent event) { |     public void onInventoryClick(InventoryClickEvent event) { | ||||||
| @@ -1063,6 +1190,7 @@ public class PlayerEventListener implements Listener { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.LOW) |     @EventHandler(priority = EventPriority.LOW) | ||||||
|  |     @SuppressWarnings("deprecation") // Paper deprecation | ||||||
|     public void onCancelledInteract(PlayerInteractEvent event) { |     public void onCancelledInteract(PlayerInteractEvent event) { | ||||||
|         if (event.isCancelled() && event.getAction() == Action.RIGHT_CLICK_AIR) { |         if (event.isCancelled() && event.getAction() == Action.RIGHT_CLICK_AIR) { | ||||||
|             Player player = event.getPlayer(); |             Player player = event.getPlayer(); | ||||||
| @@ -1127,7 +1255,7 @@ public class PlayerEventListener implements Listener { | |||||||
|                 eventType = PlayerBlockEventType.INTERACT_BLOCK; |                 eventType = PlayerBlockEventType.INTERACT_BLOCK; | ||||||
|                 blocktype1 = BukkitAdapter.asBlockType(block.getType()); |                 blocktype1 = BukkitAdapter.asBlockType(block.getType()); | ||||||
|  |  | ||||||
|                 if (blockType.isInteractable()) { |                 if (INTERACTABLE_MATERIALS != null ? INTERACTABLE_MATERIALS.contains(blockType.name()) : blockType.isInteractable()) { | ||||||
|                     if (!player.isSneaking()) { |                     if (!player.isSneaking()) { | ||||||
|                         break; |                         break; | ||||||
|                     } |                     } | ||||||
| @@ -1244,22 +1372,7 @@ public class PlayerEventListener implements Listener { | |||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|     public void onBucketEmpty(PlayerBucketEmptyEvent event) { |     public void onBucketEmpty(PlayerBucketEmptyEvent event) { | ||||||
|         BlockFace bf = event.getBlockFace(); |         final Block block = event.getBlock(); | ||||||
|         // Note: a month after Bukkit 1.14.4 released, they added the API method |  | ||||||
|         // PlayerBucketEmptyEvent#getBlock(), which returns the block the |  | ||||||
|         // bucket contents is going to be placed at. Currently we determine this |  | ||||||
|         // block ourselves to retain compatibility with 1.13. |  | ||||||
|         final Block block; |  | ||||||
|         // if the block can be waterlogged, the event might waterlog the block |  | ||||||
|         // sometimes |  | ||||||
|         if (event.getBlockClicked().getBlockData() instanceof Waterlogged waterlogged |  | ||||||
|                 && !waterlogged.isWaterlogged() && event.getBucket() != Material.LAVA_BUCKET) { |  | ||||||
|             block = event.getBlockClicked(); |  | ||||||
|         } else { |  | ||||||
|             block = event.getBlockClicked().getLocation() |  | ||||||
|                     .add(bf.getModX(), bf.getModY(), bf.getModZ()) |  | ||||||
|                     .getBlock(); |  | ||||||
|         } |  | ||||||
|         Location location = BukkitUtil.adapt(block.getLocation()); |         Location location = BukkitUtil.adapt(block.getLocation()); | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
| @@ -1612,6 +1725,11 @@ public class PlayerEventListener implements Listener { | |||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|  |             if (EntityCategories.INTERACTION.contains(entityType) && flagContainer | ||||||
|  |                     .getFlag(InteractionInteractFlag.class).getValue()) { | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |  | ||||||
|             if (EntityCategories.VILLAGER.contains(entityType) && flagContainer |             if (EntityCategories.VILLAGER.contains(entityType) && flagContainer | ||||||
|                     .getFlag(VillagerInteractFlag.class).getValue()) { |                     .getFlag(VillagerInteractFlag.class).getValue()) { | ||||||
|                 return; |                 return; | ||||||
| @@ -1855,7 +1973,9 @@ public class PlayerEventListener implements Listener { | |||||||
|  |  | ||||||
|     @EventHandler |     @EventHandler | ||||||
|     public void onPlayerTakeLecternBook(PlayerTakeLecternBookEvent event) { |     public void onPlayerTakeLecternBook(PlayerTakeLecternBookEvent event) { | ||||||
|         Location location = BukkitUtil.adapt(event.getPlayer().getLocation()); |         Player player = event.getPlayer(); | ||||||
|  |         BukkitPlayer pp = BukkitUtil.adapt(player); | ||||||
|  |         Location location = pp.getLocation(); | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
|             return; |             return; | ||||||
| @@ -1867,9 +1987,11 @@ public class PlayerEventListener implements Listener { | |||||||
|             } |             } | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         if (plot.getFlag(LecternReadBookFlag.class)) { |         if (!plot.isAdded(pp.getUUID())) { | ||||||
|             plot.debug(event.getPlayer().getName() + " could not take the book because of lectern-read-book = true"); |             if (plot.getFlag(LecternReadBookFlag.class)) { | ||||||
|             event.setCancelled(true); |                 plot.debug(event.getPlayer().getName() + " could not take the book because of lectern-read-book = true"); | ||||||
|  |                 event.setCancelled(true); | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -0,0 +1,66 @@ | |||||||
|  | /* | ||||||
|  |  * PlotSquared, a land and world management plugin for Minecraft. | ||||||
|  |  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||||
|  |  * Copyright (C) IntellectualSites team and contributors | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 3 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  | package com.plotsquared.bukkit.listener; | ||||||
|  |  | ||||||
|  | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
|  | import com.plotsquared.core.location.Location; | ||||||
|  | import com.plotsquared.core.permissions.Permission; | ||||||
|  | import com.plotsquared.core.plot.Plot; | ||||||
|  | import com.plotsquared.core.plot.PlotArea; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.EditSignFlag; | ||||||
|  | import com.plotsquared.core.util.PlotFlagUtil; | ||||||
|  | import org.bukkit.block.Sign; | ||||||
|  | import org.bukkit.event.EventHandler; | ||||||
|  | import org.bukkit.event.Listener; | ||||||
|  | import org.bukkit.event.player.PlayerSignOpenEvent; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * For events since 1.20.1 | ||||||
|  |  * @since 7.2.1 | ||||||
|  |  */ | ||||||
|  | public class PlayerEventListener1201 implements Listener { | ||||||
|  |  | ||||||
|  |     @EventHandler(ignoreCancelled = true) | ||||||
|  |     @SuppressWarnings({"removal", "UnstableApiUsage"}) // thanks Paper, thanks Spigot | ||||||
|  |     public void onPlayerSignOpenEvent(PlayerSignOpenEvent event) { | ||||||
|  |         Sign sign = event.getSign(); | ||||||
|  |         Location location = BukkitUtil.adapt(sign.getLocation()); | ||||||
|  |         PlotArea area = location.getPlotArea(); | ||||||
|  |         if (area == null) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         Plot plot = location.getOwnedPlot(); | ||||||
|  |         if (plot == null) { | ||||||
|  |             if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, EditSignFlag.class, false) | ||||||
|  |                     && !event.getPlayer().hasPermission(Permission.PERMISSION_ADMIN_INTERACT_ROAD.toString())) { | ||||||
|  |                 event.setCancelled(true); | ||||||
|  |             } | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (plot.isAdded(event.getPlayer().getUniqueId())) { | ||||||
|  |             return; // allow for added players | ||||||
|  |         } | ||||||
|  |         if (!plot.getFlag(EditSignFlag.class) | ||||||
|  |                 && !event.getPlayer().hasPermission(Permission.PERMISSION_ADMIN_INTERACT_OTHER.toString())) { | ||||||
|  |             plot.debug(event.getPlayer().getName() + " could not edit the sign because of edit-sign = false"); | ||||||
|  |             event.setCancelled(true); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -28,12 +28,14 @@ import com.plotsquared.core.player.PlotPlayer; | |||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| import com.plotsquared.core.plot.PlotArea; | import com.plotsquared.core.plot.PlotArea; | ||||||
| import com.plotsquared.core.plot.PlotHandler; | import com.plotsquared.core.plot.PlotHandler; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.FishingFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.ProjectilesFlag; | import com.plotsquared.core.plot.flag.implementations.ProjectilesFlag; | ||||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | import com.plotsquared.core.plot.world.PlotAreaManager; | ||||||
| import com.plotsquared.core.util.PlotFlagUtil; | import com.plotsquared.core.util.PlotFlagUtil; | ||||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | import net.kyori.adventure.text.minimessage.tag.Tag; | ||||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
|  | import org.bukkit.entity.FishHook; | ||||||
| import org.bukkit.entity.LivingEntity; | import org.bukkit.entity.LivingEntity; | ||||||
| import org.bukkit.entity.Player; | import org.bukkit.entity.Player; | ||||||
| import org.bukkit.entity.Projectile; | import org.bukkit.entity.Projectile; | ||||||
| @@ -132,6 +134,11 @@ public class ProjectileEventListener implements Listener { | |||||||
|                 event.setCancelled(true); |                 event.setCancelled(true); | ||||||
|             } |             } | ||||||
|         } else if (!plot.isAdded(pp.getUUID())) { |         } else if (!plot.isAdded(pp.getUUID())) { | ||||||
|  |             if (entity instanceof FishHook) { | ||||||
|  |                 if (plot.getFlag(FishingFlag.class)) { | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|             if (!plot.getFlag(ProjectilesFlag.class)) { |             if (!plot.getFlag(ProjectilesFlag.class)) { | ||||||
|                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { |                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||||
|                     pp.sendMessage( |                     pp.sendMessage( | ||||||
| @@ -187,7 +194,8 @@ public class ProjectileEventListener implements Listener { | |||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             if (plot.isAdded(pp.getUUID()) || pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER) || plot.getFlag( |             if (plot.isAdded(pp.getUUID()) || pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER) || plot.getFlag( | ||||||
|                     ProjectilesFlag.class)) { |                     ProjectilesFlag.class) || (entity instanceof FishHook && plot.getFlag( | ||||||
|  |                     FishingFlag.class))) { | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             entity.remove(); |             entity.remove(); | ||||||
|   | |||||||
| @@ -21,9 +21,14 @@ package com.plotsquared.bukkit.listener; | |||||||
| import com.google.inject.Inject; | import com.google.inject.Inject; | ||||||
| import com.plotsquared.bukkit.BukkitPlatform; | import com.plotsquared.bukkit.BukkitPlatform; | ||||||
| import com.plotsquared.bukkit.placeholder.MVdWPlaceholders; | import com.plotsquared.bukkit.placeholder.MVdWPlaceholders; | ||||||
|  | import com.plotsquared.bukkit.util.BukkitEconHandler; | ||||||
|  | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.player.ConsolePlayer; | import com.plotsquared.core.player.ConsolePlayer; | ||||||
|  | import com.plotsquared.core.util.EconHandler; | ||||||
|  | import org.apache.logging.log4j.LogManager; | ||||||
|  | import org.apache.logging.log4j.Logger; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
| import org.bukkit.event.EventHandler; | import org.bukkit.event.EventHandler; | ||||||
| import org.bukkit.event.Listener; | import org.bukkit.event.Listener; | ||||||
| @@ -32,6 +37,8 @@ import org.checkerframework.checker.nullness.qual.NonNull; | |||||||
|  |  | ||||||
| public class ServerListener implements Listener { | public class ServerListener implements Listener { | ||||||
|  |  | ||||||
|  |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + ServerListener.class.getSimpleName()); | ||||||
|  |  | ||||||
|     private final BukkitPlatform plugin; |     private final BukkitPlatform plugin; | ||||||
|  |  | ||||||
|     @Inject |     @Inject | ||||||
| @@ -45,6 +52,29 @@ public class ServerListener implements Listener { | |||||||
|             new MVdWPlaceholders(this.plugin, this.plugin.placeholderRegistry()); |             new MVdWPlaceholders(this.plugin, this.plugin.placeholderRegistry()); | ||||||
|             ConsolePlayer.getConsole().sendMessage(TranslatableCaption.of("placeholder.hooked")); |             ConsolePlayer.getConsole().sendMessage(TranslatableCaption.of("placeholder.hooked")); | ||||||
|         } |         } | ||||||
|  |         if (Settings.Enabled_Components.ECONOMY && Bukkit.getPluginManager().isPluginEnabled("Vault")) { | ||||||
|  |             EconHandler econHandler = new BukkitEconHandler(); | ||||||
|  |             try { | ||||||
|  |                 if (!econHandler.init()) { | ||||||
|  |                     LOGGER.warn("Economy is enabled but no plugin is providing an economy service. Falling back..."); | ||||||
|  |                     econHandler = EconHandler.nullEconHandler(); | ||||||
|  |                 } | ||||||
|  |             } catch (final Exception ignored) { | ||||||
|  |                 econHandler = EconHandler.nullEconHandler(); | ||||||
|  |             } | ||||||
|  |             if (PlotSquared.platform().econHandler() instanceof MutableEconHandler meh) { | ||||||
|  |                 meh.setImplementation(econHandler); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Internal use only. Required to implement lazy econ loading using Guice. | ||||||
|  |      * | ||||||
|  |      * @since 7.2.0 | ||||||
|  |      */ | ||||||
|  |     public interface MutableEconHandler { | ||||||
|  |         void setImplementation(EconHandler econHandler); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -49,9 +49,14 @@ public class SingleWorldListener implements Listener { | |||||||
|             this.methodGetHandleChunk = classCraftChunk.getMethod("getHandle").getRealMethod(); |             this.methodGetHandleChunk = classCraftChunk.getMethod("getHandle").getRealMethod(); | ||||||
|         } catch (NoSuchMethodException ignored) { |         } catch (NoSuchMethodException ignored) { | ||||||
|             try { |             try { | ||||||
|                 ReflectionUtils.RefClass classChunkStatus = getRefClass("net.minecraft.world.level.chunk.ChunkStatus"); |                 String chunkStatus = PlotSquared.platform().serverVersion()[1] < 21 | ||||||
|  |                         ? "net.minecraft.world.level.chunk" + ".ChunkStatus" | ||||||
|  |                         : "net.minecraft.world.level.chunk.status.ChunkStatus"; | ||||||
|  |                 ReflectionUtils.RefClass classChunkStatus = getRefClass(chunkStatus); | ||||||
|                 this.objChunkStatusFull = classChunkStatus.getRealClass().getField("n").get(null); |                 this.objChunkStatusFull = classChunkStatus.getRealClass().getField("n").get(null); | ||||||
|                 this.methodGetHandleChunk = classCraftChunk.getMethod("getHandle", classChunkStatus.getRealClass()).getRealMethod(); |                 this.methodGetHandleChunk = classCraftChunk | ||||||
|  |                         .getMethod("getHandle", classChunkStatus.getRealClass()) | ||||||
|  |                         .getRealMethod(); | ||||||
|             } catch (NoSuchMethodException ex) { |             } catch (NoSuchMethodException ex) { | ||||||
|                 throw new RuntimeException(ex); |                 throw new RuntimeException(ex); | ||||||
|             } |             } | ||||||
| @@ -95,7 +100,8 @@ public class SingleWorldListener implements Listener { | |||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.LOWEST) |     @EventHandler(priority = EventPriority.LOWEST) | ||||||
|     public void onChunkLoad(ChunkLoadEvent event) { |     public void onChunkLoad(ChunkLoadEvent event) { | ||||||
|         handle(event); |         // disable this for now, should address https://github.com/IntellectualSites/PlotSquared/issues/4413 | ||||||
|  |         // handle(event); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,107 @@ | |||||||
|  | /* | ||||||
|  |  * PlotSquared, a land and world management plugin for Minecraft. | ||||||
|  |  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||||
|  |  * Copyright (C) IntellectualSites team and contributors | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 3 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  | package com.plotsquared.bukkit.permissions; | ||||||
|  |  | ||||||
|  | import com.plotsquared.bukkit.player.BukkitPlayer; | ||||||
|  | import com.plotsquared.core.permissions.RangedPermissionResolver; | ||||||
|  | import com.plotsquared.core.player.PlotPlayer; | ||||||
|  | import com.plotsquared.core.util.MathMan; | ||||||
|  | import org.bukkit.permissions.PermissionAttachmentInfo; | ||||||
|  | import org.checkerframework.checker.index.qual.NonNegative; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  |  | ||||||
|  | import java.util.Set; | ||||||
|  | import java.util.stream.IntStream; | ||||||
|  |  | ||||||
|  | public class BukkitRangedPermissionResolver implements RangedPermissionResolver { | ||||||
|  |  | ||||||
|  |     private static boolean CHECK_EFFECTIVE = true; | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NonNegative int getPermissionRange( | ||||||
|  |             final @NonNull PlotPlayer<?> generic, | ||||||
|  |             final @NonNull String stub, | ||||||
|  |             final @Nullable String worldContext, | ||||||
|  |             @NonNegative final int range | ||||||
|  |     ) { | ||||||
|  |         if (!(generic instanceof BukkitPlayer player)) { | ||||||
|  |             throw new IllegalArgumentException("PlotPlayer is not a BukkitPlayer"); | ||||||
|  |         } | ||||||
|  |         if (hasWildcardRange(player, stub, worldContext)) { | ||||||
|  |             return INFINITE_RANGE_VALUE; | ||||||
|  |         } | ||||||
|  |         int max = 0; | ||||||
|  |         if (CHECK_EFFECTIVE) { | ||||||
|  |             boolean hasAny = false; | ||||||
|  |             String stubPlus = stub + "."; | ||||||
|  |             final Set<PermissionAttachmentInfo> effective = player.getPlatformPlayer().getEffectivePermissions(); | ||||||
|  |             if (!effective.isEmpty()) { | ||||||
|  |                 for (PermissionAttachmentInfo attach : effective) { | ||||||
|  |                     // Ignore all "false" permissions | ||||||
|  |                     if (!attach.getValue()) { | ||||||
|  |                         continue; | ||||||
|  |                     } | ||||||
|  |                     String permStr = attach.getPermission(); | ||||||
|  |                     if (permStr.startsWith(stubPlus)) { | ||||||
|  |                         hasAny = true; | ||||||
|  |                         String end = permStr.substring(stubPlus.length()); | ||||||
|  |                         if (MathMan.isInteger(end)) { | ||||||
|  |                             int val = Integer.parseInt(end); | ||||||
|  |                             if (val > range) { | ||||||
|  |                                 return val; | ||||||
|  |                             } | ||||||
|  |                             if (val > max) { | ||||||
|  |                                 max = val; | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 if (hasAny) { | ||||||
|  |                     return max; | ||||||
|  |                 } | ||||||
|  |                 // Workaround | ||||||
|  |                 for (PermissionAttachmentInfo attach : effective) { | ||||||
|  |                     String permStr = attach.getPermission(); | ||||||
|  |                     if (permStr.startsWith("plots.") && !permStr.equals("plots.use")) { | ||||||
|  |                         return max; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 CHECK_EFFECTIVE = false; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         for (int i = range; i > 0; i--) { | ||||||
|  |             if (player.hasPermission(worldContext, stub + "." + i)) { | ||||||
|  |                 return i; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return max; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NonNull IntStream streamFullPermissionRange( | ||||||
|  |             final @NonNull PlotPlayer<?> player, | ||||||
|  |             final @NonNull String stub, | ||||||
|  |             final @Nullable String worldContext, | ||||||
|  |             @NonNegative final int range | ||||||
|  |     ) { | ||||||
|  |         return IntStream.of(getPermissionRange(player, stub, worldContext, range)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,95 @@ | |||||||
|  | /* | ||||||
|  |  * PlotSquared, a land and world management plugin for Minecraft. | ||||||
|  |  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||||
|  |  * Copyright (C) IntellectualSites team and contributors | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 3 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  | package com.plotsquared.bukkit.permissions; | ||||||
|  |  | ||||||
|  | import com.plotsquared.core.permissions.RangedPermissionResolver; | ||||||
|  | import com.plotsquared.core.player.PlotPlayer; | ||||||
|  | import com.plotsquared.core.util.MathMan; | ||||||
|  | import net.luckperms.api.LuckPerms; | ||||||
|  | import net.luckperms.api.context.ImmutableContextSet; | ||||||
|  | import net.luckperms.api.model.user.User; | ||||||
|  | import net.luckperms.api.node.NodeType; | ||||||
|  | import net.luckperms.api.node.types.PermissionNode; | ||||||
|  | import net.luckperms.api.query.QueryOptions; | ||||||
|  | import org.bukkit.Bukkit; | ||||||
|  | import org.checkerframework.checker.index.qual.NonNegative; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  |  | ||||||
|  | import java.util.Objects; | ||||||
|  | import java.util.stream.IntStream; | ||||||
|  |  | ||||||
|  | public class LuckPermsRangedPermissionResolver implements RangedPermissionResolver { | ||||||
|  |  | ||||||
|  |     private final LuckPerms luckPerms; | ||||||
|  |  | ||||||
|  |     public LuckPermsRangedPermissionResolver() { | ||||||
|  |         this.luckPerms = Objects.requireNonNull( | ||||||
|  |                 Bukkit.getServicesManager().getRegistration(LuckPerms.class), | ||||||
|  |                 "LuckPerms is not available" | ||||||
|  |         ).getProvider(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NonNegative int getPermissionRange( | ||||||
|  |             final @NonNull PlotPlayer<?> player, | ||||||
|  |             final @NonNull String stub, | ||||||
|  |             final @Nullable String worldContext, | ||||||
|  |             final @NonNegative int range | ||||||
|  |     ) { | ||||||
|  |         // no need to use LuckPerms for basic checks | ||||||
|  |         if (this.hasWildcardRange(player, stub, worldContext)) { | ||||||
|  |             return INFINITE_RANGE_VALUE; | ||||||
|  |         } | ||||||
|  |         return this.streamFullPermissionRange(player, stub, worldContext, range) | ||||||
|  |                 .sorted() | ||||||
|  |                 .reduce((first, second) -> second) | ||||||
|  |                 .orElse(0); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @NonNull | ||||||
|  |     public IntStream streamFullPermissionRange( | ||||||
|  |             final @NonNull PlotPlayer<?> player, | ||||||
|  |             final @NonNull String stub, | ||||||
|  |             final @Nullable String worldContext, | ||||||
|  |             @NonNegative final int range | ||||||
|  |     ) { | ||||||
|  |         final User user = this.luckPerms.getUserManager().getUser(player.getUUID()); | ||||||
|  |         if (user == null) { | ||||||
|  |             throw new IllegalStateException("Luckperms User is null - is the Player online? (UUID: %s)".formatted(player.getUUID())); | ||||||
|  |         } | ||||||
|  |         final QueryOptions queryOptions = worldContext == null ? | ||||||
|  |                 QueryOptions.nonContextual() : | ||||||
|  |                 QueryOptions.contextual(ImmutableContextSet.of("world", worldContext)); | ||||||
|  |         return user.resolveInheritedNodes(queryOptions).stream() | ||||||
|  |                 // only support normal permission nodes (regex permission nodes would be a pita to support) | ||||||
|  |                 .filter(NodeType.PERMISSION::matches) | ||||||
|  |                 .map(node -> ((PermissionNode) node).getPermission()) | ||||||
|  |                 // check that the node actually has additional data after the stub | ||||||
|  |                 .filter(permission -> permission.startsWith(stub + ".")) | ||||||
|  |                 // extract the raw data after the stub | ||||||
|  |                 .map(permission -> permission.substring(stub.length() + 1)) | ||||||
|  |                 // check if data is integer and parse | ||||||
|  |                 .filter(MathMan::isInteger) | ||||||
|  |                 .mapToInt(Integer::parseInt) | ||||||
|  |                 // only use values that are positive | ||||||
|  |                 .filter(value -> value > -1); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -20,6 +20,8 @@ package com.plotsquared.bukkit.placeholder; | |||||||
|  |  | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.DoneFlag; | ||||||
|  | import com.plotsquared.core.util.query.PlotQuery; | ||||||
| import me.clip.placeholderapi.PlaceholderAPIPlugin; | import me.clip.placeholderapi.PlaceholderAPIPlugin; | ||||||
| import me.clip.placeholderapi.expansion.PlaceholderExpansion; | import me.clip.placeholderapi.expansion.PlaceholderExpansion; | ||||||
| import org.bukkit.entity.Player; | import org.bukkit.entity.Player; | ||||||
| @@ -83,6 +85,20 @@ public class PAPIPlaceholders extends PlaceholderExpansion { | |||||||
|             return String.valueOf(pl.getPlotCount(identifier)); |             return String.valueOf(pl.getPlotCount(identifier)); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         if (identifier.startsWith("base_plot_count_")) { | ||||||
|  |             identifier = identifier.substring("base_plot_count_".length()); | ||||||
|  |             if (identifier.isEmpty()) { | ||||||
|  |                 return ""; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             return String.valueOf(PlotQuery.newQuery() | ||||||
|  |                     .ownedBy(pl) | ||||||
|  |                     .inWorld(identifier) | ||||||
|  |                     .whereBasePlot() | ||||||
|  |                     .thatPasses(plot -> !DoneFlag.isDone(plot)) | ||||||
|  |                     .count()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         // PlotSquared placeholders |         // PlotSquared placeholders | ||||||
|         return PlotSquared.platform().placeholderRegistry().getPlaceholderValue(identifier, pl); |         return PlotSquared.platform().placeholderRegistry().getPlaceholderValue(identifier, pl); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -24,14 +24,13 @@ import com.plotsquared.core.PlotSquared; | |||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.events.TeleportCause; | import com.plotsquared.core.events.TeleportCause; | ||||||
| import com.plotsquared.core.location.Location; | import com.plotsquared.core.location.Location; | ||||||
| import com.plotsquared.core.permissions.Permission; |  | ||||||
| import com.plotsquared.core.permissions.PermissionHandler; | import com.plotsquared.core.permissions.PermissionHandler; | ||||||
| import com.plotsquared.core.player.ConsolePlayer; | import com.plotsquared.core.player.ConsolePlayer; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.PlotWeather; | import com.plotsquared.core.plot.PlotWeather; | ||||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | import com.plotsquared.core.plot.world.PlotAreaManager; | ||||||
| import com.plotsquared.core.util.EventDispatcher; | import com.plotsquared.core.util.EventDispatcher; | ||||||
| import com.plotsquared.core.util.MathMan; | import com.plotsquared.core.util.WorldUtil; | ||||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||||
| import com.sk89q.worldedit.extension.platform.Actor; | import com.sk89q.worldedit.extension.platform.Actor; | ||||||
| import com.sk89q.worldedit.world.item.ItemType; | import com.sk89q.worldedit.world.item.ItemType; | ||||||
| @@ -40,19 +39,17 @@ import io.papermc.lib.PaperLib; | |||||||
| import net.kyori.adventure.audience.Audience; | import net.kyori.adventure.audience.Audience; | ||||||
| import org.bukkit.GameMode; | import org.bukkit.GameMode; | ||||||
| import org.bukkit.Sound; | import org.bukkit.Sound; | ||||||
|  | import org.bukkit.SoundCategory; | ||||||
| import org.bukkit.WeatherType; | import org.bukkit.WeatherType; | ||||||
| import org.bukkit.entity.Player; | import org.bukkit.entity.Player; | ||||||
| import org.bukkit.event.Event; | import org.bukkit.event.Event; | ||||||
| import org.bukkit.event.EventException; | import org.bukkit.event.EventException; | ||||||
| import org.bukkit.event.player.PlayerTeleportEvent; | import org.bukkit.event.player.PlayerTeleportEvent; | ||||||
| import org.bukkit.permissions.PermissionAttachmentInfo; |  | ||||||
| import org.bukkit.plugin.RegisteredListener; | import org.bukkit.plugin.RegisteredListener; | ||||||
| import org.bukkit.potion.PotionEffectType; | import org.bukkit.potion.PotionEffectType; | ||||||
| import org.checkerframework.checker.index.qual.NonNegative; | import org.checkerframework.checker.index.qual.NonNegative; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.util.Arrays; |  | ||||||
| import java.util.Set; |  | ||||||
| import java.util.UUID; | import java.util.UUID; | ||||||
|  |  | ||||||
| import static com.sk89q.worldedit.world.gamemode.GameModes.ADVENTURE; | import static com.sk89q.worldedit.world.gamemode.GameModes.ADVENTURE; | ||||||
| @@ -120,6 +117,9 @@ public class BukkitPlayer extends PlotPlayer<Player> { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean canTeleport(final @NonNull Location location) { |     public boolean canTeleport(final @NonNull Location location) { | ||||||
|  |         if (!WorldUtil.isValidLocation(location)) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|         final org.bukkit.Location to = BukkitUtil.adapt(location); |         final org.bukkit.Location to = BukkitUtil.adapt(location); | ||||||
|         final org.bukkit.Location from = player.getLocation(); |         final org.bukkit.Location from = player.getLocation(); | ||||||
|         PlayerTeleportEvent event = new PlayerTeleportEvent(player, from, to); |         PlayerTeleportEvent event = new PlayerTeleportEvent(player, from, to); | ||||||
| @@ -146,79 +146,18 @@ public class BukkitPlayer extends PlotPlayer<Player> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @SuppressWarnings("StringSplitter") |  | ||||||
|     @Override |     @Override | ||||||
|     @NonNegative |     @NonNegative | ||||||
|     public int hasPermissionRange( |     public int hasPermissionRange( | ||||||
|             final @NonNull String stub, |             final @NonNull String stub, | ||||||
|             @NonNegative final int range |             @NonNegative final int range | ||||||
|     ) { |     ) { | ||||||
|         if (hasPermission(Permission.PERMISSION_ADMIN.toString())) { |         return PlotSquared.platform().rangedPermissionResolver().getPermissionRange(this, stub, null, range); | ||||||
|             return Integer.MAX_VALUE; |  | ||||||
|         } |  | ||||||
|         final String[] nodes = stub.split("\\."); |  | ||||||
|         final StringBuilder n = new StringBuilder(); |  | ||||||
|         for (int i = 0; i < (nodes.length - 1); i++) { |  | ||||||
|             n.append(nodes[i]).append("."); |  | ||||||
|             if (!stub.equals(n + Permission.PERMISSION_STAR.toString())) { |  | ||||||
|                 if (hasPermission(n + Permission.PERMISSION_STAR.toString())) { |  | ||||||
|                     return Integer.MAX_VALUE; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         if (hasPermission(stub + ".*")) { |  | ||||||
|             return Integer.MAX_VALUE; |  | ||||||
|         } |  | ||||||
|         int max = 0; |  | ||||||
|         if (CHECK_EFFECTIVE) { |  | ||||||
|             boolean hasAny = false; |  | ||||||
|             String stubPlus = stub + "."; |  | ||||||
|             final Set<PermissionAttachmentInfo> effective = player.getEffectivePermissions(); |  | ||||||
|             if (!effective.isEmpty()) { |  | ||||||
|                 for (PermissionAttachmentInfo attach : effective) { |  | ||||||
|                     // Ignore all "false" permissions |  | ||||||
|                     if (!attach.getValue()) { |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
|                     String permStr = attach.getPermission(); |  | ||||||
|                     if (permStr.startsWith(stubPlus)) { |  | ||||||
|                         hasAny = true; |  | ||||||
|                         String end = permStr.substring(stubPlus.length()); |  | ||||||
|                         if (MathMan.isInteger(end)) { |  | ||||||
|                             int val = Integer.parseInt(end); |  | ||||||
|                             if (val > range) { |  | ||||||
|                                 return val; |  | ||||||
|                             } |  | ||||||
|                             if (val > max) { |  | ||||||
|                                 max = val; |  | ||||||
|                             } |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 if (hasAny) { |  | ||||||
|                     return max; |  | ||||||
|                 } |  | ||||||
|                 // Workaround |  | ||||||
|                 for (PermissionAttachmentInfo attach : effective) { |  | ||||||
|                     String permStr = attach.getPermission(); |  | ||||||
|                     if (permStr.startsWith("plots.") && !permStr.equals("plots.use")) { |  | ||||||
|                         return max; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 CHECK_EFFECTIVE = false; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         for (int i = range; i > 0; i--) { |  | ||||||
|             if (hasPermission(stub + "." + i)) { |  | ||||||
|                 return i; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return max; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void teleport(final @NonNull Location location, final @NonNull TeleportCause cause) { |     public void teleport(final @NonNull Location location, final @NonNull TeleportCause cause) { | ||||||
|         if (Math.abs(location.getX()) >= 30000000 || Math.abs(location.getZ()) >= 30000000) { |         if (!WorldUtil.isValidLocation(location)) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         final org.bukkit.Location bukkitLocation = |         final org.bukkit.Location bukkitLocation = | ||||||
| @@ -306,18 +245,21 @@ public class BukkitPlayer extends PlotPlayer<Player> { | |||||||
|     @Override |     @Override | ||||||
|     public void playMusic(final @NonNull Location location, final @NonNull ItemType id) { |     public void playMusic(final @NonNull Location location, final @NonNull ItemType id) { | ||||||
|         if (id == ItemTypes.AIR) { |         if (id == ItemTypes.AIR) { | ||||||
|             // Let's just stop all the discs because why not? |             if (PlotSquared.platform().serverVersion()[1] >= 19) { | ||||||
|             for (final Sound sound : Arrays.stream(Sound.values()) |                 player.stopSound(SoundCategory.MUSIC); | ||||||
|                     .filter(sound -> sound.name().contains("DISC")).toList()) { |                 return; | ||||||
|                 player.stopSound(sound); |  | ||||||
|             } |             } | ||||||
|             // this.player.playEffect(BukkitUtil.getLocation(location), Effect.RECORD_PLAY, Material.AIR); |             // 1.18 and downwards require a specific Sound to stop (even tho the packet does not??) | ||||||
|         } else { |             for (final Sound sound : Sound.values()) { | ||||||
|             // this.player.playEffect(BukkitUtil.getLocation(location), Effect.RECORD_PLAY, id.to(Material.class)); |                 if (sound.name().startsWith("MUSIC_DISC")) { | ||||||
|             this.player.playSound(BukkitUtil.adapt(location), |                     this.player.stopSound(sound, SoundCategory.MUSIC); | ||||||
|                     Sound.valueOf(BukkitAdapter.adapt(id).name()), Float.MAX_VALUE, 1f |                 } | ||||||
|             ); |             } | ||||||
|  |             return; | ||||||
|         } |         } | ||||||
|  |         this.player.playSound(BukkitUtil.adapt(location), Sound.valueOf(BukkitAdapter.adapt(id).name()), | ||||||
|  |                 SoundCategory.MUSIC, Float.MAX_VALUE, 1f | ||||||
|  |         ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @SuppressWarnings("deprecation") // Needed for Spigot compatibility |     @SuppressWarnings("deprecation") // Needed for Spigot compatibility | ||||||
|   | |||||||
| @@ -30,6 +30,8 @@ import com.plotsquared.core.util.task.TaskTime; | |||||||
| import com.sk89q.worldedit.math.BlockVector2; | import com.sk89q.worldedit.math.BlockVector2; | ||||||
| import com.sk89q.worldedit.world.World; | import com.sk89q.worldedit.world.World; | ||||||
| import io.papermc.lib.PaperLib; | import io.papermc.lib.PaperLib; | ||||||
|  | import org.apache.logging.log4j.LogManager; | ||||||
|  | import org.apache.logging.log4j.Logger; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
| import org.bukkit.Chunk; | import org.bukkit.Chunk; | ||||||
| import org.bukkit.plugin.Plugin; | import org.bukkit.plugin.Plugin; | ||||||
| @@ -41,6 +43,8 @@ import java.util.LinkedList; | |||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Queue; | import java.util.Queue; | ||||||
| import java.util.concurrent.LinkedBlockingQueue; | import java.util.concurrent.LinkedBlockingQueue; | ||||||
|  | import java.util.concurrent.TimeUnit; | ||||||
|  | import java.util.concurrent.TimeoutException; | ||||||
| import java.util.concurrent.atomic.AtomicInteger; | import java.util.concurrent.atomic.AtomicInteger; | ||||||
| import java.util.function.Consumer; | import java.util.function.Consumer; | ||||||
|  |  | ||||||
| @@ -55,6 +59,8 @@ import java.util.function.Consumer; | |||||||
|  **/ |  **/ | ||||||
| public final class BukkitChunkCoordinator extends ChunkCoordinator { | public final class BukkitChunkCoordinator extends ChunkCoordinator { | ||||||
|  |  | ||||||
|  |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + BukkitChunkCoordinator.class.getSimpleName()); | ||||||
|  |  | ||||||
|     private final List<ProgressSubscriber> progressSubscribers = new LinkedList<>(); |     private final List<ProgressSubscriber> progressSubscribers = new LinkedList<>(); | ||||||
|  |  | ||||||
|     private final Queue<BlockVector2> requestedChunks; |     private final Queue<BlockVector2> requestedChunks; | ||||||
| @@ -70,6 +76,7 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | |||||||
|     private final AtomicInteger expectedSize; |     private final AtomicInteger expectedSize; | ||||||
|     private final AtomicInteger loadingChunks = new AtomicInteger(); |     private final AtomicInteger loadingChunks = new AtomicInteger(); | ||||||
|     private final boolean forceSync; |     private final boolean forceSync; | ||||||
|  |     private final boolean shouldGen; | ||||||
|  |  | ||||||
|     private int batchSize; |     private int batchSize; | ||||||
|     private PlotSquaredTask task; |     private PlotSquaredTask task; | ||||||
| @@ -87,7 +94,8 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | |||||||
|             @Assisted final @NonNull Consumer<Throwable> throwableConsumer, |             @Assisted final @NonNull Consumer<Throwable> throwableConsumer, | ||||||
|             @Assisted("unloadAfter") final boolean unloadAfter, |             @Assisted("unloadAfter") final boolean unloadAfter, | ||||||
|             @Assisted final @NonNull Collection<ProgressSubscriber> progressSubscribers, |             @Assisted final @NonNull Collection<ProgressSubscriber> progressSubscribers, | ||||||
|             @Assisted("forceSync") final boolean forceSync |             @Assisted("forceSync") final boolean forceSync, | ||||||
|  |             @Assisted("shouldGen") final boolean shouldGen | ||||||
|     ) { |     ) { | ||||||
|         this.requestedChunks = new LinkedBlockingQueue<>(requestedChunks); |         this.requestedChunks = new LinkedBlockingQueue<>(requestedChunks); | ||||||
|         this.availableChunks = new LinkedBlockingQueue<>(); |         this.availableChunks = new LinkedBlockingQueue<>(); | ||||||
| @@ -103,6 +111,7 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | |||||||
|         this.bukkitWorld = Bukkit.getWorld(world.getName()); |         this.bukkitWorld = Bukkit.getWorld(world.getName()); | ||||||
|         this.progressSubscribers.addAll(progressSubscribers); |         this.progressSubscribers.addAll(progressSubscribers); | ||||||
|         this.forceSync = forceSync; |         this.forceSync = forceSync; | ||||||
|  |         this.shouldGen = shouldGen; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -212,18 +221,28 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | |||||||
|      * Requests a batch of chunks to be loaded |      * Requests a batch of chunks to be loaded | ||||||
|      */ |      */ | ||||||
|     private void requestBatch() { |     private void requestBatch() { | ||||||
|         BlockVector2 chunk; |         for (int i = 0; i < this.batchSize && this.requestedChunks.peek() != null; i++) { | ||||||
|         for (int i = 0; i < this.batchSize && (chunk = this.requestedChunks.poll()) != null; i++) { |  | ||||||
|             // This required PaperLib to be bumped to version 1.0.4 to mark the request as urgent |             // This required PaperLib to be bumped to version 1.0.4 to mark the request as urgent | ||||||
|  |             final BlockVector2 chunk = this.requestedChunks.poll(); | ||||||
|             loadingChunks.incrementAndGet(); |             loadingChunks.incrementAndGet(); | ||||||
|             PaperLib |             PaperLib | ||||||
|                     .getChunkAtAsync(this.bukkitWorld, chunk.getX(), chunk.getZ(), true, true) |                     .getChunkAtAsync(this.bukkitWorld, chunk.getX(), chunk.getZ(), shouldGen, true) | ||||||
|  |                     .orTimeout(10L, TimeUnit.SECONDS) | ||||||
|                     .whenComplete((chunkObject, throwable) -> { |                     .whenComplete((chunkObject, throwable) -> { | ||||||
|                         loadingChunks.decrementAndGet(); |                         loadingChunks.decrementAndGet(); | ||||||
|                         if (throwable != null) { |                         if (throwable != null) { | ||||||
|                             throwable.printStackTrace(); |                             if (throwable instanceof TimeoutException) { | ||||||
|                             // We want one less because this couldn't be processed |                                 LOGGER.warn("Timed out awaiting chunk load {}", chunk); | ||||||
|                             this.expectedSize.decrementAndGet(); |                                 this.requestedChunks.offer(chunk); | ||||||
|  |                             } else { | ||||||
|  |                                 LOGGER.error("Failed to load chunk {}", chunk, throwable); | ||||||
|  |                                 // We want one less because this couldn't be processed | ||||||
|  |                                 this.expectedSize.decrementAndGet(); | ||||||
|  |                             } | ||||||
|  |                         } else if (chunkObject == null) { | ||||||
|  |                             if (shouldGen) { | ||||||
|  |                                 LOGGER.error("Null chunk returned for chunk at {}", chunk); | ||||||
|  |                             } | ||||||
|                         } else if (PlotSquared.get().isMainThread(Thread.currentThread())) { |                         } else if (PlotSquared.get().isMainThread(Thread.currentThread())) { | ||||||
|                             this.processChunk(chunkObject); |                             this.processChunk(chunkObject); | ||||||
|                         } else { |                         } else { | ||||||
|   | |||||||
| @@ -62,19 +62,28 @@ public class BukkitQueueCoordinator extends BasicQueueCoordinator { | |||||||
|     private static final SideEffectSet EDGE_LIGHTING_SIDE_EFFECT_SET; |     private static final SideEffectSet EDGE_LIGHTING_SIDE_EFFECT_SET; | ||||||
|  |  | ||||||
|     static { |     static { | ||||||
|         NO_SIDE_EFFECT_SET = SideEffectSet.none().with(SideEffect.LIGHTING, SideEffect.State.OFF).with( |         NO_SIDE_EFFECT_SET = enableNetworkIfNeeded() | ||||||
|                 SideEffect.NEIGHBORS, |                 .with(SideEffect.LIGHTING, SideEffect.State.OFF) | ||||||
|                 SideEffect.State.OFF |                 .with(SideEffect.NEIGHBORS, SideEffect.State.OFF); | ||||||
|         ); |         EDGE_SIDE_EFFECT_SET = NO_SIDE_EFFECT_SET | ||||||
|         EDGE_SIDE_EFFECT_SET = SideEffectSet.none().with(SideEffect.UPDATE, SideEffect.State.ON).with( |                 .with(SideEffect.UPDATE, SideEffect.State.ON) | ||||||
|                 SideEffect.NEIGHBORS, |                 .with(SideEffect.NEIGHBORS, SideEffect.State.ON); | ||||||
|                 SideEffect.State.ON |         LIGHTING_SIDE_EFFECT_SET = NO_SIDE_EFFECT_SET | ||||||
|         ); |                 .with(SideEffect.NEIGHBORS, SideEffect.State.OFF); | ||||||
|         LIGHTING_SIDE_EFFECT_SET = SideEffectSet.none().with(SideEffect.NEIGHBORS, SideEffect.State.OFF); |         EDGE_LIGHTING_SIDE_EFFECT_SET = NO_SIDE_EFFECT_SET | ||||||
|         EDGE_LIGHTING_SIDE_EFFECT_SET = SideEffectSet.none().with(SideEffect.UPDATE, SideEffect.State.ON).with( |                 .with(SideEffect.UPDATE, SideEffect.State.ON) | ||||||
|                 SideEffect.NEIGHBORS, |                 .with(SideEffect.NEIGHBORS, SideEffect.State.ON); | ||||||
|                 SideEffect.State.ON |     } | ||||||
|         ); |  | ||||||
|  |     // make sure block changes are sent | ||||||
|  |     private static SideEffectSet enableNetworkIfNeeded() { | ||||||
|  |         SideEffect network; | ||||||
|  |         try { | ||||||
|  |             network = SideEffect.valueOf("NETWORK"); | ||||||
|  |         } catch (IllegalArgumentException ignored) { | ||||||
|  |             return SideEffectSet.none(); | ||||||
|  |         } | ||||||
|  |         return SideEffectSet.none().with(network, SideEffect.State.ON); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private org.bukkit.World bukkitWorld; |     private org.bukkit.World bukkitWorld; | ||||||
| @@ -229,6 +238,7 @@ public class BukkitQueueCoordinator extends BasicQueueCoordinator { | |||||||
|                         .unloadAfter(isUnloadAfter()) |                         .unloadAfter(isUnloadAfter()) | ||||||
|                         .withProgressSubscribers(getProgressSubscribers()) |                         .withProgressSubscribers(getProgressSubscribers()) | ||||||
|                         .forceSync(isForceSync()) |                         .forceSync(isForceSync()) | ||||||
|  |                         .shouldGen(isShouldGen()) | ||||||
|                         .build(); |                         .build(); | ||||||
|         return super.enqueue(); |         return super.enqueue(); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -49,6 +49,7 @@ import org.bukkit.entity.Arrow; | |||||||
| import org.bukkit.entity.Creature; | import org.bukkit.entity.Creature; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
| import org.bukkit.entity.EntityType; | import org.bukkit.entity.EntityType; | ||||||
|  | import org.bukkit.entity.Firework; | ||||||
| import org.bukkit.entity.Player; | import org.bukkit.entity.Player; | ||||||
| import org.bukkit.entity.Projectile; | import org.bukkit.entity.Projectile; | ||||||
| import org.bukkit.event.entity.EntityDamageEvent; | import org.bukkit.event.entity.EntityDamageEvent; | ||||||
| @@ -341,8 +342,7 @@ public class BukkitEntityUtil { | |||||||
|         } |         } | ||||||
|         //disable the firework damage. too much of a headache to support at the moment. |         //disable the firework damage. too much of a headache to support at the moment. | ||||||
|         if (vplot != null) { |         if (vplot != null) { | ||||||
|             if (EntityDamageEvent.DamageCause.ENTITY_EXPLOSION == cause |             if (EntityDamageEvent.DamageCause.ENTITY_EXPLOSION == cause && damager instanceof Firework) { | ||||||
|                     && damager.getType() == EntityType.FIREWORK) { |  | ||||||
|                 return false; |                 return false; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -44,6 +44,7 @@ import java.util.stream.IntStream; | |||||||
| @Singleton | @Singleton | ||||||
| public class BukkitInventoryUtil extends InventoryUtil { | public class BukkitInventoryUtil extends InventoryUtil { | ||||||
|  |  | ||||||
|  |     @SuppressWarnings("deprecation") // Paper deprecation | ||||||
|     private static @Nullable ItemStack getItem(PlotItemStack item) { |     private static @Nullable ItemStack getItem(PlotItemStack item) { | ||||||
|         if (item == null) { |         if (item == null) { | ||||||
|             return null; |             return null; | ||||||
|   | |||||||
| @@ -67,6 +67,7 @@ public class BukkitSetupUtils extends SetupUtils { | |||||||
|         this.worldFile = worldFile; |         this.worldFile = worldFile; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @SuppressWarnings("deprecation") // Paper deprecation | ||||||
|     @Override |     @Override | ||||||
|     public void updateGenerators(final boolean force) { |     public void updateGenerators(final boolean force) { | ||||||
|         if (loaded && !SetupUtils.generators.isEmpty() && !force) { |         if (loaded && !SetupUtils.generators.isEmpty() && !force) { | ||||||
|   | |||||||
| @@ -75,6 +75,7 @@ import org.bukkit.entity.FallingBlock; | |||||||
| import org.bukkit.entity.Firework; | import org.bukkit.entity.Firework; | ||||||
| import org.bukkit.entity.Ghast; | import org.bukkit.entity.Ghast; | ||||||
| import org.bukkit.entity.Hanging; | import org.bukkit.entity.Hanging; | ||||||
|  | import org.bukkit.entity.Interaction; | ||||||
| import org.bukkit.entity.IronGolem; | import org.bukkit.entity.IronGolem; | ||||||
| import org.bukkit.entity.Item; | import org.bukkit.entity.Item; | ||||||
| import org.bukkit.entity.LightningStrike; | import org.bukkit.entity.LightningStrike; | ||||||
| @@ -261,6 +262,11 @@ public class BukkitUtil extends WorldUtil { | |||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean isSmallBlock(Location location) { | ||||||
|  |         return adapt(location).getBlock().getBoundingBox().getHeight() < 0.25; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     @NonNegative |     @NonNegative | ||||||
|     public int getHighestBlockSynchronous(final @NonNull String world, final int x, final int z) { |     public int getHighestBlockSynchronous(final @NonNull String world, final int x, final int z) { | ||||||
| @@ -432,6 +438,7 @@ public class BukkitUtil extends WorldUtil { | |||||||
|     @Override |     @Override | ||||||
|     public @NonNull Set<com.sk89q.worldedit.world.entity.EntityType> getTypesInCategory(final @NonNull String category) { |     public @NonNull Set<com.sk89q.worldedit.world.entity.EntityType> getTypesInCategory(final @NonNull String category) { | ||||||
|         final Collection<Class<?>> allowedInterfaces = new HashSet<>(); |         final Collection<Class<?>> allowedInterfaces = new HashSet<>(); | ||||||
|  |         final int[] version = PlotSquared.platform().serverVersion(); | ||||||
|         switch (category) { |         switch (category) { | ||||||
|             case "animal" -> { |             case "animal" -> { | ||||||
|                 allowedInterfaces.add(IronGolem.class); |                 allowedInterfaces.add(IronGolem.class); | ||||||
| @@ -439,7 +446,7 @@ public class BukkitUtil extends WorldUtil { | |||||||
|                 allowedInterfaces.add(Animals.class); |                 allowedInterfaces.add(Animals.class); | ||||||
|                 allowedInterfaces.add(WaterMob.class); |                 allowedInterfaces.add(WaterMob.class); | ||||||
|                 allowedInterfaces.add(Ambient.class); |                 allowedInterfaces.add(Ambient.class); | ||||||
|                 if (PlotSquared.platform().serverVersion()[1] >= 19) { |                 if (version[1] >= 19) { | ||||||
|                     allowedInterfaces.add(Allay.class); |                     allowedInterfaces.add(Allay.class); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -470,6 +477,11 @@ public class BukkitUtil extends WorldUtil { | |||||||
|                 allowedInterfaces.add(Firework.class); |                 allowedInterfaces.add(Firework.class); | ||||||
|             } |             } | ||||||
|             case "player" -> allowedInterfaces.add(Player.class); |             case "player" -> allowedInterfaces.add(Player.class); | ||||||
|  |             case "interaction" -> { | ||||||
|  |                 if ((version[1] > 19) || (version[1] == 19 && version[2] >= 4)) { | ||||||
|  |                     allowedInterfaces.add(Interaction.class); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|             default -> LOGGER.error("Unknown entity category requested: {}", category); |             default -> LOGGER.error("Unknown entity category requested: {}", category); | ||||||
|         } |         } | ||||||
|         final Set<com.sk89q.worldedit.world.entity.EntityType> types = new HashSet<>(); |         final Set<com.sk89q.worldedit.world.entity.EntityType> types = new HashSet<>(); | ||||||
|   | |||||||
| @@ -23,11 +23,15 @@ import com.plotsquared.core.location.World; | |||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
|  | import java.lang.ref.SoftReference; | ||||||
|  | import java.lang.ref.WeakReference; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
|  |  | ||||||
| public class BukkitWorld implements World<org.bukkit.World> { | public class BukkitWorld implements World<org.bukkit.World> { | ||||||
|  |  | ||||||
|     private static final Map<String, BukkitWorld> worldMap = Maps.newHashMap(); |     // Upon world unload we should probably have the P2 BukkitWorld be GCed relatively eagerly, thus freeing the bukkit world. | ||||||
|  |     //  We also want to avoid circumstances where a bukkit world has been GCed, but a P2 BukkitWorld has not | ||||||
|  |     private static final Map<String, WeakReference<BukkitWorld>> worldMap = Maps.newHashMap(); | ||||||
|     private static final boolean HAS_MIN_Y; |     private static final boolean HAS_MIN_Y; | ||||||
|  |  | ||||||
|     static { |     static { | ||||||
| @@ -41,10 +45,11 @@ public class BukkitWorld implements World<org.bukkit.World> { | |||||||
|         HAS_MIN_Y = temp; |         HAS_MIN_Y = temp; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private final org.bukkit.World world; |     // We want to allow GC to remove bukkit worlds, but not too eagerly | ||||||
|  |     private final SoftReference<org.bukkit.World> world; | ||||||
|  |  | ||||||
|     private BukkitWorld(final org.bukkit.World world) { |     private BukkitWorld(final org.bukkit.World world) { | ||||||
|         this.world = world; |         this.world = new SoftReference<>(world); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -68,12 +73,13 @@ public class BukkitWorld implements World<org.bukkit.World> { | |||||||
|      * @return World instance |      * @return World instance | ||||||
|      */ |      */ | ||||||
|     public static @NonNull BukkitWorld of(final org.bukkit.World world) { |     public static @NonNull BukkitWorld of(final org.bukkit.World world) { | ||||||
|         BukkitWorld bukkitWorld = worldMap.get(world.getName()); |         WeakReference<BukkitWorld> bukkitWorldRef = worldMap.get(world.getName()); | ||||||
|         if (bukkitWorld != null && bukkitWorld.getPlatformWorld().equals(world)) { |         BukkitWorld bukkitWorld; | ||||||
|  |         if (bukkitWorldRef != null && (bukkitWorld = bukkitWorldRef.get()) != null && world.equals(bukkitWorld.world.get())) { | ||||||
|             return bukkitWorld; |             return bukkitWorld; | ||||||
|         } |         } | ||||||
|         bukkitWorld = new BukkitWorld(world); |         bukkitWorld = new BukkitWorld(world); | ||||||
|         worldMap.put(world.getName(), bukkitWorld); |         worldMap.put(world.getName(), new WeakReference<>(bukkitWorld)); | ||||||
|         return bukkitWorld; |         return bukkitWorld; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -97,22 +103,26 @@ public class BukkitWorld implements World<org.bukkit.World> { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public org.bukkit.World getPlatformWorld() { |     public org.bukkit.World getPlatformWorld() { | ||||||
|         return this.world; |         org.bukkit.World world = this.world.get(); | ||||||
|  |         if (world == null) { | ||||||
|  |             throw new IllegalStateException("Bukkit platform world was unloaded from memory"); | ||||||
|  |         } | ||||||
|  |         return world; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public @NonNull String getName() { |     public @NonNull String getName() { | ||||||
|         return this.world.getName(); |         return this.getPlatformWorld().getName(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public int getMinHeight() { |     public int getMinHeight() { | ||||||
|         return getMinWorldHeight(world); |         return getMinWorldHeight(getPlatformWorld()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public int getMaxHeight() { |     public int getMaxHeight() { | ||||||
|         return getMaxWorldHeight(world) - 1; |         return getMaxWorldHeight(getPlatformWorld()) - 1; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|   | |||||||
| @@ -44,6 +44,24 @@ public class TranslationUpdateManager { | |||||||
|         String minheightReplacement = "minheight"; |         String minheightReplacement = "minheight"; | ||||||
|         String maxHeight = "maxHeight"; |         String maxHeight = "maxHeight"; | ||||||
|         String maxheightReplacement = "maxheight"; |         String maxheightReplacement = "maxheight"; | ||||||
|  |         String usedGrants = "usedGrants"; | ||||||
|  |         String usedGrantsReplacement = "used_grants"; | ||||||
|  |         String remainingGrants = "remainingGrants"; | ||||||
|  |         String remainingGrantsReplacement = "remaining_grants"; | ||||||
|  |         String minimumRadius = "minimumRadius"; | ||||||
|  |         String minimumRadiusReplacement = "minimum_radius"; | ||||||
|  |         String maximumMoves = "maximumMoves"; | ||||||
|  |         String maximumMovesReplacement = "maximum_moves"; | ||||||
|  |         String userMove = "userMove"; | ||||||
|  |         String userMoveReplacement = "user_move"; | ||||||
|  |  | ||||||
|  |         // tag opening / closing characters are important, as the locale keys exist as well, which should not be replaced | ||||||
|  |         String listInfoUnknown = "<info.unknown>"; | ||||||
|  |         String listInfoUnknownReplacement = "<unknown>"; | ||||||
|  |         String listInfoServer = "<info.server>"; | ||||||
|  |         String listInfoServerReplacement = "<server>"; | ||||||
|  |         String listInfoEveryone = "<info.everyone>"; | ||||||
|  |         String listInfoEveryoneReplacement = "<everyone>"; | ||||||
|  |  | ||||||
|         try (Stream<Path> paths = Files.walk(Paths.get(PlotSquared.platform().getDirectory().toPath().resolve("lang").toUri()))) { |         try (Stream<Path> paths = Files.walk(Paths.get(PlotSquared.platform().getDirectory().toPath().resolve("lang").toUri()))) { | ||||||
|             paths |             paths | ||||||
| @@ -53,6 +71,14 @@ public class TranslationUpdateManager { | |||||||
|                         replaceInFile(p, suggestCommand, suggestCommandReplacement); |                         replaceInFile(p, suggestCommand, suggestCommandReplacement); | ||||||
|                         replaceInFile(p, minHeight, minheightReplacement); |                         replaceInFile(p, minHeight, minheightReplacement); | ||||||
|                         replaceInFile(p, maxHeight, maxheightReplacement); |                         replaceInFile(p, maxHeight, maxheightReplacement); | ||||||
|  |                         replaceInFile(p, usedGrants, usedGrantsReplacement); | ||||||
|  |                         replaceInFile(p, remainingGrants, remainingGrantsReplacement); | ||||||
|  |                         replaceInFile(p, minimumRadius, minimumRadiusReplacement); | ||||||
|  |                         replaceInFile(p, maximumMoves, maximumMovesReplacement); | ||||||
|  |                         replaceInFile(p, userMove, userMoveReplacement); | ||||||
|  |                         replaceInFile(p, listInfoUnknown, listInfoUnknownReplacement); | ||||||
|  |                         replaceInFile(p, listInfoServer, listInfoServerReplacement); | ||||||
|  |                         replaceInFile(p, listInfoEveryone, listInfoEveryoneReplacement); | ||||||
|                     }); |                     }); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -35,7 +35,7 @@ import org.bukkit.scheduler.BukkitTask; | |||||||
| import javax.net.ssl.HttpsURLConnection; | import javax.net.ssl.HttpsURLConnection; | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
| import java.io.InputStreamReader; | import java.io.InputStreamReader; | ||||||
| import java.net.URL; | import java.net.URI; | ||||||
|  |  | ||||||
| public class UpdateUtility implements Listener { | public class UpdateUtility implements Listener { | ||||||
|  |  | ||||||
| @@ -59,8 +59,9 @@ public class UpdateUtility implements Listener { | |||||||
|     public void updateChecker() { |     public void updateChecker() { | ||||||
|         task = Bukkit.getScheduler().runTaskTimerAsynchronously(this.javaPlugin, () -> { |         task = Bukkit.getScheduler().runTaskTimerAsynchronously(this.javaPlugin, () -> { | ||||||
|             try { |             try { | ||||||
|                 HttpsURLConnection connection = (HttpsURLConnection) new URL( |                 HttpsURLConnection connection = (HttpsURLConnection) URI.create( | ||||||
|                         "https://api.spigotmc.org/simple/0.1/index.php?action=getResource&id=77506") |                         "https://api.spigotmc.org/simple/0.2/index.php?action=getResource&id=77506") | ||||||
|  |                         .toURL() | ||||||
|                         .openConnection(); |                         .openConnection(); | ||||||
|                 connection.setRequestMethod("GET"); |                 connection.setRequestMethod("GET"); | ||||||
|                 JsonObject result = new JsonParser() |                 JsonObject result = new JsonParser() | ||||||
|   | |||||||
| @@ -37,7 +37,9 @@ import com.sk89q.worldedit.regions.CuboidRegion; | |||||||
| import com.sk89q.worldedit.world.biome.BiomeType; | import com.sk89q.worldedit.world.biome.BiomeType; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
| import org.checkerframework.checker.nullness.qual.Nullable; | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  | import org.jetbrains.annotations.NotNull; | ||||||
|  |  | ||||||
|  | import java.util.Objects; | ||||||
| import java.util.Set; | import java.util.Set; | ||||||
|  |  | ||||||
| public class FaweRegionManager extends BukkitRegionManager { | public class FaweRegionManager extends BukkitRegionManager { | ||||||
| @@ -59,7 +61,10 @@ public class FaweRegionManager extends BukkitRegionManager { | |||||||
|             @Nullable PlotPlayer<?> actor, |             @Nullable PlotPlayer<?> actor, | ||||||
|             @Nullable QueueCoordinator queue |             @Nullable QueueCoordinator queue | ||||||
|     ) { |     ) { | ||||||
|         return delegate.setCuboids(area, regions, blocks, minY, maxY, queue.getCompleteTask()); |         return delegate.setCuboids( | ||||||
|  |                 area, regions, blocks, minY, maxY, | ||||||
|  |                 Objects.requireNonNullElseGet(queue, area::getQueue).getCompleteTask() | ||||||
|  |         ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -111,7 +116,7 @@ public class FaweRegionManager extends BukkitRegionManager { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean regenerateRegion(final Location pos1, final Location pos2, boolean ignore, final Runnable whenDone) { |     public boolean regenerateRegion(final @NotNull Location pos1, final @NotNull Location pos2, boolean ignore, final Runnable whenDone) { | ||||||
|         return delegate.regenerateRegion(pos1, pos2, ignore, whenDone); |         return delegate.regenerateRegion(pos1, pos2, ignore, whenDone); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,76 +0,0 @@ | |||||||
| # Contributor Covenant Code of Conduct |  | ||||||
|  |  | ||||||
| ## Our Pledge |  | ||||||
|  |  | ||||||
| In the interest of fostering an open and welcoming environment, we as |  | ||||||
| contributors and maintainers pledge to making participation in our project and |  | ||||||
| our community a harassment-free experience for everyone, regardless of age, body |  | ||||||
| size, disability, ethnicity, sex characteristics, gender identity and expression, |  | ||||||
| level of experience, education, socio-economic status, nationality, personal |  | ||||||
| appearance, race, religion, or sexual identity and orientation. |  | ||||||
|  |  | ||||||
| ## Our Standards |  | ||||||
|  |  | ||||||
| Examples of behavior that contributes to creating a positive environment |  | ||||||
| include: |  | ||||||
|  |  | ||||||
| * Using welcoming and inclusive language |  | ||||||
| * Being respectful of differing viewpoints and experiences |  | ||||||
| * Gracefully accepting constructive criticism |  | ||||||
| * Focusing on what is best for the community |  | ||||||
| * Showing empathy towards other community members |  | ||||||
|  |  | ||||||
| Examples of unacceptable behavior by participants include: |  | ||||||
|  |  | ||||||
| * The use of sexualized language or imagery and unwelcome sexual attention or |  | ||||||
|   advances |  | ||||||
| * Trolling, insulting/derogatory comments, and personal or political attacks |  | ||||||
| * Public or private harassment |  | ||||||
| * Publishing others' private information, such as a physical or electronic |  | ||||||
|   address, without explicit permission |  | ||||||
| * Other conduct which could reasonably be considered inappropriate in a |  | ||||||
|   professional setting |  | ||||||
|  |  | ||||||
| ## Our Responsibilities |  | ||||||
|  |  | ||||||
| Project maintainers are responsible for clarifying the standards of acceptable |  | ||||||
| behavior and are expected to take appropriate and fair corrective action in |  | ||||||
| response to any instances of unacceptable behavior. |  | ||||||
|  |  | ||||||
| Project maintainers 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, or to ban temporarily or |  | ||||||
| permanently any contributor for other behaviors that they deem inappropriate, |  | ||||||
| threatening, offensive, or harmful. |  | ||||||
|  |  | ||||||
| ## Scope |  | ||||||
|  |  | ||||||
| This Code of Conduct applies both within project spaces and in public spaces |  | ||||||
| when an individual is representing the project or its community. Examples of |  | ||||||
| representing a project or community include using an official project e-mail |  | ||||||
| address, posting via an official social media account, or acting as an appointed |  | ||||||
| representative at an online or offline event. Representation of a project may be |  | ||||||
| further defined and clarified by project maintainers. |  | ||||||
|  |  | ||||||
| ## Enforcement |  | ||||||
|  |  | ||||||
| Instances of abusive, harassing, or otherwise unacceptable behavior may be |  | ||||||
| reported by contacting the project team at contact<at>intellectualsites.com. All |  | ||||||
| complaints will be reviewed and investigated and will result in a response that |  | ||||||
| is deemed necessary and appropriate to the circumstances. The project team is |  | ||||||
| obligated to maintain confidentiality with regard to the reporter of an incident. |  | ||||||
| Further details of specific enforcement policies may be posted separately. |  | ||||||
|  |  | ||||||
| Project maintainers who do not follow or enforce the Code of Conduct in good |  | ||||||
| faith may face temporary or permanent repercussions as determined by other |  | ||||||
| members of the project's leadership. |  | ||||||
|  |  | ||||||
| ## Attribution |  | ||||||
|  |  | ||||||
| This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4, |  | ||||||
| available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html |  | ||||||
|  |  | ||||||
| [homepage]: https://www.contributor-covenant.org |  | ||||||
|  |  | ||||||
| For answers to common questions about this code of conduct, see |  | ||||||
| https://www.contributor-covenant.org/faq |  | ||||||
| @@ -2,18 +2,18 @@ import java.time.format.DateTimeFormatter | |||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|     // Expected everywhere. |     // Expected everywhere. | ||||||
|     compileOnlyApi("org.checkerframework:checker-qual") |     compileOnlyApi(libs.checkerqual) | ||||||
|  |  | ||||||
|     // Minecraft expectations |     // Minecraft expectations | ||||||
|     compileOnlyApi("com.google.code.gson:gson") |     compileOnlyApi(libs.gson) | ||||||
|     compileOnly("com.google.guava:guava") |     compileOnly(libs.guava) | ||||||
|  |  | ||||||
|     // Platform expectations |     // Platform expectations | ||||||
|     compileOnlyApi("org.yaml:snakeyaml") |     compileOnlyApi(libs.snakeyaml) | ||||||
|  |  | ||||||
|     // Adventure |     // Adventure | ||||||
|     api("net.kyori:adventure-api") |     api(libs.adventureApi) | ||||||
|     api("net.kyori:adventure-text-minimessage") |     api(libs.adventureMiniMessage) | ||||||
|  |  | ||||||
|     // Guice |     // Guice | ||||||
|     api(libs.guice) { |     api(libs.guice) { | ||||||
| @@ -31,19 +31,19 @@ dependencies { | |||||||
|         exclude(group = "dummypermscompat") |         exclude(group = "dummypermscompat") | ||||||
|     } |     } | ||||||
|     testImplementation(libs.worldeditCore) |     testImplementation(libs.worldeditCore) | ||||||
|     compileOnly("com.fastasyncworldedit:FastAsyncWorldEdit-Core") { isTransitive = false } |     compileOnly(libs.faweBukkit) { isTransitive = false } | ||||||
|     testImplementation("com.fastasyncworldedit:FastAsyncWorldEdit-Core") { isTransitive = false } |     testImplementation(libs.faweCore) { isTransitive = false } | ||||||
|  |  | ||||||
|     // Logging |     // Logging | ||||||
|     compileOnlyApi("org.apache.logging.log4j:log4j-api") |     compileOnlyApi(libs.log4j) | ||||||
|  |  | ||||||
|     // Other libraries |     // Other libraries | ||||||
|     api(libs.prtree) |     api(libs.prtree) | ||||||
|     api(libs.aopalliance) |     api(libs.aopalliance) | ||||||
|     api(libs.cloudServices) |     api(libs.cloudServices) | ||||||
|     api(libs.arkitektonika) |     api(libs.arkitektonika) | ||||||
|     api("com.intellectualsites.paster:Paster") |     api(libs.paster) | ||||||
|     api("com.intellectualsites.informative-annotations:informative-annotations") |     api(libs.informativeAnnotations) | ||||||
| } | } | ||||||
|  |  | ||||||
| tasks.processResources { | tasks.processResources { | ||||||
| @@ -57,8 +57,8 @@ tasks.processResources { | |||||||
|  |  | ||||||
|     doLast { |     doLast { | ||||||
|         copy { |         copy { | ||||||
|             from(File("$rootDir/LICENSE")) |             from(layout.buildDirectory.file("$rootDir/LICENSE")) | ||||||
|             into("$buildDir/resources/main/") |             into(layout.buildDirectory.dir("resources/main")) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -68,16 +68,16 @@ tasks { | |||||||
|         val isRelease = if (rootProject.version.toString().endsWith("-SNAPSHOT")) "TODO" else rootProject.version.toString() |         val isRelease = if (rootProject.version.toString().endsWith("-SNAPSHOT")) "TODO" else rootProject.version.toString() | ||||||
|         val opt = options as StandardJavadocDocletOptions |         val opt = options as StandardJavadocDocletOptions | ||||||
|         opt.links("https://docs.enginehub.org/javadoc/com.sk89q.worldedit/worldedit-core/" + libs.worldeditCore.get().versionConstraint.toString()) |         opt.links("https://docs.enginehub.org/javadoc/com.sk89q.worldedit/worldedit-core/" + libs.worldeditCore.get().versionConstraint.toString()) | ||||||
|         opt.links("https://jd.advntr.dev/api/4.14.0/") |         opt.links("https://jd.advntr.dev/api/" + libs.adventureApi.get().versionConstraint.toString()) | ||||||
|         opt.links("https://jd.advntr.dev/text-minimessage/4.14.0/") |         opt.links("https://jd.advntr.dev/text-minimessage/" + libs.adventureApi.get().versionConstraint.toString()) | ||||||
|         opt.links("https://google.github.io/guice/api-docs/" + libs.guice.get().versionConstraint.toString() + "/javadoc/") |         opt.links("https://google.github.io/guice/api-docs/" + libs.guice.get().versionConstraint.toString() + "/javadoc/") | ||||||
|         opt.links("https://checkerframework.org/api/") |         opt.links("https://checkerframework.org/api/") | ||||||
|         opt.links("https://javadoc.io/doc/com.intellectualsites.informative-annotations/informative-annotations/latest/") |  | ||||||
|         opt.isLinkSource = true |         opt.isLinkSource = true | ||||||
|         opt.bottom(File("$rootDir/javadocfooter.html").readText()) |         opt.bottom(File("$rootDir/javadocfooter.html").readText()) | ||||||
|         opt.isUse = true |         opt.isUse = true | ||||||
|         opt.encoding("UTF-8") |         opt.encoding("UTF-8") | ||||||
|         opt.keyWords() |         opt.keyWords() | ||||||
|         opt.addStringOption("-since", isRelease) |         opt.addStringOption("-since", isRelease) | ||||||
|  |         opt.noTimestamp() | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -31,6 +31,7 @@ import com.plotsquared.core.generator.IndependentPlotGenerator; | |||||||
| import com.plotsquared.core.inject.annotations.DefaultGenerator; | import com.plotsquared.core.inject.annotations.DefaultGenerator; | ||||||
| import com.plotsquared.core.location.World; | import com.plotsquared.core.location.World; | ||||||
| import com.plotsquared.core.permissions.PermissionHandler; | import com.plotsquared.core.permissions.PermissionHandler; | ||||||
|  | import com.plotsquared.core.permissions.RangedPermissionResolver; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.expiration.ExpireManager; | import com.plotsquared.core.plot.expiration.ExpireManager; | ||||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | import com.plotsquared.core.plot.world.PlotAreaManager; | ||||||
| @@ -120,6 +121,14 @@ public interface PlotPlatform<P> extends LocaleHolder { | |||||||
|      */ |      */ | ||||||
|     @NonNull String serverImplementation(); |     @NonNull String serverImplementation(); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Gets the server brand name | ||||||
|  |      * | ||||||
|  |      * @return server brand | ||||||
|  |      * @since 7.5.3 | ||||||
|  |      */ | ||||||
|  |     @NonNull String serverBrand(); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Gets the native server code package prefix. |      * Gets the native server code package prefix. | ||||||
|      * |      * | ||||||
| @@ -341,6 +350,10 @@ public interface PlotPlatform<P> extends LocaleHolder { | |||||||
|         return injector().getInstance(PermissionHandler.class); |         return injector().getInstance(PermissionHandler.class); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     default @NonNull RangedPermissionResolver rangedPermissionResolver() { | ||||||
|  |         return injector().getInstance(RangedPermissionResolver.class); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the {@link ServicePipeline} implementation |      * Get the {@link ServicePipeline} implementation | ||||||
|      * |      * | ||||||
|   | |||||||
| @@ -84,7 +84,7 @@ import java.io.InputStream; | |||||||
| import java.io.InputStreamReader; | import java.io.InputStreamReader; | ||||||
| import java.io.ObjectInputStream; | import java.io.ObjectInputStream; | ||||||
| import java.io.ObjectOutputStream; | import java.io.ObjectOutputStream; | ||||||
| import java.net.MalformedURLException; | import java.net.URI; | ||||||
| import java.net.URISyntaxException; | import java.net.URISyntaxException; | ||||||
| import java.net.URL; | import java.net.URL; | ||||||
| import java.nio.file.Files; | import java.nio.file.Files; | ||||||
| @@ -206,14 +206,16 @@ public class PlotSquared { | |||||||
|         GlobalFlagContainer.setup(); |         GlobalFlagContainer.setup(); | ||||||
|  |  | ||||||
|         try { |         try { | ||||||
|             new ReflectionUtils(this.platform.serverNativePackage()); |             String ver = this.platform.serverNativePackage(); | ||||||
|  |             new ReflectionUtils(ver.isEmpty() ? null : ver); | ||||||
|             try { |             try { | ||||||
|                 URL logurl = PlotSquared.class.getProtectionDomain().getCodeSource().getLocation(); |                 URL logurl = PlotSquared.class.getProtectionDomain().getCodeSource().getLocation(); | ||||||
|                 this.jarFile = new File( |                 this.jarFile = new File( | ||||||
|                         new URL(logurl.toURI().toString().split("\\!")[0].replaceAll("jar:file", "file")) |                         URI.create( | ||||||
|                                 .toURI().getPath()); |                                 logurl.toURI().toString().split("\\!")[0].replaceAll("jar:file", "file")) | ||||||
|             } catch (MalformedURLException | URISyntaxException | SecurityException e) { |                                 .getPath()); | ||||||
|                 e.printStackTrace(); |             } catch (URISyntaxException | SecurityException e) { | ||||||
|  |                 LOGGER.error(e); | ||||||
|                 this.jarFile = new File(this.platform.getDirectory().getParentFile(), "PlotSquared.jar"); |                 this.jarFile = new File(this.platform.getDirectory().getParentFile(), "PlotSquared.jar"); | ||||||
|                 if (!this.jarFile.exists()) { |                 if (!this.jarFile.exists()) { | ||||||
|                     this.jarFile = new File( |                     this.jarFile = new File( | ||||||
| @@ -237,7 +239,7 @@ public class PlotSquared { | |||||||
|             copyFile("skyblock.template", Settings.Paths.TEMPLATES); |             copyFile("skyblock.template", Settings.Paths.TEMPLATES); | ||||||
|             showDebug(); |             showDebug(); | ||||||
|         } catch (Throwable e) { |         } catch (Throwable e) { | ||||||
|             e.printStackTrace(); |             LOGGER.error(e); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -794,9 +796,8 @@ public class PlotSquared { | |||||||
|         if (world.equals("CheckingPlotSquaredGenerator")) { |         if (world.equals("CheckingPlotSquaredGenerator")) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         if (!this.getPlotAreaManager().addWorld(world)) { |         // Don't check the return result -> breaks runtime loading of single plot areas on creation | ||||||
|             return; |         this.getPlotAreaManager().addWorld(world); | ||||||
|         } |  | ||||||
|         Set<String> worlds; |         Set<String> worlds; | ||||||
|         if (this.worldConfiguration.contains("worlds")) { |         if (this.worldConfiguration.contains("worlds")) { | ||||||
|             worlds = this.worldConfiguration.getConfigurationSection("worlds").getKeys(false); |             worlds = this.worldConfiguration.getConfigurationSection("worlds").getKeys(false); | ||||||
| @@ -1281,7 +1282,7 @@ public class PlotSquared { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Setup the database connection. |      * Set up the database connection. | ||||||
|      */ |      */ | ||||||
|     public void setupDatabase() { |     public void setupDatabase() { | ||||||
|         try { |         try { | ||||||
|   | |||||||
| @@ -19,6 +19,7 @@ | |||||||
| package com.plotsquared.core.command; | package com.plotsquared.core.command; | ||||||
|  |  | ||||||
| import com.google.inject.Inject; | import com.google.inject.Inject; | ||||||
|  | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.database.DBFunc; | import com.plotsquared.core.database.DBFunc; | ||||||
| @@ -101,9 +102,14 @@ public class Add extends Command { | |||||||
|                                 Permission.PERMISSION_ADMIN_COMMAND_TRUST))) { |                                 Permission.PERMISSION_ADMIN_COMMAND_TRUST))) { | ||||||
|                             player.sendMessage( |                             player.sendMessage( | ||||||
|                                     TranslatableCaption.of("errors.invalid_player"), |                                     TranslatableCaption.of("errors.invalid_player"), | ||||||
|                                     TagResolver.resolver("value", Tag.inserting( |                                     PlotSquared | ||||||
|                                             PlayerManager.resolveName(uuid).toComponent(player) |                                             .platform() | ||||||
|                                     )) |                                             .playerManager() | ||||||
|  |                                             .getUsernameCaption(uuid) | ||||||
|  |                                             .thenApply(caption -> TagResolver.resolver( | ||||||
|  |                                                     "value", | ||||||
|  |                                                     Tag.inserting(caption.toComponent(player)) | ||||||
|  |                                             )) | ||||||
|                             ); |                             ); | ||||||
|                             iterator.remove(); |                             iterator.remove(); | ||||||
|                             continue; |                             continue; | ||||||
| @@ -111,9 +117,11 @@ public class Add extends Command { | |||||||
|                         if (plot.isOwner(uuid)) { |                         if (plot.isOwner(uuid)) { | ||||||
|                             player.sendMessage( |                             player.sendMessage( | ||||||
|                                     TranslatableCaption.of("member.already_added"), |                                     TranslatableCaption.of("member.already_added"), | ||||||
|                                     TagResolver.resolver("player", Tag.inserting( |                                     PlotSquared.platform().playerManager().getUsernameCaption(uuid) | ||||||
|                                             PlayerManager.resolveName(uuid).toComponent(player) |                                             .thenApply(caption -> TagResolver.resolver( | ||||||
|                                     )) |                                                     "player", | ||||||
|  |                                                     Tag.inserting(caption.toComponent(player)) | ||||||
|  |                                             )) | ||||||
|                             ); |                             ); | ||||||
|                             iterator.remove(); |                             iterator.remove(); | ||||||
|                             continue; |                             continue; | ||||||
| @@ -121,9 +129,11 @@ public class Add extends Command { | |||||||
|                         if (plot.getMembers().contains(uuid)) { |                         if (plot.getMembers().contains(uuid)) { | ||||||
|                             player.sendMessage( |                             player.sendMessage( | ||||||
|                                     TranslatableCaption.of("member.already_added"), |                                     TranslatableCaption.of("member.already_added"), | ||||||
|                                     TagResolver.resolver("player", Tag.inserting( |                                     PlotSquared.platform().playerManager().getUsernameCaption(uuid) | ||||||
|                                             PlayerManager.resolveName(uuid).toComponent(player) |                                             .thenApply(caption -> TagResolver.resolver( | ||||||
|                                     )) |                                                     "player", | ||||||
|  |                                                     Tag.inserting(caption.toComponent(player)) | ||||||
|  |                                             )) | ||||||
|                             ); |                             ); | ||||||
|                             iterator.remove(); |                             iterator.remove(); | ||||||
|                             continue; |                             continue; | ||||||
|   | |||||||
| @@ -184,6 +184,7 @@ public class Area extends SubCommand { | |||||||
|                         CuboidRegion.makeCuboid(playerSelectedRegion) |                         CuboidRegion.makeCuboid(playerSelectedRegion) | ||||||
|                 ).length != 0) { |                 ).length != 0) { | ||||||
|                     player.sendMessage(TranslatableCaption.of("single.single_area_overlapping")); |                     player.sendMessage(TranslatableCaption.of("single.single_area_overlapping")); | ||||||
|  |                     return false; | ||||||
|                 } |                 } | ||||||
|                 // Alter the region |                 // Alter the region | ||||||
|                 final BlockVector3 playerSelectionMin = playerSelectedRegion.getMinimumPoint(); |                 final BlockVector3 playerSelectionMin = playerSelectedRegion.getMinimumPoint(); | ||||||
|   | |||||||
| @@ -131,8 +131,8 @@ public class Auto extends SubCommand { | |||||||
|                         player.sendMessage( |                         player.sendMessage( | ||||||
|                                 TranslatableCaption.of("economy.removed_granted_plot"), |                                 TranslatableCaption.of("economy.removed_granted_plot"), | ||||||
|                                 TagResolver.builder() |                                 TagResolver.builder() | ||||||
|                                         .tag("usedGrants", Tag.inserting(Component.text(grantedPlots - left))) |                                         .tag("used_grants", Tag.inserting(Component.text(grantedPlots - left))) | ||||||
|                                         .tag("remainingGrants", Tag.inserting(Component.text(left))) |                                         .tag("remaining_grants", Tag.inserting(Component.text(left))) | ||||||
|                                         .build() |                                         .build() | ||||||
|                         ); |                         ); | ||||||
|                     } |                     } | ||||||
|   | |||||||
| @@ -21,8 +21,9 @@ package com.plotsquared.core.command; | |||||||
| import com.google.inject.Inject; | import com.google.inject.Inject; | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.events.PlotFlagRemoveEvent; | import com.plotsquared.core.events.PlayerBuyPlotEvent; | ||||||
| import com.plotsquared.core.events.Result; | import com.plotsquared.core.events.Result; | ||||||
|  | import com.plotsquared.core.player.OfflinePlotPlayer; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| import com.plotsquared.core.plot.PlotArea; | import com.plotsquared.core.plot.PlotArea; | ||||||
| @@ -88,22 +89,30 @@ public class Buy extends Command { | |||||||
|                 TranslatableCaption.of("permission.cant_claim_more_plots"), |                 TranslatableCaption.of("permission.cant_claim_more_plots"), | ||||||
|                 TagResolver.resolver("amount", Tag.inserting(Component.text(player.getAllowedPlots()))) |                 TagResolver.resolver("amount", Tag.inserting(Component.text(player.getAllowedPlots()))) | ||||||
|         ); |         ); | ||||||
|         double price = plot.getFlag(PriceFlag.class); |         double priceFlag = plot.getFlag(PriceFlag.class); | ||||||
|         if (price <= 0) { |         if (priceFlag <= 0) { | ||||||
|             throw new CommandException(TranslatableCaption.of("economy.not_for_sale")); |             throw new CommandException(TranslatableCaption.of("economy.not_for_sale")); | ||||||
|         } |         } | ||||||
|         checkTrue( |         checkTrue( | ||||||
|                 this.econHandler.isSupported(), |                 this.econHandler.isSupported(), | ||||||
|                 TranslatableCaption.of("economy.vault_or_consumer_null") |                 TranslatableCaption.of("economy.vault_or_consumer_null") | ||||||
|         ); |         ); | ||||||
|         checkTrue( |  | ||||||
|                 this.econHandler.getMoney(player) >= price, |         PlayerBuyPlotEvent event = this.eventDispatcher.callPlayerBuyPlot(player, plot, priceFlag); | ||||||
|                 TranslatableCaption.of("economy.cannot_afford_plot"), |         if (event.getEventResult() == Result.DENY) { | ||||||
|                 TagResolver.builder() |             throw new CommandException(TranslatableCaption.of("economy.cannot_buy_blocked")); | ||||||
|                         .tag("money", Tag.inserting(Component.text(this.econHandler.format(price)))) |         } | ||||||
|                         .tag("balance", Tag.inserting(Component.text(this.econHandler.format(this.econHandler.getMoney(player))))) |  | ||||||
|                         .build() |         double price = event.getEventResult() == Result.FORCE ? 0 : event.price(); | ||||||
|         ); |         if (this.econHandler.getMoney(player) < price) { | ||||||
|  |             throw new CommandException( | ||||||
|  |                     TranslatableCaption.of("economy.cannot_afford_plot"), | ||||||
|  |                     TagResolver.builder() | ||||||
|  |                             .tag("money", Tag.inserting(Component.text(this.econHandler.format(price)))) | ||||||
|  |                             .tag("balance", Tag.inserting(Component.text(this.econHandler.format(this.econHandler.getMoney(player))))) | ||||||
|  |                             .build() | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|         this.econHandler.withdrawMoney(player, price); |         this.econHandler.withdrawMoney(player, price); | ||||||
|         // Failure |         // Failure | ||||||
|         // Success |         // Success | ||||||
| @@ -113,7 +122,8 @@ public class Buy extends Command { | |||||||
|                     TagResolver.resolver("money", Tag.inserting(Component.text(this.econHandler.format(price)))) |                     TagResolver.resolver("money", Tag.inserting(Component.text(this.econHandler.format(price)))) | ||||||
|             ); |             ); | ||||||
|  |  | ||||||
|             this.econHandler.depositMoney(PlotSquared.platform().playerManager().getOfflinePlayer(plot.getOwnerAbs()), price); |             OfflinePlotPlayer previousOwner = PlotSquared.platform().playerManager().getOfflinePlayer(plot.getOwnerAbs()); | ||||||
|  |             this.econHandler.depositMoney(previousOwner, price); | ||||||
|  |  | ||||||
|             PlotPlayer<?> owner = PlotSquared.platform().playerManager().getPlayerIfExists(plot.getOwnerAbs()); |             PlotPlayer<?> owner = PlotSquared.platform().playerManager().getPlayerIfExists(plot.getOwnerAbs()); | ||||||
|             if (owner != null) { |             if (owner != null) { | ||||||
| @@ -127,16 +137,17 @@ public class Buy extends Command { | |||||||
|                 ); |                 ); | ||||||
|             } |             } | ||||||
|             PlotFlag<?, ?> plotFlag = plot.getFlagContainer().getFlag(PriceFlag.class); |             PlotFlag<?, ?> plotFlag = plot.getFlagContainer().getFlag(PriceFlag.class); | ||||||
|             PlotFlagRemoveEvent event = this.eventDispatcher.callFlagRemove(plotFlag, plot); |             if (this.eventDispatcher.callFlagRemove(plotFlag, plot).getEventResult() != Result.DENY) { | ||||||
|             if (event.getEventResult() != Result.DENY) { |                 plot.removeFlag(plotFlag); | ||||||
|                 plot.removeFlag(event.getFlag()); |  | ||||||
|             } |             } | ||||||
|             plot.setOwner(player.getUUID()); |             plot.setOwner(player.getUUID()); | ||||||
|             plot.getPlotModificationManager().setSign(player.getName()); |             plot.getPlotModificationManager().setSign(player.getName()); | ||||||
|             player.sendMessage( |             player.sendMessage( | ||||||
|                     TranslatableCaption.of("working.claimed"), |                     TranslatableCaption.of("working.claimed"), | ||||||
|  |                     TagResolver.resolver("world", Tag.inserting(Component.text(plot.getArea().getWorldName()))), | ||||||
|                     TagResolver.resolver("plot", Tag.inserting(Component.text(plot.getId().toString()))) |                     TagResolver.resolver("plot", Tag.inserting(Component.text(plot.getId().toString()))) | ||||||
|             ); |             ); | ||||||
|  |             this.eventDispatcher.callPostPlayerBuyPlot(player, previousOwner, plot, price); | ||||||
|             whenDone.run(Buy.this, CommandResult.SUCCESS); |             whenDone.run(Buy.this, CommandResult.SUCCESS); | ||||||
|         }, () -> { |         }, () -> { | ||||||
|             this.econHandler.depositMoney(player, price); |             this.econHandler.depositMoney(player, price); | ||||||
|   | |||||||
| @@ -186,14 +186,14 @@ public class Claim extends SubCommand { | |||||||
|                 player.sendMessage( |                 player.sendMessage( | ||||||
|                         TranslatableCaption.of("economy.removed_granted_plot"), |                         TranslatableCaption.of("economy.removed_granted_plot"), | ||||||
|                         TagResolver.builder() |                         TagResolver.builder() | ||||||
|                                 .tag("usedGrants", Tag.inserting(Component.text(grants - 1))) |                                 .tag("used_grants", Tag.inserting(Component.text(grants - 1))) | ||||||
|                                 .tag("remainingGrants", Tag.inserting(Component.text(grants))) |                                 .tag("remaining_grants", Tag.inserting(Component.text(grants))) | ||||||
|                                 .build() |                                 .build() | ||||||
|                 ); |                 ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         if (!player.hasPermission(Permission.PERMISSION_ADMIN_BYPASS_BORDER)) { |         if (!player.hasPermission(Permission.PERMISSION_ADMIN_BYPASS_BORDER)) { | ||||||
|             int border = area.getBorder(); |             int border = area.getBorder(false); | ||||||
|             if (border != Integer.MAX_VALUE && plot.getDistanceFromOrigin() > border && !force) { |             if (border != Integer.MAX_VALUE && plot.getDistanceFromOrigin() > border && !force) { | ||||||
|                 player.sendMessage(TranslatableCaption.of("border.denied")); |                 player.sendMessage(TranslatableCaption.of("border.denied")); | ||||||
|                 return false; |                 return false; | ||||||
|   | |||||||
| @@ -131,10 +131,12 @@ public class Clear extends Command { | |||||||
|                     player.sendMessage( |                     player.sendMessage( | ||||||
|                             TranslatableCaption.of("working.clearing_done"), |                             TranslatableCaption.of("working.clearing_done"), | ||||||
|                             TagResolver.builder() |                             TagResolver.builder() | ||||||
|  |                                     .tag("world", Tag.inserting(Component.text(plot.getArea().getWorldName()))) | ||||||
|                                     .tag("amount", Tag.inserting(Component.text(System.currentTimeMillis() - start))) |                                     .tag("amount", Tag.inserting(Component.text(System.currentTimeMillis() - start))) | ||||||
|                                     .tag("plot", Tag.inserting(Component.text(plot.getId().toString()))) |                                     .tag("plot", Tag.inserting(Component.text(plot.getId().toString()))) | ||||||
|                                     .build() |                                     .build() | ||||||
|                     ); |                     ); | ||||||
|  |                     this.eventDispatcher.callPostPlotClear(player, plot); | ||||||
|                 })); |                 })); | ||||||
|                 if (!result) { |                 if (!result) { | ||||||
|                     player.sendMessage(TranslatableCaption.of("errors.wait_for_timer")); |                     player.sendMessage(TranslatableCaption.of("errors.wait_for_timer")); | ||||||
|   | |||||||
| @@ -24,7 +24,9 @@ import com.plotsquared.core.configuration.caption.TranslatableCaption; | |||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| import net.kyori.adventure.text.minimessage.MiniMessage; | import net.kyori.adventure.text.minimessage.MiniMessage; | ||||||
|  | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  | import org.jetbrains.annotations.NotNull; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * CommandCategory. |  * CommandCategory. | ||||||
| @@ -82,7 +84,7 @@ public enum CommandCategory implements Caption { | |||||||
|     // TODO this method shouldn't be invoked |     // TODO this method shouldn't be invoked | ||||||
|     @Deprecated |     @Deprecated | ||||||
|     @Override |     @Override | ||||||
|     public String toString() { |     public @NotNull String toString() { | ||||||
|         return this.caption.getComponent(LocaleHolder.console()); |         return this.caption.getComponent(LocaleHolder.console()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -97,6 +99,14 @@ public enum CommandCategory implements Caption { | |||||||
|         return MiniMessage.miniMessage().deserialize(getComponent(localeHolder)); |         return MiniMessage.miniMessage().deserialize(getComponent(localeHolder)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NonNull Component toComponent( | ||||||
|  |             @NonNull final LocaleHolder localeHolder, | ||||||
|  |             final @NonNull TagResolver @NonNull ... tagResolvers | ||||||
|  |     ) { | ||||||
|  |         return MiniMessage.miniMessage().deserialize(getComponent(localeHolder)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Checks if a player has access to this command category |      * Checks if a player has access to this command category | ||||||
|      * |      * | ||||||
| @@ -108,4 +118,5 @@ public enum CommandCategory implements Caption { | |||||||
|         return !MainCommand.getInstance().getCommands(this, player).isEmpty(); |         return !MainCommand.getInstance().getCommands(this, player).isEmpty(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -256,11 +256,11 @@ public class Condense extends SubCommand { | |||||||
|                 player.sendMessage(TranslatableCaption.of("condense.default_eval")); |                 player.sendMessage(TranslatableCaption.of("condense.default_eval")); | ||||||
|                 player.sendMessage( |                 player.sendMessage( | ||||||
|                         TranslatableCaption.of("condense.minimum_radius"), |                         TranslatableCaption.of("condense.minimum_radius"), | ||||||
|                         TagResolver.resolver("minimumRadius", Tag.inserting(Component.text(minimumRadius))) |                         TagResolver.resolver("minimum_radius", Tag.inserting(Component.text(minimumRadius))) | ||||||
|                 ); |                 ); | ||||||
|                 player.sendMessage( |                 player.sendMessage( | ||||||
|                         TranslatableCaption.of("condense.maximum_moved"), |                         TranslatableCaption.of("condense.maximum_moved"), | ||||||
|                         TagResolver.resolver("maxMove", Tag.inserting(Component.text(maxMove))) |                         TagResolver.resolver("maximum_moves", Tag.inserting(Component.text(maxMove))) | ||||||
|                 ); |                 ); | ||||||
|                 player.sendMessage(TranslatableCaption.of("condense.input_eval")); |                 player.sendMessage(TranslatableCaption.of("condense.input_eval")); | ||||||
|                 player.sendMessage( |                 player.sendMessage( | ||||||
| @@ -269,7 +269,7 @@ public class Condense extends SubCommand { | |||||||
|                 ); |                 ); | ||||||
|                 player.sendMessage( |                 player.sendMessage( | ||||||
|                         TranslatableCaption.of("condense.estimated_moves"), |                         TranslatableCaption.of("condense.estimated_moves"), | ||||||
|                         TagResolver.resolver("userMove", Tag.inserting(Component.text(userMove))) |                         TagResolver.resolver("user_move", Tag.inserting(Component.text(userMove))) | ||||||
|                 ); |                 ); | ||||||
|                 player.sendMessage(TranslatableCaption.of("condense.eta")); |                 player.sendMessage(TranslatableCaption.of("condense.eta")); | ||||||
|                 player.sendMessage(TranslatableCaption.of("condense.radius_measured")); |                 player.sendMessage(TranslatableCaption.of("condense.radius_measured")); | ||||||
|   | |||||||
| @@ -29,7 +29,6 @@ import com.plotsquared.core.util.WorldUtil; | |||||||
| import com.plotsquared.core.util.entity.EntityCategories; | import com.plotsquared.core.util.entity.EntityCategories; | ||||||
| import com.plotsquared.core.util.entity.EntityCategory; | import com.plotsquared.core.util.entity.EntityCategory; | ||||||
| import com.plotsquared.core.util.query.PlotQuery; | import com.plotsquared.core.util.query.PlotQuery; | ||||||
| import com.plotsquared.core.util.task.TaskManager; |  | ||||||
| import com.plotsquared.core.uuid.UUIDMapping; | import com.plotsquared.core.uuid.UUIDMapping; | ||||||
| import com.sk89q.worldedit.world.entity.EntityType; | import com.sk89q.worldedit.world.entity.EntityType; | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| @@ -71,7 +70,7 @@ public class Debug extends SubCommand { | |||||||
|                     TranslatableCaption.of("commandconfig.command_syntax"), |                     TranslatableCaption.of("commandconfig.command_syntax"), | ||||||
|                     TagResolver.resolver( |                     TagResolver.resolver( | ||||||
|                             "value", |                             "value", | ||||||
|                             Tag.inserting(Component.text("/plot debug <loadedchunks | player | debug-players | entitytypes | msg>")) |                             Tag.inserting(Component.text("/plot debug <player | debug-players | entitytypes | msg>")) | ||||||
|                     ) |                     ) | ||||||
|             ); |             ); | ||||||
|         } |         } | ||||||
| @@ -85,16 +84,6 @@ public class Debug extends SubCommand { | |||||||
|                 return true; |                 return true; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         if (args.length > 0 && "loadedchunks".equalsIgnoreCase(args[0])) { |  | ||||||
|             final long start = System.currentTimeMillis(); |  | ||||||
|             player.sendMessage(TranslatableCaption.of("debug.fetching_loaded_chunks")); |  | ||||||
|             TaskManager.runTaskAsync(() -> player.sendMessage(StaticCaption |  | ||||||
|                     .of("Loaded chunks: " + this.worldUtil |  | ||||||
|                             .getChunkChunks(player.getLocation().getWorldName()) |  | ||||||
|                             .size() + " (" + (System.currentTimeMillis() |  | ||||||
|                             - start) + "ms) using thread: " + Thread.currentThread().getName()))); |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
|         if (args.length > 0 && "uuids".equalsIgnoreCase(args[0])) { |         if (args.length > 0 && "uuids".equalsIgnoreCase(args[0])) { | ||||||
|             final Collection<UUIDMapping> mappings = PlotSquared.get().getImpromptuUUIDPipeline().getAllImmediately(); |             final Collection<UUIDMapping> mappings = PlotSquared.get().getImpromptuUUIDPipeline().getAllImmediately(); | ||||||
|             player.sendMessage( |             player.sendMessage( | ||||||
| @@ -196,7 +185,7 @@ public class Debug extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public Collection<Command> tab(final PlotPlayer<?> player, String[] args, boolean space) { |     public Collection<Command> tab(final PlotPlayer<?> player, String[] args, boolean space) { | ||||||
|         return Stream.of("loadedchunks", "debug-players", "entitytypes") |         return Stream.of("debug-players", "entitytypes") | ||||||
|                 .filter(value -> value.startsWith(args[0].toLowerCase(Locale.ENGLISH))) |                 .filter(value -> value.startsWith(args[0].toLowerCase(Locale.ENGLISH))) | ||||||
|                 .map(value -> new Command(null, false, value, "plots.admin", RequiredType.NONE, null) { |                 .map(value -> new Command(null, false, value, "plots.admin", RequiredType.NONE, null) { | ||||||
|                 }).collect(Collectors.toList()); |                 }).collect(Collectors.toList()); | ||||||
|   | |||||||
| @@ -86,7 +86,8 @@ public class DebugPaste extends SubCommand { | |||||||
|                 b.append("# WorldEdit implementation:\n"); |                 b.append("# WorldEdit implementation:\n"); | ||||||
|                 b.append(PlotSquared.platform().worldEditImplementations()).append("\n\n"); |                 b.append(PlotSquared.platform().worldEditImplementations()).append("\n\n"); | ||||||
|                 b.append("# Server Information\n"); |                 b.append("# Server Information\n"); | ||||||
|                 b.append("Server Version: ").append(PlotSquared.platform().serverImplementation()) |                 b.append("Server Version: ").append(PlotSquared.platform().serverBrand()).append(": ") | ||||||
|  |                         .append(PlotSquared.platform().serverImplementation()).append("\n") | ||||||
|                         .append("\n"); |                         .append("\n"); | ||||||
|                 b.append("online_mode: ").append(!Settings.UUID.OFFLINE).append(';') |                 b.append("online_mode: ").append(!Settings.UUID.OFFLINE).append(';') | ||||||
|                         .append(!Settings.UUID.OFFLINE).append('\n'); |                         .append(!Settings.UUID.OFFLINE).append('\n'); | ||||||
|   | |||||||
| @@ -96,6 +96,7 @@ public class DebugRoadRegen extends SubCommand { | |||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); | ||||||
|  |             return false; | ||||||
|         } |         } | ||||||
|         Plot plot = player.getCurrentPlot(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|   | |||||||
| @@ -124,6 +124,7 @@ public class Delete extends SubCommand { | |||||||
|                                 "amount", |                                 "amount", | ||||||
|                                 Tag.inserting(Component.text(String.valueOf(System.currentTimeMillis() - start))) |                                 Tag.inserting(Component.text(String.valueOf(System.currentTimeMillis() - start))) | ||||||
|                         ), |                         ), | ||||||
|  |                         TagResolver.resolver("world", Tag.inserting(Component.text(plotArea.getWorldName()))), | ||||||
|                         TagResolver.resolver("plot", Tag.inserting(Component.text(plot.getId().toString()))) |                         TagResolver.resolver("plot", Tag.inserting(Component.text(plot.getId().toString()))) | ||||||
|                 ); |                 ); | ||||||
|                 eventDispatcher.callPostDelete(plot); |                 eventDispatcher.callPostDelete(plot); | ||||||
|   | |||||||
| @@ -117,10 +117,11 @@ public class Deny extends SubCommand { | |||||||
|                     } else if (plot.getDenied().contains(uuid)) { |                     } else if (plot.getDenied().contains(uuid)) { | ||||||
|                         player.sendMessage( |                         player.sendMessage( | ||||||
|                                 TranslatableCaption.of("member.already_added"), |                                 TranslatableCaption.of("member.already_added"), | ||||||
|                                 TagResolver.resolver( |                                 PlotSquared.platform().playerManager().getUsernameCaption(uuid) | ||||||
|  |                                         .thenApply(caption -> TagResolver.resolver( | ||||||
|                                         "player", |                                         "player", | ||||||
|                                         Tag.inserting(PlayerManager.resolveName(uuid).toComponent(player)) |                                         Tag.inserting(caption.toComponent(player)) | ||||||
|                                 ) |                                 )) | ||||||
|                         ); |                         ); | ||||||
|                         return; |                         return; | ||||||
|                     } else { |                     } else { | ||||||
|   | |||||||
| @@ -94,7 +94,7 @@ public class Done extends SubCommand { | |||||||
|                 TagResolver.resolver("plot", Tag.inserting(Component.text(plot.getId().toString()))) |                 TagResolver.resolver("plot", Tag.inserting(Component.text(plot.getId().toString()))) | ||||||
|         ); |         ); | ||||||
|         final Settings.Auto_Clear doneRequirements = Settings.AUTO_CLEAR.get("done"); |         final Settings.Auto_Clear doneRequirements = Settings.AUTO_CLEAR.get("done"); | ||||||
|         if (PlotSquared.platform().expireManager() == null || doneRequirements == null) { |         if (PlotSquared.platform().expireManager() == null || doneRequirements == null || player.hasPermission(Permission.PERMISSION_ADMIN_COMMAND_DONE)) { | ||||||
|             finish(plot, player, true); |             finish(plot, player, true); | ||||||
|             plot.removeRunning(); |             plot.removeRunning(); | ||||||
|         } else { |         } else { | ||||||
|   | |||||||
| @@ -20,7 +20,6 @@ package com.plotsquared.core.command; | |||||||
|  |  | ||||||
| import com.google.inject.Inject; | import com.google.inject.Inject; | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.configuration.caption.StaticCaption; |  | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| @@ -136,7 +135,9 @@ public class Download extends SubCommand { | |||||||
|                     } |                     } | ||||||
|                     player.sendMessage( |                     player.sendMessage( | ||||||
|                             TranslatableCaption.of("web.generation_link_success_legacy_world"), |                             TranslatableCaption.of("web.generation_link_success_legacy_world"), | ||||||
|                             TagResolver.resolver("url", Tag.inserting(Component.text(url.toString()))) |                             TagResolver.builder() | ||||||
|  |                                     .tag("url", Tag.preProcessParsed(url.toString())) | ||||||
|  |                                     .build() | ||||||
|                     ); |                     ); | ||||||
|                 } |                 } | ||||||
|             }); |             }); | ||||||
| @@ -200,7 +201,6 @@ public class Download extends SubCommand { | |||||||
|                                                     .tag("delete", Tag.preProcessParsed("Not available")) |                                                     .tag("delete", Tag.preProcessParsed("Not available")) | ||||||
|                                                     .build() |                                                     .build() | ||||||
|                                     ); |                                     ); | ||||||
|                                     player.sendMessage(StaticCaption.of(value.toString())); |  | ||||||
|                                 } |                                 } | ||||||
|                             } |                             } | ||||||
|                     )); |                     )); | ||||||
|   | |||||||
| @@ -103,9 +103,10 @@ public final class FlagCommand extends Command { | |||||||
|         if (flag instanceof IntegerFlag && MathMan.isInteger(value)) { |         if (flag instanceof IntegerFlag && MathMan.isInteger(value)) { | ||||||
|             try { |             try { | ||||||
|                 int numeric = Integer.parseInt(value); |                 int numeric = Integer.parseInt(value); | ||||||
|  |                 // Getting full permission without ".<amount>" at the end | ||||||
|                 perm = perm.substring(0, perm.length() - value.length() - 1); |                 perm = perm.substring(0, perm.length() - value.length() - 1); | ||||||
|                 boolean result = false; |                 boolean result = false; | ||||||
|                 if (numeric > 0) { |                 if (numeric >= 0) { | ||||||
|                     int checkRange = PlotSquared.get().getPlatform().equalsIgnoreCase("bukkit") ? |                     int checkRange = PlotSquared.get().getPlatform().equalsIgnoreCase("bukkit") ? | ||||||
|                             numeric : |                             numeric : | ||||||
|                             Settings.Limit.MAX_PLOTS; |                             Settings.Limit.MAX_PLOTS; | ||||||
|   | |||||||
| @@ -113,38 +113,34 @@ public class Help extends Command { | |||||||
|             } |             } | ||||||
|             if (cat == null && page == 0) { |             if (cat == null && page == 0) { | ||||||
|                 TextComponent.Builder builder = Component.text(); |                 TextComponent.Builder builder = Component.text(); | ||||||
|                 builder.append(MINI_MESSAGE.deserialize(TranslatableCaption.of("help.help_header").getComponent(player))); |                 builder.append(TranslatableCaption.of("help.help_header").toComponent(player)); | ||||||
|                 for (CommandCategory c : CommandCategory.values()) { |                 for (CommandCategory c : CommandCategory.values()) { | ||||||
|                     if (!c.canAccess(player)) { |                     if (!c.canAccess(player)) { | ||||||
|                         continue; |                         continue; | ||||||
|                     } |                     } | ||||||
|                     builder.append(Component.newline()).append(MINI_MESSAGE |                     builder.append(Component.newline()); | ||||||
|                             .deserialize( |                     builder.append(TranslatableCaption.of("help.help_info_item").toComponent( | ||||||
|                                     TranslatableCaption.of("help.help_info_item").getComponent(player), |                             player, TagResolver.builder() | ||||||
|                                     TagResolver.builder() |                                     .tag("command", Tag.inserting(Component.text("/plot help"))) | ||||||
|                                             .tag("command", Tag.inserting(Component.text("/plot help"))) |                                     .tag("category", Tag.inserting(Component.text(c.name().toLowerCase()))) | ||||||
|                                             .tag("category", Tag.inserting(Component.text(c.name().toLowerCase()))) |                                     .tag("category_desc", Tag.inserting(c.toComponent(player))) | ||||||
|                                             .tag("category_desc", Tag.inserting(c.toComponent(player))) |                                     .build() | ||||||
|                                             .build() |                     )); | ||||||
|                             )); |  | ||||||
|                 } |                 } | ||||||
|                 builder.append(Component.newline()).append(MINI_MESSAGE |                 builder.append(Component.newline()); | ||||||
|                         .deserialize( |                 builder.append(TranslatableCaption.of("help.help_info_item").toComponent( | ||||||
|                                 TranslatableCaption.of("help.help_info_item").getComponent(player), |                         player, TagResolver.builder() | ||||||
|                                 TagResolver.builder() |                                 .tag("command", Tag.inserting(Component.text("/plot help"))) | ||||||
|                                         .tag("command", Tag.inserting(Component.text("/plot help"))) |                                 .tag("category", Tag.inserting(Component.text("all"))) | ||||||
|                                         .tag("category", Tag.inserting(Component.text("all"))) |                                 .tag( | ||||||
|                                         .tag( |                                         "category_desc", Tag.inserting(TranslatableCaption | ||||||
|                                                 "category_desc", |                                                 .of("help.help_display_all_commands") | ||||||
|                                                 Tag.inserting(TranslatableCaption |                                                 .toComponent(player)) | ||||||
|                                                         .of("help.help_display_all_commands") |                                 ) | ||||||
|                                                         .toComponent(player)) |                                 .build() | ||||||
|                                         ) |                 )); | ||||||
|                                         .build() |                 builder.append(Component.newline()); | ||||||
|                         )); |                 builder.append(TranslatableCaption.of("help.help_footer").toComponent(player)); | ||||||
|                 builder.append(Component.newline()).append(MINI_MESSAGE.deserialize(TranslatableCaption |  | ||||||
|                         .of("help.help_footer") |  | ||||||
|                         .getComponent(player))); |  | ||||||
|                 player.sendMessage(StaticCaption.of(MINI_MESSAGE.serialize(builder.asComponent()))); |                 player.sendMessage(StaticCaption.of(MINI_MESSAGE.serialize(builder.asComponent()))); | ||||||
|                 return true; |                 return true; | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -153,7 +153,7 @@ public class Kick extends SubCommand { | |||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             return Collections.emptyList(); |             return Collections.emptyList(); | ||||||
|         } |         } | ||||||
|         return TabCompletions.completePlayersInPlot(plot, String.join(",", args).trim(), |         return TabCompletions.completePlayersInPlot(player, plot, String.join(",", args).trim(), | ||||||
|                 Collections.singletonList(player.getName()) |                 Collections.singletonList(player.getName()) | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -465,7 +465,7 @@ public class ListCmd extends SubCommand { | |||||||
|                 TextComponent.Builder builder = Component.text(); |                 TextComponent.Builder builder = Component.text(); | ||||||
|                 if (plot.getFlag(ServerPlotFlag.class)) { |                 if (plot.getFlag(ServerPlotFlag.class)) { | ||||||
|                     TagResolver serverResolver = TagResolver.resolver( |                     TagResolver serverResolver = TagResolver.resolver( | ||||||
|                             "info.server", |                             "server", | ||||||
|                             Tag.inserting(TranslatableCaption.of("info.server").toComponent(player)) |                             Tag.inserting(TranslatableCaption.of("info.server").toComponent(player)) | ||||||
|                     ); |                     ); | ||||||
|                     builder.append(MINI_MESSAGE.deserialize(server, serverResolver)); |                     builder.append(MINI_MESSAGE.deserialize(server, serverResolver)); | ||||||
| @@ -483,13 +483,13 @@ public class ListCmd extends SubCommand { | |||||||
|                                 builder.append(MINI_MESSAGE.deserialize(online, resolver)); |                                 builder.append(MINI_MESSAGE.deserialize(online, resolver)); | ||||||
|                             } else if (uuidMapping.username().equalsIgnoreCase("unknown")) { |                             } else if (uuidMapping.username().equalsIgnoreCase("unknown")) { | ||||||
|                                 TagResolver unknownResolver = TagResolver.resolver( |                                 TagResolver unknownResolver = TagResolver.resolver( | ||||||
|                                         "info.unknown", |                                         "unknown", | ||||||
|                                         Tag.inserting(TranslatableCaption.of("info.unknown").toComponent(player)) |                                         Tag.inserting(TranslatableCaption.of("info.unknown").toComponent(player)) | ||||||
|                                 ); |                                 ); | ||||||
|                                 builder.append(MINI_MESSAGE.deserialize(unknown, unknownResolver)); |                                 builder.append(MINI_MESSAGE.deserialize(unknown, unknownResolver)); | ||||||
|                             } else if (uuidMapping.uuid().equals(DBFunc.EVERYONE)) { |                             } else if (uuidMapping.uuid().equals(DBFunc.EVERYONE)) { | ||||||
|                                 TagResolver everyoneResolver = TagResolver.resolver( |                                 TagResolver everyoneResolver = TagResolver.resolver( | ||||||
|                                         "info.everyone", |                                         "everyone", | ||||||
|                                         Tag.inserting(TranslatableCaption.of("info.everyone").toComponent(player)) |                                         Tag.inserting(TranslatableCaption.of("info.everyone").toComponent(player)) | ||||||
|                                 ); |                                 ); | ||||||
|                                 builder.append(MINI_MESSAGE.deserialize(everyone, everyoneResolver)); |                                 builder.append(MINI_MESSAGE.deserialize(everyone, everyoneResolver)); | ||||||
| @@ -517,6 +517,7 @@ public class ListCmd extends SubCommand { | |||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|                 finalResolver.tag("players", Tag.inserting(builder.asComponent())); |                 finalResolver.tag("players", Tag.inserting(builder.asComponent())); | ||||||
|  |                 finalResolver.tag("size", Tag.inserting(Component.text(plot.getConnectedPlots().size()))); | ||||||
|                 caption.set(TranslatableCaption.of("info.plot_list_item")); |                 caption.set(TranslatableCaption.of("info.plot_list_item")); | ||||||
|                 caption.setTagResolvers(finalResolver.build()); |                 caption.setTagResolvers(finalResolver.build()); | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -41,6 +41,7 @@ import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | |||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.net.MalformedURLException; | import java.net.MalformedURLException; | ||||||
|  | import java.net.URI; | ||||||
| import java.net.URL; | import java.net.URL; | ||||||
| import java.util.Collections; | import java.util.Collections; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| @@ -116,7 +117,7 @@ public class Load extends SubCommand { | |||||||
|                     } |                     } | ||||||
|                     final URL url; |                     final URL url; | ||||||
|                     try { |                     try { | ||||||
|                         url = new URL(Settings.Web.URL + "saves/" + player.getUUID() + '/' + schematic); |                         url = URI.create(Settings.Web.URL + "saves/" + player.getUUID() + '/' + schematic).toURL(); | ||||||
|                     } catch (MalformedURLException e) { |                     } catch (MalformedURLException e) { | ||||||
|                         e.printStackTrace(); |                         e.printStackTrace(); | ||||||
|                         player.sendMessage(TranslatableCaption.of("web.load_failed")); |                         player.sendMessage(TranslatableCaption.of("web.load_failed")); | ||||||
|   | |||||||
| @@ -44,8 +44,13 @@ import org.apache.logging.log4j.Logger; | |||||||
| import java.util.Arrays; | import java.util.Arrays; | ||||||
| import java.util.LinkedList; | import java.util.LinkedList; | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  | import java.util.Objects; | ||||||
|  | import java.util.Optional; | ||||||
| import java.util.concurrent.CompletableFuture; | import java.util.concurrent.CompletableFuture; | ||||||
|  |  | ||||||
|  | import javax.annotation.Nonnull; | ||||||
|  | import javax.annotation.Nullable; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * PlotSquared command class. |  * PlotSquared command class. | ||||||
|  */ |  */ | ||||||
| @@ -147,8 +152,7 @@ public class MainCommand extends Command { | |||||||
|                 try { |                 try { | ||||||
|                     injector.getInstance(command); |                     injector.getInstance(command); | ||||||
|                 } catch (final Exception e) { |                 } catch (final Exception e) { | ||||||
|                     LOGGER.error("Failed to register command {}", command.getCanonicalName()); |                     LOGGER.error("Failed to register command {}", command.getCanonicalName(), e); | ||||||
|                     e.printStackTrace(); |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
| @@ -236,110 +240,171 @@ public class MainCommand extends Command { | |||||||
|             RunnableVal3<Command, Runnable, Runnable> confirm, |             RunnableVal3<Command, Runnable, Runnable> confirm, | ||||||
|             RunnableVal2<Command, CommandResult> whenDone |             RunnableVal2<Command, CommandResult> whenDone | ||||||
|     ) { |     ) { | ||||||
|         // Optional command scope // |         prepareArguments(new CommandExecutionData(player, args, confirm, whenDone, null)) | ||||||
|         Location location = null; |                 .thenCompose(executionData -> { | ||||||
|         Plot plot = null; |                     if (executionData.isEmpty()) { | ||||||
|         boolean tp = false; |                         return CompletableFuture.completedFuture(false); | ||||||
|         if (args.length >= 2) { |  | ||||||
|             PlotArea area = player.getApplicablePlotArea(); |  | ||||||
|             Plot newPlot = Plot.fromString(area, args[0]); |  | ||||||
|             if (newPlot != null && (player instanceof ConsolePlayer || newPlot.getArea() |  | ||||||
|                     .equals(area) || player.hasPermission(Permission.PERMISSION_ADMIN) |  | ||||||
|                     || player.hasPermission(Permission.PERMISSION_ADMIN_AREA_SUDO)) |  | ||||||
|                     && !newPlot.isDenied(player.getUUID())) { |  | ||||||
|                 final Location newLoc; |  | ||||||
|                 if (newPlot.getArea() instanceof SinglePlotArea) { |  | ||||||
|                     newLoc = newPlot.isLoaded() ? newPlot.getCenterSynchronous() : Location.at("", 0, 0, 0); |  | ||||||
|                 } else { |  | ||||||
|                     newLoc = newPlot.getCenterSynchronous(); |  | ||||||
|                 } |  | ||||||
|                 if (player.canTeleport(newLoc)) { |  | ||||||
|                     // Save meta |  | ||||||
|                     try (final MetaDataAccess<Location> locationMetaDataAccess |  | ||||||
|                                  = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LOCATION)) { |  | ||||||
|                         location = locationMetaDataAccess.get().orElse(null); |  | ||||||
|                         locationMetaDataAccess.set(newLoc); |  | ||||||
|                     } |                     } | ||||||
|                     try (final MetaDataAccess<Plot> plotMetaDataAccess |                     var data = executionData.get(); | ||||||
|                                  = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { |                     try { | ||||||
|                         plot = plotMetaDataAccess.get().orElse(null); |                         return super.execute(data.player(), data.args(), data.confirm(), data.whenDone()); | ||||||
|                         plotMetaDataAccess.set(newPlot); |                     } catch (CommandException e) { | ||||||
|                     } |                         throw e; | ||||||
|                     tp = true; |                     } catch (Throwable e) { | ||||||
|                 } else { |                         LOGGER.error("A error occurred while executing plot command", e); | ||||||
|                     player.sendMessage(TranslatableCaption.of("border.denied")); |                         String message = e.getMessage(); | ||||||
|                 } |                         if (message != null) { | ||||||
|                 // Trim command |                             data.player().sendMessage( | ||||||
|                 args = Arrays.copyOfRange(args, 1, args.length); |                                     TranslatableCaption.of("errors.error"), | ||||||
|             } |                                     TagResolver.resolver("value", Tag.inserting(Component.text(message))) | ||||||
|             if (args.length >= 2 && !args[0].isEmpty() && args[0].charAt(0) == '-') { |                             ); | ||||||
|                 if ("f".equals(args[0].substring(1))) { |                         } else { | ||||||
|                     confirm = new RunnableVal3<>() { |                             data.player().sendMessage( | ||||||
|                         @Override |                                     TranslatableCaption.of("errors.error_console")); | ||||||
|                         public void run(Command cmd, Runnable success, Runnable failure) { |  | ||||||
|                             if (area != null && PlotSquared.platform().econHandler().isEnabled(area)) { |  | ||||||
|                                 PlotExpression priceEval = |  | ||||||
|                                         area.getPrices().get(cmd.getFullId()); |  | ||||||
|                                 double price = priceEval != null ? priceEval.evaluate(0d) : 0d; |  | ||||||
|                                 if (price != 0d |  | ||||||
|                                         && PlotSquared.platform().econHandler().getMoney(player) < price) { |  | ||||||
|                                     if (failure != null) { |  | ||||||
|                                         failure.run(); |  | ||||||
|                                     } |  | ||||||
|                                     return; |  | ||||||
|                                 } |  | ||||||
|                             } |  | ||||||
|                             if (success != null) { |  | ||||||
|                                 success.run(); |  | ||||||
|                             } |  | ||||||
|                         } |                         } | ||||||
|                     }; |                     } finally { | ||||||
|                     args = Arrays.copyOfRange(args, 1, args.length); |                         if (data.postCommandData() != null) { | ||||||
|                 } else { |                             resetCommandScope(data.player(), data.postCommandData()); | ||||||
|                     player.sendMessage(TranslatableCaption.of("errors.invalid_command_flag")); |                         } | ||||||
|                     return CompletableFuture.completedFuture(false); |                     } | ||||||
|                 } |                     return CompletableFuture.completedFuture(true); | ||||||
|             } |                 }); | ||||||
|         } |  | ||||||
|         try { |  | ||||||
|             super.execute(player, args, confirm, whenDone); |  | ||||||
|         } catch (CommandException e) { |  | ||||||
|             throw e; |  | ||||||
|         } catch (Throwable e) { |  | ||||||
|             e.printStackTrace(); |  | ||||||
|             String message = e.getMessage(); |  | ||||||
|             if (message != null) { |  | ||||||
|                 player.sendMessage( |  | ||||||
|                         TranslatableCaption.of("errors.error"), |  | ||||||
|                         TagResolver.resolver("value", Tag.inserting(Component.text(message))) |  | ||||||
|                 ); |  | ||||||
|             } else { |  | ||||||
|                 player.sendMessage( |  | ||||||
|                         TranslatableCaption.of("errors.error_console")); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         // Reset command scope // |  | ||||||
|         if (tp && !(player instanceof ConsolePlayer)) { |  | ||||||
|             try (final MetaDataAccess<Location> locationMetaDataAccess |  | ||||||
|                          = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LOCATION)) { |  | ||||||
|                 if (location == null) { |  | ||||||
|                     locationMetaDataAccess.remove(); |  | ||||||
|                 } else { |  | ||||||
|                     locationMetaDataAccess.set(location); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             try (final MetaDataAccess<Plot> plotMetaDataAccess |  | ||||||
|                          = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { |  | ||||||
|                 if (plot == null) { |  | ||||||
|                     plotMetaDataAccess.remove(); |  | ||||||
|                 } else { |  | ||||||
|                     plotMetaDataAccess.set(plot); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return CompletableFuture.completedFuture(true); |         return CompletableFuture.completedFuture(true); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private CompletableFuture<Optional<CommandExecutionData>> prepareArguments(CommandExecutionData data) { | ||||||
|  |         if (data.args().length >= 2) { | ||||||
|  |             PlotArea area = data.player().getApplicablePlotArea(); | ||||||
|  |             Plot newPlot = Plot.fromString(area, data.args()[0]); | ||||||
|  |             return preparePlotArgument(newPlot, data, area) | ||||||
|  |                     .thenApply(d -> d.flatMap(x -> prepareFlagArgument(x, area))); | ||||||
|  |         } else { | ||||||
|  |             return CompletableFuture.completedFuture(Optional.of(data)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private CompletableFuture<Optional<CommandExecutionData>> preparePlotArgument(@Nullable Plot newPlot, | ||||||
|  |                                                                         @Nonnull CommandExecutionData data, | ||||||
|  |                                                                                   @Nullable PlotArea area) { | ||||||
|  |         if (newPlot != null && (data.player() instanceof ConsolePlayer | ||||||
|  |                 || (area != null && area.equals(newPlot.getArea())) | ||||||
|  |                 || data.player().hasPermission(Permission.PERMISSION_ADMIN) | ||||||
|  |                 || data.player().hasPermission(Permission.PERMISSION_ADMIN_AREA_SUDO)) | ||||||
|  |                 && !newPlot.isDenied(data.player().getUUID())) { | ||||||
|  |             return fetchPlotCenterLocation(newPlot) | ||||||
|  |                     .thenApply(newLoc -> { | ||||||
|  |                         if (!data.player().canTeleport(newLoc)) { | ||||||
|  |                             data.player().sendMessage(TranslatableCaption.of("border.denied")); | ||||||
|  |                             return Optional.empty(); | ||||||
|  |                         } | ||||||
|  |                         // Save meta | ||||||
|  |                         var originalCommandMeta = setCommandScope(data.player(), new TemporaryCommandMeta(newLoc, newPlot)); | ||||||
|  |                         return Optional.of(new CommandExecutionData( | ||||||
|  |                                 data.player(), | ||||||
|  |                                 Arrays.copyOfRange(data.args(), 1, data.args().length), // Trimmed command | ||||||
|  |                                 data.confirm(), | ||||||
|  |                                 data.whenDone(), | ||||||
|  |                                 originalCommandMeta | ||||||
|  |                         )); | ||||||
|  |                     }); | ||||||
|  |         } | ||||||
|  |         return CompletableFuture.completedFuture(Optional.of(data)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private Optional<CommandExecutionData> prepareFlagArgument(@Nonnull CommandExecutionData data, @Nonnull PlotArea area) { | ||||||
|  |         if (data.args().length >= 2 && !data.args()[0].isEmpty() && data.args()[0].charAt(0) == '-') { | ||||||
|  |             if ("f".equals(data.args()[0].substring(1))) { | ||||||
|  |                 return Optional.of(new CommandExecutionData( | ||||||
|  |                         data.player(), | ||||||
|  |                         Arrays.copyOfRange(data.args(), 1, data.args().length), // Trimmed command | ||||||
|  |                         createForcedConfirmation(data.player(), area), | ||||||
|  |                         data.whenDone(), | ||||||
|  |                         data.postCommandData() | ||||||
|  |                 )); | ||||||
|  |             } else { | ||||||
|  |                 data.player().sendMessage(TranslatableCaption.of("errors.invalid_command_flag")); | ||||||
|  |                 return Optional.empty(); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return Optional.of(data); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private CompletableFuture<Location> fetchPlotCenterLocation(Plot plot) { | ||||||
|  |         if (plot.getArea() instanceof SinglePlotArea && !plot.isLoaded()) { | ||||||
|  |             return CompletableFuture.completedFuture(Location.at("", 0, 0, 0)); | ||||||
|  |         } | ||||||
|  |         CompletableFuture<Location> future = new CompletableFuture<>(); | ||||||
|  |         plot.getCenter(future::complete); | ||||||
|  |         return future; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private @Nonnull RunnableVal3<Command, Runnable, Runnable> createForcedConfirmation(@Nonnull PlotPlayer<?> player, | ||||||
|  |                                                                                         @Nullable PlotArea area) { | ||||||
|  |         return new RunnableVal3<>() { | ||||||
|  |             @Override | ||||||
|  |             public void run(Command cmd, Runnable success, Runnable failure) { | ||||||
|  |                 if (area != null && PlotSquared.platform().econHandler().isEnabled(area) | ||||||
|  |                         && Optional.of(area.getPrices().get(cmd.getFullId())) | ||||||
|  |                         .map(priceEval -> priceEval.evaluate(0d)) | ||||||
|  |                         .filter(price -> price != 0d) | ||||||
|  |                         .filter(price -> PlotSquared.platform().econHandler().getMoney(player) < price) | ||||||
|  |                         .isPresent()) { | ||||||
|  |                     if (failure != null) { | ||||||
|  |                         failure.run(); | ||||||
|  |                     } | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |                 if (success != null) { | ||||||
|  |                     success.run(); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private @Nonnull TemporaryCommandMeta setCommandScope(@Nonnull PlotPlayer<?> player, @Nonnull TemporaryCommandMeta commandMeta) { | ||||||
|  |         Objects.requireNonNull(commandMeta.location()); | ||||||
|  |         Objects.requireNonNull(commandMeta.plot()); | ||||||
|  |         Location location; | ||||||
|  |         Plot plot; | ||||||
|  |         try (final MetaDataAccess<Location> locationMetaDataAccess | ||||||
|  |                      = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LOCATION)) { | ||||||
|  |             location = locationMetaDataAccess.get().orElse(null); | ||||||
|  |             locationMetaDataAccess.set(commandMeta.location()); | ||||||
|  |         } | ||||||
|  |         try (final MetaDataAccess<Plot> plotMetaDataAccess | ||||||
|  |                      = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { | ||||||
|  |             plot = plotMetaDataAccess.get().orElse(null); | ||||||
|  |             plotMetaDataAccess.set(commandMeta.plot()); | ||||||
|  |         } | ||||||
|  |         return new TemporaryCommandMeta(location, plot); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void resetCommandScope(@Nonnull PlotPlayer<?> player, @Nonnull TemporaryCommandMeta commandMeta) { | ||||||
|  |         try (final MetaDataAccess<Location> locationMetaDataAccess | ||||||
|  |                      = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LOCATION)) { | ||||||
|  |             if (commandMeta.location() == null) { | ||||||
|  |                 locationMetaDataAccess.remove(); | ||||||
|  |             } else { | ||||||
|  |                 locationMetaDataAccess.set(commandMeta.location()); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         try (final MetaDataAccess<Plot> plotMetaDataAccess | ||||||
|  |                      = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { | ||||||
|  |             if (commandMeta.plot() == null) { | ||||||
|  |                 plotMetaDataAccess.remove(); | ||||||
|  |             } else { | ||||||
|  |                 plotMetaDataAccess.set(commandMeta.plot()); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private record CommandExecutionData(@Nonnull PlotPlayer<?> player, @Nonnull String[] args, | ||||||
|  |                     @Nonnull RunnableVal3<Command, Runnable, Runnable> confirm, | ||||||
|  |                     @Nonnull RunnableVal2<Command, CommandResult> whenDone, | ||||||
|  |                     @Nullable TemporaryCommandMeta postCommandData) {} | ||||||
|  |  | ||||||
|  |     private record TemporaryCommandMeta(@Nullable Location location, @Nullable Plot plot) {} | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean canExecute(PlotPlayer<?> player, boolean message) { |     public boolean canExecute(PlotPlayer<?> player, boolean message) { | ||||||
|         return true; |         return true; | ||||||
|   | |||||||
| @@ -56,9 +56,13 @@ public class Music extends SubCommand { | |||||||
|             .asList("music_disc_13", "music_disc_cat", "music_disc_blocks", "music_disc_chirp", |             .asList("music_disc_13", "music_disc_cat", "music_disc_blocks", "music_disc_chirp", | ||||||
|                     "music_disc_far", "music_disc_mall", "music_disc_mellohi", "music_disc_stal", |                     "music_disc_far", "music_disc_mall", "music_disc_mellohi", "music_disc_stal", | ||||||
|                     "music_disc_strad", "music_disc_ward", "music_disc_11", "music_disc_wait", "music_disc_otherside", |                     "music_disc_strad", "music_disc_ward", "music_disc_11", "music_disc_wait", "music_disc_otherside", | ||||||
|                     "music_disc_pigstep", "music_disc_5", "music_disc_relic" |                     "music_disc_pigstep", "music_disc_5", "music_disc_relic", "music_disc_creator", | ||||||
|  |                     "music_disc_creator_music_box", "music_disc_precipice" | ||||||
|             ); |             ); | ||||||
|  |  | ||||||
|  |     // make sure all discs and the bedrock ("cancel") fit into the inventory | ||||||
|  |     private static final int INVENTORY_ROWS = (int) Math.ceil((DISCS.size() + 1) / 9.0); | ||||||
|  |  | ||||||
|     private final InventoryUtil inventoryUtil; |     private final InventoryUtil inventoryUtil; | ||||||
|     private final EventDispatcher eventDispatcher; |     private final EventDispatcher eventDispatcher; | ||||||
|  |  | ||||||
| @@ -93,7 +97,7 @@ public class Music extends SubCommand { | |||||||
|         PlotInventory inv = new PlotInventory( |         PlotInventory inv = new PlotInventory( | ||||||
|                 this.inventoryUtil, |                 this.inventoryUtil, | ||||||
|                 player, |                 player, | ||||||
|                 2, |                 INVENTORY_ROWS, | ||||||
|                 TranslatableCaption.of("plotjukebox.jukebox_header").getComponent(player) |                 TranslatableCaption.of("plotjukebox.jukebox_header").getComponent(player) | ||||||
|         ) { |         ) { | ||||||
|             @Override |             @Override | ||||||
|   | |||||||
| @@ -31,7 +31,6 @@ import com.plotsquared.core.player.PlayerMetaDataKeys; | |||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| import com.plotsquared.core.util.EventDispatcher; | import com.plotsquared.core.util.EventDispatcher; | ||||||
| import com.plotsquared.core.util.PlayerManager; |  | ||||||
| import com.plotsquared.core.util.TabCompletions; | import com.plotsquared.core.util.TabCompletions; | ||||||
| import com.plotsquared.core.util.task.TaskManager; | import com.plotsquared.core.util.task.TaskManager; | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| @@ -136,10 +135,11 @@ public class Owner extends SetCommand { | |||||||
|             if (plot.isOwner(uuid)) { |             if (plot.isOwner(uuid)) { | ||||||
|                 player.sendMessage( |                 player.sendMessage( | ||||||
|                         TranslatableCaption.of("member.already_owner"), |                         TranslatableCaption.of("member.already_owner"), | ||||||
|                         TagResolver.resolver( |                         PlotSquared.platform().playerManager().getUsernameCaption(uuid) | ||||||
|  |                                 .thenApply(caption -> TagResolver.resolver( | ||||||
|                                 "player", |                                 "player", | ||||||
|                                 Tag.inserting(PlayerManager.resolveName(uuid, false).toComponent(player)) |                                 Tag.inserting(caption.toComponent(player)) | ||||||
|                         ) |                         )) | ||||||
|                 ); |                 ); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
| @@ -147,10 +147,11 @@ public class Owner extends SetCommand { | |||||||
|                 if (other == null) { |                 if (other == null) { | ||||||
|                     player.sendMessage( |                     player.sendMessage( | ||||||
|                             TranslatableCaption.of("errors.invalid_player_offline"), |                             TranslatableCaption.of("errors.invalid_player_offline"), | ||||||
|                             TagResolver.resolver( |                             PlotSquared.platform().playerManager().getUsernameCaption(uuid) | ||||||
|  |                                     .thenApply(caption -> TagResolver.resolver( | ||||||
|                                     "player", |                                     "player", | ||||||
|                                     Tag.inserting(PlayerManager.resolveName(uuid).toComponent(player)) |                                     Tag.inserting(caption.toComponent(player)) | ||||||
|                             ) |                             )) | ||||||
|                     ); |                     ); | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|   | |||||||
| @@ -100,23 +100,25 @@ public class Remove extends SubCommand { | |||||||
|                             count++; |                             count++; | ||||||
|                         } |                         } | ||||||
|                     } else if (uuid == DBFunc.EVERYONE) { |                     } else if (uuid == DBFunc.EVERYONE) { | ||||||
|  |                         count += plot.getTrusted().size(); | ||||||
|                         if (plot.removeTrusted(uuid)) { |                         if (plot.removeTrusted(uuid)) { | ||||||
|                             this.eventDispatcher.callTrusted(player, plot, uuid, false); |                             this.eventDispatcher.callTrusted(player, plot, uuid, false); | ||||||
|                             count++; |                         } | ||||||
|                         } else if (plot.removeMember(uuid)) { |                         count += plot.getMembers().size(); | ||||||
|  |                         if (plot.removeMember(uuid)) { | ||||||
|                             this.eventDispatcher.callMember(player, plot, uuid, false); |                             this.eventDispatcher.callMember(player, plot, uuid, false); | ||||||
|                             count++; |                         } | ||||||
|                         } else if (plot.removeDenied(uuid)) { |                         count += plot.getDenied().size(); | ||||||
|  |                         if (plot.removeDenied(uuid)) { | ||||||
|                             this.eventDispatcher.callDenied(player, plot, uuid, false); |                             this.eventDispatcher.callDenied(player, plot, uuid, false); | ||||||
|                             count++; |  | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             if (count == 0) { |             if (count == 0) { | ||||||
|                 player.sendMessage( |                 player.sendMessage( | ||||||
|                         TranslatableCaption.of("errors.invalid_player"), |                         TranslatableCaption.of("member.player_not_removed"), | ||||||
|                         TagResolver.resolver("value", Tag.inserting(Component.text(args[0]))) |                         TagResolver.resolver("player", Tag.inserting(Component.text(args[0]))) | ||||||
|                 ); |                 ); | ||||||
|             } else { |             } else { | ||||||
|                 player.sendMessage( |                 player.sendMessage( | ||||||
|   | |||||||
| @@ -40,6 +40,7 @@ import net.kyori.adventure.text.minimessage.tag.Tag; | |||||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
|  | import java.net.URI; | ||||||
| import java.net.URL; | import java.net.URL; | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
| @@ -130,8 +131,7 @@ public class SchematicCmd extends SubCommand { | |||||||
|                     if (location.startsWith("url:")) { |                     if (location.startsWith("url:")) { | ||||||
|                         try { |                         try { | ||||||
|                             UUID uuid = UUID.fromString(location.substring(4)); |                             UUID uuid = UUID.fromString(location.substring(4)); | ||||||
|                             URL base = new URL(Settings.Web.URL); |                             URL url = URI.create(Settings.Web.URL + "uploads/" + uuid + ".schematic").toURL(); | ||||||
|                             URL url = new URL(base, "uploads/" + uuid + ".schematic"); |  | ||||||
|                             schematic = this.schematicHandler.getSchematic(url); |                             schematic = this.schematicHandler.getSchematic(url); | ||||||
|                         } catch (Exception e) { |                         } catch (Exception e) { | ||||||
|                             e.printStackTrace(); |                             e.printStackTrace(); | ||||||
|   | |||||||
| @@ -19,6 +19,7 @@ | |||||||
| package com.plotsquared.core.command; | package com.plotsquared.core.command; | ||||||
|  |  | ||||||
| import com.google.inject.Inject; | import com.google.inject.Inject; | ||||||
|  | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.database.DBFunc; | import com.plotsquared.core.database.DBFunc; | ||||||
| @@ -103,10 +104,11 @@ public class Trust extends Command { | |||||||
|                             player.hasPermission(Permission.PERMISSION_TRUST_EVERYONE) || player.hasPermission(Permission.PERMISSION_ADMIN_COMMAND_TRUST))) { |                             player.hasPermission(Permission.PERMISSION_TRUST_EVERYONE) || player.hasPermission(Permission.PERMISSION_ADMIN_COMMAND_TRUST))) { | ||||||
|                         player.sendMessage( |                         player.sendMessage( | ||||||
|                                 TranslatableCaption.of("errors.invalid_player"), |                                 TranslatableCaption.of("errors.invalid_player"), | ||||||
|                                 TagResolver.resolver( |                                 PlotSquared.platform().playerManager().getUsernameCaption(uuid) | ||||||
|  |                                         .thenApply(caption -> TagResolver.resolver( | ||||||
|                                         "value", |                                         "value", | ||||||
|                                         Tag.inserting(PlayerManager.resolveName(uuid).toComponent(player)) |                                         Tag.inserting(caption.toComponent(player)) | ||||||
|                                 ) |                                 )) | ||||||
|                         ); |                         ); | ||||||
|                         iterator.remove(); |                         iterator.remove(); | ||||||
|                         continue; |                         continue; | ||||||
| @@ -114,10 +116,11 @@ public class Trust extends Command { | |||||||
|                     if (currentPlot.isOwner(uuid)) { |                     if (currentPlot.isOwner(uuid)) { | ||||||
|                         player.sendMessage( |                         player.sendMessage( | ||||||
|                                 TranslatableCaption.of("member.already_added"), |                                 TranslatableCaption.of("member.already_added"), | ||||||
|                                 TagResolver.resolver( |                                 PlotSquared.platform().playerManager().getUsernameCaption(uuid) | ||||||
|                                         "value", |                                         .thenApply(caption -> TagResolver.resolver( | ||||||
|                                         Tag.inserting(PlayerManager.resolveName(uuid).toComponent(player)) |                                         "player", | ||||||
|                                 ) |                                         Tag.inserting(caption.toComponent(player)) | ||||||
|  |                                 )) | ||||||
|                         ); |                         ); | ||||||
|                         iterator.remove(); |                         iterator.remove(); | ||||||
|                         continue; |                         continue; | ||||||
| @@ -125,10 +128,11 @@ public class Trust extends Command { | |||||||
|                     if (currentPlot.getTrusted().contains(uuid)) { |                     if (currentPlot.getTrusted().contains(uuid)) { | ||||||
|                         player.sendMessage( |                         player.sendMessage( | ||||||
|                                 TranslatableCaption.of("member.already_added"), |                                 TranslatableCaption.of("member.already_added"), | ||||||
|                                 TagResolver.resolver( |                                 PlotSquared.platform().playerManager().getUsernameCaption(uuid) | ||||||
|                                         "value", |                                         .thenApply(caption -> TagResolver.resolver( | ||||||
|                                         Tag.inserting(PlayerManager.resolveName(uuid).toComponent(player)) |                                         "player", | ||||||
|                                 ) |                                         Tag.inserting(caption.toComponent(player)) | ||||||
|  |                                 )) | ||||||
|                         ); |                         ); | ||||||
|                         iterator.remove(); |                         iterator.remove(); | ||||||
|                         continue; |                         continue; | ||||||
|   | |||||||
| @@ -77,6 +77,7 @@ public class Visit extends Command { | |||||||
|             query.whereBasePlot(); |             query.whereBasePlot(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         // without specified argument | ||||||
|         if (page == Integer.MIN_VALUE) { |         if (page == Integer.MIN_VALUE) { | ||||||
|             page = 1; |             page = 1; | ||||||
|         } |         } | ||||||
| @@ -94,10 +95,15 @@ public class Visit extends Command { | |||||||
|  |  | ||||||
|         final List<Plot> plots = query.asList(); |         final List<Plot> plots = query.asList(); | ||||||
|  |  | ||||||
|  |         // Conversion of reversed page argument | ||||||
|  |         if (page < 0) { | ||||||
|  |             page = (plots.size() + 1) + page; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         if (plots.isEmpty()) { |         if (plots.isEmpty()) { | ||||||
|             player.sendMessage(TranslatableCaption.of("invalid.found_no_plots")); |             player.sendMessage(TranslatableCaption.of("invalid.found_no_plots")); | ||||||
|             return; |             return; | ||||||
|         } else if (plots.size() < page || page < 1) { |         } else if (page > plots.size() || page < 1) { | ||||||
|             player.sendMessage( |             player.sendMessage( | ||||||
|                     TranslatableCaption.of("invalid.number_not_in_range"), |                     TranslatableCaption.of("invalid.number_not_in_range"), | ||||||
|                     TagResolver.builder() |                     TagResolver.builder() | ||||||
| @@ -188,34 +194,22 @@ public class Visit extends Command { | |||||||
|         int page = Integer.MIN_VALUE; |         int page = Integer.MIN_VALUE; | ||||||
|  |  | ||||||
|         switch (args.length) { |         switch (args.length) { | ||||||
|             // /p v <user> <area> <page> |             // /p v <player> <area> <page> | ||||||
|             case 3: |             case 3: | ||||||
|                 if (!MathMan.isInteger(args[2])) { |                 if (isInvalidPageNr(args[2])) { | ||||||
|                     player.sendMessage( |                     sendInvalidPageNrMsg(player); | ||||||
|                             TranslatableCaption.of("invalid.not_valid_number"), |  | ||||||
|                             TagResolver.resolver("value", Tag.inserting(Component.text("(1, ∞)"))) |  | ||||||
|                     ); |  | ||||||
|                     player.sendMessage( |  | ||||||
|                             TranslatableCaption.of("commandconfig.command_syntax"), |  | ||||||
|                             TagResolver.resolver("value", Tag.inserting(Component.text(getUsage()))) |  | ||||||
|                     ); |  | ||||||
|                     return CompletableFuture.completedFuture(false); |                     return CompletableFuture.completedFuture(false); | ||||||
|                 } |                 } | ||||||
|                 page = Integer.parseInt(args[2]); |                 page = getPageNr(args[2]); | ||||||
|                 // /p v <name> <area> [page] |                 // /p v <player> <area> [page] | ||||||
|                 // /p v <name> [page] |                 // /p v <player> [page] | ||||||
|             case 2: |             case 2: | ||||||
|                 if (page != Integer.MIN_VALUE || !MathMan.isInteger(args[1])) { |                 // If "case 3" is already through or the argument is not a page number: | ||||||
|  |                 // -> /p v <player> <area> [page] | ||||||
|  |                 if (page != Integer.MIN_VALUE || isInvalidPageNr(args[1])) { | ||||||
|                     sortByArea = this.plotAreaManager.getPlotAreaByString(args[1]); |                     sortByArea = this.plotAreaManager.getPlotAreaByString(args[1]); | ||||||
|                     if (sortByArea == null) { |                     if (sortByArea == null) { | ||||||
|                         player.sendMessage( |                         sendInvalidPageNrMsg(player); | ||||||
|                                 TranslatableCaption.of("invalid.not_valid_number"), |  | ||||||
|                                 TagResolver.resolver("value", Tag.inserting(Component.text("(1, ∞)"))) |  | ||||||
|                         ); |  | ||||||
|                         player.sendMessage( |  | ||||||
|                                 TranslatableCaption.of("commandconfig.command_syntax"), |  | ||||||
|                                 TagResolver.resolver("value", Tag.inserting(Component.text(getUsage()))) |  | ||||||
|                         ); |  | ||||||
|                         return CompletableFuture.completedFuture(false); |                         return CompletableFuture.completedFuture(false); | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
| @@ -249,16 +243,13 @@ public class Visit extends Command { | |||||||
|                     }); |                     }); | ||||||
|                     break; |                     break; | ||||||
|                 } |                 } | ||||||
|                 try { |                 // -> /p v <player> <page> | ||||||
|                     page = Integer.parseInt(args[1]); |                 if (isInvalidPageNr(args[1])) { | ||||||
|                 } catch (NumberFormatException ignored) { |                     sendInvalidPageNrMsg(player); | ||||||
|                     player.sendMessage( |  | ||||||
|                             TranslatableCaption.of("invalid.not_a_number"), |  | ||||||
|                             TagResolver.resolver("value", Tag.inserting(Component.text(args[1]))) |  | ||||||
|                     ); |  | ||||||
|                     return CompletableFuture.completedFuture(false); |                     return CompletableFuture.completedFuture(false); | ||||||
|                 } |                 } | ||||||
|                 // /p v <name> [page] |                 page = getPageNr(args[1]); | ||||||
|  |                 // /p v <player> [page] | ||||||
|                 // /p v <uuid> [page] |                 // /p v <uuid> [page] | ||||||
|                 // /p v <plot> [page] |                 // /p v <plot> [page] | ||||||
|                 // /p v <alias> |                 // /p v <alias> | ||||||
| @@ -326,6 +317,35 @@ public class Visit extends Command { | |||||||
|         return CompletableFuture.completedFuture(true); |         return CompletableFuture.completedFuture(true); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private boolean isInvalidPageNr(String arg) { | ||||||
|  |         if (MathMan.isInteger(arg)) { | ||||||
|  |             return false; | ||||||
|  |         } else if (arg.equals("last") || arg.equals("n")) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private int getPageNr(String arg) { | ||||||
|  |         if (MathMan.isInteger(arg)) { | ||||||
|  |             return Integer.parseInt(arg); | ||||||
|  |         } else if (arg.equals("last") || arg.equals("n")) { | ||||||
|  |             return -1; | ||||||
|  |         } | ||||||
|  |         return Integer.MIN_VALUE; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void sendInvalidPageNrMsg(PlotPlayer<?> player) { | ||||||
|  |         player.sendMessage( | ||||||
|  |                 TranslatableCaption.of("invalid.not_valid_number"), | ||||||
|  |                 TagResolver.resolver("value", Tag.inserting(Component.text("(1, ∞)"))) | ||||||
|  |         ); | ||||||
|  |         player.sendMessage( | ||||||
|  |                 TranslatableCaption.of("commandconfig.command_syntax"), | ||||||
|  |                 TagResolver.resolver("value", Tag.inserting(Component.text(getUsage()))) | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public Collection<Command> tab(PlotPlayer<?> player, String[] args, boolean space) { |     public Collection<Command> tab(PlotPlayer<?> player, String[] args, boolean space) { | ||||||
|         final List<Command> completions = new ArrayList<>(); |         final List<Command> completions = new ArrayList<>(); | ||||||
| @@ -334,6 +354,7 @@ public class Visit extends Command { | |||||||
|             case 1 -> { |             case 1 -> { | ||||||
|                 completions.addAll( |                 completions.addAll( | ||||||
|                         TabCompletions.completeAreas(args[1])); |                         TabCompletions.completeAreas(args[1])); | ||||||
|  |                 completions.addAll(TabCompletions.asCompletions("last")); | ||||||
|                 if (args[1].isEmpty()) { |                 if (args[1].isEmpty()) { | ||||||
|                     // if no input is given, only suggest 1 - 3 |                     // if no input is given, only suggest 1 - 3 | ||||||
|                     completions.addAll( |                     completions.addAll( | ||||||
| @@ -344,6 +365,7 @@ public class Visit extends Command { | |||||||
|                         TabCompletions.completeNumbers(args[1], 10, 999)); |                         TabCompletions.completeNumbers(args[1], 10, 999)); | ||||||
|             } |             } | ||||||
|             case 2 -> { |             case 2 -> { | ||||||
|  |                 completions.addAll(TabCompletions.asCompletions("last")); | ||||||
|                 if (args[2].isEmpty()) { |                 if (args[2].isEmpty()) { | ||||||
|                     // if no input is given, only suggest 1 - 3 |                     // if no input is given, only suggest 1 - 3 | ||||||
|                     completions.addAll( |                     completions.addAll( | ||||||
|   | |||||||
| @@ -26,6 +26,7 @@ import org.apache.logging.log4j.Logger; | |||||||
|  |  | ||||||
| import java.io.File; | import java.io.File; | ||||||
| import java.io.PrintWriter; | import java.io.PrintWriter; | ||||||
|  | import java.lang.annotation.Documented; | ||||||
| import java.lang.annotation.ElementType; | import java.lang.annotation.ElementType; | ||||||
| import java.lang.annotation.Retention; | import java.lang.annotation.Retention; | ||||||
| import java.lang.annotation.RetentionPolicy; | import java.lang.annotation.RetentionPolicy; | ||||||
| @@ -372,6 +373,7 @@ public class Config { | |||||||
|      */ |      */ | ||||||
|     @Retention(RetentionPolicy.RUNTIME) |     @Retention(RetentionPolicy.RUNTIME) | ||||||
|     @Target({ElementType.FIELD, ElementType.TYPE}) |     @Target({ElementType.FIELD, ElementType.TYPE}) | ||||||
|  |     @Documented | ||||||
|     public @interface Comment { |     public @interface Comment { | ||||||
|  |  | ||||||
|         String[] value(); |         String[] value(); | ||||||
|   | |||||||
| @@ -43,6 +43,11 @@ public class Settings extends Config { | |||||||
|             "Leave it off if you don't need it, it can spam your console."}) |             "Leave it off if you don't need it, it can spam your console."}) | ||||||
|     public static boolean DEBUG = true; |     public static boolean DEBUG = true; | ||||||
|  |  | ||||||
|  |     @Comment({"The activity of high-frequency event listener can be deactivated here to improve the server performance. ", | ||||||
|  |             "Affected settings: 'redstone' settings here below. Affected flags: 'disable-physics', 'redstone'. ", | ||||||
|  |             "Only deactivate this setting if you do not need any of the mentioned settings or flags."}) | ||||||
|  |     public static boolean HIGH_FREQUENCY_LISTENER = true; | ||||||
|  |  | ||||||
|     @Create // This value will be generated automatically |     @Create // This value will be generated automatically | ||||||
|     public static ConfigBlock<Auto_Clear> AUTO_CLEAR = null; |     public static ConfigBlock<Auto_Clear> AUTO_CLEAR = null; | ||||||
|     // A ConfigBlock is a section that can have multiple instances e.g. multiple expiry tasks |     // A ConfigBlock is a section that can have multiple instances e.g. multiple expiry tasks | ||||||
| @@ -520,9 +525,9 @@ public class Settings extends Config { | |||||||
|     public static final class Limit { |     public static final class Limit { | ||||||
|  |  | ||||||
|         @Comment("Should the limit be global (over multiple worlds)") |         @Comment("Should the limit be global (over multiple worlds)") | ||||||
|         public static boolean GLOBAL = |         public static boolean GLOBAL = false; | ||||||
|                 false; |  | ||||||
|         @Comment({"The max range of permissions to check for, e.g. plots.plot.127", |         @Comment({"The max range of integer permissions to check for, e.g. 'plots.plot.127' or 'plots.set.flag.mob-cap.127'", | ||||||
|                 "The value covers the permission range to check, you need to assign the permission to players/groups still", |                 "The value covers the permission range to check, you need to assign the permission to players/groups still", | ||||||
|                 "Modifying the value does NOT change the amount of plots players can claim"}) |                 "Modifying the value does NOT change the amount of plots players can claim"}) | ||||||
|         public static int MAX_PLOTS = 127; |         public static int MAX_PLOTS = 127; | ||||||
| @@ -577,6 +582,8 @@ public class Settings extends Config { | |||||||
|         public static boolean PER_WORLD_VISIT = false; |         public static boolean PER_WORLD_VISIT = false; | ||||||
|         @Comment("Search merged plots for having multiple owners when using the visit command") |         @Comment("Search merged plots for having multiple owners when using the visit command") | ||||||
|         public static boolean VISIT_MERGED_OWNERS = true; |         public static boolean VISIT_MERGED_OWNERS = true; | ||||||
|  |         @Comment("Allows to teleport based on block size instead to spawn on the highest block at the home command") | ||||||
|  |         public static boolean SIZED_BASED = true; | ||||||
|  |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -646,6 +653,8 @@ public class Settings extends Config { | |||||||
|         public static boolean PAPER_LISTENERS = true; |         public static boolean PAPER_LISTENERS = true; | ||||||
|         @Comment("Prevent entities from leaving plots") |         @Comment("Prevent entities from leaving plots") | ||||||
|         public static boolean ENTITY_PATHING = true; |         public static boolean ENTITY_PATHING = true; | ||||||
|  |         @Comment("Prevent entities from leaving plots, even by pushing or pulling") | ||||||
|  |         public static boolean ENTITY_MOVEMENT = false; | ||||||
|         @Comment( |         @Comment( | ||||||
|                 "Cancel entity spawns when the chunk is loaded if the PlotArea's mob spawning is off") |                 "Cancel entity spawns when the chunk is loaded if the PlotArea's mob spawning is off") | ||||||
|         public static boolean CANCEL_CHUNK_SPAWN = true; |         public static boolean CANCEL_CHUNK_SPAWN = true; | ||||||
| @@ -723,6 +732,13 @@ public class Settings extends Config { | |||||||
|  |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Comment("Settings related to flags") | ||||||
|  |     public static final class Flags { | ||||||
|  |  | ||||||
|  |         @Comment("If \"instabreak\" should consider the used tool.") | ||||||
|  |         public static boolean INSTABREAK_CONSIDER_TOOL = false; | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Comment({"Enable or disable parts of the plugin", |     @Comment({"Enable or disable parts of the plugin", | ||||||
|             "Note: A cache will use some memory if enabled"}) |             "Note: A cache will use some memory if enabled"}) | ||||||
| @@ -815,4 +831,17 @@ public class Settings extends Config { | |||||||
|  |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Comment({"Permission-Resolver specific settings"}) | ||||||
|  |     public static final class Permissions { | ||||||
|  |  | ||||||
|  |         @Comment({ | ||||||
|  |                 "use the new LuckPerms resolver for ranged permissions (e.g., plots.plot.<number>).", | ||||||
|  |                 "in contrary to the default resolver, this resolver can handle inheritance of permissions.", | ||||||
|  |                 "e.g. if the player has multiple matching permissions, for example, due to multiple group memberships " + | ||||||
|  |                         "(plots.plot.5 & plots.plot.15), this resolver would use 15 as the limit as opposed to the previous 5." | ||||||
|  |         }) | ||||||
|  |         public static boolean USE_LUCKPERMS_RANGE_RESOLVER = false; | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -46,7 +46,7 @@ public class Storage extends Config { | |||||||
|         public static String PASSWORD = "password"; |         public static String PASSWORD = "password"; | ||||||
|         public static String DATABASE = "plot_db"; |         public static String DATABASE = "plot_db"; | ||||||
|  |  | ||||||
|         @Comment("Set additional properties: https://goo.gl/wngtN8") |         @Comment("Set additional properties: https://dev.mysql.com/doc/connector-j/en/connector-j-reference-configuration-properties.html") | ||||||
|         public static List<String> |         public static List<String> | ||||||
|                 PROPERTIES = new ArrayList<>(Collections.singletonList("useSSL=false")); |                 PROPERTIES = new ArrayList<>(Collections.singletonList("useSSL=false")); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -20,6 +20,7 @@ package com.plotsquared.core.configuration.caption; | |||||||
|  |  | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| import net.kyori.adventure.text.ComponentLike; | import net.kyori.adventure.text.ComponentLike; | ||||||
|  | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -44,4 +45,16 @@ public interface Caption { | |||||||
|      */ |      */ | ||||||
|     @NonNull Component toComponent(@NonNull LocaleHolder localeHolder); |     @NonNull Component toComponent(@NonNull LocaleHolder localeHolder); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get the Adventure {@link ComponentLike} for this caption while applying custom {@link TagResolver} | ||||||
|  |      * (apart from the default {@code core.prefix}) | ||||||
|  |      * @param localeHolder Local holder | ||||||
|  |      * @param tagResolvers custom tag resolvers to replace placeholders / parameters | ||||||
|  |      * @return {@link ComponentLike} | ||||||
|  |      * @since TODO | ||||||
|  |      */ | ||||||
|  |     @NonNull Component toComponent(@NonNull LocaleHolder localeHolder, @NonNull TagResolver @NonNull... tagResolvers); | ||||||
|  |  | ||||||
|  |     @NonNull String toString(); | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -21,6 +21,7 @@ package com.plotsquared.core.configuration.caption; | |||||||
| import com.google.common.base.Preconditions; | import com.google.common.base.Preconditions; | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| import net.kyori.adventure.text.minimessage.MiniMessage; | import net.kyori.adventure.text.minimessage.MiniMessage; | ||||||
|  | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| public final class StaticCaption implements Caption { | public final class StaticCaption implements Caption { | ||||||
| @@ -51,4 +52,17 @@ public final class StaticCaption implements Caption { | |||||||
|         return MiniMessage.miniMessage().deserialize(this.value); |         return MiniMessage.miniMessage().deserialize(this.value); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NonNull Component toComponent( | ||||||
|  |             @NonNull final LocaleHolder localeHolder, | ||||||
|  |             final @NonNull TagResolver @NonNull ... tagResolvers | ||||||
|  |     ) { | ||||||
|  |         return MiniMessage.miniMessage().deserialize(this.value, tagResolvers); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NonNull String toString() { | ||||||
|  |         return "StaticCaption(" + value + ")"; | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -25,7 +25,9 @@ import net.kyori.adventure.text.minimessage.MiniMessage; | |||||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | import net.kyori.adventure.text.minimessage.tag.Tag; | ||||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  | import org.jetbrains.annotations.NotNull; | ||||||
|  |  | ||||||
|  | import java.util.Arrays; | ||||||
| import java.util.Locale; | import java.util.Locale; | ||||||
| import java.util.regex.Pattern; | import java.util.regex.Pattern; | ||||||
|  |  | ||||||
| @@ -95,13 +97,23 @@ public final class TranslatableCaption implements NamespacedCaption { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public @NonNull Component toComponent(@NonNull final LocaleHolder localeHolder) { |     public @NonNull Component toComponent(@NonNull final LocaleHolder localeHolder) { | ||||||
|  |         return this.toComponent(localeHolder, new TagResolver[0]); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NonNull Component toComponent( | ||||||
|  |             @NonNull final LocaleHolder localeHolder, | ||||||
|  |             final @NonNull TagResolver @NonNull ... tagResolvers | ||||||
|  |     ) { | ||||||
|         if (getKey().equals("core.prefix")) { |         if (getKey().equals("core.prefix")) { | ||||||
|             return MiniMessage.miniMessage().deserialize(getComponent(localeHolder)); |             return MiniMessage.miniMessage().deserialize(getComponent(localeHolder)); | ||||||
|         } |         } | ||||||
|         return MiniMessage.miniMessage().deserialize(getComponent(localeHolder), TagResolver.resolver( |         TagResolver[] finalResolvers = Arrays.copyOf(tagResolvers, tagResolvers.length + 1); | ||||||
|  |         finalResolvers[finalResolvers.length - 1] = TagResolver.resolver( | ||||||
|                 "prefix", |                 "prefix", | ||||||
|                 Tag.inserting(TranslatableCaption.of("core.prefix").toComponent(localeHolder)) |                 Tag.inserting(TranslatableCaption.of("core.prefix").toComponent(localeHolder)) | ||||||
|         )); |         ); | ||||||
|  |         return MiniMessage.miniMessage().deserialize(getComponent(localeHolder), finalResolvers); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -132,4 +144,9 @@ public final class TranslatableCaption implements NamespacedCaption { | |||||||
|         return Objects.hashCode(this.getNamespace(), this.getKey()); |         return Objects.hashCode(this.getNamespace(), this.getKey()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NotNull String toString() { | ||||||
|  |         return "TranslatableCaption(" + getNamespace() + ":" + getKey() + ")"; | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -130,6 +130,7 @@ public class SQLManager implements AbstractDB { | |||||||
|     public volatile ConcurrentHashMap<PlotCluster, Queue<UniqueStatement>> clusterTasks; |     public volatile ConcurrentHashMap<PlotCluster, Queue<UniqueStatement>> clusterTasks; | ||||||
|     // Private |     // Private | ||||||
|     private Connection connection; |     private Connection connection; | ||||||
|  |     private boolean supportsGetGeneratedKeys; | ||||||
|     private boolean closed = false; |     private boolean closed = false; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -154,6 +155,8 @@ public class SQLManager implements AbstractDB { | |||||||
|         this.worldConfiguration = worldConfiguration; |         this.worldConfiguration = worldConfiguration; | ||||||
|         this.database = database; |         this.database = database; | ||||||
|         this.connection = database.openConnection(); |         this.connection = database.openConnection(); | ||||||
|  |         final DatabaseMetaData databaseMetaData = this.connection.getMetaData(); | ||||||
|  |         this.supportsGetGeneratedKeys = databaseMetaData.supportsGetGeneratedKeys(); | ||||||
|         this.mySQL = database instanceof MySQL; |         this.mySQL = database instanceof MySQL; | ||||||
|         this.globalTasks = new ConcurrentLinkedQueue<>(); |         this.globalTasks = new ConcurrentLinkedQueue<>(); | ||||||
|         this.notifyTasks = new ConcurrentLinkedQueue<>(); |         this.notifyTasks = new ConcurrentLinkedQueue<>(); | ||||||
| @@ -161,6 +164,14 @@ public class SQLManager implements AbstractDB { | |||||||
|         this.playerTasks = new ConcurrentHashMap<>(); |         this.playerTasks = new ConcurrentHashMap<>(); | ||||||
|         this.clusterTasks = new ConcurrentHashMap<>(); |         this.clusterTasks = new ConcurrentHashMap<>(); | ||||||
|         this.prefix = prefix; |         this.prefix = prefix; | ||||||
|  |  | ||||||
|  |         if (mySQL && !supportsGetGeneratedKeys) { | ||||||
|  |             String driver = databaseMetaData.getDriverName(); | ||||||
|  |             String driverVersion = databaseMetaData.getDriverVersion(); | ||||||
|  |             throw new SQLException("Database Driver for MySQL does not support Statement#getGeneratedKeys - which breaks " + | ||||||
|  |                     "PlotSquared functionality (Using " + driver + ":" + driverVersion + ")"); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         this.SET_OWNER = "UPDATE `" + this.prefix |         this.SET_OWNER = "UPDATE `" + this.prefix | ||||||
|                 + "plot` SET `owner` = ? WHERE `plot_id_x` = ? AND `plot_id_z` = ? AND `world` = ?"; |                 + "plot` SET `owner` = ? WHERE `plot_id_x` = ? AND `plot_id_z` = ? AND `world` = ?"; | ||||||
|         this.GET_ALL_PLOTS = |         this.GET_ALL_PLOTS = | ||||||
| @@ -171,20 +182,32 @@ public class SQLManager implements AbstractDB { | |||||||
|                 "INSERT INTO `" + this.prefix + "plot_settings` (`plot_plot_id`) values "; |                 "INSERT INTO `" + this.prefix + "plot_settings` (`plot_plot_id`) values "; | ||||||
|         this.CREATE_TIERS = |         this.CREATE_TIERS = | ||||||
|                 "INSERT INTO `" + this.prefix + "plot_%tier%` (`plot_plot_id`, `user_uuid`) values "; |                 "INSERT INTO `" + this.prefix + "plot_%tier%` (`plot_plot_id`, `user_uuid`) values "; | ||||||
|         this.CREATE_PLOT = "INSERT INTO `" + this.prefix |         String tempCreatePlot = "INSERT INTO `" + this.prefix | ||||||
|                 + "plot`(`plot_id_x`, `plot_id_z`, `owner`, `world`, `timestamp`) VALUES(?, ?, ?, ?, ?)"; |                 + "plot`(`plot_id_x`, `plot_id_z`, `owner`, `world`, `timestamp`) VALUES(?, ?, ?, ?, ?)"; | ||||||
|  |         if (!supportsGetGeneratedKeys) { | ||||||
|  |             tempCreatePlot += " RETURNING `id`"; | ||||||
|  |         } | ||||||
|  |         this.CREATE_PLOT = tempCreatePlot; | ||||||
|         if (mySQL) { |         if (mySQL) { | ||||||
|             this.CREATE_PLOT_SAFE = "INSERT IGNORE INTO `" + this.prefix |             this.CREATE_PLOT_SAFE = "INSERT IGNORE INTO `" + this.prefix | ||||||
|                     + "plot`(`plot_id_x`, `plot_id_z`, `owner`, `world`, `timestamp`) SELECT ?, ?, ?, ?, ? FROM DUAL WHERE NOT EXISTS (SELECT null FROM `" |                     + "plot`(`plot_id_x`, `plot_id_z`, `owner`, `world`, `timestamp`) SELECT ?, ?, ?, ?, ? FROM DUAL WHERE NOT EXISTS (SELECT null FROM `" | ||||||
|                     + this.prefix + "plot` WHERE `world` = ? AND `plot_id_x` = ? AND `plot_id_z` = ?)"; |                     + this.prefix + "plot` WHERE `world` = ? AND `plot_id_x` = ? AND `plot_id_z` = ?)"; | ||||||
|         } else { |         } else { | ||||||
|             this.CREATE_PLOT_SAFE = "INSERT INTO `" + this.prefix |             String tempCreatePlotSafe = "INSERT INTO `" + this.prefix | ||||||
|                     + "plot`(`plot_id_x`, `plot_id_z`, `owner`, `world`, `timestamp`) SELECT ?, ?, ?, ?, ? WHERE NOT EXISTS (SELECT null FROM `" |                     + "plot`(`plot_id_x`, `plot_id_z`, `owner`, `world`, `timestamp`) SELECT ?, ?, ?, ?, ? WHERE NOT EXISTS (SELECT null FROM `" | ||||||
|                     + this.prefix + "plot` WHERE `world` = ? AND `plot_id_x` = ? AND `plot_id_z` = ?)"; |                     + this.prefix + "plot` WHERE `world` = ? AND `plot_id_x` = ? AND `plot_id_z` = ?)"; | ||||||
|  |             if (!supportsGetGeneratedKeys) { | ||||||
|  |                 tempCreatePlotSafe += " RETURNING `id`"; | ||||||
|  |             } | ||||||
|  |             this.CREATE_PLOT_SAFE = tempCreatePlotSafe; | ||||||
|         } |         } | ||||||
|         this.CREATE_CLUSTER = "INSERT INTO `" + this.prefix |         String tempCreateCluster = "INSERT INTO `" + this.prefix | ||||||
|                 + "cluster`(`pos1_x`, `pos1_z`, `pos2_x`, `pos2_z`, `owner`, `world`) VALUES(?, ?, ?, ?, ?, ?)"; |                 + "cluster`(`pos1_x`, `pos1_z`, `pos2_x`, `pos2_z`, `owner`, `world`) VALUES(?, ?, ?, ?, ?, ?)"; | ||||||
|  |         if (!supportsGetGeneratedKeys) { | ||||||
|  |             tempCreateCluster += " RETURNING `id`"; | ||||||
|  |         } | ||||||
|  |         this.CREATE_CLUSTER = tempCreateCluster; | ||||||
|  |  | ||||||
|         try { |         try { | ||||||
|             createTables(); |             createTables(); | ||||||
|         } catch (SQLException e) { |         } catch (SQLException e) { | ||||||
| @@ -1073,9 +1096,8 @@ public class SQLManager implements AbstractDB { | |||||||
|  |  | ||||||
|             @Override |             @Override | ||||||
|             public void addBatch(PreparedStatement statement) throws SQLException { |             public void addBatch(PreparedStatement statement) throws SQLException { | ||||||
|                 int inserted = statement.executeUpdate(); |                 if (statement.execute() || statement.getUpdateCount() > 0) { | ||||||
|                 if (inserted > 0) { |                     try (ResultSet keys = supportsGetGeneratedKeys ? statement.getGeneratedKeys() : statement.getResultSet()) { | ||||||
|                     try (ResultSet keys = statement.getGeneratedKeys()) { |  | ||||||
|                         if (keys.next()) { |                         if (keys.next()) { | ||||||
|                             plot.temp = keys.getInt(1); |                             plot.temp = keys.getInt(1); | ||||||
|                             addPlotTask(plot, new UniqueStatement( |                             addPlotTask(plot, new UniqueStatement( | ||||||
| @@ -1145,8 +1167,8 @@ public class SQLManager implements AbstractDB { | |||||||
|  |  | ||||||
|             @Override |             @Override | ||||||
|             public void addBatch(PreparedStatement statement) throws SQLException { |             public void addBatch(PreparedStatement statement) throws SQLException { | ||||||
|                 statement.executeUpdate(); |                 statement.execute(); | ||||||
|                 try (ResultSet keys = statement.getGeneratedKeys()) { |                 try (ResultSet keys = supportsGetGeneratedKeys ? statement.getGeneratedKeys() : statement.getResultSet()) { | ||||||
|                     if (keys.next()) { |                     if (keys.next()) { | ||||||
|                         plot.temp = keys.getInt(1); |                         plot.temp = keys.getInt(1); | ||||||
|                     } |                     } | ||||||
| @@ -2379,7 +2401,8 @@ public class SQLManager implements AbstractDB { | |||||||
|         addPlotTask(plot, new UniqueStatement("setPosition") { |         addPlotTask(plot, new UniqueStatement("setPosition") { | ||||||
|             @Override |             @Override | ||||||
|             public void set(PreparedStatement statement) throws SQLException { |             public void set(PreparedStatement statement) throws SQLException { | ||||||
|                 statement.setString(1, position == null ? "" : position); |                 // Please see the table creation statement. There is the default value of "default" | ||||||
|  |                 statement.setString(1, position == null ? "DEFAULT" : position); | ||||||
|                 statement.setInt(2, getId(plot)); |                 statement.setInt(2, getId(plot)); | ||||||
|             } |             } | ||||||
|  |  | ||||||
| @@ -3058,8 +3081,8 @@ public class SQLManager implements AbstractDB { | |||||||
|  |  | ||||||
|             @Override |             @Override | ||||||
|             public void addBatch(PreparedStatement statement) throws SQLException { |             public void addBatch(PreparedStatement statement) throws SQLException { | ||||||
|                 statement.executeUpdate(); |                 statement.execute(); | ||||||
|                 try (ResultSet keys = statement.getGeneratedKeys()) { |                 try (ResultSet keys = supportsGetGeneratedKeys ? statement.getGeneratedKeys() : statement.getResultSet()) { | ||||||
|                     if (keys.next()) { |                     if (keys.next()) { | ||||||
|                         cluster.temp = keys.getInt(1); |                         cluster.temp = keys.getInt(1); | ||||||
|                     } |                     } | ||||||
|   | |||||||
| @@ -18,17 +18,34 @@ | |||||||
|  */ |  */ | ||||||
| package com.plotsquared.core.events; | package com.plotsquared.core.events; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * PlotSquared event with {@link Result} to cancel, force, or allow. |  * PlotSquared event with {@link Result} to cancel, force, or allow. | ||||||
|  */ |  */ | ||||||
| public interface CancellablePlotEvent { | public interface CancellablePlotEvent { | ||||||
|  |  | ||||||
|     Result getEventResult(); |     /** | ||||||
|  |      * The currently set {@link Result} for this event (as set by potential previous event listeners). | ||||||
|  |      * | ||||||
|  |      * @return the current result. | ||||||
|  |      */ | ||||||
|  |     @Nullable Result getEventResult(); | ||||||
|  |  | ||||||
|     void setEventResult(Result eventResult); |     /** | ||||||
|  |      * Set the {@link Result} for this event. | ||||||
|  |      * | ||||||
|  |      * @param eventResult the new result. | ||||||
|  |      */ | ||||||
|  |     void setEventResult(@Nullable Result eventResult); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @deprecated No usage and not null-safe | ||||||
|  |      */ | ||||||
|  |     @Deprecated(since = "7.3.2") | ||||||
|     default int getEventResultRaw() { |     default int getEventResultRaw() { | ||||||
|         return getEventResult().getValue(); |         return getEventResult() != null ? getEventResult().getValue() : -1; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,88 @@ | |||||||
|  | /* | ||||||
|  |  * PlotSquared, a land and world management plugin for Minecraft. | ||||||
|  |  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||||
|  |  * Copyright (C) IntellectualSites team and contributors | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 3 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  | package com.plotsquared.core.events; | ||||||
|  |  | ||||||
|  | import com.plotsquared.core.player.PlotPlayer; | ||||||
|  | import com.plotsquared.core.plot.Plot; | ||||||
|  | import org.checkerframework.checker.index.qual.NonNegative; | ||||||
|  | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Called when a user attempts to buy a plot. | ||||||
|  |  * <p> | ||||||
|  |  * Setting the {@link #setEventResult(Result) Result} to {@link Result#FORCE} ignores the price and players account balance and does not charge the | ||||||
|  |  * player anything. {@link Result#DENY} blocks the purchase completely, {@link Result#ACCEPT} and {@code null} do not modify | ||||||
|  |  * the behaviour. | ||||||
|  |  * <p> | ||||||
|  |  * Setting the {@link #setPrice(double) price} to {@code 0} makes the plot practically free. | ||||||
|  |  * | ||||||
|  |  * @since 7.3.2 | ||||||
|  |  */ | ||||||
|  | public class PlayerBuyPlotEvent extends PlotPlayerEvent implements CancellablePlotEvent { | ||||||
|  |  | ||||||
|  |     private Result result; | ||||||
|  |     private double price; | ||||||
|  |  | ||||||
|  |     public PlayerBuyPlotEvent(final PlotPlayer<?> plotPlayer, final Plot plot, @NonNegative final double price) { | ||||||
|  |         super(plotPlayer, plot); | ||||||
|  |         this.price = price; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Sets the price required to buy the plot. | ||||||
|  |      * | ||||||
|  |      * @param price the new price. | ||||||
|  |      * @since 7.3.2 | ||||||
|  |      */ | ||||||
|  |     public void setPrice(@NonNegative final double price) { | ||||||
|  |         //noinspection ConstantValue - the annotation does not ensure a non-negative runtime value | ||||||
|  |         if (price < 0) { | ||||||
|  |             throw new IllegalArgumentException("price must be non-negative"); | ||||||
|  |         } | ||||||
|  |         this.price = price; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Returns the currently set price required to buy the plot. | ||||||
|  |      * | ||||||
|  |      * @return the price. | ||||||
|  |      * @since 7.3.2 | ||||||
|  |      */ | ||||||
|  |     public @NonNegative double price() { | ||||||
|  |         return price; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * {@inheritDoc} | ||||||
|  |      */ | ||||||
|  |     @Override | ||||||
|  |     public void setEventResult(@Nullable final Result eventResult) { | ||||||
|  |         this.result = eventResult; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * {@inheritDoc} | ||||||
|  |      */ | ||||||
|  |     @Override | ||||||
|  |     public @Nullable Result getEventResult() { | ||||||
|  |         return this.result; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -24,7 +24,7 @@ import com.plotsquared.core.plot.Plot; | |||||||
| public class PlayerEnterPlotEvent extends PlotPlayerEvent { | public class PlayerEnterPlotEvent extends PlotPlayerEvent { | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Called when a player leaves a plot. |      * PlayerEnterPlotEvent: Called when a player enters a plot | ||||||
|      * |      * | ||||||
|      * @param player Player that entered the plot |      * @param player Player that entered the plot | ||||||
|      * @param plot   Plot that was entered |      * @param plot   Plot that was entered | ||||||
|   | |||||||
| @@ -0,0 +1,77 @@ | |||||||
|  | /* | ||||||
|  |  * PlotSquared, a land and world management plugin for Minecraft. | ||||||
|  |  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||||
|  |  * Copyright (C) IntellectualSites team and contributors | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 3 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  | package com.plotsquared.core.events; | ||||||
|  |  | ||||||
|  | import com.plotsquared.core.player.PlotPlayer; | ||||||
|  | import org.checkerframework.checker.index.qual.NonNegative; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Called every time after PlotSquared calculated a players plot limit based on their permission. | ||||||
|  |  * <p> | ||||||
|  |  * May be used to grant a player more plots based on another rank or bought feature. | ||||||
|  |  * | ||||||
|  |  * @since 7.3.0 | ||||||
|  |  */ | ||||||
|  | public class PlayerPlotLimitEvent { | ||||||
|  |  | ||||||
|  |     private final PlotPlayer<?> player; | ||||||
|  |  | ||||||
|  |     private int limit; | ||||||
|  |  | ||||||
|  |     public PlayerPlotLimitEvent(@NonNull final PlotPlayer<?> player, @NonNegative final int limit) { | ||||||
|  |         this.player = player; | ||||||
|  |         this.limit = limit; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Overrides the previously calculated or set plot limit for {@link #player()}. | ||||||
|  |      * | ||||||
|  |      * @param limit The amount of plots a player may claim. Must be {@code 0} or greater. | ||||||
|  |      * @since 7.3.0 | ||||||
|  |      */ | ||||||
|  |     public void limit(@NonNegative final int limit) { | ||||||
|  |         if (limit < 0) { | ||||||
|  |             throw new IllegalArgumentException("Player plot limit must be greater or equal 0"); | ||||||
|  |         } | ||||||
|  |         this.limit = limit; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Returns the previous set limit, if none was overridden before this event handler the default limit based on the players | ||||||
|  |      * permissions node is returned. | ||||||
|  |      * | ||||||
|  |      * @return The currently defined plot limit of this player. | ||||||
|  |      * @since 7.3.0 | ||||||
|  |      */ | ||||||
|  |     public @NonNegative int limit() { | ||||||
|  |         return limit; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * The player for which the limit is queried. | ||||||
|  |      * | ||||||
|  |      * @return the player. | ||||||
|  |      * @since 7.3.0 | ||||||
|  |      */ | ||||||
|  |     public @NonNull PlotPlayer<?> player() { | ||||||
|  |         return player; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -21,21 +21,26 @@ package com.plotsquared.core.events; | |||||||
| import com.plotsquared.core.location.Location; | import com.plotsquared.core.location.Location; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
|  | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  |  | ||||||
|  | import java.util.function.UnaryOperator; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Called when a player teleports to a plot |  * Called when a player teleports to a plot | ||||||
|  */ |  */ | ||||||
| public class PlayerTeleportToPlotEvent extends PlotPlayerEvent implements CancellablePlotEvent { | public class PlayerTeleportToPlotEvent extends PlotPlayerEvent implements CancellablePlotEvent { | ||||||
|  |  | ||||||
|     private final Location from; |  | ||||||
|     private final TeleportCause cause; |     private final TeleportCause cause; | ||||||
|     private Result eventResult; |     private Result eventResult; | ||||||
|  |     private final Location from; | ||||||
|  |     private UnaryOperator<Location> locationTransformer; | ||||||
|  |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * PlayerTeleportToPlotEvent: Called when a player teleports to a plot |      * PlayerTeleportToPlotEvent: Called when a player teleports to a plot | ||||||
|      * |      * | ||||||
|      * @param player That was teleported |      * @param player That was teleported | ||||||
|      * @param from   Start location |      * @param from   The origin location, from where the teleport was triggered (players location most likely) | ||||||
|      * @param plot   Plot to which the player was teleported |      * @param plot   Plot to which the player was teleported | ||||||
|      * @param cause  Why the teleport is being completed |      * @param cause  Why the teleport is being completed | ||||||
|      * @since 6.1.0 |      * @since 6.1.0 | ||||||
| @@ -57,7 +62,8 @@ public class PlayerTeleportToPlotEvent extends PlotPlayerEvent implements Cancel | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the from location |      * Get the location, from where the teleport was triggered | ||||||
|  |      * (the players current location when executing the home command for example) | ||||||
|      * |      * | ||||||
|      * @return Location |      * @return Location | ||||||
|      */ |      */ | ||||||
| @@ -65,6 +71,27 @@ public class PlayerTeleportToPlotEvent extends PlotPlayerEvent implements Cancel | |||||||
|         return this.from; |         return this.from; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Gets the currently applied {@link UnaryOperator<Location> transformer} or null, if none was set | ||||||
|  |      * | ||||||
|  |      * @return LocationTransformer | ||||||
|  |      * @since 7.2.1 | ||||||
|  |      */ | ||||||
|  |     public @Nullable UnaryOperator<Location> getLocationTransformer() { | ||||||
|  |         return this.locationTransformer; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Sets the {@link UnaryOperator<Location> transformer} to mutate the location where the player will be teleported to. | ||||||
|  |      * May be {@code null}, if any previous set transformations should be discarded. | ||||||
|  |      * | ||||||
|  |      * @param locationTransformer The new transformer | ||||||
|  |      * @since 7.2.1 | ||||||
|  |      */ | ||||||
|  |     public void setLocationTransformer(@Nullable UnaryOperator<Location> locationTransformer) { | ||||||
|  |         this.locationTransformer = locationTransformer; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public Result getEventResult() { |     public Result getEventResult() { | ||||||
|         return eventResult; |         return eventResult; | ||||||
|   | |||||||
| @@ -0,0 +1,64 @@ | |||||||
|  | /* | ||||||
|  |  * PlotSquared, a land and world management plugin for Minecraft. | ||||||
|  |  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||||
|  |  * Copyright (C) IntellectualSites team and contributors | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 3 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  | package com.plotsquared.core.events.post; | ||||||
|  |  | ||||||
|  | import com.plotsquared.core.events.PlotPlayerEvent; | ||||||
|  | import com.plotsquared.core.player.OfflinePlotPlayer; | ||||||
|  | import com.plotsquared.core.player.PlotPlayer; | ||||||
|  | import com.plotsquared.core.plot.Plot; | ||||||
|  | import org.checkerframework.checker.index.qual.NonNegative; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Called after a player has successfully bought a plot. | ||||||
|  |  * | ||||||
|  |  * @since 7.3.2 | ||||||
|  |  */ | ||||||
|  | public class PostPlayerBuyPlotEvent extends PlotPlayerEvent { | ||||||
|  |  | ||||||
|  |     private final OfflinePlotPlayer previousOwner; | ||||||
|  |     private final double price; | ||||||
|  |  | ||||||
|  |     public PostPlayerBuyPlotEvent( | ||||||
|  |             final PlotPlayer<?> plotPlayer, final OfflinePlotPlayer previousOwner, final Plot plot, | ||||||
|  |             @NonNegative final double price | ||||||
|  |     ) { | ||||||
|  |         super(plotPlayer, plot); | ||||||
|  |         this.previousOwner = previousOwner; | ||||||
|  |         this.price = price; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * The previous owner of the bought plot. | ||||||
|  |      * | ||||||
|  |      * @return the previous owner. | ||||||
|  |      */ | ||||||
|  |     public OfflinePlotPlayer previousOwner() { | ||||||
|  |         return previousOwner; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Returns the price after potential modifications by {@link com.plotsquared.core.events.PlayerBuyPlotEvent}. | ||||||
|  |      * | ||||||
|  |      * @return the price the player had to pay to buy the plot. | ||||||
|  |      */ | ||||||
|  |     public double price() { | ||||||
|  |         return price; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,43 @@ | |||||||
|  | /* | ||||||
|  |  * PlotSquared, a land and world management plugin for Minecraft. | ||||||
|  |  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||||
|  |  * Copyright (C) IntellectualSites team and contributors | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 3 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  | package com.plotsquared.core.events.post; | ||||||
|  |  | ||||||
|  | import com.plotsquared.core.events.PlotPlayerEvent; | ||||||
|  | import com.plotsquared.core.player.PlotPlayer; | ||||||
|  | import com.plotsquared.core.plot.Plot; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Called after a {@link Plot} was cleared. | ||||||
|  |  * | ||||||
|  |  * @since 7.3.2 | ||||||
|  |  */ | ||||||
|  | public class PostPlotClearEvent extends PlotPlayerEvent { | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Instantiate a new PostPlotClearEvent. | ||||||
|  |      * | ||||||
|  |      * @param plotPlayer The {@link PlotPlayer} that initiated the clear. | ||||||
|  |      * @param plot       The clearing plot. | ||||||
|  |      */ | ||||||
|  |     public PostPlotClearEvent(final PlotPlayer<?> plotPlayer, final Plot plot) { | ||||||
|  |         super(plotPlayer, plot); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -143,7 +143,7 @@ public class ClassicPlotManager extends SquarePlotManager { | |||||||
|                     classicPlotWorld, |                     classicPlotWorld, | ||||||
|                     plot.getRegions(), |                     plot.getRegions(), | ||||||
|                     blocks, |                     blocks, | ||||||
|                     classicPlotWorld.getMinBuildHeight(), |                     classicPlotWorld.getMinComponentHeight(), | ||||||
|                     classicPlotWorld.getMaxBuildHeight() - 1, |                     classicPlotWorld.getMaxBuildHeight() - 1, | ||||||
|                     actor, |                     actor, | ||||||
|                     queue |                     queue | ||||||
| @@ -204,7 +204,7 @@ public class ClassicPlotManager extends SquarePlotManager { | |||||||
|                     classicPlotWorld, |                     classicPlotWorld, | ||||||
|                     plot.getRegions(), |                     plot.getRegions(), | ||||||
|                     blocks, |                     blocks, | ||||||
|                     classicPlotWorld.getMinBuildHeight(), |                     classicPlotWorld.getMinComponentHeight(), | ||||||
|                     classicPlotWorld.PLOT_HEIGHT - 1, |                     classicPlotWorld.PLOT_HEIGHT - 1, | ||||||
|                     actor, |                     actor, | ||||||
|                     queue |                     queue | ||||||
| @@ -379,7 +379,7 @@ public class ClassicPlotManager extends SquarePlotManager { | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         int yStart = classicPlotWorld.getMinBuildHeight() + (classicPlotWorld.PLOT_BEDROCK ? 1 : 0); |         int yStart = classicPlotWorld.getMinComponentHeight(); | ||||||
|         if (!plot.isMerged(Direction.NORTH)) { |         if (!plot.isMerged(Direction.NORTH)) { | ||||||
|             int z = bot.getZ(); |             int z = bot.getZ(); | ||||||
|             for (int x = bot.getX(); x < top.getX(); x++) { |             for (int x = bot.getX(); x < top.getX(); x++) { | ||||||
|   | |||||||
| @@ -52,6 +52,7 @@ public abstract class ClassicPlotWorld extends SquarePlotWorld { | |||||||
|     public BlockBucket ROAD_BLOCK = new BlockBucket(BlockTypes.QUARTZ_BLOCK); |     public BlockBucket ROAD_BLOCK = new BlockBucket(BlockTypes.QUARTZ_BLOCK); | ||||||
|     public boolean PLOT_BEDROCK = true; |     public boolean PLOT_BEDROCK = true; | ||||||
|     public boolean PLACE_TOP_BLOCK = true; |     public boolean PLACE_TOP_BLOCK = true; | ||||||
|  |     public boolean COMPONENT_BELOW_BEDROCK = false; | ||||||
|  |  | ||||||
|     public ClassicPlotWorld( |     public ClassicPlotWorld( | ||||||
|             final @NonNull String worldName, |             final @NonNull String worldName, | ||||||
| @@ -129,6 +130,9 @@ public abstract class ClassicPlotWorld extends SquarePlotWorld { | |||||||
|                 ), |                 ), | ||||||
|                 new ConfigurationNode("plot.bedrock", this.PLOT_BEDROCK, TranslatableCaption.of("setup.bedrock_boolean"), |                 new ConfigurationNode("plot.bedrock", this.PLOT_BEDROCK, TranslatableCaption.of("setup.bedrock_boolean"), | ||||||
|                         ConfigurationUtil.BOOLEAN |                         ConfigurationUtil.BOOLEAN | ||||||
|  |                 ), | ||||||
|  |                 new ConfigurationNode("world.component_below_bedrock", this.COMPONENT_BELOW_BEDROCK, TranslatableCaption.of( | ||||||
|  |                         "setup.component_below_bedrock_boolean"), ConfigurationUtil.BOOLEAN | ||||||
|                 )}; |                 )}; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -150,6 +154,14 @@ public abstract class ClassicPlotWorld extends SquarePlotWorld { | |||||||
|         this.PLACE_TOP_BLOCK = config.getBoolean("wall.place_top_block"); |         this.PLACE_TOP_BLOCK = config.getBoolean("wall.place_top_block"); | ||||||
|         this.WALL_HEIGHT = Math.min(getMaxGenHeight() - (PLACE_TOP_BLOCK ? 1 : 0), config.getInt("wall.height")); |         this.WALL_HEIGHT = Math.min(getMaxGenHeight() - (PLACE_TOP_BLOCK ? 1 : 0), config.getInt("wall.height")); | ||||||
|         this.CLAIMED_WALL_BLOCK = createCheckedBlockBucket(config.getString("wall.block_claimed"), CLAIMED_WALL_BLOCK); |         this.CLAIMED_WALL_BLOCK = createCheckedBlockBucket(config.getString("wall.block_claimed"), CLAIMED_WALL_BLOCK); | ||||||
|  |         this.COMPONENT_BELOW_BEDROCK = config.getBoolean("world.component_below_bedrock"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public int getMinComponentHeight() { | ||||||
|  |         return COMPONENT_BELOW_BEDROCK && getMinGenHeight() >= getMinBuildHeight() | ||||||
|  |                 ? getMinGenHeight() + (PLOT_BEDROCK ? 1 : 0) | ||||||
|  |                 : getMinBuildHeight(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     int schematicStartHeight() { |     int schematicStartHeight() { | ||||||
|   | |||||||
| @@ -189,7 +189,7 @@ public class HybridPlotWorld extends ClassicPlotWorld { | |||||||
|                 } |                 } | ||||||
|                 Object value; |                 Object value; | ||||||
|                 try { |                 try { | ||||||
|                     final boolean accessible = field.isAccessible(); |                     final boolean accessible = field.canAccess(this); | ||||||
|                     field.setAccessible(true); |                     field.setAccessible(true); | ||||||
|                     value = field.get(this); |                     value = field.get(this); | ||||||
|                     field.setAccessible(accessible); |                     field.setAccessible(accessible); | ||||||
|   | |||||||
| @@ -77,6 +77,10 @@ public class HybridUtils { | |||||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + HybridUtils.class.getSimpleName()); |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + HybridUtils.class.getSimpleName()); | ||||||
|     private static final BlockState AIR = BlockTypes.AIR.getDefaultState(); |     private static final BlockState AIR = BlockTypes.AIR.getDefaultState(); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Deprecated and likely to be removed in a future release. | ||||||
|  |      */ | ||||||
|  |     @Deprecated(forRemoval = true, since = "7.0.0") | ||||||
|     public static HybridUtils manager; |     public static HybridUtils manager; | ||||||
|     public static Set<BlockVector2> regions; |     public static Set<BlockVector2> regions; | ||||||
|     public static int height; |     public static int height; | ||||||
| @@ -428,6 +432,7 @@ public class HybridUtils { | |||||||
|                 if (!UPDATE) { |                 if (!UPDATE) { | ||||||
|                     Iterator<BlockVector2> iter = chunks.iterator(); |                     Iterator<BlockVector2> iter = chunks.iterator(); | ||||||
|                     QueueCoordinator queue = blockQueue.getNewQueue(worldUtil.getWeWorld(area.getWorldName())); |                     QueueCoordinator queue = blockQueue.getNewQueue(worldUtil.getWeWorld(area.getWorldName())); | ||||||
|  |                     queue.setShouldGen(false); | ||||||
|                     while (iter.hasNext()) { |                     while (iter.hasNext()) { | ||||||
|                         BlockVector2 chunk = iter.next(); |                         BlockVector2 chunk = iter.next(); | ||||||
|                         iter.remove(); |                         iter.remove(); | ||||||
| @@ -470,6 +475,7 @@ public class HybridUtils { | |||||||
|                                     Iterator<BlockVector2> iterator = chunks.iterator(); |                                     Iterator<BlockVector2> iterator = chunks.iterator(); | ||||||
|                                     if (chunks.size() >= 32) { |                                     if (chunks.size() >= 32) { | ||||||
|                                         QueueCoordinator queue = blockQueue.getNewQueue(worldUtil.getWeWorld(area.getWorldName())); |                                         QueueCoordinator queue = blockQueue.getNewQueue(worldUtil.getWeWorld(area.getWorldName())); | ||||||
|  |                                         queue.setShouldGen(false); | ||||||
|                                         for (int i = 0; i < 32; i++) { |                                         for (int i = 0; i < 32; i++) { | ||||||
|                                             final BlockVector2 chunk = iterator.next(); |                                             final BlockVector2 chunk = iterator.next(); | ||||||
|                                             iterator.remove(); |                                             iterator.remove(); | ||||||
| @@ -483,6 +489,7 @@ public class HybridUtils { | |||||||
|                                         return null; |                                         return null; | ||||||
|                                     } |                                     } | ||||||
|                                     QueueCoordinator queue = blockQueue.getNewQueue(worldUtil.getWeWorld(area.getWorldName())); |                                     QueueCoordinator queue = blockQueue.getNewQueue(worldUtil.getWeWorld(area.getWorldName())); | ||||||
|  |                                     queue.setShouldGen(false); | ||||||
|                                     while (!chunks.isEmpty()) { |                                     while (!chunks.isEmpty()) { | ||||||
|                                         final BlockVector2 chunk = iterator.next(); |                                         final BlockVector2 chunk = iterator.next(); | ||||||
|                                         iterator.remove(); |                                         iterator.remove(); | ||||||
| @@ -498,7 +505,6 @@ public class HybridUtils { | |||||||
|                                 return; |                                 return; | ||||||
|                             } |                             } | ||||||
|                         } catch (Exception e) { |                         } catch (Exception e) { | ||||||
|                             e.printStackTrace(); |  | ||||||
|                             Iterator<BlockVector2> iterator = HybridUtils.regions.iterator(); |                             Iterator<BlockVector2> iterator = HybridUtils.regions.iterator(); | ||||||
|                             BlockVector2 loc = iterator.next(); |                             BlockVector2 loc = iterator.next(); | ||||||
|                             iterator.remove(); |                             iterator.remove(); | ||||||
| @@ -506,7 +512,8 @@ public class HybridUtils { | |||||||
|                                     "Error! Could not update '{}/region/r.{}.{}.mca' (Corrupt chunk?)", |                                     "Error! Could not update '{}/region/r.{}.{}.mca' (Corrupt chunk?)", | ||||||
|                                     area.getWorldHash(), |                                     area.getWorldHash(), | ||||||
|                                     loc.getX(), |                                     loc.getX(), | ||||||
|                                     loc.getZ() |                                     loc.getZ(), | ||||||
|  |                                     e | ||||||
|                             ); |                             ); | ||||||
|                         } |                         } | ||||||
|                         TaskManager.runTaskLater(task, TaskTime.seconds(1L)); |                         TaskManager.runTaskLater(task, TaskTime.seconds(1L)); | ||||||
| @@ -554,7 +561,7 @@ public class HybridUtils { | |||||||
|                                 try { |                                 try { | ||||||
|                                     plotworld.setupSchematics(); |                                     plotworld.setupSchematics(); | ||||||
|                                 } catch (SchematicHandler.UnsupportedFormatException e) { |                                 } catch (SchematicHandler.UnsupportedFormatException e) { | ||||||
|                                     e.printStackTrace(); |                                     LOGGER.error(e); | ||||||
|                                 } |                                 } | ||||||
|                             }); |                             }); | ||||||
|                 }); |                 }); | ||||||
|   | |||||||
| @@ -40,7 +40,8 @@ public interface ChunkCoordinatorFactory { | |||||||
|             final @NonNull Consumer<Throwable> throwableConsumer, |             final @NonNull Consumer<Throwable> throwableConsumer, | ||||||
|             @Assisted("unloadAfter") final boolean unloadAfter, |             @Assisted("unloadAfter") final boolean unloadAfter, | ||||||
|             final @NonNull Collection<ProgressSubscriber> progressSubscribers, |             final @NonNull Collection<ProgressSubscriber> progressSubscribers, | ||||||
|             @Assisted("forceSync") final boolean forceSync |             @Assisted("forceSync") final boolean forceSync, | ||||||
|  |             @Assisted("shouldGen") final boolean shouldGen | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -55,7 +55,6 @@ import com.plotsquared.core.plot.flag.implementations.TitlesFlag; | |||||||
| import com.plotsquared.core.plot.flag.implementations.WeatherFlag; | import com.plotsquared.core.plot.flag.implementations.WeatherFlag; | ||||||
| import com.plotsquared.core.plot.flag.types.TimedFlag; | import com.plotsquared.core.plot.flag.types.TimedFlag; | ||||||
| import com.plotsquared.core.util.EventDispatcher; | import com.plotsquared.core.util.EventDispatcher; | ||||||
| import com.plotsquared.core.util.PlayerManager; |  | ||||||
| import com.plotsquared.core.util.task.TaskManager; | import com.plotsquared.core.util.task.TaskManager; | ||||||
| import com.plotsquared.core.util.task.TaskTime; | import com.plotsquared.core.util.task.TaskTime; | ||||||
| import com.sk89q.worldedit.world.gamemode.GameMode; | import com.sk89q.worldedit.world.gamemode.GameMode; | ||||||
| @@ -63,7 +62,6 @@ import com.sk89q.worldedit.world.gamemode.GameModes; | |||||||
| import com.sk89q.worldedit.world.item.ItemType; | import com.sk89q.worldedit.world.item.ItemType; | ||||||
| import com.sk89q.worldedit.world.item.ItemTypes; | import com.sk89q.worldedit.world.item.ItemTypes; | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| import net.kyori.adventure.text.ComponentLike; |  | ||||||
| import net.kyori.adventure.text.minimessage.MiniMessage; | import net.kyori.adventure.text.minimessage.MiniMessage; | ||||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | import net.kyori.adventure.text.minimessage.tag.Tag; | ||||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| @@ -77,6 +75,7 @@ import java.util.List; | |||||||
| import java.util.Map; | import java.util.Map; | ||||||
| import java.util.Optional; | import java.util.Optional; | ||||||
| import java.util.UUID; | import java.util.UUID; | ||||||
|  | import java.util.concurrent.CompletableFuture; | ||||||
|  |  | ||||||
| public class PlotListener { | public class PlotListener { | ||||||
|  |  | ||||||
| @@ -107,6 +106,10 @@ public class PlotListener { | |||||||
|                             iterator.remove(); |                             iterator.remove(); | ||||||
|                             continue; |                             continue; | ||||||
|                         } |                         } | ||||||
|  |                         // Don't attempt to heal dead players - they will get stuck in the abyss (#4406) | ||||||
|  |                         if (PlotSquared.platform().worldUtil().getHealth(player) <= 0) { | ||||||
|  |                             continue; | ||||||
|  |                         } | ||||||
|                         double level = PlotSquared.platform().worldUtil().getHealth(player); |                         double level = PlotSquared.platform().worldUtil().getHealth(player); | ||||||
|                         if (level != value.max) { |                         if (level != value.max) { | ||||||
|                             PlotSquared.platform().worldUtil().setHealth(player, Math.min(level + value.amount, value.max)); |                             PlotSquared.platform().worldUtil().setHealth(player, Math.min(level + value.amount, value.max)); | ||||||
| @@ -321,22 +324,27 @@ public class PlotListener { | |||||||
|                         } |                         } | ||||||
|                         if ((lastPlot != null) && plot.getId().equals(lastPlot.getId()) && plot.hasOwner()) { |                         if ((lastPlot != null) && plot.getId().equals(lastPlot.getId()) && plot.hasOwner()) { | ||||||
|                             final UUID plotOwner = plot.getOwnerAbs(); |                             final UUID plotOwner = plot.getOwnerAbs(); | ||||||
|                             ComponentLike owner = PlayerManager.resolveName(plotOwner, true).toComponent(player); |  | ||||||
|                             Caption header = fromFlag ? StaticCaption.of(title) : TranslatableCaption.of("titles" + |                             Caption header = fromFlag ? StaticCaption.of(title) : TranslatableCaption.of("titles" + | ||||||
|                                     ".title_entered_plot"); |                                     ".title_entered_plot"); | ||||||
|                             Caption subHeader = fromFlag ? StaticCaption.of(subtitle) : TranslatableCaption.of("titles" + |                             Caption subHeader = fromFlag ? StaticCaption.of(subtitle) : TranslatableCaption.of("titles" + | ||||||
|                                     ".title_entered_plot_sub"); |                                     ".title_entered_plot_sub"); | ||||||
|                             TagResolver resolver = TagResolver.builder() |  | ||||||
|                                     .tag("plot", Tag.inserting(Component.text(lastPlot.getId().toString()))) |                             CompletableFuture<TagResolver> future = PlotSquared.platform().playerManager() | ||||||
|                                     .tag("world", Tag.inserting(Component.text(player.getLocation().getWorldName()))) |                                     .getUsernameCaption(plotOwner).thenApply(caption -> TagResolver.builder() | ||||||
|                                     .tag("owner", Tag.inserting(owner)) |                                             .tag("owner", Tag.inserting(caption.toComponent(player))) | ||||||
|                                     .tag("alias", Tag.inserting(Component.text(plot.getAlias()))) |                                             .tag("plot", Tag.inserting(Component.text(lastPlot.getId().toString()))) | ||||||
|                                     .build(); |                                             .tag("world", Tag.inserting(Component.text(player.getLocation().getWorldName()))) | ||||||
|                             if (Settings.Titles.TITLES_AS_ACTIONBAR) { |                                             .tag("alias", Tag.inserting(Component.text(plot.getAlias()))) | ||||||
|                                 player.sendActionBar(header, resolver); |                                             .build() | ||||||
|                             } else { |                                     ); | ||||||
|                                 player.sendTitle(header, subHeader, resolver); |  | ||||||
|                             } |                             future.whenComplete((tagResolver, throwable) -> { | ||||||
|  |                                 if (Settings.Titles.TITLES_AS_ACTIONBAR) { | ||||||
|  |                                     player.sendActionBar(header, tagResolver); | ||||||
|  |                                 } else { | ||||||
|  |                                     player.sendTitle(header, subHeader, tagResolver); | ||||||
|  |                                 } | ||||||
|  |                             }); | ||||||
|                         } |                         } | ||||||
|                     }, TaskTime.seconds(1L)); |                     }, TaskTime.seconds(1L)); | ||||||
|                 } |                 } | ||||||
| @@ -360,7 +368,6 @@ public class PlotListener { | |||||||
|     public boolean plotExit(final PlotPlayer<?> player, Plot plot) { |     public boolean plotExit(final PlotPlayer<?> player, Plot plot) { | ||||||
|         try (final MetaDataAccess<Plot> lastPlot = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { |         try (final MetaDataAccess<Plot> lastPlot = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { | ||||||
|             final Plot previous = lastPlot.remove(); |             final Plot previous = lastPlot.remove(); | ||||||
|             this.eventDispatcher.callLeave(player, plot); |  | ||||||
|  |  | ||||||
|             List<StatusEffect> effects = playerEffects.remove(player.getUUID()); |             List<StatusEffect> effects = playerEffects.remove(player.getUUID()); | ||||||
|             if (effects != null) { |             if (effects != null) { | ||||||
| @@ -463,6 +470,8 @@ public class PlotListener { | |||||||
|                 feedRunnable.remove(player.getUUID()); |                 feedRunnable.remove(player.getUUID()); | ||||||
|                 healRunnable.remove(player.getUUID()); |                 healRunnable.remove(player.getUUID()); | ||||||
|             } |             } | ||||||
|  |         } finally { | ||||||
|  |             this.eventDispatcher.callLeave(player, plot); | ||||||
|         } |         } | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -55,6 +55,25 @@ public enum Direction { | |||||||
|         return NORTH; |         return NORTH; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * {@return the opposite direction} | ||||||
|  |      * If this is {@link Direction#ALL}, then {@link Direction#ALL} is returned. | ||||||
|  |      * @since 7.2.0 | ||||||
|  |      */ | ||||||
|  |     public Direction opposite() { | ||||||
|  |         return switch (this) { | ||||||
|  |             case ALL -> ALL; | ||||||
|  |             case NORTH -> SOUTH; | ||||||
|  |             case EAST -> WEST; | ||||||
|  |             case SOUTH -> NORTH; | ||||||
|  |             case WEST -> EAST; | ||||||
|  |             case NORTHEAST -> SOUTHWEST; | ||||||
|  |             case SOUTHEAST -> NORTHWEST; | ||||||
|  |             case SOUTHWEST -> NORTHEAST; | ||||||
|  |             case NORTHWEST -> SOUTHEAST; | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public int getIndex() { |     public int getIndex() { | ||||||
|         return index; |         return index; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -59,6 +59,9 @@ public enum Permission implements ComponentLike { | |||||||
|     PERMISSION_ADMIN_DESTROY_VEHICLE_UNOWNED("plots.admin.vehicle.break.unowned"), |     PERMISSION_ADMIN_DESTROY_VEHICLE_UNOWNED("plots.admin.vehicle.break.unowned"), | ||||||
|     PERMISSION_ADMIN_DESTROY_VEHICLE_OTHER("plots.admin.vehicle.break.other"), |     PERMISSION_ADMIN_DESTROY_VEHICLE_OTHER("plots.admin.vehicle.break.other"), | ||||||
|     PERMISSION_ADMIN_PVE("plots.admin.pve"), |     PERMISSION_ADMIN_PVE("plots.admin.pve"), | ||||||
|  |     PERMISSION_ADMIN_PLACE_VEHICLE_ROAD("plots.admin.vehicle.place.road"), | ||||||
|  |     PERMISSION_ADMIN_PLACE_VEHICLE_UNOWNED("plots.admin.vehicle.place.unowned"), | ||||||
|  |     PERMISSION_ADMIN_PLACE_VEHICLE_OTHER("plots.admin.vehicle.place.other"), | ||||||
|     PERMISSION_ADMIN_PVP("plots.admin.pvp"), |     PERMISSION_ADMIN_PVP("plots.admin.pvp"), | ||||||
|     PERMISSION_ADMIN_BUILD_ROAD("plots.admin.build.road"), |     PERMISSION_ADMIN_BUILD_ROAD("plots.admin.build.road"), | ||||||
|     PERMISSION_ADMIN_PROJECTILE_ROAD("plots.admin.projectile.road"), |     PERMISSION_ADMIN_PROJECTILE_ROAD("plots.admin.projectile.road"), | ||||||
|   | |||||||
| @@ -100,6 +100,7 @@ public interface PermissionHolder { | |||||||
|         } |         } | ||||||
|         String[] nodes = stub.split("\\."); |         String[] nodes = stub.split("\\."); | ||||||
|         StringBuilder builder = new StringBuilder(); |         StringBuilder builder = new StringBuilder(); | ||||||
|  |         // Wildcard check from less specific permission to more specific permission | ||||||
|         for (int i = 0; i < (nodes.length - 1); i++) { |         for (int i = 0; i < (nodes.length - 1); i++) { | ||||||
|             builder.append(nodes[i]).append("."); |             builder.append(nodes[i]).append("."); | ||||||
|             if (!stub.equals(builder + Permission.PERMISSION_STAR.toString())) { |             if (!stub.equals(builder + Permission.PERMISSION_STAR.toString())) { | ||||||
| @@ -108,6 +109,7 @@ public interface PermissionHolder { | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |         // Wildcard check for the full permission | ||||||
|         if (hasPermission(stub + ".*")) { |         if (hasPermission(stub + ".*")) { | ||||||
|             return Integer.MAX_VALUE; |             return Integer.MAX_VALUE; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -0,0 +1,126 @@ | |||||||
|  | /* | ||||||
|  |  * PlotSquared, a land and world management plugin for Minecraft. | ||||||
|  |  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||||
|  |  * Copyright (C) IntellectualSites team and contributors | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 3 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  | package com.plotsquared.core.permissions; | ||||||
|  |  | ||||||
|  | import com.plotsquared.core.player.PlotPlayer; | ||||||
|  | import org.checkerframework.checker.index.qual.NonNegative; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  |  | ||||||
|  | import java.util.stream.IntStream; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Represents a resolver for ranged permissions. Return values depend on the actual implementation (see Bukkit module). | ||||||
|  |  * <br> | ||||||
|  |  * Even though this interface is not linked to platform implementations by design, implementation-specific details are added to | ||||||
|  |  * the Javadocs. | ||||||
|  |  * | ||||||
|  |  * @since TODO | ||||||
|  |  */ | ||||||
|  | public interface RangedPermissionResolver { | ||||||
|  |  | ||||||
|  |     int INFINITE_RANGE_VALUE = Integer.MAX_VALUE; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Gets the applicable range value of a player for a specific permission stub | ||||||
|  |      * ({@code plots.plot} would check for {@code plots.plot.<numeric>}). | ||||||
|  |      * <br> | ||||||
|  |      * The standard bukkit implementation would return the lowest numeric value, while the LuckPerms specific resolver would | ||||||
|  |      * try returning the highest possible value. | ||||||
|  |      * | ||||||
|  |      * @param player       the permission holder | ||||||
|  |      * @param stub         the permission stub to check against | ||||||
|  |      * @param worldContext the optional world context of the action requiring the range | ||||||
|  |      * @param range        the maximum permission range to check against (for the default bukkit resolver) | ||||||
|  |      * @return the applicable range value of the player for the given permission stub | ||||||
|  |      * @since TODO | ||||||
|  |      */ | ||||||
|  |     @NonNegative | ||||||
|  |     int getPermissionRange( | ||||||
|  |             final @NonNull PlotPlayer<?> player, | ||||||
|  |             final @NonNull String stub, | ||||||
|  |             final @Nullable String worldContext, | ||||||
|  |             final @NonNegative int range | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Gets a stream of all applicable permission range values for the given stub. The stream is unordered by default. If a | ||||||
|  |      * specific order is needed, use the stateful {@link IntStream#sorted()} operation. | ||||||
|  |      * <br> | ||||||
|  |      * The standard bukkit implementation will only return a stream containing a single value equal to | ||||||
|  |      * {@link #getPermissionRange(PlotPlayer, String, String, int)}. For LuckPerms, all applicable node values will be in the | ||||||
|  |      * stream. | ||||||
|  |      * | ||||||
|  |      * @param player       the permission holder | ||||||
|  |      * @param stub         the permission stub to check against | ||||||
|  |      * @param worldContext the optional world context of the action requiring the range | ||||||
|  |      * @param range        the maximum permission range to check against (for the default bukkit resolver) | ||||||
|  |      * @return a stream of all applicable permission range values for the given stub | ||||||
|  |      * @since TODO | ||||||
|  |      */ | ||||||
|  |     @NonNull | ||||||
|  |     IntStream streamFullPermissionRange( | ||||||
|  |             final @NonNull PlotPlayer<?> player, | ||||||
|  |             final @NonNull String stub, | ||||||
|  |             final @Nullable String worldContext, | ||||||
|  |             final @NonNegative int range | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Checks if the given player has a wildcard range for the given permission stub. | ||||||
|  |      * <br> | ||||||
|  |      * For example, if checking for the stub {@code plots.plot}, this method would check for: | ||||||
|  |      * <ul> | ||||||
|  |      *     <li>{@code *}</li> | ||||||
|  |      *     <li>{@code plots.admin}</li> | ||||||
|  |      *     <li>{@code plots.plot.*}</li> | ||||||
|  |      *     <li>{@code plots.*}</li> | ||||||
|  |      * </ul> | ||||||
|  |      * | ||||||
|  |      * @param player       the permission holder | ||||||
|  |      * @param stub         the permission stub to check against | ||||||
|  |      * @param worldContext the optional world context of the action requiring the range | ||||||
|  |      * @return {@code true} if the player has a wildcard range for the given permission stub, else {@code false} | ||||||
|  |      * @since TODO | ||||||
|  |      */ | ||||||
|  |     default boolean hasWildcardRange( | ||||||
|  |             final @NonNull PlotPlayer<?> player, | ||||||
|  |             final @NonNull String stub, | ||||||
|  |             final @Nullable String worldContext | ||||||
|  |     ) { | ||||||
|  |         if (player.hasPermission(Permission.PERMISSION_STAR) || | ||||||
|  |                 player.hasPermission(Permission.PERMISSION_ADMIN) || | ||||||
|  |                 player.hasPermission(worldContext, stub + ".*")) { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |         String node = stub; | ||||||
|  |         while (true) { | ||||||
|  |             int lastIndex = node.lastIndexOf('.'); | ||||||
|  |             if (lastIndex == -1) { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             node = node.substring(0, lastIndex); | ||||||
|  |             if (player.hasPermission(worldContext, node + ".*")) { | ||||||
|  |                 return true; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -25,9 +25,9 @@ import java.util.UUID; | |||||||
| public interface OfflinePlotPlayer extends PermissionHolder { | public interface OfflinePlotPlayer extends PermissionHolder { | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Gets the {@code UUID} of this player |      * Returns the UUID of the player. | ||||||
|      * |      * | ||||||
|      * @return the player {@link UUID} |      * @return the UUID of the player | ||||||
|      */ |      */ | ||||||
|     UUID getUUID(); |     UUID getUUID(); | ||||||
|  |  | ||||||
| @@ -39,9 +39,9 @@ public interface OfflinePlotPlayer extends PermissionHolder { | |||||||
|     long getLastPlayed(); |     long getLastPlayed(); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Gets the name of this player. |      * Returns the name of the player. | ||||||
|      * |      * | ||||||
|      * @return the player name |      * @return the name of the player | ||||||
|      */ |      */ | ||||||
|     String getName(); |     String getName(); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -80,6 +80,7 @@ import java.util.Map; | |||||||
| import java.util.Queue; | import java.util.Queue; | ||||||
| import java.util.Set; | import java.util.Set; | ||||||
| import java.util.UUID; | import java.util.UUID; | ||||||
|  | import java.util.concurrent.CompletableFuture; | ||||||
| import java.util.concurrent.ConcurrentHashMap; | import java.util.concurrent.ConcurrentHashMap; | ||||||
| import java.util.concurrent.atomic.AtomicInteger; | import java.util.concurrent.atomic.AtomicInteger; | ||||||
|  |  | ||||||
| @@ -273,8 +274,9 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|         return this.meta == null ? null : this.meta.remove(key); |         return this.meta == null ? null : this.meta.remove(key); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * This player's name. |      * Returns the name of the player. | ||||||
|      * |      * | ||||||
|      * @return the name of the player |      * @return the name of the player | ||||||
|      */ |      */ | ||||||
| @@ -304,7 +306,8 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|      * @return number of allowed plots within the scope (globally, or in the player's current world as defined in the settings.yml) |      * @return number of allowed plots within the scope (globally, or in the player's current world as defined in the settings.yml) | ||||||
|      */ |      */ | ||||||
|     public int getAllowedPlots() { |     public int getAllowedPlots() { | ||||||
|         return hasPermissionRange("plots.plot", Settings.Limit.MAX_PLOTS); |         final int calculatedLimit = hasPermissionRange("plots.plot", Settings.Limit.MAX_PLOTS); | ||||||
|  |         return this.eventDispatcher.callPlayerPlotLimit(this, calculatedLimit).limit(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -441,7 +444,7 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get this player's UUID. |      * Get this player's UUID. | ||||||
|      * === !IMPORTANT ===<br> |      * <p>=== !IMPORTANT ===</p> | ||||||
|      * The UUID is dependent on the mode chosen in the settings.yml and may not be the same as Bukkit has |      * The UUID is dependent on the mode chosen in the settings.yml and may not be the same as Bukkit has | ||||||
|      * (especially if using an old version of Bukkit that does not support UUIDs) |      * (especially if using an old version of Bukkit that does not support UUIDs) | ||||||
|      * |      * | ||||||
| @@ -880,7 +883,7 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|         final Component titleComponent = MiniMessage.miniMessage().deserialize(title.getComponent(this), replacements); |         final Component titleComponent = MiniMessage.miniMessage().deserialize(title.getComponent(this), replacements); | ||||||
|         final Component subtitleComponent = |         final Component subtitleComponent = | ||||||
|                 MiniMessage.miniMessage().deserialize(subtitle.getComponent(this), replacements); |                 MiniMessage.miniMessage().deserialize(subtitle.getComponent(this), replacements); | ||||||
|         final Title.Times times = Title.Times.of( |         final Title.Times times = Title.Times.times( | ||||||
|                 Duration.of(Settings.Titles.TITLES_FADE_IN * 50L, ChronoUnit.MILLIS), |                 Duration.of(Settings.Titles.TITLES_FADE_IN * 50L, ChronoUnit.MILLIS), | ||||||
|                 Duration.of(Settings.Titles.TITLES_STAY * 50L, ChronoUnit.MILLIS), |                 Duration.of(Settings.Titles.TITLES_STAY * 50L, ChronoUnit.MILLIS), | ||||||
|                 Duration.of(Settings.Titles.TITLES_FADE_OUT * 50L, ChronoUnit.MILLIS) |                 Duration.of(Settings.Titles.TITLES_FADE_OUT * 50L, ChronoUnit.MILLIS) | ||||||
| @@ -952,6 +955,54 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Sends a message to the command caller, when the future is resolved | ||||||
|  |      * | ||||||
|  |      * @param caption          Caption to send | ||||||
|  |      * @param asyncReplacement Async variable replacement | ||||||
|  |      * @return A Future to be resolved, after the message was sent | ||||||
|  |      * @since 7.1.0 | ||||||
|  |      */ | ||||||
|  |     public final CompletableFuture<Void> sendMessage( | ||||||
|  |             @NonNull Caption caption, | ||||||
|  |             CompletableFuture<@NonNull TagResolver> asyncReplacement | ||||||
|  |     ) { | ||||||
|  |         return sendMessage(caption, new CompletableFuture[]{asyncReplacement}); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Sends a message to the command caller, when all futures are resolved | ||||||
|  |      * | ||||||
|  |      * @param caption           Caption to send | ||||||
|  |      * @param asyncReplacements Async variable replacements | ||||||
|  |      * @param replacements      Sync variable replacements | ||||||
|  |      * @return A Future to be resolved, after the message was sent | ||||||
|  |      * @since 7.1.0 | ||||||
|  |      */ | ||||||
|  |     public final CompletableFuture<Void> sendMessage( | ||||||
|  |             @NonNull Caption caption, | ||||||
|  |             CompletableFuture<@NonNull TagResolver>[] asyncReplacements, | ||||||
|  |             @NonNull TagResolver... replacements | ||||||
|  |     ) { | ||||||
|  |         return CompletableFuture.allOf(asyncReplacements).whenComplete((unused, throwable) -> { | ||||||
|  |             Set<TagResolver> resolvers = new HashSet<>(Arrays.asList(replacements)); | ||||||
|  |             if (throwable != null) { | ||||||
|  |                 sendMessage( | ||||||
|  |                         TranslatableCaption.of("errors.error"), | ||||||
|  |                         TagResolver.resolver("value", Tag.inserting( | ||||||
|  |                                 Component.text("Failed to resolve asynchronous caption replacements") | ||||||
|  |                         )) | ||||||
|  |                 ); | ||||||
|  |                 LOGGER.error("Failed to resolve asynchronous tagresolver(s) for " + caption, throwable); | ||||||
|  |             } else { | ||||||
|  |                 for (final CompletableFuture<TagResolver> asyncReplacement : asyncReplacements) { | ||||||
|  |                     resolvers.add(asyncReplacement.join()); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             sendMessage(caption, resolvers.toArray(TagResolver[]::new)); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     // Redefine from PermissionHolder as it's required from CommandCaller |     // Redefine from PermissionHolder as it's required from CommandCaller | ||||||
|     @Override |     @Override | ||||||
|     public boolean hasPermission(@NonNull String permission) { |     public boolean hasPermission(@NonNull String permission) { | ||||||
|   | |||||||
| @@ -29,6 +29,7 @@ import com.plotsquared.core.configuration.caption.CaptionUtility; | |||||||
| import com.plotsquared.core.configuration.caption.StaticCaption; | import com.plotsquared.core.configuration.caption.StaticCaption; | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.database.DBFunc; | import com.plotsquared.core.database.DBFunc; | ||||||
|  | import com.plotsquared.core.events.PlayerTeleportToPlotEvent; | ||||||
| import com.plotsquared.core.events.Result; | import com.plotsquared.core.events.Result; | ||||||
| import com.plotsquared.core.events.TeleportCause; | import com.plotsquared.core.events.TeleportCause; | ||||||
| import com.plotsquared.core.generator.ClassicPlotWorld; | import com.plotsquared.core.generator.ClassicPlotWorld; | ||||||
| @@ -85,6 +86,7 @@ import java.util.ArrayDeque; | |||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
| import java.util.Collections; | import java.util.Collections; | ||||||
|  | import java.util.Deque; | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.HashSet; | import java.util.HashSet; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| @@ -641,35 +643,22 @@ public class Plot { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Gets a immutable set of owner UUIDs for a plot (supports multi-owner mega-plots). |      * Gets an immutable set of owner UUIDs for a plot (supports multi-owner mega-plots). | ||||||
|      * <p> |      * <p> | ||||||
|      * This method cannot be used to add or remove owners from a plot. |      * This method cannot be used to add or remove owners from a plot. | ||||||
|      * </p> |      * </p> | ||||||
|      * |      * | ||||||
|      * @return Immutable view of plot owners |      * @return Immutable set of plot owners | ||||||
|      */ |      */ | ||||||
|     public @NonNull Set<UUID> getOwners() { |     public @NonNull Set<UUID> getOwners() { | ||||||
|         if (this.getOwner() == null) { |         ImmutableSet.Builder<UUID> owners = ImmutableSet.builder(); | ||||||
|             return ImmutableSet.of(); |         for (Plot plot : getConnectedPlots()) { | ||||||
|         } |             UUID owner = plot.getOwner(); | ||||||
|         if (isMerged()) { |             if (owner != null) { | ||||||
|             Set<Plot> plots = getConnectedPlots(); |                 owners.add(owner); | ||||||
|             Plot[] array = plots.toArray(new Plot[0]); |  | ||||||
|             ImmutableSet.Builder<UUID> owners = ImmutableSet.builder(); |  | ||||||
|             UUID last = this.getOwner(); |  | ||||||
|             owners.add(this.getOwner()); |  | ||||||
|             for (final Plot current : array) { |  | ||||||
|                 if (current.getOwner() == null) { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (last == null || current.getOwner().getMostSignificantBits() != last.getMostSignificantBits()) { |  | ||||||
|                     owners.add(current.getOwner()); |  | ||||||
|                     last = current.getOwner(); |  | ||||||
|                 } |  | ||||||
|             } |             } | ||||||
|             return owners.build(); |  | ||||||
|         } |         } | ||||||
|         return ImmutableSet.of(this.getOwner()); |         return owners.build(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -1418,6 +1407,9 @@ public class Plot { | |||||||
|                 ); |                 ); | ||||||
|             } |             } | ||||||
|             Location location = toHomeLocation(bottom, home); |             Location location = toHomeLocation(bottom, home); | ||||||
|  |             if (Settings.Teleport.SIZED_BASED && this.worldUtil.isSmallBlock(location) && this.worldUtil.isSmallBlock(location.add(0,1,0))) { | ||||||
|  |                 return location; | ||||||
|  |             } | ||||||
|             if (!this.worldUtil.getBlockSynchronous(location).getBlockType().getMaterial().isAir()) { |             if (!this.worldUtil.getBlockSynchronous(location).getBlockType().getMaterial().isAir()) { | ||||||
|                 location = location.withY( |                 location = location.withY( | ||||||
|                         Math.max(1 + this.worldUtil.getHighestBlockSynchronous( |                         Math.max(1 + this.worldUtil.getHighestBlockSynchronous( | ||||||
| @@ -1451,15 +1443,21 @@ public class Plot { | |||||||
|             } |             } | ||||||
|             Location bottom = this.getBottomAbs(); |             Location bottom = this.getBottomAbs(); | ||||||
|             Location location = toHomeLocation(bottom, home); |             Location location = toHomeLocation(bottom, home); | ||||||
|             this.worldUtil.getBlock(location, block -> { |             if (Settings.Teleport.SIZED_BASED && this.worldUtil.isSmallBlock(location) && this.worldUtil.isSmallBlock(location.add(0,1,0))) { | ||||||
|                 if (!block.getBlockType().getMaterial().isAir()) { |                 result.accept(location); | ||||||
|                     this.worldUtil.getHighestBlock(this.getWorldName(), location.getX(), location.getZ(), |             } else { | ||||||
|                             y -> result.accept(location.withY(Math.max(1 + y, bottom.getY()))) |                 this.worldUtil.getBlock(location, block -> { | ||||||
|                     ); |  | ||||||
|                 } else { |                     if (!block.getBlockType().getMaterial().isAir()) { | ||||||
|                     result.accept(location); |                         this.worldUtil.getHighestBlock(this.getWorldName(), location.getX(), location.getZ(), | ||||||
|                 } |                                 y -> result.accept(location.withY(Math.max(1 + y, bottom.getY()))) | ||||||
|             }); |                         ); | ||||||
|  |                     } else { | ||||||
|  |                         result.accept(location); | ||||||
|  |                     } | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -1481,7 +1479,7 @@ public class Plot { | |||||||
|      */ |      */ | ||||||
|     public void setHome(BlockLoc location) { |     public void setHome(BlockLoc location) { | ||||||
|         Plot plot = this.getBasePlot(false); |         Plot plot = this.getBasePlot(false); | ||||||
|         if (BlockLoc.ZERO.equals(location) || BlockLoc.MINY.equals(location)) { |         if (location != null && (BlockLoc.ZERO.equals(location) || BlockLoc.MINY.equals(location))) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         plot.getSettings().setPosition(location); |         plot.getSettings().setPosition(location); | ||||||
| @@ -1719,6 +1717,7 @@ public class Plot { | |||||||
|         } |         } | ||||||
|         player.sendMessage( |         player.sendMessage( | ||||||
|                 TranslatableCaption.of("working.claimed"), |                 TranslatableCaption.of("working.claimed"), | ||||||
|  |                 TagResolver.resolver("world", Tag.inserting(Component.text(this.getWorldName()))), | ||||||
|                 TagResolver.resolver("plot", Tag.inserting(Component.text(this.getId().toString()))) |                 TagResolver.resolver("plot", Tag.inserting(Component.text(this.getId().toString()))) | ||||||
|         ); |         ); | ||||||
|         if (teleport) { |         if (teleport) { | ||||||
| @@ -2193,6 +2192,9 @@ public class Plot { | |||||||
|      * @return if the given player can claim the plot |      * @return if the given player can claim the plot | ||||||
|      */ |      */ | ||||||
|     public boolean canClaim(@NonNull PlotPlayer<?> player) { |     public boolean canClaim(@NonNull PlotPlayer<?> player) { | ||||||
|  |         if (!WorldUtil.isValidLocation(getBottomAbs())) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|         PlotCluster cluster = this.getCluster(); |         PlotCluster cluster = this.getCluster(); | ||||||
|         if (cluster != null) { |         if (cluster != null) { | ||||||
|             if (!cluster.isAdded(player.getUUID()) && !player.hasPermission("plots.admin.command.claim")) { |             if (!cluster.isAdded(player.getUUID()) && !player.hasPermission("plots.admin.command.claim")) { | ||||||
| @@ -2283,8 +2285,8 @@ public class Plot { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Gets a set of plots connected (and including) this plot<br> |      * Gets a set of plots connected (and including) this plot. | ||||||
|      * - This result is cached globally |      * The returned set is immutable. | ||||||
|      * |      * | ||||||
|      * @return a Set of Plots connected to this Plot |      * @return a Set of Plots connected to this Plot | ||||||
|      */ |      */ | ||||||
| @@ -2295,117 +2297,75 @@ public class Plot { | |||||||
|         if (!this.isMerged()) { |         if (!this.isMerged()) { | ||||||
|             return Collections.singleton(this); |             return Collections.singleton(this); | ||||||
|         } |         } | ||||||
|  |         Plot basePlot = getBasePlot(false); | ||||||
|  |         if (this.connectedCache == null && this != basePlot) { | ||||||
|  |             // share cache between connected plots | ||||||
|  |             Set<Plot> connectedPlots = basePlot.getConnectedPlots(); | ||||||
|  |             this.connectedCache = connectedPlots; | ||||||
|  |             return connectedPlots; | ||||||
|  |         } | ||||||
|         if (this.connectedCache != null && this.connectedCache.contains(this)) { |         if (this.connectedCache != null && this.connectedCache.contains(this)) { | ||||||
|             return this.connectedCache; |             return this.connectedCache; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         HashSet<Plot> tmpSet = new HashSet<>(); |         Set<Plot> tmpSet = new HashSet<>(); | ||||||
|         tmpSet.add(this); |         tmpSet.add(this); | ||||||
|         Plot tmp; |         HashSet<Plot> queueCache = new HashSet<>(); | ||||||
|         HashSet<Object> queuecache = new HashSet<>(); |  | ||||||
|         ArrayDeque<Plot> frontier = new ArrayDeque<>(); |         ArrayDeque<Plot> frontier = new ArrayDeque<>(); | ||||||
|         if (this.isMerged(Direction.NORTH)) { |         computeDirectMerged(queueCache, frontier, Direction.NORTH); | ||||||
|             tmp = this.area.getPlotAbs(this.id.getRelative(Direction.NORTH)); |         computeDirectMerged(queueCache, frontier, Direction.EAST); | ||||||
|             if (!tmp.isMerged(Direction.SOUTH)) { |         computeDirectMerged(queueCache, frontier, Direction.SOUTH); | ||||||
|                 // invalid merge |         computeDirectMerged(queueCache, frontier, Direction.WEST); | ||||||
|                 if (tmp.isOwnerAbs(this.getOwnerAbs())) { |  | ||||||
|                     tmp.getSettings().setMerged(Direction.SOUTH, true); |  | ||||||
|                     DBFunc.setMerged(tmp, tmp.getSettings().getMerged()); |  | ||||||
|                 } else { |  | ||||||
|                     this.getSettings().setMerged(Direction.NORTH, false); |  | ||||||
|                     DBFunc.setMerged(this, this.getSettings().getMerged()); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             queuecache.add(tmp); |  | ||||||
|             frontier.add(tmp); |  | ||||||
|         } |  | ||||||
|         if (this.isMerged(Direction.EAST)) { |  | ||||||
|             tmp = this.area.getPlotAbs(this.id.getRelative(Direction.EAST)); |  | ||||||
|             assert tmp != null; |  | ||||||
|             if (!tmp.isMerged(Direction.WEST)) { |  | ||||||
|                 // invalid merge |  | ||||||
|                 if (tmp.isOwnerAbs(this.getOwnerAbs())) { |  | ||||||
|                     tmp.getSettings().setMerged(Direction.WEST, true); |  | ||||||
|                     DBFunc.setMerged(tmp, tmp.getSettings().getMerged()); |  | ||||||
|                 } else { |  | ||||||
|                     this.getSettings().setMerged(Direction.EAST, false); |  | ||||||
|                     DBFunc.setMerged(this, this.getSettings().getMerged()); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             queuecache.add(tmp); |  | ||||||
|             frontier.add(tmp); |  | ||||||
|         } |  | ||||||
|         if (this.isMerged(Direction.SOUTH)) { |  | ||||||
|             tmp = this.area.getPlotAbs(this.id.getRelative(Direction.SOUTH)); |  | ||||||
|             assert tmp != null; |  | ||||||
|             if (!tmp.isMerged(Direction.NORTH)) { |  | ||||||
|                 // invalid merge |  | ||||||
|                 if (tmp.isOwnerAbs(this.getOwnerAbs())) { |  | ||||||
|                     tmp.getSettings().setMerged(Direction.NORTH, true); |  | ||||||
|                     DBFunc.setMerged(tmp, tmp.getSettings().getMerged()); |  | ||||||
|                 } else { |  | ||||||
|                     this.getSettings().setMerged(Direction.SOUTH, false); |  | ||||||
|                     DBFunc.setMerged(this, this.getSettings().getMerged()); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             queuecache.add(tmp); |  | ||||||
|             frontier.add(tmp); |  | ||||||
|         } |  | ||||||
|         if (this.isMerged(Direction.WEST)) { |  | ||||||
|             tmp = this.area.getPlotAbs(this.id.getRelative(Direction.WEST)); |  | ||||||
|             if (!tmp.isMerged(Direction.EAST)) { |  | ||||||
|                 // invalid merge |  | ||||||
|                 if (tmp.isOwnerAbs(this.getOwnerAbs())) { |  | ||||||
|                     tmp.getSettings().setMerged(Direction.EAST, true); |  | ||||||
|                     DBFunc.setMerged(tmp, tmp.getSettings().getMerged()); |  | ||||||
|                 } else { |  | ||||||
|                     this.getSettings().setMerged(Direction.WEST, false); |  | ||||||
|                     DBFunc.setMerged(this, this.getSettings().getMerged()); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             queuecache.add(tmp); |  | ||||||
|             frontier.add(tmp); |  | ||||||
|         } |  | ||||||
|         Plot current; |         Plot current; | ||||||
|         while ((current = frontier.poll()) != null) { |         while ((current = frontier.poll()) != null) { | ||||||
|             if (!current.hasOwner() || current.settings == null) { |             if (!current.hasOwner() || current.settings == null) { | ||||||
|                 continue; |                 continue; | ||||||
|             } |             } | ||||||
|             tmpSet.add(current); |             tmpSet.add(current); | ||||||
|             queuecache.remove(current); |             queueCache.remove(current); | ||||||
|             if (current.isMerged(Direction.NORTH)) { |             addIfIncluded(current, Direction.NORTH, queueCache, tmpSet, frontier); | ||||||
|                 tmp = current.area.getPlotAbs(current.id.getRelative(Direction.NORTH)); |             addIfIncluded(current, Direction.EAST, queueCache, tmpSet, frontier); | ||||||
|                 if (tmp != null && !queuecache.contains(tmp) && !tmpSet.contains(tmp)) { |             addIfIncluded(current, Direction.SOUTH, queueCache, tmpSet, frontier); | ||||||
|                     queuecache.add(tmp); |             addIfIncluded(current, Direction.WEST, queueCache, tmpSet, frontier); | ||||||
|                     frontier.add(tmp); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             if (current.isMerged(Direction.EAST)) { |  | ||||||
|                 tmp = current.area.getPlotAbs(current.id.getRelative(Direction.EAST)); |  | ||||||
|                 if (tmp != null && !queuecache.contains(tmp) && !tmpSet.contains(tmp)) { |  | ||||||
|                     queuecache.add(tmp); |  | ||||||
|                     frontier.add(tmp); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             if (current.isMerged(Direction.SOUTH)) { |  | ||||||
|                 tmp = current.area.getPlotAbs(current.id.getRelative(Direction.SOUTH)); |  | ||||||
|                 if (tmp != null && !queuecache.contains(tmp) && !tmpSet.contains(tmp)) { |  | ||||||
|                     queuecache.add(tmp); |  | ||||||
|                     frontier.add(tmp); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             if (current.isMerged(Direction.WEST)) { |  | ||||||
|                 tmp = current.area.getPlotAbs(current.id.getRelative(Direction.WEST)); |  | ||||||
|                 if (tmp != null && !queuecache.contains(tmp) && !tmpSet.contains(tmp)) { |  | ||||||
|                     queuecache.add(tmp); |  | ||||||
|                     frontier.add(tmp); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|  |         tmpSet = Set.copyOf(tmpSet); | ||||||
|         this.connectedCache = tmpSet; |         this.connectedCache = tmpSet; | ||||||
|         return tmpSet; |         return tmpSet; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private void computeDirectMerged(Set<Plot> queueCache, Deque<Plot> frontier, Direction direction) { | ||||||
|  |         if (this.isMerged(direction)) { | ||||||
|  |             Plot tmp = this.area.getPlotAbs(this.id.getRelative(direction)); | ||||||
|  |             assert tmp != null; | ||||||
|  |             if (!tmp.isMerged(direction.opposite())) { | ||||||
|  |                 // invalid merge | ||||||
|  |                 if (tmp.isOwnerAbs(this.getOwnerAbs())) { | ||||||
|  |                     tmp.getSettings().setMerged(direction.opposite(), true); | ||||||
|  |                     DBFunc.setMerged(tmp, tmp.getSettings().getMerged()); | ||||||
|  |                 } else { | ||||||
|  |                     this.getSettings().setMerged(direction, false); | ||||||
|  |                     DBFunc.setMerged(this, this.getSettings().getMerged()); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             queueCache.add(tmp); | ||||||
|  |             frontier.add(tmp); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void addIfIncluded( | ||||||
|  |             Plot current, Direction | ||||||
|  |             direction, Set<Plot> queueCache, Set<Plot> tmpSet, Deque<Plot> frontier | ||||||
|  |     ) { | ||||||
|  |         if (!current.isMerged(direction)) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         Plot tmp = current.area.getPlotAbs(current.id.getRelative(direction)); | ||||||
|  |         if (tmp != null && !queueCache.contains(tmp) && !tmpSet.contains(tmp)) { | ||||||
|  |             queueCache.add(tmp); | ||||||
|  |             frontier.add(tmp); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * This will combine each plot into effective rectangular regions<br> |      * This will combine each plot into effective rectangular regions<br> | ||||||
|      * - This result is cached globally<br> |      * - This result is cached globally<br> | ||||||
| @@ -2614,8 +2574,15 @@ public class Plot { | |||||||
|      */ |      */ | ||||||
|     public void teleportPlayer(final PlotPlayer<?> player, TeleportCause cause, Consumer<Boolean> resultConsumer) { |     public void teleportPlayer(final PlotPlayer<?> player, TeleportCause cause, Consumer<Boolean> resultConsumer) { | ||||||
|         Plot plot = this.getBasePlot(false); |         Plot plot = this.getBasePlot(false); | ||||||
|         Result result = this.eventDispatcher.callTeleport(player, player.getLocation(), plot, cause).getEventResult(); |         if ((getArea() == null || !(getArea() instanceof SinglePlotArea)) && !WorldUtil.isValidLocation(plot.getBottomAbs())) { | ||||||
|         if (result == Result.DENY) { |             // prevent from teleporting into unsafe regions | ||||||
|  |             player.sendMessage(TranslatableCaption.of("border.denied")); | ||||||
|  |             resultConsumer.accept(false); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         PlayerTeleportToPlotEvent event = this.eventDispatcher.callTeleport(player, player.getLocation(), plot, cause); | ||||||
|  |         if (event.getEventResult() == Result.DENY) { | ||||||
|             player.sendMessage( |             player.sendMessage( | ||||||
|                     TranslatableCaption.of("events.event_denied"), |                     TranslatableCaption.of("events.event_denied"), | ||||||
|                     TagResolver.resolver("value", Tag.inserting(Component.text("Teleport"))) |                     TagResolver.resolver("value", Tag.inserting(Component.text("Teleport"))) | ||||||
| @@ -2623,7 +2590,10 @@ public class Plot { | |||||||
|             resultConsumer.accept(false); |             resultConsumer.accept(false); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         final Consumer<Location> locationConsumer = location -> { |  | ||||||
|  |         final Consumer<Location> locationConsumer = calculatedLocation -> { | ||||||
|  |             Location location = event.getLocationTransformer() == null ? calculatedLocation : | ||||||
|  |                     Objects.requireNonNullElse(event.getLocationTransformer().apply(calculatedLocation), calculatedLocation); | ||||||
|             if (Settings.Teleport.DELAY == 0 || player.hasPermission("plots.teleport.delay.bypass")) { |             if (Settings.Teleport.DELAY == 0 || player.hasPermission("plots.teleport.delay.bypass")) { | ||||||
|                 player.sendMessage(TranslatableCaption.of("teleport.teleported_to_plot")); |                 player.sendMessage(TranslatableCaption.of("teleport.teleported_to_plot")); | ||||||
|                 player.teleport(location, cause); |                 player.teleport(location, cause); | ||||||
| @@ -2679,6 +2649,11 @@ public class Plot { | |||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get the maximum distance of the plot from x=0, z=0. | ||||||
|  |      * | ||||||
|  |      * @return max block distance from 0,0 | ||||||
|  |      */ | ||||||
|     public int getDistanceFromOrigin() { |     public int getDistanceFromOrigin() { | ||||||
|         Location bot = getManager().getPlotBottomLocAbs(id); |         Location bot = getManager().getPlotBottomLocAbs(id); | ||||||
|         Location top = getManager().getPlotTopLocAbs(id); |         Location top = getManager().getPlotTopLocAbs(id); | ||||||
| @@ -2692,7 +2667,7 @@ public class Plot { | |||||||
|      * Expands the world border to include this plot if it is beyond the current border. |      * Expands the world border to include this plot if it is beyond the current border. | ||||||
|      */ |      */ | ||||||
|     public void updateWorldBorder() { |     public void updateWorldBorder() { | ||||||
|         int border = this.area.getBorder(); |         int border = this.area.getBorder(false); | ||||||
|         if (border == Integer.MAX_VALUE) { |         if (border == Integer.MAX_VALUE) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -51,6 +51,8 @@ import com.plotsquared.core.util.MathMan; | |||||||
| import com.plotsquared.core.util.PlotExpression; | import com.plotsquared.core.util.PlotExpression; | ||||||
| import com.plotsquared.core.util.RegionUtil; | import com.plotsquared.core.util.RegionUtil; | ||||||
| import com.plotsquared.core.util.StringMan; | import com.plotsquared.core.util.StringMan; | ||||||
|  | import com.plotsquared.core.util.task.TaskManager; | ||||||
|  | import com.plotsquared.core.util.task.TaskTime; | ||||||
| import com.sk89q.worldedit.math.BlockVector2; | import com.sk89q.worldedit.math.BlockVector2; | ||||||
| import com.sk89q.worldedit.math.BlockVector3; | import com.sk89q.worldedit.math.BlockVector3; | ||||||
| import com.sk89q.worldedit.regions.CuboidRegion; | import com.sk89q.worldedit.regions.CuboidRegion; | ||||||
| @@ -145,6 +147,7 @@ public abstract class PlotArea implements ComponentLike { | |||||||
|     private Map<String, PlotExpression> prices = new HashMap<>(); |     private Map<String, PlotExpression> prices = new HashMap<>(); | ||||||
|     private List<String> schematics = new ArrayList<>(); |     private List<String> schematics = new ArrayList<>(); | ||||||
|     private boolean worldBorder = false; |     private boolean worldBorder = false; | ||||||
|  |     private int borderSize = 1; | ||||||
|     private boolean useEconomy = false; |     private boolean useEconomy = false; | ||||||
|     private int hash; |     private int hash; | ||||||
|     private CuboidRegion region; |     private CuboidRegion region; | ||||||
| @@ -354,6 +357,7 @@ public abstract class PlotArea implements ComponentLike { | |||||||
|         this.plotChat = config.getBoolean("chat.enabled"); |         this.plotChat = config.getBoolean("chat.enabled"); | ||||||
|         this.forcingPlotChat = config.getBoolean("chat.forced"); |         this.forcingPlotChat = config.getBoolean("chat.forced"); | ||||||
|         this.worldBorder = config.getBoolean("world.border"); |         this.worldBorder = config.getBoolean("world.border"); | ||||||
|  |         this.borderSize = config.getInt("world.border_size"); | ||||||
|         this.maxBuildHeight = config.getInt("world.max_height"); |         this.maxBuildHeight = config.getInt("world.max_height"); | ||||||
|         this.minBuildHeight = config.getInt("world.min_height"); |         this.minBuildHeight = config.getInt("world.min_height"); | ||||||
|         this.minGenHeight = config.getInt("world.min_gen_height"); |         this.minGenHeight = config.getInt("world.min_gen_height"); | ||||||
| @@ -391,6 +395,28 @@ public abstract class PlotArea implements ComponentLike { | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         this.spawnEggs = config.getBoolean("event.spawn.egg"); | ||||||
|  |         this.spawnCustom = config.getBoolean("event.spawn.custom"); | ||||||
|  |         this.spawnBreeding = config.getBoolean("event.spawn.breeding"); | ||||||
|  |  | ||||||
|  |         if (PlotSquared.get().isWeInitialised()) { | ||||||
|  |             loadFlags(config); | ||||||
|  |         } else { | ||||||
|  |             ConsolePlayer.getConsole().sendMessage( | ||||||
|  |                     TranslatableCaption.of("flags.delaying_loading_area_flags"), | ||||||
|  |                     TagResolver.resolver("area", Tag.inserting(Component.text(this.id == null ? this.worldName : this.id))) | ||||||
|  |             ); | ||||||
|  |             TaskManager.runTaskLater(() -> loadFlags(config), TaskTime.ticks(1)); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         loadConfiguration(config); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void loadFlags(ConfigurationSection config) { | ||||||
|  |         ConsolePlayer.getConsole().sendMessage( | ||||||
|  |                 TranslatableCaption.of("flags.loading_area_flags"), | ||||||
|  |                 TagResolver.resolver("area", Tag.inserting(Component.text(this.id == null ? this.worldName : this.id))) | ||||||
|  |         ); | ||||||
|         List<String> flags = config.getStringList("flags.default"); |         List<String> flags = config.getStringList("flags.default"); | ||||||
|         if (flags.isEmpty()) { |         if (flags.isEmpty()) { | ||||||
|             flags = config.getStringList("flags"); |             flags = config.getStringList("flags"); | ||||||
| @@ -411,10 +437,6 @@ public abstract class PlotArea implements ComponentLike { | |||||||
|                 TagResolver.resolver("flags", Tag.inserting(Component.text(flags.toString()))) |                 TagResolver.resolver("flags", Tag.inserting(Component.text(flags.toString()))) | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         this.spawnEggs = config.getBoolean("event.spawn.egg"); |  | ||||||
|         this.spawnCustom = config.getBoolean("event.spawn.custom"); |  | ||||||
|         this.spawnBreeding = config.getBoolean("event.spawn.breeding"); |  | ||||||
|  |  | ||||||
|         List<String> roadflags = config.getStringList("road.flags"); |         List<String> roadflags = config.getStringList("road.flags"); | ||||||
|         if (roadflags.isEmpty()) { |         if (roadflags.isEmpty()) { | ||||||
|             roadflags = new ArrayList<>(); |             roadflags = new ArrayList<>(); | ||||||
| @@ -426,14 +448,12 @@ public abstract class PlotArea implements ComponentLike { | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         this.roadFlags = roadflags.size() > 0; |         this.roadFlags = !roadflags.isEmpty(); | ||||||
|         parseFlags(this.getRoadFlagContainer(), roadflags); |         parseFlags(this.getRoadFlagContainer(), roadflags); | ||||||
|         ConsolePlayer.getConsole().sendMessage( |         ConsolePlayer.getConsole().sendMessage( | ||||||
|                 TranslatableCaption.of("flags.road_flags"), |                 TranslatableCaption.of("flags.road_flags"), | ||||||
|                 TagResolver.resolver("flags", Tag.inserting(Component.text(roadflags.toString()))) |                 TagResolver.resolver("flags", Tag.inserting(Component.text(roadflags.toString()))) | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         loadConfiguration(config); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public abstract void loadConfiguration(ConfigurationSection config); |     public abstract void loadConfiguration(ConfigurationSection config); | ||||||
| @@ -471,6 +491,7 @@ public abstract class PlotArea implements ComponentLike { | |||||||
|         options.put("event.spawn.custom", this.isSpawnCustom()); |         options.put("event.spawn.custom", this.isSpawnCustom()); | ||||||
|         options.put("event.spawn.breeding", this.isSpawnBreeding()); |         options.put("event.spawn.breeding", this.isSpawnBreeding()); | ||||||
|         options.put("world.border", this.hasWorldBorder()); |         options.put("world.border", this.hasWorldBorder()); | ||||||
|  |         options.put("world.border_size", this.getBorderSize()); | ||||||
|         options.put("home.default", "side"); |         options.put("home.default", "side"); | ||||||
|         String position = config.getString( |         String position = config.getString( | ||||||
|                 "home.nonmembers", |                 "home.nonmembers", | ||||||
| @@ -658,10 +679,8 @@ public abstract class PlotArea implements ComponentLike { | |||||||
|                     TranslatableCaption.of("height.height_limit"), |                     TranslatableCaption.of("height.height_limit"), | ||||||
|                     TagResolver.builder() |                     TagResolver.builder() | ||||||
|                             .tag("minheight", Tag.inserting(Component.text(minBuildHeight))) |                             .tag("minheight", Tag.inserting(Component.text(minBuildHeight))) | ||||||
|                             .tag( |                             .tag("maxheight", Tag.inserting(Component.text(maxBuildHeight))) | ||||||
|                                     "maxheight", |                             .build() | ||||||
|                                     Tag.inserting(Component.text(maxBuildHeight)) |  | ||||||
|                             ).build() |  | ||||||
|             ); |             ); | ||||||
|             // Return true if "failed" as the method will always be inverted otherwise |             // Return true if "failed" as the method will always be inverted otherwise | ||||||
|             return true; |             return true; | ||||||
| @@ -919,7 +938,9 @@ public abstract class PlotArea implements ComponentLike { | |||||||
|      * Get the plot border distance for a world<br> |      * Get the plot border distance for a world<br> | ||||||
|      * |      * | ||||||
|      * @return The border distance or Integer.MAX_VALUE if no border is set |      * @return The border distance or Integer.MAX_VALUE if no border is set | ||||||
|  |      * @deprecated Use {@link PlotArea#getBorder(boolean)} | ||||||
|      */ |      */ | ||||||
|  |     @Deprecated(forRemoval = true, since = "7.2.0") | ||||||
|     public int getBorder() { |     public int getBorder() { | ||||||
|         final Integer meta = (Integer) getMeta("worldBorder"); |         final Integer meta = (Integer) getMeta("worldBorder"); | ||||||
|         if (meta != null) { |         if (meta != null) { | ||||||
| @@ -933,6 +954,27 @@ public abstract class PlotArea implements ComponentLike { | |||||||
|         return Integer.MAX_VALUE; |         return Integer.MAX_VALUE; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get the plot border distance for a world, specifying whether the returned value should include the world.border-size | ||||||
|  |      * value. This is a player-traversable area, where plots cannot be claimed | ||||||
|  |      * | ||||||
|  |      * @param getExtended If the extra border given by world.border-size should be included | ||||||
|  |      * @return Border distance of Integer.MAX_VALUE if no border is set | ||||||
|  |      * @since 7.2.0 | ||||||
|  |      */ | ||||||
|  |     public int getBorder(boolean getExtended) { | ||||||
|  |         final Integer meta = (Integer) getMeta("worldBorder"); | ||||||
|  |         if (meta != null) { | ||||||
|  |             int border = meta + 1; | ||||||
|  |             if (border == 0) { | ||||||
|  |                 return Integer.MAX_VALUE; | ||||||
|  |             } else { | ||||||
|  |                 return getExtended ? border + borderSize : border; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return Integer.MAX_VALUE; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Setup the plot border for a world (usually done when the world is created). |      * Setup the plot border for a world (usually done when the world is created). | ||||||
|      */ |      */ | ||||||
| @@ -1192,6 +1234,16 @@ public abstract class PlotArea implements ComponentLike { | |||||||
|         return worldBorder; |         return worldBorder; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get the "extra border" size of the plot area. | ||||||
|  |      * | ||||||
|  |      * @return Plot area extra border size | ||||||
|  |      * @since 7.2.0 | ||||||
|  |      */ | ||||||
|  |     public int getBorderSize() { | ||||||
|  |         return borderSize; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get whether plot signs are allowed or not. |      * Get whether plot signs are allowed or not. | ||||||
|      * |      * | ||||||
| @@ -1398,6 +1450,24 @@ public abstract class PlotArea implements ComponentLike { | |||||||
|         this.defaultHome = defaultHome; |         this.defaultHome = defaultHome; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get the maximum height that changes to plot components (wall filling, air, all etc.) may operate to | ||||||
|  |      * | ||||||
|  |      * @since 7.3.4 | ||||||
|  |      */ | ||||||
|  |     public int getMaxComponentHeight() { | ||||||
|  |         return this.maxBuildHeight; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get the minimum height that changes to plot components (wall filling, air, all etc.) may operate to | ||||||
|  |      * | ||||||
|  |      * @since 7.3.4 | ||||||
|  |      */ | ||||||
|  |     public int getMinComponentHeight() { | ||||||
|  |         return this.minBuildHeight; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the maximum height players may build in. Exclusive. |      * Get the maximum height players may build in. Exclusive. | ||||||
|      */ |      */ | ||||||
|   | |||||||
| @@ -26,8 +26,8 @@ import java.util.Iterator; | |||||||
| import java.util.NoSuchElementException; | import java.util.NoSuchElementException; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Plot (X,Y) tuples for plot locations |  * The PlotId class represents a Plot's x and y coordinates within a {@link PlotArea}. PlotId x,y values do not correspond to Block locations. | ||||||
|  * within a plot area |  * A PlotId instance can be created using the {@link #of(int, int)} method or parsed from a string using the {@link #fromString(String)} method. | ||||||
|  */ |  */ | ||||||
| public final class PlotId { | public final class PlotId { | ||||||
|  |  | ||||||
| @@ -36,10 +36,10 @@ public final class PlotId { | |||||||
|     private final int hash; |     private final int hash; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * PlotId class (PlotId x,y values do not correspond to Block locations) |      * Constructs a new PlotId with the given x and y coordinates. | ||||||
|      * |      * | ||||||
|      * @param x The plot x coordinate |      * @param x the x-coordinate of the plot | ||||||
|      * @param y The plot y coordinate |      * @param y the y-coordinate of the plot | ||||||
|      */ |      */ | ||||||
|     private PlotId(final int x, final int y) { |     private PlotId(final int x, final int y) { | ||||||
|         this.x = x; |         this.x = x; | ||||||
| @@ -48,11 +48,11 @@ public final class PlotId { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Create a new plot ID instance |      * Returns a new PlotId instance with the specified x and y coordinates. | ||||||
|      * |      * | ||||||
|      * @param x The plot x coordinate |      * @param x the x-coordinate of the plot | ||||||
|      * @param y The plot y coordinate |      * @param y the y-coordinate of the plot | ||||||
|      * @return a new PlotId at x,y |      * @return a new PlotId instance with the specified x and y coordinates | ||||||
|      */ |      */ | ||||||
|     public static @NonNull PlotId of(final int x, final int y) { |     public static @NonNull PlotId of(final int x, final int y) { | ||||||
|         return new PlotId(x, y); |         return new PlotId(x, y); | ||||||
| @@ -74,10 +74,13 @@ public final class PlotId { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Attempt to parse a plot ID from a string |      * Returns a PlotId object from the given string, or null if the string is invalid. | ||||||
|  |      * The string should be in the format "x;y" where x and y are integers. | ||||||
|  |      * The string can also contain any combination of the characters ";_,." | ||||||
|  |      * as delimiters. | ||||||
|      * |      * | ||||||
|      * @param string ID string |      * @param string the string to parse | ||||||
|      * @return Plot ID, or {@code null} if none could be parsed |      * @return a PlotId object parsed from the given string, or null if the string is invalid | ||||||
|      */ |      */ | ||||||
|     public static @Nullable PlotId fromStringOrNull(final @NonNull String string) { |     public static @Nullable PlotId fromStringOrNull(final @NonNull String string) { | ||||||
|         final String[] parts = string.split("[;_,.]"); |         final String[] parts = string.split("[;_,.]"); | ||||||
| @@ -95,39 +98,39 @@ public final class PlotId { | |||||||
|         return of(x, y); |         return of(x, y); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Gets the PlotId from the HashCode<br> |      * Returns a new PlotId instance from the given hash. | ||||||
|      * Note: Only accurate for small x,z values (short) |  | ||||||
|      * |      * | ||||||
|      * @param hash ID hash |      * @param hash the hash to unpair | ||||||
|      * @return Plot ID |      * @return a new PlotId instance | ||||||
|      */ |      */ | ||||||
|     public static @NonNull PlotId unpair(final int hash) { |     public static @NonNull PlotId unpair(final int hash) { | ||||||
|         return PlotId.of(hash >> 16, hash & 0xFFFF); |         return PlotId.of(hash >> 16, hash & 0xFFFF); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the ID X component |      * Returns the x-coordinate of this Plot ID. | ||||||
|      * |      * | ||||||
|      * @return X component |      * @return the x-coordinate of this Plot ID | ||||||
|      */ |      */ | ||||||
|     public int getX() { |     public int getX() { | ||||||
|         return this.x; |         return this.x; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the ID Y component |      * Returns the y-coordinate of this Plot ID. | ||||||
|      * |      * | ||||||
|      * @return Y component |      * @return the y-coordinate of this Plot ID | ||||||
|      */ |      */ | ||||||
|     public int getY() { |     public int getY() { | ||||||
|         return this.y; |         return this.y; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the next plot ID for claiming purposes |      * Returns the next Plot ID for claiming purposes based on the current Plot ID. | ||||||
|      * |      * | ||||||
|      * @return Next plot ID |      * @return the next Plot ID | ||||||
|      */ |      */ | ||||||
|     public @NonNull PlotId getNextId() { |     public @NonNull PlotId getNextId() { | ||||||
|         final int absX = Math.abs(x); |         final int absX = Math.abs(x); | ||||||
| @@ -159,10 +162,11 @@ public final class PlotId { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the PlotId in a relative direction |      * Returns a new Plot ID in the specified relative direction based on the | ||||||
|  |      * current Plot ID. | ||||||
|      * |      * | ||||||
|      * @param direction Direction |      * @param direction the direction in which to get the relative Plot ID | ||||||
|      * @return Relative plot ID |      * @return the relative Plot ID | ||||||
|      */ |      */ | ||||||
|     public @NonNull PlotId getRelative(final @NonNull Direction direction) { |     public @NonNull PlotId getRelative(final @NonNull Direction direction) { | ||||||
|         return switch (direction) { |         return switch (direction) { | ||||||
| @@ -193,10 +197,11 @@ public final class PlotId { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get a String representation of the plot ID where the |      * Returns a string representation of this Plot ID in the format "x;y". | ||||||
|      * components are separated by ";" |  | ||||||
|      * |      * | ||||||
|      * @return {@code x + ";" + y} |      * <p> The format is {@code x + ";" + y} | ||||||
|  |      * | ||||||
|  |      * @return a string representation of this Plot ID | ||||||
|      */ |      */ | ||||||
|     @Override |     @Override | ||||||
|     public @NonNull String toString() { |     public @NonNull String toString() { | ||||||
| @@ -204,41 +209,40 @@ public final class PlotId { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get a String representation of the plot ID where the |      * Returns a string representation of this Plot ID with the specified separator. | ||||||
|      * components are separated by a specified string |      * <p> | ||||||
|  |      * The format is {@code x + separator + y} | ||||||
|      * |      * | ||||||
|      * @param separator Separator |      * @param separator the separator to use between the X and Y coordinates | ||||||
|      * @return {@code x + separator + y} |      * @return a string representation of this Plot ID with the specified separator | ||||||
|      */ |      */ | ||||||
|     public @NonNull String toSeparatedString(String separator) { |     public @NonNull String toSeparatedString(String separator) { | ||||||
|         return this.getX() + separator + this.getY(); |         return this.getX() + separator + this.getY(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get a String representation of the plot ID where the |      * Returns a string representation of this Plot ID in the format "x,y". | ||||||
|      * components are separated by "," |  | ||||||
|      * |      * | ||||||
|      * @return {@code x + "," + y} |      * @return a string representation of this Plot ID | ||||||
|      */ |      */ | ||||||
|     public @NonNull String toCommaSeparatedString() { |     public @NonNull String toCommaSeparatedString() { | ||||||
|         return this.getX() + "," + this.getY(); |         return this.getX() + "," + this.getY(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get a String representation of the plot ID where the |      * Returns a string representation of this Plot ID in the format "x_y". | ||||||
|      * components are separated by "_" |  | ||||||
|      * |      * | ||||||
|      * @return {@code x + "_" + y} |      * @return a string representation of this Plot ID | ||||||
|      */ |      */ | ||||||
|  |  | ||||||
|     public @NonNull String toUnderscoreSeparatedString() { |     public @NonNull String toUnderscoreSeparatedString() { | ||||||
|         return this.getX() + "_" + this.getY(); |         return this.getX() + "_" + this.getY(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get a String representation of the plot ID where the |      * Returns a string representation of this Plot ID in the format "x-y". | ||||||
|      * components are separated by "-" |  | ||||||
|      * |      * | ||||||
|      * @return {@code x + "-" + y} |      * @return a string representation of this Plot ID | ||||||
|      */ |      */ | ||||||
|     public @NonNull String toDashSeparatedString() { |     public @NonNull String toDashSeparatedString() { | ||||||
|         return this.getX() + "-" + this.getY(); |         return this.getX() + "-" + this.getY(); | ||||||
| @@ -250,6 +254,10 @@ public final class PlotId { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * An iterator that iterates over a range of {@link PlotId}s. | ||||||
|  |      * The range is defined by a start and end {@link PlotId}. | ||||||
|  |      */ | ||||||
|     public static final class PlotRangeIterator implements Iterator<PlotId>, Iterable<PlotId> { |     public static final class PlotRangeIterator implements Iterator<PlotId>, Iterable<PlotId> { | ||||||
|  |  | ||||||
|         private final PlotId start; |         private final PlotId start; | ||||||
| @@ -265,6 +273,13 @@ public final class PlotId { | |||||||
|             this.y = this.start.getY(); |             this.y = this.start.getY(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         /** | ||||||
|  |          * Returns a new {@link PlotRangeIterator} that iterates over the range of Plots between the specified start and end Plots (inclusive). | ||||||
|  |          * | ||||||
|  |          * @param start the starting Plot of the range | ||||||
|  |          * @param end the ending Plot of the range | ||||||
|  |          * @return a new {@link PlotRangeIterator} that iterates over the range of Plots between the specified start and end Plots (inclusive) | ||||||
|  |          */ | ||||||
|         public static PlotRangeIterator range(final @NonNull PlotId start, final @NonNull PlotId end) { |         public static PlotRangeIterator range(final @NonNull PlotId start, final @NonNull PlotId end) { | ||||||
|             return new PlotRangeIterator(start, end); |             return new PlotRangeIterator(start, end); | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -67,14 +67,25 @@ public class PlotItemStack { | |||||||
|         return this.type; |         return this.type; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Returns the number of items in this stack. | ||||||
|  |      * Valid values range from 1-255. | ||||||
|  |      * | ||||||
|  |      * @return the amount of items in this stack | ||||||
|  |      */ | ||||||
|     public int getAmount() { |     public int getAmount() { | ||||||
|         return amount; |         return amount; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Returns the given name of this stack of items. The name is displayed when | ||||||
|  |      * hovering over the item. | ||||||
|  |      * | ||||||
|  |      * @return the given name of this stack of items | ||||||
|  |      */ | ||||||
|     public String getName() { |     public String getName() { | ||||||
|         return name; |         return name; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public String[] getLore() { |     public String[] getLore() { | ||||||
|         return lore; |         return lore; | ||||||
|     } |     } | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user