mirror of
				https://github.com/IntellectualSites/PlotSquared.git
				synced 2025-10-25 07:33:44 +02:00 
			
		
		
		
	Compare commits
	
		
			841 Commits
		
	
	
		
			legacy/v5
			...
			fix/v6/mis
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 764d3b8166 | ||
|   | 59e0b4b67a | ||
|   | 5fc153d896 | ||
|   | a003836dbc | ||
|   | 42bf413528 | ||
|   | 827f46566c | ||
|   | 6f4d2f6d5a | ||
|   | 6073b96317 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 74a490f9f0 | ||
|   | 3a752db698 | ||
|   | 7cdde5a822 | ||
|   | 8b6e7b2f2b | ||
|   | 06f0b42a97 | ||
|   | 9a85080bf6 | ||
|   | d14d2caa2d | ||
|   | bcfe7c58dd | ||
|   | 289f0f3bfd | ||
|   | 6fc4005c72 | ||
|   | 8f7c2e4c02 | ||
|   | b43d08f4aa | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2c55c6a92e | ||
|   | 31f48cc7d1 | ||
|   | 4b77f1ab28 | ||
|   | 7cc38b5fa8 | ||
|   | 0c76d08b10 | ||
|   | 9c474570c8 | ||
|   | 7e1d56c849 | ||
|   | 07d0f124b4 | ||
|   | 8d6c621763 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cef7098014 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e1efa7266e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f363941719 | ||
|   | 25095f4546 | ||
|   | fd0c2050d8 | ||
|   | 07fdc94dd8 | ||
|   | b501a81e21 | ||
|   | eb334b3aac | ||
|   | a77a51b6f2 | ||
|   | 7d981bc610 | ||
|   | 3eb485e900 | ||
|   | 9f4af889f7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2559c889e2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2cf268b99c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e6fad8309c | ||
|   | 7551450cf9 | ||
|   | e042566bb7 | ||
|   | 1f26808c7b | ||
|   | a7026047d0 | ||
|   | af2613202d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3da1e9255a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9394906a79 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d881cb6084 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b9d43897af | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 76b58485a9 | ||
|   | 7002df0bc7 | ||
|   | 958aac3cb1 | ||
|   | a93402e27b | ||
|   | 92f41f43c5 | ||
|   | 392ee9fa07 | ||
|   | 8859871e89 | ||
|   | 93d99630a9 | ||
|   | 47ae79e123 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f9ad00c2c8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 566af259db | ||
|   | d9a6431078 | ||
|   | e9c9375e78 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7c99c8aabf | ||
|   | 88eb8f88a7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fd118c2c37 | ||
|   | 858b6b5471 | ||
|   | f086826942 | ||
|   | 9c84dc2bc0 | ||
|   | cba0f3ac1f | ||
|   | b9a130ab00 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 688056352b | ||
|   | 2a40f52dea | ||
|   | 3ed369de19 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | bec51401a3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 109f884b5f | ||
|   | 8f3fa419c4 | ||
|   | a9f08bc885 | ||
|   | a8923ec729 | ||
|   | 4eafd043a3 | ||
|   | 826cc8c835 | ||
|   | d1dbf777a4 | ||
|   | 823e78377b | ||
|   | 87a2e81ec3 | ||
|   | 88775334b7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e126547e97 | ||
|   | c1163cdb2e | ||
|   | 30d06b7801 | ||
|   | a5cf7bf2c2 | ||
|   | 3d4d413de8 | ||
|   | b97e843849 | ||
|   | 48c7a3e94e | ||
|   | 7263290bbe | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e99c4e3289 | ||
|   | 87cb26189f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2dae336a5f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 258ed1469b | ||
|   | 6f3eabba0e | ||
|   | 804228fa6c | ||
|   | be85708e5a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f2368f97df | ||
|   | df87bcb743 | ||
|   | 5653d0f175 | ||
|   | b022a97196 | ||
|   | cb596ee6ea | ||
|   | 4e391136c0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a6a0e1d12f | ||
|   | d2776eed5a | ||
|   | 6528c60f4d | ||
|   | 7ee46be2ac | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 13bc231091 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 55d1a40394 | ||
|   | d70f99b489 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fde9735da2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 20f0aa3656 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0b7bf2d26d | ||
|   | f10e38af63 | ||
|   | 21727ebfc7 | ||
|   | e322ee85fd | ||
|   | 541255fe7e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 20c2f36f6c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 744f7a18ae | ||
|   | 18eece0b71 | ||
|   | 4f12dcf966 | ||
|   | ee832528db | ||
|   | bb5d6c91da | ||
|   | 6d4adecb32 | ||
|   | c37b13dcb3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 086dac2ea4 | ||
|   | b2b8598b5b | ||
|   | 61e5d9f1b6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c8989b3332 | ||
|   | 9f54472dd6 | ||
|   | 115932c65a | ||
|   | 04821f777e | ||
|   | cfae804780 | ||
|   | 4568b17c54 | ||
|   | 7bc76a3824 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4b8180807c | ||
|   | d77a8c8fa4 | ||
|   | 27a8fcd739 | ||
|   | c25a0c65a2 | ||
|   | b224a8d1b8 | ||
|   | 87d134215f | ||
|   | 93e280dba4 | ||
|   | 732f79bbfc | ||
|   | 8e5e33eec2 | ||
|   | da3fb1abec | ||
|   | 62d6b21081 | ||
|   | 4caaead7ee | ||
|   | be38086f6c | ||
|   | c18b0f943c | ||
|   | f036980dc5 | ||
|   | 0e4f54e9c0 | ||
|   | 7a3d5bd737 | ||
|   | 2ab1973668 | ||
|   | 387c6e79ba | ||
|   | e3e00d34ef | ||
|   | 40e651188e | ||
|   | b2915d9bd7 | ||
|   | 6ec136a03d | ||
|   | a8fc6662e0 | ||
|   | fb8e749411 | ||
|   | 4b26a7e300 | ||
|   | 8a53b41b52 | ||
|   | daa8ffc38f | ||
|   | 34f0236e9f | ||
|   | 8f0ae7e51d | ||
|   | 2988ad6b11 | ||
|   | 252d6a2866 | ||
|   | b8c9ae2b0b | ||
|   | bfe2dda5b1 | ||
|   | 33a79595af | ||
|   | eb2848e5d7 | ||
|   | da4ae9f4f5 | ||
|   | b841a7c03b | ||
|   | 4ee5e3d134 | ||
|   | 0e05935ad4 | ||
|   | 78a775530a | ||
|   | dd36036334 | ||
|   | 3b2a04e4db | ||
|   | 5ab8d50b86 | ||
|   | 391476ba26 | ||
|   | 25a69f4ca0 | ||
|   | c8c946cc10 | ||
|   | a405611b3a | ||
|   | fd4cec39c4 | ||
|   | e7fa057abe | ||
|   | a86c06bb81 | ||
|   | 9afaead81c | ||
|   | 6e7bd0a536 | ||
|   | 89e17bb468 | ||
|   | 8fc9e0b13f | ||
|   | 6ad51bba65 | ||
|   | 9d396fa91f | ||
|   | a8c3ac6cee | ||
|   | 9a9365a04d | ||
|   | c9b012ea5b | ||
|   | 9dd35e87d6 | ||
|   | 82f0b4194f | ||
|   | 3b27b6c850 | ||
|   | 1e41734826 | ||
|   | a83a79c17e | ||
|   | 8a77d5a450 | ||
|   | 2676b76107 | ||
|   | 51ff043af9 | ||
|   | fbde60fcf9 | ||
|   | a0557112a1 | ||
|   | 6d71e0f8be | ||
|   | 62e17f5fc5 | ||
|   | 613a92eeb9 | ||
|   | 31522b6502 | ||
|   | 2e64ac6d25 | ||
|   | fe65a0c548 | ||
|   | cc60d50dd2 | ||
|   | a797d748ea | ||
|   | 2d1e3b8c03 | ||
|   | 635e18e7ed | ||
|   | b9cb6842e2 | ||
|   | 991703914c | ||
|   | fdb6367351 | ||
|   | bbee330f4f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8b8ac76852 | ||
|   | 3b5b51b40c | ||
|   | 5395ddc6a3 | ||
|   | 60c701d20b | ||
|   | eb11d9e3aa | ||
|   | 7898313b0b | ||
|   | 8e23b10f7c | ||
|   | c46647d491 | ||
|   | 013e8fcda5 | ||
|   | f4ff9c0336 | ||
|   | 7c4d96e176 | ||
|   | 2c920bd016 | ||
|   | a5452012b5 | ||
|   | 43dd35f54b | ||
|   | 67bbd64fa1 | ||
|   | 8f48b83c79 | ||
|   | 6a57a5c369 | ||
|   | 20df062d9f | ||
|   | 9fa0a8b27b | ||
|   | 9acf9f4f19 | ||
|   | 0e84cf085b | ||
|   | ca9e371067 | ||
|   | 56920fe833 | ||
|   | f2e6fd9692 | ||
|   | b2966cbc45 | ||
|   | 45c566859e | ||
|   | 654f7ed3ae | ||
|   | f1ef07bee0 | ||
|   | 3e28d39285 | ||
|   | 8eb8c24209 | ||
|   | 4dbbf8f599 | ||
|   | a3ebabacb8 | ||
|   | 440afcca5d | ||
|   | 8220d4aa31 | ||
|   | b196c9ce8e | ||
|   | a9d896eb45 | ||
|   | 3cd9b76805 | ||
|   | 942d799c9c | ||
|   | f0cbc4f23e | ||
|   | 9bd14b142b | ||
|   | 4ba1217b84 | ||
|   | 50d4353045 | ||
|   | 19e97a7738 | ||
|   | 2ac5fe45e3 | ||
|   | 7f3d3ecb00 | ||
|   | 331a6ea1b3 | ||
|   | 8f2f673438 | ||
|   | 66e12e2a0c | ||
|   | 2b3ecc178a | ||
|   | 9b043b7444 | ||
|   | 24945efc7d | ||
|   | fa14c40283 | ||
|   | d8f19cec66 | ||
|   | 955341ff9f | ||
|   | cb342de934 | ||
|   | ea0342f100 | ||
|   | cd57741a51 | ||
|   | f4552e358d | ||
|   | 1dc225362d | ||
|   | a6b3e97cda | ||
|   | 8a244d12fc | ||
|   | 5a5d1f5e62 | ||
|   | 21214814b7 | ||
|   | 3ac93dcfca | ||
|   | c509882a18 | ||
|   | c146d6207d | ||
|   | 2045a4988d | ||
|   | dc7e6010cb | ||
|   | b1fc17e045 | ||
|   | daf0e6e1ba | ||
|   | 6f263735f0 | ||
|   | 1d2aa74e51 | ||
|   | a69b1d895c | ||
|   | 399d77c60f | ||
|   | b768ee6d41 | ||
|   | 58faffed6d | ||
|   | 87706d471b | ||
|   | eb7eb15ee7 | ||
|   | a17085bb18 | ||
|   | cde27899dd | ||
|   | 93ff778de0 | ||
|   | 17f91c8eb9 | ||
|   | 7775cd0073 | ||
|   | cc48f273c0 | ||
|   | c8406681d7 | ||
|   | a05dd19986 | ||
|   | aae350c1b3 | ||
|   | c48559410f | ||
|   | 9c59bfde52 | ||
|   | eb9d7e97d9 | ||
|   | 24eebe6711 | ||
|   | a2b8e4493f | ||
|   | 7c4a85325c | ||
|   | 94ba90d694 | ||
|   | e97bc25fb8 | ||
|   | 73936bb10d | ||
|   | 23e65ac5d3 | ||
|   | 9dc0396411 | ||
|   | cc6db9ae9e | ||
|   | 3748d8e246 | ||
|   | 0341111f8f | ||
|   | a7c4b40fcc | ||
|   | e7fb3571ac | ||
|   | 58ad9db5ed | ||
|   | 6fd55b47c2 | ||
|   | 3d7c191d1c | ||
|   | 9344e1ca79 | ||
|   | 68eb5cd74f | ||
|   | ca7ac71e76 | ||
|   | 3a280f3b6a | ||
|   | 19cea0e129 | ||
|   | 2c40648a60 | ||
|   | 48815649a8 | ||
|   | f690e14a14 | ||
|   | b37ad79e50 | ||
|   | a990205d41 | ||
|   | 891230c78e | ||
|   | 8bfdf8ecf2 | ||
|   | 78655102b9 | ||
|   | 491a171409 | ||
|   | b3aec8f839 | ||
|   | 0106a4222d | ||
|   | 11af33f2d5 | ||
|   | 812442fbbc | ||
|   | 1750dd3b97 | ||
|   | ffe298710d | ||
|   | 361b936aa0 | ||
|   | 9503b0bfc4 | ||
|   | de597391dc | ||
|   | 59599261ff | ||
|   | e1fb8c1ae5 | ||
|   | ba4146f82c | ||
|   | a5c43bb823 | ||
|   | 050cf3edb3 | ||
|   | 84d5ebfa41 | ||
|   | 6bc4e5b45c | ||
|   | 496fe09f78 | ||
|   | dd9edb68b2 | ||
|   | b02177e1d0 | ||
|   | 600d38d3e2 | ||
|   | 07fd7e497f | ||
|   | 567f1d4247 | ||
|   | 3f05dfe4bf | ||
|   | fbebcf57c1 | ||
|   | 30d3a458e2 | ||
|   | 7591d88d00 | ||
|   | 22baabf751 | ||
|   | 0cdda8b0ae | ||
|   | 295a63087f | ||
|   | 797d3ed362 | ||
|   | 31de7de385 | ||
|   | 5780ad370a | ||
|   | 3833d2cd83 | ||
|   | 38b60205e8 | ||
|   | 027456fd77 | ||
|   | 10e2d65221 | ||
|   | da45813a06 | ||
|   | 6cc9b5c62b | ||
|   | 1172e02f1b | ||
|   | 46fbc05040 | ||
|   | fee1ffa2e9 | ||
|   | 12d260ca82 | ||
|   | 82e90553bc | ||
|   | 18c1a0e4f6 | ||
|   | aca2d2e510 | ||
|   | a75db92007 | ||
|   | aaf2fa8807 | ||
|   | 86e8457574 | ||
|   | 8292ed9e31 | ||
|   | 57fc51d013 | ||
|   | 8b332adbe7 | ||
|   | ae0ee1ebdd | ||
|   | 3cea734b9b | ||
|   | 7ac3f7ca03 | ||
|   | 0a5c73478d | ||
|   | 495952acb0 | ||
|   | 25a8f1522a | ||
|   | f13f7ce129 | ||
|   | cc90127af9 | ||
|   | 98e8476a88 | ||
|   | ec8bdc9af6 | ||
|   | 9598416932 | ||
|   | 4c0bc79e49 | ||
|   | 01dd2d8097 | ||
|   | e09444d94f | ||
|   | 0c76833997 | ||
|   | ad99ca1723 | ||
|   | f47561b580 | ||
|   | 4f60da292a | ||
|   | 70fb86a1c3 | ||
|   | e8642df674 | ||
|   | 3702caa25f | ||
|   | 4200bbfd0a | ||
|   | 0fcca4c141 | ||
|   | fafdae9ba9 | ||
|   | 1d0721034d | ||
|   | 3cfbe9585a | ||
|   | 71305e636c | ||
|   | ce23c153ee | ||
|   | 019da4d2f4 | ||
|   | ffc31f565b | ||
|   | 7f01f2d716 | ||
|   | bf20b0dd79 | ||
|   | 867826759b | ||
|   | 3be370071c | ||
|   | 07491d8028 | ||
|   | 7f1baca872 | ||
|   | 8c3a0a8275 | ||
|   | 397692d113 | ||
|   | 035ecc1517 | ||
|   | bb52301200 | ||
|   | f4adf5d7e7 | ||
|   | 14fa738fbe | ||
|   | 4da7aa38ea | ||
|   | 09c84e25bf | ||
|   | 224e3b6ad4 | ||
|   | 992a683ba2 | ||
|   | e8e116312a | ||
|   | edd9ae5118 | ||
|   | 23ae3b5830 | ||
|   | 6746aab7ef | ||
|   | cf70efabb0 | ||
|   | 6eb2958afc | ||
|   | 18f13b7d15 | ||
|   | 0bb959d26a | ||
|   | b4f3e85001 | ||
|   | 25100c46f7 | ||
|   | 540f4c0958 | ||
|   | 8930ebe572 | ||
|   | 8f93f2b6d2 | ||
|   | 9ba91889e0 | ||
|   | 67683fbddb | ||
|   | 824f9d7ad0 | ||
|   | 3e99f90eb1 | ||
|   | 3ca8a94419 | ||
|   | 714837dd00 | ||
|   | bc1d3ddd5d | ||
|   | bf7b75b619 | ||
|   | 469d6ab907 | ||
|   | a4dd5bb62f | ||
|   | 60859de0bd | ||
|   | ff70d5db14 | ||
|   | c8ad936d26 | ||
|   | 60b5f5fe48 | ||
|   | 14ec7fb816 | ||
|   | b4ea230ff1 | ||
|   | 410879d154 | ||
|   | f14121e99c | ||
|   | 19d7444946 | ||
|   | d9aaed0035 | ||
|   | cfd3c159f9 | ||
|   | 4ff50525ed | ||
|   | 34e02e9f3a | ||
|   | 62690c2a53 | ||
|   | 5d30e0854f | ||
|   | ec683cdf88 | ||
|   | 39feed6fac | ||
|   | 209d52e920 | ||
|   | 4595534a6f | ||
|   | be9d9264ae | ||
|   | a8495c67d4 | ||
|   | eec369ea4f | ||
|   | abed07b613 | ||
|   | 6cfdedb823 | ||
|   | db903ebb0a | ||
|   | e12c83fd83 | ||
|   | 12861f6c07 | ||
|   | 52065b8313 | ||
|   | f6ff843cfb | ||
|   | aedf402c17 | ||
|   | 0745b03271 | ||
|   | bb5c0de367 | ||
|   | 9478251d44 | ||
|   | 47cb85d3db | ||
|   | ce7ceccc1c | ||
|   | f33fc092e3 | ||
|   | 0ff8abcd6f | ||
|   | 092690e9f0 | ||
|   | 5b11175cf3 | ||
|   | 7a203a12a3 | ||
|   | f7dd8af35e | ||
|   | 7a9b3442d2 | ||
|   | 81c72ee5d9 | ||
|   | fed7f89f96 | ||
|   | aefa629509 | ||
|   | bc32581cbd | ||
|   | 3f7f6af051 | ||
|   | ee9b2e8bf8 | ||
|   | 0b968abfe4 | ||
|   | 7e7ecd6a0f | ||
|   | 7e9a9e14c7 | ||
|   | 333493f351 | ||
|   | 38333a6d11 | ||
|   | a1bfc1cb26 | ||
|   | 191666a2f2 | ||
|   | aeeae7c167 | ||
|   | 08c7d4d15e | ||
|   | 29b92659c7 | ||
|   | 209f25a5cf | ||
|   | 4622f4b51d | ||
|   | 4d51dc9b6c | ||
|   | 76b992509b | ||
|   | 925978db8f | ||
|   | c0e90ddb33 | ||
|   | a7a29eaf97 | ||
|   | a451d2d6f2 | ||
|   | 914429ab4f | ||
|   | f721a7c66c | ||
|   | bba499649b | ||
|   | f9cd54c445 | ||
|   | 0d89125a8e | ||
|   | 6b37d678df | ||
|   | 02698b3a0e | ||
|   | 6f0512c18c | ||
|   | e46b38b301 | ||
|   | 5841817fb5 | ||
|   | e28fb49015 | ||
|   | 841eef6a52 | ||
|   | 2eee1ef203 | ||
|   | cd605f4996 | ||
|   | 54a8482a88 | ||
|   | e15b5a07fb | ||
|   | c198305b5e | ||
|   | a637a8970b | ||
|   | d97f68eec0 | ||
|   | c3069a425d | ||
|   | 9b1dc7d42e | ||
|   | ec23fcdd75 | ||
|   | 544d57c720 | ||
|   | f80e8c8d11 | ||
|   | 2468be4736 | ||
|   | 4e835ed3ff | ||
|   | 41f494fbff | ||
|   | e974fe5dc7 | ||
|   | 85a23442cc | ||
|   | a822a70f00 | ||
|   | 819902c24e | ||
|   | 554efbb057 | ||
|   | 1552a8e74b | ||
|   | f0e9a8c5fe | ||
|   | 9e85748b2e | ||
|   | a98b23af02 | ||
|   | b12e9832c9 | ||
|   | 0f9554c717 | ||
|   | cba2474df7 | ||
|   | e924a1920e | ||
|   | 311a2ddc75 | ||
|   | 551d1d9f1a | ||
|   | a5dea9e7f6 | ||
|   | 02bd83bd90 | ||
|   | 31ea7297c2 | ||
|   | 26a99a122a | ||
|   | 116b870152 | ||
|   | 667c7b07b6 | ||
|   | 45e75cc06e | ||
|   | cee0bbb805 | ||
|   | 77eb75fcc6 | ||
|   | 55211907f2 | ||
|   | e1e7cd1479 | ||
|   | 2d3c729215 | ||
|   | f391cfd432 | ||
|   | 352136f0c6 | ||
|   | b6e7f90f6a | ||
|   | 86cfeb76b6 | ||
|   | d91357a807 | ||
|   | 3c8d7a808b | ||
|   | d6a80c7ea5 | ||
|   | 05a15ac689 | ||
|   | 580cc359e5 | ||
|   | 630284e7ae | ||
|   | cc562033e7 | ||
|   | fabb9b6fd7 | ||
|   | e8ffcaae46 | ||
|   | 47d9895077 | ||
|   | aacf15757f | ||
|   | 914b44069b | ||
|   | 4789327378 | ||
|   | c55f2945ac | ||
|   | c6a368d6f2 | ||
|   | 7cd1e8ea76 | ||
|   | 351ae1b2c7 | ||
|   | 457f3e25cc | ||
|   | 41c670450b | ||
|   | f636db49f7 | ||
|   | 505bba7612 | ||
|   | 72507aba4b | ||
|   | 9b086b3f2a | ||
|   | f9d7d2d1dd | ||
|   | e9efa3f2d3 | ||
|   | 665a72a08f | ||
|   | a4c9ed90b7 | ||
|   | 28d6d4db92 | ||
|   | 12def37194 | ||
|   | ed2302e545 | ||
|   | 618adb913d | ||
|   | 4a220f9bfb | ||
|   | 87f0b1fc97 | ||
|   | 5b27b652e7 | ||
|   | 556ff0baf4 | ||
|   | 8cc536b20f | ||
|   | 5e20c871a5 | ||
|   | 56fb892818 | ||
|   | 7f104c249a | ||
|   | 0eeface374 | ||
|   | ffe1e1e40d | ||
|   | 5efdef19cf | ||
|   | c9fae6a070 | ||
|   | 2b17f730d4 | ||
|   | 6e16ef1246 | ||
|   | 7bad242944 | ||
|   | 74a5c48214 | ||
|   | 11af906c79 | ||
|   | c31c4b4286 | ||
|   | 7f29b5d1e8 | ||
|   | 6012705e95 | ||
|   | 21a97863ac | ||
|   | 50cfecb3fd | ||
|   | 1461804039 | ||
|   | 6970dfa5f8 | ||
|   | 18f630ba15 | ||
|   | f68eb9c778 | ||
|   | 4a960d9f2c | ||
|   | f1e3902fea | ||
|   | 41a623a643 | ||
|   | d3fe1d3b2b | ||
|   | 3fd3baaa47 | ||
|   | 38988b4819 | ||
|   | 704e92c3d0 | ||
|   | 97b1a60ae8 | ||
|   | 3288721259 | ||
|   | b3ddabda29 | ||
|   | b8b3e1e72a | ||
|   | 615a5212bf | ||
|   | 221d299052 | ||
|   | 3180d2ddf2 | ||
|   | 7aaa075ba8 | ||
|   | 4fb590889a | ||
|   | 115ca903b2 | ||
|   | e90170a384 | ||
|   | 1d0760c630 | ||
|   | af44fe74ff | ||
|   | 72c0021306 | ||
|   | 18918eb3a3 | ||
|   | 416e181992 | ||
|   | d7a49e3316 | ||
|   | cf13f13fbf | ||
|   | f53a8d3b06 | ||
|   | 8c5c1922c9 | ||
|   | 707559e023 | ||
|   | 97a2ccbfe0 | ||
|   | 95f2a7a908 | ||
|   | e53c53bb4a | ||
|   | d906a85095 | ||
|   | dc1f669a1f | ||
|   | 0ea862b572 | ||
|   | fe7a57b2b7 | ||
|   | c9c62a1083 | ||
|   | 5608c5a901 | ||
|   | 722361aedb | ||
|   | d2f40612f4 | ||
|   | 6a63e5bb51 | ||
|   | ad40ab7e1e | ||
|   | 05ce67fbfe | ||
|   | 555860cf83 | ||
|   | 2575787bd0 | ||
|   | 2cfb646065 | ||
|   | a1da452c41 | ||
|   | 420e38bead | ||
|   | 41c4018a27 | ||
|   | 4d4f755607 | ||
|   | d8e5daf12b | ||
|   | c3322021d0 | ||
|   | 45cc88091e | ||
|   | 27498f68fb | ||
|   | d8e80daa93 | ||
|   | 384a6730a4 | ||
|   | a7de76d150 | ||
|   | 2154e237ff | ||
|   | 5fda3e9765 | ||
|   | b302bb9379 | ||
|   | bfbb81030f | ||
|   | 63ce3292aa | ||
|   | b9a8846ee9 | ||
|   | 6ae1d28f80 | ||
|   | 17d358f8fe | ||
|   | cfd71457d2 | ||
|   | 87285e08dc | ||
|   | b3a63123fd | ||
|   | 7fc8238fb0 | ||
|   | 32a0765484 | ||
|   | c5e1b87c61 | ||
|   | 603d1ee9bb | ||
|   | 86a0816b26 | ||
|   | 287cb0f5ad | ||
|   | 3009db3aa7 | ||
|   | 32e095ce15 | ||
|   | 9d6744ec15 | ||
|   | 532f2caa37 | ||
|   | da2fa24e74 | ||
|   | 8d04728ebc | ||
|   | 2aa5c276c9 | ||
|   | 6dba31b257 | ||
|   | 6040e1bb5e | ||
|   | 20e9d13f60 | ||
|   | 5f76cc4f7b | ||
|   | 186a810bf6 | ||
|   | 207e56969b | ||
|   | 94b6a27cf3 | ||
|   | 21693e344c | ||
|   | d2443f6de8 | ||
|   | ed27422e69 | ||
|   | d24c89405a | ||
|   | 2d518dfe2b | ||
|   | f357fa74f3 | ||
|   | 335cf5d2e9 | ||
|   | ed3eedd238 | ||
|   | 23783b8b0b | ||
|   | 193054f1fc | ||
|   | 656700b5be | ||
|   | 3e84f3b3a4 | ||
|   | 5360df6012 | ||
|   | 39fdaa367c | ||
|   | 7e0cdb12d7 | ||
|   | 03983e8886 | ||
|   | 57af50ed49 | ||
|   | e80ade65c7 | ||
|   | 4d24112ec7 | ||
|   | 1a269a1227 | ||
|   | d077fafd29 | ||
|   | 5478e70576 | ||
|   | 66b94ab9f1 | ||
|   | 74c6fc954e | ||
|   | 22bd2eb238 | ||
|   | 2a72f274c9 | ||
|   | 3a1b56494a | ||
|   | 932052772c | ||
|   | 43d058d3db | ||
|   | 020b030667 | ||
|   | 5a3eacde0b | ||
|   | 7f412f5472 | ||
|   | 33b6ef66d6 | ||
|   | 9fefe57c90 | ||
|   | ed77522c08 | ||
|   | 09aca839a8 | ||
|   | def9a1bcf8 | ||
|   | c0a0d36e5d | ||
|   | b68d7150f1 | ||
|   | 198bcfdf4d | ||
|   | 8eb903ad72 | ||
|   | c853147635 | ||
|   | 0294397021 | ||
|   | 97fed21811 | ||
|   | 510ea56431 | ||
|   | 7897d78f0d | ||
|   | 25a58a5c46 | ||
|   | f9e5fd714d | ||
|   | 55eefd09da | ||
|   | 0ce1f3e0f6 | ||
|   | 99be181aea | ||
|   | fbf6a3517d | ||
|   | 1dc4da8beb | ||
|   | e72ce9c90e | ||
|   | 33f2ff7e6d | ||
|   | 6664d49928 | ||
|   | d648a6d3db | ||
|   | a1f262b5b2 | ||
|   | b109b76120 | ||
|   | 1388f280e6 | ||
|   | d141b040d6 | ||
|   | e84d82315c | ||
|   | c6962ef4d2 | ||
|   | f93714a44e | ||
|   | 564efd77f7 | ||
|   | 5442c7cc2e | ||
|   | 0fa5a16cd0 | ||
|   | 1881cdc9ab | ||
|   | fb2aa44f78 | ||
|   | 5cdb3f4fd5 | ||
|   | 5a5c5721cc | ||
|   | 916675fb08 | ||
|   | 6f6cb4b630 | ||
|   | c0f69f321d | ||
|   | 55bf41d2da | ||
|   | 7687d7705b | ||
|   | 21ad9a36c8 | ||
|   | d00dc658df | ||
|   | 2dab7c8dda | ||
|   | c37cc40ad9 | ||
|   | 5341015cb1 | ||
|   | cf1b027db9 | ||
|   | db37077af7 | ||
|   | 3fa532a3c0 | ||
|   | 9792d4cc4b | ||
|   | 090bd69be5 | ||
|   | c36e311520 | ||
|   | 51bd21a464 | ||
|   | 196df855ac | ||
|   | d76c9dad52 | ||
|   | 298e65a394 | ||
|   | 078b776f60 | ||
|   | 4d61a345c6 | ||
|   | dc0358957e | ||
|   | 42d648e338 | ||
|   | 1c254984c1 | ||
|   | f287cc34e7 | ||
|   | c8a8806e7d | ||
|   | d061f728bd | ||
|   | 6bf5bc60d1 | ||
|   | c784c69eb3 | ||
|   | e3759d059a | ||
|   | 35cff29917 | ||
|   | f80cadcd7f | ||
|   | 600f757046 | ||
|   | a2aaa3633a | ||
|   | 3b793929d8 | ||
|   | 4b997d42df | 
							
								
								
									
										1013
									
								
								.editorconfig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1013
									
								
								.editorconfig
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										12
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,12 +0,0 @@ | ||||
| # These are supported funding model platforms | ||||
|  | ||||
| github: [NotMyFault, dordsor21, SirYwell] | ||||
| patreon: IntellectualSites # Replace with a single Patreon username | ||||
| open_collective: IntellectualSites | ||||
| ko_fi: # Replace with a single Ko-fi username | ||||
| tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel | ||||
| community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry | ||||
| liberapay: # Replace with a single Liberapay username | ||||
| issuehunt: # Replace with a single IssueHunt username | ||||
| otechie: # Replace with a single Otechie username | ||||
| custom: https://www.paypal.me/AlexanderBrandes # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2'] | ||||
							
								
								
									
										10
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							| @@ -7,7 +7,8 @@ body: | ||||
|     attributes: | ||||
|       value: | | ||||
|         Thanks for taking the time to fill out this bug report for PlotSquared! Fill out the following form to your best ability to help us fix the problem. | ||||
|         Only use this if you're absolutely sure that you found a bug and can reproduce it. For anything else, use: [our Discord server](https://discord.gg/intellectualsites) or [the wiki](https://github.com/IntellectualSites/PlotSquared/wiki). | ||||
|         Only use this if you're absolutely sure that you found a bug and can reproduce it. For anything else, use: [our Discord server](https://discord.gg/intellectualsites) or [the wiki](https://github.com/IntellectualSites/PlotSquared-Documentation/wiki). | ||||
|         Do NOT use the public issue tracker to report security vulnerabilities! They are disclosed using [this](https://forms.gle/btgdRn9yhGtzEiGW8) form! | ||||
|  | ||||
|   - type: dropdown | ||||
|     attributes: | ||||
| @@ -17,8 +18,6 @@ body: | ||||
|       options: | ||||
|         - Paper | ||||
|         - Spigot | ||||
|         - Tuinity | ||||
|         - Purpur | ||||
|     validations: | ||||
|       required: true | ||||
|  | ||||
| @@ -28,6 +27,9 @@ body: | ||||
|       description: Which server version version you using? If your server version is not listed, it is not supported. Update to a supported version first. | ||||
|       multiple: false | ||||
|       options: | ||||
|         - '1.18.1' | ||||
|         - '1.18' | ||||
|         - '1.17.1' | ||||
|         - '1.16.5' | ||||
|         - '1.15.2' | ||||
|         - '1.14.4' | ||||
| @@ -99,4 +101,4 @@ body: | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: Anything else? | ||||
|       description: You can provide additional context below. | ||||
|       description: You can provide additional context below. | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							| @@ -4,5 +4,5 @@ contact_links: | ||||
|     url: https://discord.gg/intellectualsites | ||||
|     about: Our support Discord, please ask questions and seek support here. | ||||
|   - name: PlotSquared Wiki | ||||
|     url: https://github.com/IntellectualSites/PlotSquared/wiki | ||||
|     url: https://github.com/IntellectualSites/PlotSquared-Documentation/wiki | ||||
|     about: Take a look at the wiki page for instructions how to setup PlotSquared and use its commands. | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/ISSUE_TEMPLATE/feature_request.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/ISSUE_TEMPLATE/feature_request.yml
									
									
									
									
										vendored
									
									
								
							| @@ -7,7 +7,7 @@ body: | ||||
|     attributes: | ||||
|       value: | | ||||
|         Thanks for taking the time to fill out this feature request for PlotSquared! Fill out the following form to your best ability to help us understand your feature request and greately improve the change of it getting added. | ||||
|         For anything else than a feature request, use: [our Discord server](https://discord.gg/intellectualsites) or [the wiki](https://github.com/IntellectualSites/PlotSquared/wiki). | ||||
|         For anything else than a feature request, use: [our Discord server](https://discord.gg/intellectualsites) or [the wiki](https://github.com/IntellectualSites/PlotSquared-Documentation/wiki). | ||||
|  | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|   | ||||
							
								
								
									
										17
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										17
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,17 +0,0 @@ | ||||
| ## Overview | ||||
| <!--  Please describe which issue this Pull Request targets | ||||
|  | ||||
| If there is no issue, please create one so we can look into it before approving your PR. | ||||
| You can do so here: https://github.com/IntellectualSites/PlotSquared/issues | ||||
| --> | ||||
|  | ||||
| **Fixes {Link to issue}** | ||||
|  | ||||
| ## Description | ||||
|  | ||||
| ## Checklist | ||||
| <!-- Make sure you have completed the following steps (put an "X" between of brackets): --> | ||||
| - [] I included all information required in the sections above | ||||
| - [] I tested my changes and approved their functionality | ||||
| - [] I ensured my changes do not break other parts of the code | ||||
| - [] I read and followed the [contribution guidelines](https://github.com/IntellectualSites/PlotSquared/blob/v5/CONTRIBUTING.md) | ||||
							
								
								
									
										1
									
								
								.github/release-drafter.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.github/release-drafter.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| _extends: .github | ||||
							
								
								
									
										2
									
								
								.github/stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/stale.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,4 @@ | ||||
| daysUntilStale: 60 | ||||
| daysUntilStale: 30 | ||||
| daysUntilClose: 7 | ||||
| only: issues | ||||
| exemptLabels: | ||||
|   | ||||
							
								
								
									
										53
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										53
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,20 +1,41 @@ | ||||
| name: "build" | ||||
| name: build | ||||
|  | ||||
| on: [ pull_request, push ] | ||||
|  | ||||
| on: ["pull_request", "push"] | ||||
|   | ||||
| jobs: | ||||
|   build: | ||||
|     strategy: | ||||
|       matrix: | ||||
|         java: ["1.8", "11"] | ||||
|         os: ["ubuntu-18.04"] | ||||
|     runs-on: "${{ matrix.os }}" | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - name: "Checkout Repository" | ||||
|       uses: "actions/checkout@v2.3.4" | ||||
|     - name: "Setup JDK ${{ matrix.java }}" | ||||
|       uses: "actions/setup-java@v1.4.3" | ||||
|       with: | ||||
|           java-version: "${{ matrix.java }}" | ||||
|     - name: "Clean Build" | ||||
|       run: "./gradlew clean build" | ||||
|       - name: Checkout Repository | ||||
|         uses: actions/checkout@v2.4.0 | ||||
|       - name: Validate Gradle Wrapper" | ||||
|         uses: gradle/wrapper-validation-action@v1.0.4 | ||||
|       - name: Setup Java | ||||
|         uses: actions/setup-java@v2.5.0 | ||||
|         with: | ||||
|           distribution: temurin | ||||
|           java-version: 17 | ||||
|       - name: Clean Build | ||||
|         run: ./gradlew clean build | ||||
|       - name: Determine release status | ||||
|         if: ${{ runner.os == 'Linux' }} | ||||
|         run: | | ||||
|           if [ "$(./gradlew properties | awk '/^version:/ { print $2; }' | grep '\-SNAPSHOT')" ]; then | ||||
|             echo "STATUS=snapshot" >> $GITHUB_ENV | ||||
|           else | ||||
|             echo "STATUS=release" >> $GITHUB_ENV | ||||
|           fi | ||||
|       - name: Publish Release | ||||
|         if: ${{ runner.os == 'Linux' && env.STATUS == 'release' && github.event_name == 'push' && github.ref == 'refs/heads/v6'}} | ||||
|         run: ./gradlew publishToSonatype closeAndReleaseSonatypeStagingRepository | ||||
|         env: | ||||
|           ORG_GRADLE_PROJECT_sonatypeUsername: ${{ secrets.SONATYPE_USERNAME }} | ||||
|           ORG_GRADLE_PROJECT_sonatypePassword: ${{ secrets.SONATYPE_PASSWORD }} | ||||
|           ORG_GRADLE_PROJECT_signingKey: ${{ secrets.SIGNING_KEY }} | ||||
|           ORG_GRADLE_PROJECT_signingPassword: ${{ secrets.SIGNING_PASSWORD }} | ||||
|       - name: Publish Snapshot | ||||
|         if: ${{ runner.os == 'Linux' && env.STATUS != 'release' && github.event_name == 'push' && github.ref == 'refs/heads/v6' }} | ||||
|         run: ./gradlew publishToSonatype | ||||
|         env: | ||||
|           ORG_GRADLE_PROJECT_sonatypeUsername: ${{ secrets.SONATYPE_USERNAME }} | ||||
|           ORG_GRADLE_PROJECT_sonatypePassword: ${{ secrets.SONATYPE_PASSWORD }} | ||||
|   | ||||
							
								
								
									
										20
									
								
								.github/workflows/rebase.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								.github/workflows/rebase.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| name: Rebase Pull Request | ||||
| on: | ||||
|   issue_comment: | ||||
|     types: [created] | ||||
|  | ||||
| jobs: | ||||
|   rebase: | ||||
|     name: Rebase | ||||
|     if: github.event.issue.pull_request != '' && contains(github.event.comment.body, '/rebase') && github.event.comment.author_association == 'MEMBER' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout Repository | ||||
|         uses: actions/checkout@v2.4.0 | ||||
|         with: | ||||
|           token: ${{ secrets.REBASE_TOKEN }} | ||||
|           fetch-depth: 0 | ||||
|       - name: Automatic Rebase | ||||
|         uses: cirrus-actions/rebase@1.5 | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.REBASE_TOKEN }} | ||||
							
								
								
									
										18
									
								
								.github/workflows/release-drafter.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								.github/workflows/release-drafter.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| name: draft release | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - v6 | ||||
|   pull_request: | ||||
|     types: [ opened, reopened, synchronize ] | ||||
|   pull_request_target: | ||||
|     types: [ opened, reopened, synchronize ] | ||||
|  | ||||
| jobs: | ||||
|   update_release_draft: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: release-drafter/release-drafter@v5.17.5 | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
							
								
								
									
										12
									
								
								.github/workflows/validate-gradle-wrapper.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/validate-gradle-wrapper.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,12 +0,0 @@ | ||||
| name: "validate gradle wrapper" | ||||
|  | ||||
| on: ["pull_request", "push"] | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     runs-on: "ubuntu-18.04" | ||||
|     steps: | ||||
|       - name: "Checkout Repository" | ||||
|         uses: "actions/checkout@v2.3.4" | ||||
|       - name: "Validate Gradle Wrapper" | ||||
|         uses: "gradle/wrapper-validation-action@v1.0.3" | ||||
							
								
								
									
										8
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -2,11 +2,7 @@ | ||||
| *.cmd | ||||
| *.sh | ||||
| *.prefs | ||||
| Sponge/build | ||||
| Core/build | ||||
| Bukkit/build | ||||
| Nukkit/build | ||||
| buildSrc/ | ||||
|  | ||||
| ### Maven ### | ||||
| /mvn | ||||
| @@ -134,12 +130,8 @@ local.properties | ||||
| # STS (Spring Tool Suite) | ||||
| .springBeans | ||||
| /target/ | ||||
| Nukkit/build/classes/ | ||||
| Nukkit/build/dependency-cache/ | ||||
| checkstyle.xml | ||||
| classes/ | ||||
| p2error.txt | ||||
| *.bat | ||||
| Nukkit/build/resources/main/plugin.yml | ||||
| docs/ | ||||
| build/ | ||||
|   | ||||
							
								
								
									
										4
									
								
								.lift.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								.lift.toml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| jdkVersion = "17" | ||||
| build = "gradle clean build -x test" | ||||
| tools = ["findsecbugs", "ErrorProne", "Semgrep", "Detekt", "Infer"] | ||||
| ignoreRules = ["CatchAndPrintStackTrace", "ReferenceEquality", "FallThrough", "FutureReturnValueIgnored", "MixedMutabilityReturnType", "EmptyCatch", "MissingCasesInEnumSwitch", "OperatorPrecedence", "StaticAssignmentInConstructor", "ReferenceEquality", "EqualsHashCode", "EqualsGetClass", "TypeParameterUnusedInFormals", "StringSplitter", "InlineMeSuggester", "NULL_DEREFERENCE"] | ||||
							
								
								
									
										13
									
								
								.whitesource
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								.whitesource
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| { | ||||
|   "scanSettings": { | ||||
|     "baseBranches": ["v6"] | ||||
|   }, | ||||
|   "checkRunSettings": { | ||||
|     "vulnerableCheckRunConclusionLevel": "success", | ||||
|     "displayMode": "diff" | ||||
|   }, | ||||
|   "issueSettings": { | ||||
|     "minSeverityLevel": "LOW" | ||||
|   }, | ||||
|   "enableRenovate": "true" | ||||
| } | ||||
| @@ -1,125 +0,0 @@ | ||||
| plugins { | ||||
|     id "com.github.johnrengelman.shadow" | ||||
| } | ||||
| repositories { | ||||
|     maven { url = "https://hub.spigotmc.org/nexus/content/repositories/snapshots/" } | ||||
|     maven { url = "https://oss.sonatype.org/content/repositories/snapshots/" } | ||||
|     maven { url = "https://jitpack.io" } | ||||
|     maven { url = "https://repo.codemc.org/repository/maven-public" } | ||||
|     maven { url = "https://repo.extendedclip.com/content/repositories/placeholderapi/" } | ||||
|     maven { | ||||
|         name = "papermc" | ||||
|         url = "https://papermc.io/repo/repository/maven-public/" | ||||
|     } | ||||
|     maven { url = "https://ci.ender.zone/plugin/repository/everything/" } | ||||
|     maven { url = "https://mvn.intellectualsites.com/content/repositories/releases" } | ||||
|     maven { url = "https://mvn.intellectualsites.com/content/repositories/thirdparty"} | ||||
|     maven { url = "https://mvn.intellectualsites.com/content/repositories/snapshots" } | ||||
|     maven { url = "http://repo.mvdw-software.be/content/groups/public/" } | ||||
|     mavenLocal() | ||||
| } | ||||
|  | ||||
|  | ||||
| dependencies { | ||||
|     implementation(project(":PlotSquared-Core")) | ||||
|     compile("org.bstats:bstats-bukkit:1.7") | ||||
|     compile(project(":PlotSquared-Core")) | ||||
|     compile("com.destroystokyo.paper:paper-api:1.16.4-R0.1-SNAPSHOT") | ||||
|     implementation("org.spigotmc:spigot-api:1.16.4-R0.1-SNAPSHOT") | ||||
|     compile(group: "com.sk89q.worldedit", name: "worldedit-bukkit", version: "7.2.0") { | ||||
|         exclude(module: "bukkit") | ||||
|     } | ||||
|     compile("io.papermc:paperlib:1.0.5") | ||||
|     implementation("net.kyori:text-adapter-bukkit:3.0.3") | ||||
|     compile("com.github.MilkBowl:VaultAPI:1.7") { | ||||
|         exclude(module: "bukkit") | ||||
|     } | ||||
|     implementation("me.clip:placeholderapi:2.10.6") | ||||
|     implementation("net.luckperms:api:5.1") | ||||
|     implementation("net.ess3:EssentialsX:2.18.0") { | ||||
|         exclude(group: "io.papermc", module: "paperlib") | ||||
|     } | ||||
|     compile("se.hyperver.hyperverse:Core:0.6.0-SNAPSHOT"){ transitive = false } | ||||
|     compile('com.sk89q:squirrelid:1.0.0-SNAPSHOT'){ transitive = false } | ||||
|     compile('be.maximvdw:MVdWPlaceholderAPI:3.1.1'){ transitive = false } | ||||
|     implementation("org.incendo.serverlib:ServerLib:2.2.0") | ||||
| } | ||||
|  | ||||
| sourceCompatibility = 1.8 | ||||
| targetCompatibility = 1.8 | ||||
|  | ||||
| processResources { | ||||
|     from("src/main/resources") { | ||||
|         include("plugin.yml") | ||||
|         expand( | ||||
|                 name: project.parent.name, | ||||
|                 version: project.parent.version | ||||
|         ) | ||||
|     } | ||||
| } | ||||
|  | ||||
| //noinspection GroovyAssignabilityCheck | ||||
| jar.archiveFileName = "PlotSquared-Bukkit-${project.parent.version}.jar" | ||||
| jar.destinationDirectory = file("../mvn/com/plotsquared/PlotSquared-Bukkit/" + project.parent.version) | ||||
| task createPom { | ||||
|     doLast { | ||||
|         pom { | ||||
|             project { | ||||
|                 groupId = rootProject.group | ||||
|                 artifactId = "PlotSquared-Bukkit" | ||||
|                 version = rootProject.version | ||||
|             } | ||||
|         }.writeTo("../mvn/com/plotsquared/PlotSquared-Bukkit/${project.parent.version}/PlotSquared-Bukkit-${project.parent.version}.pom") | ||||
|         pom { | ||||
|             project { | ||||
|                 groupId = rootProject.group | ||||
|                 artifactId = "PlotSquared-Bukkit" | ||||
|                 version = "latest" | ||||
|             } | ||||
|         }.writeTo("../mvn/com/plotsquared/PlotSquared-Bukkit/latest/PlotSquared-Bukkit-latest.pom") | ||||
|          .writeTo("pom.xml") | ||||
|     } | ||||
| } | ||||
|  | ||||
| task copyFiles { | ||||
|     doLast { | ||||
|         copy { | ||||
|             from("../mvn/com/plotsquared/PlotSquared-Bukkit/${project.parent.version}/") | ||||
|             into("../mvn/com/plotsquared/PlotSquared-Bukkit/latest/") | ||||
|             include("PlotSquared-Bukkit*.jar") | ||||
|             rename("PlotSquared-Bukkit-${project.parent.version}.jar", "PlotSquared-Bukkit-latest.jar") | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| shadowJar { | ||||
|     dependencies { | ||||
|         include(dependency(":PlotSquared-Core")) | ||||
|         include(dependency("io.papermc:paperlib:1.0.5")) | ||||
|         include(dependency("net.kyori:text-adapter-bukkit:3.0.3")) | ||||
|         include(dependency("org.bstats:bstats-bukkit:1.7")) | ||||
|         include(dependency("org.khelekore:prtree:1.7.0-SNAPSHOT")) | ||||
|         include(dependency("com.sk89q:squirrelid:1.0.0-SNAPSHOT")) | ||||
|         include(dependency("com.intellectualsites.paster:Paster:1.0.2-SNAPSHOT")) | ||||
|         include(dependency("org.incendo.serverlib:ServerLib:2.2.0")) | ||||
|     } | ||||
|     relocate('net.kyori.text', 'com.plotsquared.formatting.text') | ||||
|     relocate("io.papermc.lib", "com.plotsquared.bukkit.paperlib") | ||||
|     relocate("org.bstats", "com.plotsquared.metrics") | ||||
|     relocate('com.sk89q.squirrelid', 'com.plotsquared.squirrelid') | ||||
|     relocate('org.khelekore.prtree', 'com.plotsquared.prtree') | ||||
|     relocate('com.intellectualsites.paster', 'com.plotsquared.core.paster') | ||||
|     relocate('org.incendo.serverlib', 'com.plotsquared.bukkit') | ||||
|     archiveFileName = "${project.name}-${parent.version}.jar" | ||||
|     destinationDirectory = file "../target" | ||||
| } | ||||
|  | ||||
| shadowJar.doLast { | ||||
|     task -> | ||||
|         ant.checksum file: task.archivePath | ||||
| } | ||||
|  | ||||
| build.dependsOn(shadowJar) | ||||
| build.finalizedBy(copyFiles) | ||||
| copyFiles.dependsOn(createPom) | ||||
							
								
								
									
										105
									
								
								Bukkit/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								Bukkit/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,105 @@ | ||||
| import com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar | ||||
|  | ||||
| repositories { | ||||
|     maven { | ||||
|         name = "PlaceholderAPI" | ||||
|         url = uri("https://repo.extendedclip.com/content/repositories/placeholderapi/") | ||||
|     } | ||||
|  | ||||
|     maven { | ||||
|         name = "PaperMC" | ||||
|         url = uri("https://papermc.io/repo/repository/maven-public/") | ||||
|     } | ||||
|  | ||||
|     maven { | ||||
|         name = "EssentialsX" | ||||
|         url = uri("https://repo.essentialsx.net/releases/") | ||||
|     } | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     api(projects.plotSquaredCore) | ||||
|  | ||||
|     // Metrics | ||||
|     implementation(libs.bstats) | ||||
|  | ||||
|     // Paper | ||||
|     compileOnly(libs.paper) | ||||
|     implementation(libs.paperlib) | ||||
|  | ||||
|     // Plugins | ||||
|     compileOnly(libs.worldeditBukkit) { | ||||
|         exclude(group = "org.bukkit") | ||||
|         exclude(group = "org.spigotmc") | ||||
|     } | ||||
|     compileOnly(libs.fastasyncworldeditBukkit) { isTransitive = false } | ||||
|     testImplementation(libs.fastasyncworldeditBukkit) { isTransitive = false } | ||||
|     compileOnly(libs.vault) { | ||||
|         exclude(group = "org.bukkit") | ||||
|     } | ||||
|     compileOnly(libs.placeholderapi) | ||||
|     compileOnly(libs.luckperms) | ||||
|     compileOnly(libs.essentialsx) | ||||
|     compileOnly(libs.mvdwapi) { isTransitive = false } | ||||
|  | ||||
|     // Other libraries | ||||
|     implementation(libs.squirrelid) { isTransitive = false } | ||||
|     implementation(libs.serverlib) | ||||
|  | ||||
|     // Our libraries | ||||
|     implementation(libs.arkitektonika) | ||||
|     implementation(libs.http4j) | ||||
|     implementation(libs.paster) | ||||
|  | ||||
|     // Adventure | ||||
|     implementation(libs.adventurePlatformBukkit) | ||||
| } | ||||
|  | ||||
| tasks.processResources { | ||||
|     filesMatching("plugin.yml") { | ||||
|         expand("version" to project.version) | ||||
|     } | ||||
| } | ||||
|  | ||||
| tasks.named<ShadowJar>("shadowJar") { | ||||
|     dependencies { | ||||
|         exclude(dependency("org.checkerframework:")) | ||||
|     } | ||||
|  | ||||
|     relocate("net.kyori.adventure", "com.plotsquared.core.configuration.adventure") | ||||
|     relocate("net.kyori.examination", "com.plotsquared.core.configuration.examination") | ||||
|     relocate("io.papermc.lib", "com.plotsquared.bukkit.paperlib") | ||||
|     relocate("org.bstats", "com.plotsquared.metrics") | ||||
|     relocate("org.enginehub", "com.plotsquared.squirrelid") | ||||
|     relocate("org.khelekore.prtree", "com.plotsquared.prtree") | ||||
|     relocate("com.google.inject", "com.plotsquared.google") | ||||
|     relocate("org.aopalliance", "com.plotsquared.core.aopalliance") | ||||
|     relocate("cloud.commandframework.services", "com.plotsquared.core.services") | ||||
|     relocate("io.leangen.geantyref", "com.plotsquared.core.geantyref") | ||||
|     relocate("com.intellectualsites.arkitektonika", "com.plotsquared.core.arkitektonika") | ||||
|     relocate("com.intellectualsites.http", "com.plotsquared.core.http") | ||||
|     relocate("com.intellectualsites.paster", "com.plotsquared.core.paster") | ||||
|     relocate("org.incendo.serverlib", "com.plotsquared.bukkit.serverlib") | ||||
|     relocate("org.jetbrains", "com.plotsquared.core.annotations") | ||||
|     relocate("org.intellij.lang", "com.plotsquared.core.intellij.annotations") | ||||
|     relocate("javax.annotation", "com.plotsquared.core.annotation") | ||||
|     relocate("com.google.code.findbugs", "com.plotsquared.core.findbugs") | ||||
|     relocate("javax.inject", "com.plotsquared.core.annotation.inject") | ||||
|  | ||||
|     // Get rid of all the libs which are 100% unused. | ||||
|     minimize() | ||||
|  | ||||
|     mergeServiceFiles() | ||||
| } | ||||
|  | ||||
| tasks { | ||||
|     withType<Javadoc> { | ||||
|         val opt = options as StandardJavadocDocletOptions | ||||
|         opt.links("https://papermc.io/javadocs/paper/1.17/") | ||||
|         opt.links("https://docs.enginehub.org/javadoc/com.sk89q.worldedit/worldedit-core/7.2.7/") | ||||
|         opt.links("https://docs.enginehub.org/javadoc/com.sk89q.worldedit/worldedit-bukkit/7.2.7/") | ||||
|         opt.links("https://jd.adventure.kyori.net/api/4.9.3/") | ||||
|         opt.links("https://google.github.io/guice/api-docs/5.0.1/javadoc/") | ||||
|         opt.links("https://checkerframework.org/api/") | ||||
|     } | ||||
| } | ||||
							
								
								
									
										198
									
								
								Bukkit/pom.xml
									
									
									
									
									
								
							
							
						
						
									
										198
									
								
								Bukkit/pom.xml
									
									
									
									
									
								
							| @@ -1,198 +0,0 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd" xmlns="http://maven.apache.org/POM/4.0.0" | ||||
|     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> | ||||
|   <modelVersion>4.0.0</modelVersion> | ||||
|   <groupId>com.plotsquared</groupId> | ||||
|   <artifactId>PlotSquared-Bukkit</artifactId> | ||||
|   <version>latest</version> | ||||
|   <dependencies> | ||||
|     <dependency> | ||||
|       <groupId>org.json</groupId> | ||||
|       <artifactId>json</artifactId> | ||||
|       <version>20200518</version> | ||||
|       <scope>compile</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>org.bstats</groupId> | ||||
|       <artifactId>bstats-bukkit</artifactId> | ||||
|       <version>1.7</version> | ||||
|       <scope>compile</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>com.plotsquared</groupId> | ||||
|       <artifactId>PlotSquared-Core</artifactId> | ||||
|       <version>5.13.13</version> | ||||
|       <scope>compile</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>com.destroystokyo.paper</groupId> | ||||
|       <artifactId>paper-api</artifactId> | ||||
|       <version>1.16.4-R0.1-SNAPSHOT</version> | ||||
|       <scope>compile</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>com.sk89q.worldedit</groupId> | ||||
|       <artifactId>worldedit-bukkit</artifactId> | ||||
|       <version>7.2.0</version> | ||||
|       <scope>compile</scope> | ||||
|       <exclusions> | ||||
|         <exclusion> | ||||
|           <artifactId>bukkit</artifactId> | ||||
|           <groupId>*</groupId> | ||||
|         </exclusion> | ||||
|       </exclusions> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>io.papermc</groupId> | ||||
|       <artifactId>paperlib</artifactId> | ||||
|       <version>1.0.5</version> | ||||
|       <scope>compile</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>com.github.MilkBowl</groupId> | ||||
|       <artifactId>VaultAPI</artifactId> | ||||
|       <version>1.7</version> | ||||
|       <scope>compile</scope> | ||||
|       <exclusions> | ||||
|         <exclusion> | ||||
|           <artifactId>bukkit</artifactId> | ||||
|           <groupId>*</groupId> | ||||
|         </exclusion> | ||||
|       </exclusions> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>se.hyperver.hyperverse</groupId> | ||||
|       <artifactId>Core</artifactId> | ||||
|       <version>0.6.0-SNAPSHOT</version> | ||||
|       <scope>compile</scope> | ||||
|       <exclusions> | ||||
|         <exclusion> | ||||
|           <artifactId>*</artifactId> | ||||
|           <groupId>*</groupId> | ||||
|         </exclusion> | ||||
|       </exclusions> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>com.sk89q</groupId> | ||||
|       <artifactId>squirrelid</artifactId> | ||||
|       <version>1.0.0-SNAPSHOT</version> | ||||
|       <scope>compile</scope> | ||||
|       <exclusions> | ||||
|         <exclusion> | ||||
|           <artifactId>*</artifactId> | ||||
|           <groupId>*</groupId> | ||||
|         </exclusion> | ||||
|       </exclusions> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>be.maximvdw</groupId> | ||||
|       <artifactId>MVdWPlaceholderAPI</artifactId> | ||||
|       <version>3.1.1</version> | ||||
|       <scope>compile</scope> | ||||
|       <exclusions> | ||||
|         <exclusion> | ||||
|           <artifactId>*</artifactId> | ||||
|           <groupId>*</groupId> | ||||
|         </exclusion> | ||||
|       </exclusions> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>com.sk89q.worldedit</groupId> | ||||
|       <artifactId>worldedit-core</artifactId> | ||||
|       <version>7.2.0</version> | ||||
|       <scope>runtime</scope> | ||||
|       <exclusions> | ||||
|         <exclusion> | ||||
|           <artifactId>dummypermscompat</artifactId> | ||||
|           <groupId>*</groupId> | ||||
|         </exclusion> | ||||
|         <exclusion> | ||||
|           <artifactId>bukkit-classloader-check</artifactId> | ||||
|           <groupId>*</groupId> | ||||
|         </exclusion> | ||||
|         <exclusion> | ||||
|           <artifactId>mockito-core</artifactId> | ||||
|           <groupId>*</groupId> | ||||
|         </exclusion> | ||||
|       </exclusions> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>net.kyori</groupId> | ||||
|       <artifactId>text-api</artifactId> | ||||
|       <version>3.0.2</version> | ||||
|       <scope>runtime</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>net.kyori</groupId> | ||||
|       <artifactId>text-serializer-gson</artifactId> | ||||
|       <version>3.0.2</version> | ||||
|       <scope>runtime</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>net.kyori</groupId> | ||||
|       <artifactId>text-serializer-legacy</artifactId> | ||||
|       <version>3.0.2</version> | ||||
|       <scope>runtime</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>net.kyori</groupId> | ||||
|       <artifactId>text-serializer-plain</artifactId> | ||||
|       <version>3.0.2</version> | ||||
|       <scope>runtime</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>com.google.guava</groupId> | ||||
|       <artifactId>guava</artifactId> | ||||
|       <version>21.0</version> | ||||
|       <scope>runtime</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>org.spigotmc</groupId> | ||||
|       <artifactId>spigot-api</artifactId> | ||||
|       <version>1.16.4-R0.1-SNAPSHOT</version> | ||||
|       <scope>runtime</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>net.kyori</groupId> | ||||
|       <artifactId>text-adapter-bukkit</artifactId> | ||||
|       <version>3.0.3</version> | ||||
|       <scope>runtime</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>me.clip</groupId> | ||||
|       <artifactId>placeholderapi</artifactId> | ||||
|       <version>2.10.6</version> | ||||
|       <scope>runtime</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>net.luckperms</groupId> | ||||
|       <artifactId>api</artifactId> | ||||
|       <version>5.1</version> | ||||
|       <scope>runtime</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>net.ess3</groupId> | ||||
|       <artifactId>EssentialsX</artifactId> | ||||
|       <version>2.18.0</version> | ||||
|       <scope>runtime</scope> | ||||
|       <exclusions> | ||||
|         <exclusion> | ||||
|           <artifactId>paperlib</artifactId> | ||||
|           <groupId>io.papermc</groupId> | ||||
|         </exclusion> | ||||
|       </exclusions> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>org.incendo.serverlib</groupId> | ||||
|       <artifactId>ServerLib</artifactId> | ||||
|       <version>2.2.0</version> | ||||
|       <scope>runtime</scope> | ||||
|     </dependency> | ||||
|     <dependency> | ||||
|       <groupId>junit</groupId> | ||||
|       <artifactId>junit</artifactId> | ||||
|       <version>4.13</version> | ||||
|       <scope>test</scope> | ||||
|     </dependency> | ||||
|   </dependencies> | ||||
| </project> | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,12 +21,13 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit; | ||||
|  | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.command.MainCommand; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.player.ConsolePlayer; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import org.bukkit.command.Command; | ||||
| @@ -42,35 +43,43 @@ import java.util.ArrayList; | ||||
| import java.util.Collection; | ||||
| import java.util.Collections; | ||||
| import java.util.List; | ||||
| import java.util.Locale; | ||||
|  | ||||
| public class BukkitCommand implements CommandExecutor, TabCompleter { | ||||
|  | ||||
|     @Override | ||||
|     public boolean onCommand(CommandSender commandSender, Command command, String commandLabel, | ||||
|         String[] args) { | ||||
|     public boolean onCommand( | ||||
|             CommandSender commandSender, Command command, String commandLabel, | ||||
|             String[] args | ||||
|     ) { | ||||
|         if (commandSender instanceof Player) { | ||||
|             return MainCommand.onCommand(BukkitUtil.getPlayer((Player) commandSender), args); | ||||
|             return MainCommand.onCommand(BukkitUtil.adapt((Player) commandSender), args); | ||||
|         } | ||||
|         if (commandSender instanceof ConsoleCommandSender | ||||
|             || commandSender instanceof ProxiedCommandSender | ||||
|             || commandSender instanceof RemoteConsoleCommandSender) { | ||||
|                 || commandSender instanceof ProxiedCommandSender | ||||
|                 || commandSender instanceof RemoteConsoleCommandSender) { | ||||
|             return MainCommand.onCommand(ConsolePlayer.getConsole(), args); | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public List<String> onTabComplete(CommandSender commandSender, Command command, String s, | ||||
|         String[] args) { | ||||
|     public List<String> onTabComplete( | ||||
|             CommandSender commandSender, Command command, String label, | ||||
|             String[] args | ||||
|     ) { | ||||
|         if (!(commandSender instanceof Player)) { | ||||
|             return null; | ||||
|         } | ||||
|         PlotPlayer player = BukkitUtil.getPlayer((Player) commandSender); | ||||
|         PlotPlayer<?> player = BukkitUtil.adapt((Player) commandSender); | ||||
|         if (args.length == 0) { | ||||
|             return Collections.singletonList("plots"); | ||||
|         } | ||||
|         if (!Settings.Enabled_Components.TAB_COMPLETED_ALIASES.contains(label.toLowerCase(Locale.ENGLISH))) { | ||||
|             return List.of(); | ||||
|         } | ||||
|         Collection<com.plotsquared.core.command.Command> objects = | ||||
|             MainCommand.getInstance().tab(player, args, s.endsWith(" ")); | ||||
|                 MainCommand.getInstance().tab(player, args, label.endsWith(" ")); | ||||
|         if (objects == null) { | ||||
|             return null; | ||||
|         } | ||||
| @@ -80,4 +89,5 @@ public class BukkitCommand implements CommandExecutor, TabCompleter { | ||||
|         } | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,132 +0,0 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  * | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.chat; | ||||
|  | ||||
| import org.apache.commons.lang.Validate; | ||||
|  | ||||
| import java.lang.reflect.Array; | ||||
| import java.util.Arrays; | ||||
| import java.util.Collection; | ||||
|  | ||||
| /** | ||||
|  * Represents a wrapper around an array class of an arbitrary reference type, | ||||
|  * which properly implements "value" hash code and equality functions. | ||||
|  * <p> | ||||
|  * This class is intended for use as a key to a map. | ||||
|  * </p> | ||||
|  * | ||||
|  * @param <E> The type of elements in the array. | ||||
|  * @author Glen Husman | ||||
|  * @see Arrays | ||||
|  */ | ||||
| public final class ArrayWrapper<E> { | ||||
|  | ||||
|     private E[] _array; | ||||
|  | ||||
|     /** | ||||
|      * Creates an array wrapper with some elements. | ||||
|      * | ||||
|      * @param elements The elements of the array. | ||||
|      */ | ||||
|     public ArrayWrapper(E... elements) { | ||||
|         setArray(elements); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Converts an iterable element collection to an array of elements. | ||||
|      * The iteration order of the specified object will be used as the array element order. | ||||
|      * | ||||
|      * @param list The iterable of objects which will be converted to an array. | ||||
|      * @param c    The type of the elements of the array. | ||||
|      * @return An array of elements in the specified iterable. | ||||
|      */ | ||||
|     @SuppressWarnings("unchecked") public static <T> T[] toArray(Iterable<? extends T> list, | ||||
|         Class<T> c) { | ||||
|         int size = -1; | ||||
|         if (list instanceof Collection<?>) { | ||||
|             @SuppressWarnings("rawtypes") Collection coll = (Collection) list; | ||||
|             size = coll.size(); | ||||
|         } | ||||
|  | ||||
|  | ||||
|         if (size < 0) { | ||||
|             size = 0; | ||||
|             // Ugly hack: Count it ourselves | ||||
|             for (@SuppressWarnings("unused") T element : list) { | ||||
|                 size++; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         T[] result = (T[]) Array.newInstance(c, size); | ||||
|         int i = 0; | ||||
|         for (T element : list) { // Assumes iteration order is consistent | ||||
|             result[i++] = element; // Assign array element at index THEN increment counter | ||||
|         } | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Retrieves a reference to the wrapped array instance. | ||||
|      * | ||||
|      * @return The array wrapped by this instance. | ||||
|      */ | ||||
|     public E[] getArray() { | ||||
|         return _array; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set this wrapper to wrap a new array instance. | ||||
|      * | ||||
|      * @param array The new wrapped array. | ||||
|      */ | ||||
|     public void setArray(E[] array) { | ||||
|         Validate.notNull(array, "The array must not be null."); | ||||
|         _array = array; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Determines if this object has a value equivalent to another object. | ||||
|      * | ||||
|      * @see Arrays#equals(Object[], Object[]) | ||||
|      */ | ||||
|     @SuppressWarnings("rawtypes") @Override public boolean equals(Object other) { | ||||
|         if (!(other instanceof ArrayWrapper)) { | ||||
|             return false; | ||||
|         } | ||||
|         return Arrays.equals(_array, ((ArrayWrapper) other)._array); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets the hash code represented by this objects value. | ||||
|      * | ||||
|      * @return This object's hash code. | ||||
|      * @see Arrays#hashCode(Object[]) | ||||
|      */ | ||||
|     @Override public int hashCode() { | ||||
|         return Arrays.hashCode(_array); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,944 +0,0 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  * | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.chat; | ||||
|  | ||||
| import com.google.gson.JsonArray; | ||||
| import com.google.gson.JsonElement; | ||||
| import com.google.gson.JsonObject; | ||||
| import com.google.gson.JsonParser; | ||||
| import com.google.gson.stream.JsonWriter; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.ChatColor; | ||||
| import org.bukkit.Material; | ||||
| import org.bukkit.Statistic; | ||||
| import org.bukkit.Statistic.Type; | ||||
| import org.bukkit.command.CommandSender; | ||||
| import org.bukkit.configuration.serialization.ConfigurationSerializable; | ||||
| import org.bukkit.configuration.serialization.ConfigurationSerialization; | ||||
| import org.bukkit.entity.EntityType; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.inventory.ItemStack; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.io.StringWriter; | ||||
| import java.lang.reflect.Constructor; | ||||
| import java.lang.reflect.Field; | ||||
| import java.lang.reflect.InvocationTargetException; | ||||
| import java.lang.reflect.Method; | ||||
| import java.lang.reflect.Modifier; | ||||
| import java.util.ArrayList; | ||||
| import java.util.Arrays; | ||||
| import java.util.Collections; | ||||
| import java.util.HashMap; | ||||
| import java.util.Iterator; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.UUID; | ||||
| import java.util.logging.Level; | ||||
|  | ||||
| /** | ||||
|  * Represents a formattable message. Such messages can use elements such as colors, formatting codes, hover and click data, and other features provided by the vanilla Minecraft <a href="http://minecraft.gamepedia.com/Tellraw#Raw_JSON_Text">JSON message formatter</a>. | ||||
|  * This class allows plugins to emulate the functionality of the vanilla Minecraft <a href="http://minecraft.gamepedia.com/Commands#tellraw">tellraw command</a>. | ||||
|  * <p> | ||||
|  * This class follows the builder pattern, allowing for method chaining. | ||||
|  * It is set up such that invocations of property-setting methods will affect the current editing component, | ||||
|  * and a call to {@link #then()} or {@link #then(String)} will append a new editing component to the end of the message, | ||||
|  * optionally initializing it with text. Further property-setting method calls will affect that editing component. | ||||
|  * </p> | ||||
|  */ | ||||
| public class FancyMessage | ||||
|     implements JsonRepresentedObject, Cloneable, Iterable<MessagePart>, ConfigurationSerializable { | ||||
|  | ||||
|     private static Constructor<?> nmsPacketPlayOutChatConstructor; | ||||
|     // The ChatSerializer's instance of Gson | ||||
|     private static Object nmsChatSerializerGsonInstance; | ||||
|     private static Object chatMessageType; | ||||
|     private static Method fromJsonMethod; | ||||
|     private static JsonParser _stringParser = new JsonParser(); | ||||
|  | ||||
|     static { | ||||
|         ConfigurationSerialization.registerClass(FancyMessage.class); | ||||
|     } | ||||
|  | ||||
|     private List<MessagePart> messageParts; | ||||
|     private String jsonString; | ||||
|     private boolean dirty; | ||||
|  | ||||
|     /** | ||||
|      * Creates a JSON message with text. | ||||
|      * | ||||
|      * @param firstPartText The existing text in the message. | ||||
|      */ | ||||
|     public FancyMessage(final String firstPartText) { | ||||
|         this(TextualComponent.rawText(firstPartText)); | ||||
|     } | ||||
|  | ||||
|     private FancyMessage(final TextualComponent firstPartText) { | ||||
|         messageParts = new ArrayList<>(); | ||||
|         messageParts.add(new MessagePart(firstPartText)); | ||||
|         jsonString = null; | ||||
|         dirty = false; | ||||
|         if (nmsPacketPlayOutChatConstructor == null) { | ||||
|             try { | ||||
|                 Class<?> componentClass = Reflection.getNMSClass("IChatBaseComponent"); | ||||
|                 if (!Reflection.getVersion().startsWith("v1_16")) { // < 1.16 TODO needs to be fixed before 1.17 :P | ||||
|                     nmsPacketPlayOutChatConstructor = Reflection.getNMSClass("PacketPlayOutChat") | ||||
|                             .getDeclaredConstructor(componentClass); | ||||
|                 } else { | ||||
|                     Class<Enum> chatMessageTypeClass = (Class<Enum>) Reflection.getNMSClass("ChatMessageType"); | ||||
|                     nmsPacketPlayOutChatConstructor = Reflection.getNMSClass("PacketPlayOutChat") | ||||
|                             .getDeclaredConstructor(componentClass, chatMessageTypeClass, UUID.class); | ||||
|                     chatMessageType = Enum.valueOf(chatMessageTypeClass, "SYSTEM"); | ||||
|                 } | ||||
|                 nmsPacketPlayOutChatConstructor.setAccessible(true); | ||||
|             } catch (NoSuchMethodException e) { | ||||
|                 Bukkit.getLogger() | ||||
|                     .log(Level.SEVERE, "Could not find Minecraft method or constructor.", e); | ||||
|             } catch (SecurityException e) { | ||||
|                 Bukkit.getLogger().log(Level.WARNING, "Could not access constructor.", e); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Creates a JSON message without text. | ||||
|      */ | ||||
|     public FancyMessage() { | ||||
|         this((TextualComponent) null); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Deserializes a JSON-represented message from a mapping of key-value pairs. | ||||
|      * This is called by the Bukkit serialization API. | ||||
|      * It is not intended for direct public API consumption. | ||||
|      * | ||||
|      * @param serialized The key-value mapping which represents a fancy message. | ||||
|      */ | ||||
|     @SuppressWarnings("unchecked") public static FancyMessage deserialize( | ||||
|         Map<String, Object> serialized) { | ||||
|         FancyMessage msg = new FancyMessage(); | ||||
|         msg.messageParts = (List<MessagePart>) serialized.get("messageParts"); | ||||
|         msg.jsonString = serialized.containsKey("JSON") ? serialized.get("JSON").toString() : null; | ||||
|         msg.dirty = !serialized.containsKey("JSON"); | ||||
|         return msg; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Deserializes a fancy message from its JSON representation. This JSON representation is of the format of | ||||
|      * that returned by {@link #toJSONString()}, and is compatible with vanilla inputs. | ||||
|      * | ||||
|      * @param json The JSON string which represents a fancy message. | ||||
|      * @return A {@code FancyMessage} representing the parameterized JSON message. | ||||
|      */ | ||||
|     public static FancyMessage deserialize(String json) { | ||||
|         JsonObject serialized = _stringParser.parse(json).getAsJsonObject(); | ||||
|         JsonArray extra = serialized.getAsJsonArray("extra"); // Get the extra component | ||||
|         FancyMessage returnVal = new FancyMessage(); | ||||
|         returnVal.messageParts.clear(); | ||||
|         for (JsonElement mPrt : extra) { | ||||
|             MessagePart component = new MessagePart(); | ||||
|             JsonObject messagePart = mPrt.getAsJsonObject(); | ||||
|             for (Map.Entry<String, JsonElement> entry : messagePart.entrySet()) { | ||||
|                 // Deserialize text | ||||
|                 if (TextualComponent.isTextKey(entry.getKey())) { | ||||
|                     // The map mimics the YAML serialization, which has a "key" field and one or more "value" fields | ||||
|                     Map<String, Object> serializedMapForm = | ||||
|                         new HashMap<>(); // Must be object due to Bukkit serializer API compliance | ||||
|                     serializedMapForm.put("key", entry.getKey()); | ||||
|                     if (entry.getValue().isJsonPrimitive()) { | ||||
|                         // Assume string | ||||
|                         serializedMapForm.put("value", entry.getValue().getAsString()); | ||||
|                     } else { | ||||
|                         // Composite object, but we assume each element is a string | ||||
|                         for (Map.Entry<String, JsonElement> compositeNestedElement : entry | ||||
|                             .getValue().getAsJsonObject().entrySet()) { | ||||
|                             serializedMapForm.put("value." + compositeNestedElement.getKey(), | ||||
|                                 compositeNestedElement.getValue().getAsString()); | ||||
|                         } | ||||
|                     } | ||||
|                     component.text = TextualComponent.deserialize(serializedMapForm); | ||||
|                 } else if (MessagePart.stylesToNames.inverse().containsKey(entry.getKey())) { | ||||
|                     if (entry.getValue().getAsBoolean()) { | ||||
|                         component.styles | ||||
|                             .add(MessagePart.stylesToNames.inverse().get(entry.getKey())); | ||||
|                     } | ||||
|                 } else if (entry.getKey().equals("color")) { | ||||
|                     component.color = | ||||
|                         ChatColor.valueOf(entry.getValue().getAsString().toUpperCase()); | ||||
|                 } else if (entry.getKey().equals("clickEvent")) { | ||||
|                     JsonObject object = entry.getValue().getAsJsonObject(); | ||||
|                     component.clickActionName = object.get("action").getAsString(); | ||||
|                     component.clickActionData = object.get("value").getAsString(); | ||||
|                 } else if (entry.getKey().equals("hoverEvent")) { | ||||
|                     JsonObject object = entry.getValue().getAsJsonObject(); | ||||
|                     component.hoverActionName = object.get("action").getAsString(); | ||||
|                     if (object.get("value").isJsonPrimitive()) { | ||||
|                         // Assume string | ||||
|                         component.hoverActionData = | ||||
|                             new JsonString(object.get("value").getAsString()); | ||||
|                     } else { | ||||
|                         // Assume composite type | ||||
|                         // The only composite type we currently store is another FancyMessage | ||||
|                         // Therefore, recursion time! | ||||
|                         component.hoverActionData = deserialize(object.get("value") | ||||
|                             .toString() /* This should properly serialize the JSON object as a JSON string */); | ||||
|                     } | ||||
|                 } else if (entry.getKey().equals("insertion")) { | ||||
|                     component.insertionData = entry.getValue().getAsString(); | ||||
|                 } else if (entry.getKey().equals("with")) { | ||||
|                     for (JsonElement object : entry.getValue().getAsJsonArray()) { | ||||
|                         if (object.isJsonPrimitive()) { | ||||
|                             component.translationReplacements | ||||
|                                 .add(new JsonString(object.getAsString())); | ||||
|                         } else { | ||||
|                             // Only composite type stored in this array is - again - FancyMessages | ||||
|                             // Recurse within this function to parse this as a translation replacement | ||||
|                             component.translationReplacements.add(deserialize(object.toString())); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             returnVal.messageParts.add(component); | ||||
|         } | ||||
|         return returnVal; | ||||
|     } | ||||
|  | ||||
|     @Override public FancyMessage clone() throws CloneNotSupportedException { | ||||
|         FancyMessage instance = (FancyMessage) super.clone(); | ||||
|         instance.messageParts = new ArrayList<>(messageParts.size()); | ||||
|         for (int i = 0; i < messageParts.size(); i++) { | ||||
|             instance.messageParts.add(i, messageParts.get(i).clone()); | ||||
|         } | ||||
|         instance.dirty = false; | ||||
|         instance.jsonString = null; | ||||
|         return instance; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sets the text of the current editing component to a value. | ||||
|      * | ||||
|      * @param text The new text of the current editing component. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage text(String text) { | ||||
|         MessagePart latest = latest(); | ||||
|         latest.text = TextualComponent.rawText(text); | ||||
|         dirty = true; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sets the text of the current editing component to a value. | ||||
|      * | ||||
|      * @param text The new text of the current editing component. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage text(TextualComponent text) { | ||||
|         MessagePart latest = latest(); | ||||
|         latest.text = text; | ||||
|         dirty = true; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sets the color of the current editing component to a value. | ||||
|      * | ||||
|      * @param color The new color of the current editing component. | ||||
|      * @return This builder instance. | ||||
|      * @throws IllegalArgumentException If the specified {@code ChatColor} enumeration value is not a color (but a format value). | ||||
|      */ | ||||
|     public FancyMessage color(final ChatColor color) { | ||||
|         if (!color.isColor()) { | ||||
|             throw new IllegalArgumentException(color.name() + " is not a color"); | ||||
|         } | ||||
|         latest().color = color; | ||||
|         dirty = true; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sets the stylization of the current editing component. | ||||
|      * | ||||
|      * @param styles The array of styles to apply to the editing component. | ||||
|      * @return This builder instance. | ||||
|      * @throws IllegalArgumentException If any of the enumeration values in the array do not represent formatters. | ||||
|      */ | ||||
|     public FancyMessage style(ChatColor... styles) { | ||||
|         for (final ChatColor style : styles) { | ||||
|             if (!style.isFormat()) { | ||||
|                 throw new IllegalArgumentException(style.name() + " is not a style"); | ||||
|             } | ||||
|         } | ||||
|         latest().styles.addAll(Arrays.asList(styles)); | ||||
|         dirty = true; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to instruct the client to open a file on the client side filesystem when the currently edited part of the {@code FancyMessage} is clicked. | ||||
|      * | ||||
|      * @param path The path of the file on the client filesystem. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage file(final String path) { | ||||
|         onClick("open_file", path); | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to instruct the client to open a webpage in the client's web browser when the currently edited part of the {@code FancyMessage} is clicked. | ||||
|      * | ||||
|      * @param url The URL of the page to open when the link is clicked. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage link(final String url) { | ||||
|         onClick("open_url", url); | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to instruct the client to replace the chat input box content with the specified string when the currently edited part of the {@code FancyMessage} is clicked. | ||||
|      * The client will not immediately send the command to the server to be executed unless the client player submits the command/chat message, usually with the enter key. | ||||
|      * | ||||
|      * @param command The text to display in the chat bar of the client. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage suggest(final String command) { | ||||
|         onClick("suggest_command", command); | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to instruct the client to append the chat input box content with the specified string when the currently edited part of the {@code FancyMessage} is SHIFT-CLICKED. | ||||
|      * The client will not immediately send the command to the server to be executed unless the client player submits the command/chat message, usually with the enter key. | ||||
|      * | ||||
|      * @param command The text to append to the chat bar of the client. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage insert(final String command) { | ||||
|         latest().insertionData = command; | ||||
|         dirty = true; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to instruct the client to send the specified string to the server as a chat message when the currently edited part of the {@code FancyMessage} is clicked. | ||||
|      * The client <b>will</b> immediately send the command to the server to be executed when the editing component is clicked. | ||||
|      * | ||||
|      * @param command The text to display in the chat bar of the client. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage command(final String command) { | ||||
|         onClick("run_command", command); | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to display information about an achievement when the client hovers over the text. | ||||
|      * <p>Tooltips do not inherit display characteristics, such as color and styles, from the message component on which they are applied.</p> | ||||
|      * | ||||
|      * @param name The name of the achievement to display, excluding the "achievement." prefix. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage achievementTooltip(final String name) { | ||||
|         onHover("show_achievement", new JsonString("achievement." + name)); | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to display information about a parameterless statistic when the client hovers over the text. | ||||
|      * <p>Tooltips do not inherit display characteristics, such as color and styles, from the message component on which they are applied.</p> | ||||
|      * | ||||
|      * @param which The statistic to display. | ||||
|      * @return This builder instance. | ||||
|      * @throws IllegalArgumentException If the statistic requires a parameter which was not supplied. | ||||
|      */ | ||||
|     public FancyMessage statisticTooltip(final Statistic which) { | ||||
|         Type type = which.getType(); | ||||
|         if (type != Type.UNTYPED) { | ||||
|             throw new IllegalArgumentException( | ||||
|                 "That statistic requires an additional " + type + " parameter!"); | ||||
|         } | ||||
|         try { | ||||
|             Object statistic = Reflection | ||||
|                 .getMethod(Reflection.getOBCClass("CraftStatistic"), "getNMSStatistic", | ||||
|                     Statistic.class).invoke(null, which); | ||||
|             return achievementTooltip( | ||||
|                 (String) Reflection.getField(Reflection.getNMSClass("Statistic"), "name") | ||||
|                     .get(statistic)); | ||||
|         } catch (IllegalAccessException e) { | ||||
|             Bukkit.getLogger().log(Level.WARNING, "Could not access method.", e); | ||||
|             return this; | ||||
|         } catch (IllegalArgumentException e) { | ||||
|             Bukkit.getLogger().log(Level.WARNING, "Argument could not be passed.", e); | ||||
|             return this; | ||||
|         } catch (InvocationTargetException e) { | ||||
|             Bukkit.getLogger() | ||||
|                 .log(Level.WARNING, "A error has occurred during invoking of method.", e); | ||||
|             return this; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to display information about a statistic parameter with a material when the client hovers over the text. | ||||
|      * <p>Tooltips do not inherit display characteristics, such as color and styles, from the message component on which they are applied.</p> | ||||
|      * | ||||
|      * @param which The statistic to display. | ||||
|      * @param item  The sole material parameter to the statistic. | ||||
|      * @return This builder instance. | ||||
|      * @throws IllegalArgumentException If the statistic requires a parameter which was not supplied, or was supplied a parameter that was not required. | ||||
|      */ | ||||
|     public FancyMessage statisticTooltip(final Statistic which, Material item) { | ||||
|         Type type = which.getType(); | ||||
|         if (type == Type.UNTYPED) { | ||||
|             throw new IllegalArgumentException("That statistic needs no additional parameter!"); | ||||
|         } | ||||
|         if ((type == Type.BLOCK && item.isBlock()) || type == Type.ENTITY) { | ||||
|             throw new IllegalArgumentException( | ||||
|                 "Wrong parameter type for that statistic - needs " + type + "!"); | ||||
|         } | ||||
|         try { | ||||
|             Object statistic = Reflection | ||||
|                 .getMethod(Reflection.getOBCClass("CraftStatistic"), "getMaterialStatistic", | ||||
|                     Statistic.class, Material.class).invoke(null, which, item); | ||||
|             return achievementTooltip( | ||||
|                 (String) Reflection.getField(Reflection.getNMSClass("Statistic"), "name") | ||||
|                     .get(statistic)); | ||||
|         } catch (IllegalAccessException e) { | ||||
|             Bukkit.getLogger().log(Level.WARNING, "Could not access method.", e); | ||||
|             return this; | ||||
|         } catch (IllegalArgumentException e) { | ||||
|             Bukkit.getLogger().log(Level.WARNING, "Argument could not be passed.", e); | ||||
|             return this; | ||||
|         } catch (InvocationTargetException e) { | ||||
|             Bukkit.getLogger() | ||||
|                 .log(Level.WARNING, "A error has occurred during invoking of method.", e); | ||||
|             return this; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to display information about a statistic parameter with an entity type when the client hovers over the text. | ||||
|      * <p>Tooltips do not inherit display characteristics, such as color and styles, from the message component on which they are applied.</p> | ||||
|      * | ||||
|      * @param which  The statistic to display. | ||||
|      * @param entity The sole entity type parameter to the statistic. | ||||
|      * @return This builder instance. | ||||
|      * @throws IllegalArgumentException If the statistic requires a parameter which was not supplied, or was supplied a parameter that was not required. | ||||
|      */ | ||||
|     public FancyMessage statisticTooltip(final Statistic which, EntityType entity) { | ||||
|         Type type = which.getType(); | ||||
|         if (type == Type.UNTYPED) { | ||||
|             throw new IllegalArgumentException("That statistic needs no additional parameter!"); | ||||
|         } | ||||
|         if (type != Type.ENTITY) { | ||||
|             throw new IllegalArgumentException( | ||||
|                 "Wrong parameter type for that statistic - needs " + type + "!"); | ||||
|         } | ||||
|         try { | ||||
|             Object statistic = Reflection | ||||
|                 .getMethod(Reflection.getOBCClass("CraftStatistic"), "getEntityStatistic", | ||||
|                     Statistic.class, EntityType.class).invoke(null, which, entity); | ||||
|             return achievementTooltip( | ||||
|                 (String) Reflection.getField(Reflection.getNMSClass("Statistic"), "name") | ||||
|                     .get(statistic)); | ||||
|         } catch (IllegalAccessException e) { | ||||
|             Bukkit.getLogger().log(Level.WARNING, "Could not access method.", e); | ||||
|             return this; | ||||
|         } catch (IllegalArgumentException e) { | ||||
|             Bukkit.getLogger().log(Level.WARNING, "Argument could not be passed.", e); | ||||
|             return this; | ||||
|         } catch (InvocationTargetException e) { | ||||
|             Bukkit.getLogger() | ||||
|                 .log(Level.WARNING, "A error has occurred during invoking of method.", e); | ||||
|             return this; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to display information about an item when the client hovers over the text. | ||||
|      * <p>Tooltips do not inherit display characteristics, such as color and styles, from the message component on which they are applied.</p> | ||||
|      * | ||||
|      * @param itemJSON A string representing the JSON-serialized NBT data tag of an {@link ItemStack}. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage itemTooltip(final String itemJSON) { | ||||
|         onHover("show_item", new JsonString( | ||||
|             itemJSON)); // Seems a bit hacky, considering we have a JSON object as a parameter | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to display information about an item when the client hovers over the text. | ||||
|      * <p>Tooltips do not inherit display characteristics, such as color and styles, from the message component on which they are applied.</p> | ||||
|      * | ||||
|      * @param itemStack The stack for which to display information. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage itemTooltip(final ItemStack itemStack) { | ||||
|         try { | ||||
|             Object nmsItem = Reflection | ||||
|                 .getMethod(Reflection.getOBCClass("inventory.CraftItemStack"), "asNMSCopy", | ||||
|                     ItemStack.class).invoke(null, itemStack); | ||||
|             return itemTooltip(Reflection.getMethod(Reflection.getNMSClass("ItemStack"), "save", | ||||
|                 Reflection.getNMSClass("NBTTagCompound")) | ||||
|                 .invoke(nmsItem, Reflection.getNMSClass("NBTTagCompound").newInstance()) | ||||
|                 .toString()); | ||||
|         } catch (Exception e) { | ||||
|             e.printStackTrace(); | ||||
|             return this; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to display raw text when the client hovers over the text. | ||||
|      * <p>Tooltips do not inherit display characteristics, such as color and styles, from the message component on which they are applied.</p> | ||||
|      * | ||||
|      * @param text The text, which supports newlines, which will be displayed to the client upon hovering. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage tooltip(final String text) { | ||||
|         onHover("show_text", new JsonString(text)); | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to display raw text when the client hovers over the text. | ||||
|      * <p>Tooltips do not inherit display characteristics, such as color and styles, from the message component on which they are applied.</p> | ||||
|      * | ||||
|      * @param lines The lines of text which will be displayed to the client upon hovering. The iteration order of this object will be the order in which the lines of the tooltip are created. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage tooltip(final Iterable<String> lines) { | ||||
|         tooltip(ArrayWrapper.toArray(lines, String.class)); | ||||
|         return this; | ||||
|     } | ||||
|     /* | ||||
|  | ||||
|     /** | ||||
|      * If the text is a translatable key, and it has replaceable values, this function can be used to set the replacements that will be used in the message. | ||||
|      * @param replacements The replacements, in order, that will be used in the language-specific message. | ||||
|      * @return This builder instance. | ||||
|      */   /* ------------ | ||||
|     public FancyMessage translationReplacements(final Iterable<? extends CharSequence> replacements){ | ||||
|         for(CharSequence str : replacements){ | ||||
|             latest().translationReplacements.add(new JsonString(str)); | ||||
|         } | ||||
|  | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     */ | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to display raw text when the client hovers over the text. | ||||
|      * <p>Tooltips do not inherit display characteristics, such as color and styles, from the message component on which they are applied.</p> | ||||
|      * | ||||
|      * @param lines The lines of text which will be displayed to the client upon hovering. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage tooltip(final String... lines) { | ||||
|         StringBuilder builder = new StringBuilder(); | ||||
|         for (int i = 0; i < lines.length; i++) { | ||||
|             builder.append(lines[i]); | ||||
|             if (i != lines.length - 1) { | ||||
|                 builder.append('\n'); | ||||
|             } | ||||
|         } | ||||
|         tooltip(builder.toString()); | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to display formatted text when the client hovers over the text. | ||||
|      * <p>Tooltips do not inherit display characteristics, such as color and styles, from the message component on which they are applied.</p> | ||||
|      * | ||||
|      * @param text The formatted text which will be displayed to the client upon hovering. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage formattedTooltip(FancyMessage text) { | ||||
|         for (MessagePart component : text.messageParts) { | ||||
|             if (component.clickActionData != null && component.clickActionName != null) { | ||||
|                 throw new IllegalArgumentException("The tooltip text cannot have click data."); | ||||
|             } else if (component.hoverActionData != null && component.hoverActionName != null) { | ||||
|                 throw new IllegalArgumentException("The tooltip text cannot have a tooltip."); | ||||
|             } | ||||
|         } | ||||
|         onHover("show_text", text); | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to display the specified lines of formatted text when the client hovers over the text. | ||||
|      * <p>Tooltips do not inherit display characteristics, such as color and styles, from the message component on which they are applied.</p> | ||||
|      * | ||||
|      * @param lines The lines of formatted text which will be displayed to the client upon hovering. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage formattedTooltip(FancyMessage... lines) { | ||||
|         if (lines.length < 1) { | ||||
|             onHover(null, null); // Clear tooltip | ||||
|             return this; | ||||
|         } | ||||
|  | ||||
|         FancyMessage result = new FancyMessage(); | ||||
|         result.messageParts | ||||
|             .clear(); // Remove the one existing text component that exists by default, which destabilizes the object | ||||
|  | ||||
|         for (int i = 0; i < lines.length; i++) { | ||||
|             try { | ||||
|                 for (MessagePart component : lines[i]) { | ||||
|                     if (component.clickActionData != null && component.clickActionName != null) { | ||||
|                         throw new IllegalArgumentException( | ||||
|                             "The tooltip text cannot have click data."); | ||||
|                     } else if (component.hoverActionData != null | ||||
|                         && component.hoverActionName != null) { | ||||
|                         throw new IllegalArgumentException( | ||||
|                             "The tooltip text cannot have a tooltip."); | ||||
|                     } | ||||
|                     if (component.hasText()) { | ||||
|                         result.messageParts.add(component.clone()); | ||||
|                     } | ||||
|                 } | ||||
|                 if (i != lines.length - 1) { | ||||
|                     result.messageParts.add(new MessagePart(TextualComponent.rawText("\n"))); | ||||
|                 } | ||||
|             } catch (CloneNotSupportedException e) { | ||||
|                 Bukkit.getLogger().log(Level.WARNING, "Failed to clone object", e); | ||||
|                 return this; | ||||
|             } | ||||
|         } | ||||
|         return formattedTooltip( | ||||
|             result.messageParts.isEmpty() ? null : result); // Throws NPE if size is 0, intended | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the behavior of the current editing component to display the specified lines of formatted text when the client hovers over the text. | ||||
|      * <p>Tooltips do not inherit display characteristics, such as color and styles, from the message component on which they are applied.</p> | ||||
|      * | ||||
|      * @param lines The lines of text which will be displayed to the client upon hovering. The iteration order of this object will be the order in which the lines of the tooltip are created. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage formattedTooltip(final Iterable<FancyMessage> lines) { | ||||
|         return formattedTooltip(ArrayWrapper.toArray(lines, FancyMessage.class)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * If the text is a translatable key, and it has replaceable values, this function can be used to set the replacements that will be used in the message. | ||||
|      * | ||||
|      * @param replacements The replacements, in order, that will be used in the language-specific message. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage translationReplacements(final String... replacements) { | ||||
|         for (String str : replacements) { | ||||
|             latest().translationReplacements.add(new JsonString(str)); | ||||
|         } | ||||
|         dirty = true; | ||||
|  | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * If the text is a translatable key, and it has replaceable values, this function can be used to set the replacements that will be used in the message. | ||||
|      * | ||||
|      * @param replacements The replacements, in order, that will be used in the language-specific message. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage translationReplacements(final FancyMessage... replacements) { | ||||
|         Collections.addAll(latest().translationReplacements, replacements); | ||||
|  | ||||
|         dirty = true; | ||||
|  | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * If the text is a translatable key, and it has replaceable values, this function can be used to set the replacements that will be used in the message. | ||||
|      * | ||||
|      * @param replacements The replacements, in order, that will be used in the language-specific message. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage translationReplacements(final Iterable<FancyMessage> replacements) { | ||||
|         return translationReplacements(ArrayWrapper.toArray(replacements, FancyMessage.class)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Terminate construction of the current editing component, and begin construction of a new message component. | ||||
|      * After a successful call to this method, all setter methods will refer to a new message component, created as a result of the call to this method. | ||||
|      * | ||||
|      * @param text The text which will populate the new message component. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage then(final String text) { | ||||
|         return then(TextualComponent.rawText(text)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Terminate construction of the current editing component, and begin construction of a new message component. | ||||
|      * After a successful call to this method, all setter methods will refer to a new message component, created as a result of the call to this method. | ||||
|      * | ||||
|      * @param text The text which will populate the new message component. | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage then(final TextualComponent text) { | ||||
|         if (!latest().hasText()) { | ||||
|             throw new IllegalStateException("previous message part has no text"); | ||||
|         } | ||||
|         messageParts.add(new MessagePart(text)); | ||||
|         dirty = true; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Terminate construction of the current editing component, and begin construction of a new message component. | ||||
|      * After a successful call to this method, all setter methods will refer to a new message component, created as a result of the call to this method. | ||||
|      * | ||||
|      * @return This builder instance. | ||||
|      */ | ||||
|     public FancyMessage then() { | ||||
|         if (!latest().hasText()) { | ||||
|             throw new IllegalStateException("previous message part has no text"); | ||||
|         } | ||||
|         messageParts.add(new MessagePart()); | ||||
|         dirty = true; | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     @Override public void writeJson(JsonWriter writer) throws IOException { | ||||
|         if (messageParts.size() == 1) { | ||||
|             latest().writeJson(writer); | ||||
|         } else { | ||||
|             writer.beginObject().name("text").value("").name("extra").beginArray(); | ||||
|             for (final MessagePart part : this) { | ||||
|                 part.writeJson(writer); | ||||
|             } | ||||
|             writer.endArray().endObject(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Serialize this fancy message, converting it into syntactically-valid JSON using a {@link JsonWriter}. | ||||
|      * This JSON should be compatible with vanilla formatter commands such as {@code /tellraw}. | ||||
|      * | ||||
|      * @return The JSON string representing this object. | ||||
|      */ | ||||
|     public String toJSONString() { | ||||
|         if (!dirty && jsonString != null) { | ||||
|             return jsonString; | ||||
|         } | ||||
|         StringWriter string = new StringWriter(); | ||||
|         JsonWriter json = new JsonWriter(string); | ||||
|         try { | ||||
|             writeJson(json); | ||||
|             json.close(); | ||||
|         } catch (IOException e) { | ||||
|             throw new RuntimeException("invalid message"); | ||||
|         } | ||||
|         jsonString = string.toString(); | ||||
|         dirty = false; | ||||
|         return jsonString; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sends this message to a player. The player will receive the fully-fledged formatted display of this message. | ||||
|      * | ||||
|      * @param player The player who will receive the message. | ||||
|      */ | ||||
|     public void send(Player player) { | ||||
|         send(player, toJSONString()); | ||||
|     } | ||||
|  | ||||
|     private void send(CommandSender sender, String jsonString) { | ||||
|         if (!(sender instanceof Player)) { | ||||
|             sender.sendMessage(toOldMessageFormat()); | ||||
|             return; | ||||
|         } | ||||
|         Player player = (Player) sender; | ||||
|         try { | ||||
|             Object handle = Reflection.getHandle(player); | ||||
|             Object connection = | ||||
|                 Reflection.getField(handle.getClass(), "playerConnection").get(handle); | ||||
|             Reflection | ||||
|                 .getMethod(connection.getClass(), "sendPacket", Reflection.getNMSClass("Packet")) | ||||
|                 .invoke(connection, createChatPacket(jsonString, ((Player) sender).getUniqueId())); | ||||
|         } catch (IllegalArgumentException e) { | ||||
|             Bukkit.getLogger().log(Level.WARNING, "Argument could not be passed.", e); | ||||
|         } catch (IllegalAccessException e) { | ||||
|             Bukkit.getLogger().log(Level.WARNING, "Could not access method.", e); | ||||
|         } catch (InstantiationException e) { | ||||
|             Bukkit.getLogger().log(Level.WARNING, "Underlying class is abstract.", e); | ||||
|         } catch (InvocationTargetException e) { | ||||
|             Bukkit.getLogger() | ||||
|                 .log(Level.WARNING, "A error has occurred during invoking of method.", e); | ||||
|         } catch (NoSuchMethodException e) { | ||||
|             Bukkit.getLogger().log(Level.WARNING, "Could not find method.", e); | ||||
|         } catch (ClassNotFoundException e) { | ||||
|             Bukkit.getLogger().log(Level.WARNING, "Could not find class.", e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private Object createChatPacket(String json, UUID receiver) | ||||
|         throws IllegalArgumentException, IllegalAccessException, InstantiationException, | ||||
|         InvocationTargetException, NoSuchMethodException, ClassNotFoundException { | ||||
|         if (nmsChatSerializerGsonInstance == null) { | ||||
|             // Find the field and its value, completely bypassing obfuscation | ||||
|             Class<?> chatSerializerClazz; | ||||
|  | ||||
|             // Get the three parts of the version string (major version is currently unused) | ||||
|             // vX_Y_RZ | ||||
|             //   X = major | ||||
|             //   Y = minor | ||||
|             //   Z = revision | ||||
|             final String version = Reflection.getVersion(); | ||||
|             String[] split = | ||||
|                 version.substring(1, version.length() - 1).split("_"); // Remove trailing dot | ||||
|             //int majorVersion = Integer.parseInt(split[0]); | ||||
|             int minorVersion = Integer.parseInt(split[1]); | ||||
|             int revisionVersion = Integer.parseInt(split[2].substring(1)); // Substring to ignore R | ||||
|  | ||||
|             if (minorVersion < 8 || (minorVersion == 8 && revisionVersion == 1)) { | ||||
|                 chatSerializerClazz = Reflection.getNMSClass("ChatSerializer"); | ||||
|             } else { | ||||
|                 chatSerializerClazz = Reflection.getNMSClass("IChatBaseComponent$ChatSerializer"); | ||||
|             } | ||||
|  | ||||
|             if (chatSerializerClazz == null) { | ||||
|                 throw new ClassNotFoundException("Can't find the ChatSerializer class"); | ||||
|             } | ||||
|  | ||||
|             for (Field declaredField : chatSerializerClazz.getDeclaredFields()) { | ||||
|                 if (Modifier.isFinal(declaredField.getModifiers()) && Modifier | ||||
|                     .isStatic(declaredField.getModifiers()) && declaredField.getType().getName() | ||||
|                     .endsWith("Gson")) { | ||||
|                     // We've found our field | ||||
|                     declaredField.setAccessible(true); | ||||
|                     nmsChatSerializerGsonInstance = declaredField.get(null); | ||||
|                     fromJsonMethod = nmsChatSerializerGsonInstance.getClass() | ||||
|                         .getMethod("fromJson", String.class, Class.class); | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Since the method is so simple, and all the obfuscated methods have the same name, it's easier to reimplement 'IChatBaseComponent a(String)' than to reflectively call it | ||||
|         // Of course, the implementation may change, but fuzzy matches might break with signature changes | ||||
|         Object serializedChatComponent = fromJsonMethod.invoke(nmsChatSerializerGsonInstance, json, | ||||
|             Reflection.getNMSClass("IChatBaseComponent")); | ||||
|  | ||||
|         if (!Reflection.getVersion().startsWith("v1_16")) { | ||||
|             return nmsPacketPlayOutChatConstructor.newInstance(serializedChatComponent); | ||||
|         } else { | ||||
|             return nmsPacketPlayOutChatConstructor.newInstance(serializedChatComponent, chatMessageType, receiver); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sends this message to a command sender. | ||||
|      * If the sender is a player, they will receive the fully-fledged formatted display of this message. | ||||
|      * Otherwise, they will receive a version of this message with less formatting. | ||||
|      * | ||||
|      * @param sender The command sender who will receive the message. | ||||
|      * @see #toOldMessageFormat() | ||||
|      */ | ||||
|     public void send(CommandSender sender) { | ||||
|         send(sender, toJSONString()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Sends this message to multiple command senders. | ||||
|      * | ||||
|      * @param senders The command senders who will receive the message. | ||||
|      * @see #send(CommandSender) | ||||
|      */ | ||||
|     public void send(final Iterable<? extends CommandSender> senders) { | ||||
|         String string = toJSONString(); | ||||
|         for (final CommandSender sender : senders) { | ||||
|             send(sender, string); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Convert this message to a human-readable string with limited formatting. | ||||
|      * This method is used to send this message to clients without JSON formatting support. | ||||
|      * <p> | ||||
|      * Serialization of this message by using this message will include (in this order for each message part): | ||||
|      * <ol> | ||||
|      * <li>The color of each message part.</li> | ||||
|      * <li>The applicable stylizations for each message part.</li> | ||||
|      * <li>The core text of the message part.</li> | ||||
|      * </ol> | ||||
|      * The primary omissions are tooltips and clickable actions. Consequently, this method should be used only as a last resort. | ||||
|      * <p> | ||||
|      * Color and formatting can be removed from the returned string by using {@link ChatColor#stripColor(String)}.</p> | ||||
|      * | ||||
|      * @return A human-readable string representing limited formatting in addition to the core text of this message. | ||||
|      */ | ||||
|     public String toOldMessageFormat() { | ||||
|         StringBuilder result = new StringBuilder(); | ||||
|         for (MessagePart part : this) { | ||||
|             result.append(part.color == null ? "" : part.color); | ||||
|             for (ChatColor formatSpecifier : part.styles) { | ||||
|                 result.append(formatSpecifier); | ||||
|             } | ||||
|             result.append(part.text); | ||||
|         } | ||||
|         return result.toString(); | ||||
|     } | ||||
|  | ||||
|     private MessagePart latest() { | ||||
|         return messageParts.get(messageParts.size() - 1); | ||||
|     } | ||||
|  | ||||
|     private void onClick(final String name, final String data) { | ||||
|         final MessagePart latest = latest(); | ||||
|         latest.clickActionName = name; | ||||
|         latest.clickActionData = data; | ||||
|         dirty = true; | ||||
|     } | ||||
|  | ||||
|     private void onHover(final String name, final JsonRepresentedObject data) { | ||||
|         final MessagePart latest = latest(); | ||||
|         latest.hoverActionName = name; | ||||
|         latest.hoverActionData = data; | ||||
|         dirty = true; | ||||
|     } | ||||
|  | ||||
|     // Doc copied from interface | ||||
|     public Map<String, Object> serialize() { | ||||
|         HashMap<String, Object> map = new HashMap<>(); | ||||
|         map.put("messageParts", messageParts); | ||||
|         //		map.put("JSON", toJSONString()); | ||||
|         return map; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * <b>Internally called method. Not for API consumption.</b> | ||||
|      */ | ||||
|     public Iterator<MessagePart> iterator() { | ||||
|         return messageParts.iterator(); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,70 +0,0 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  * | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.chat; | ||||
|  | ||||
| import com.google.gson.stream.JsonWriter; | ||||
| import org.bukkit.configuration.serialization.ConfigurationSerializable; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * Represents a JSON string value. | ||||
|  * Writes by this object will not write name values nor begin/end objects in the JSON stream. | ||||
|  * All writes merely write the represented string value. | ||||
|  */ | ||||
| final class JsonString implements JsonRepresentedObject, ConfigurationSerializable { | ||||
|  | ||||
|     private String _value; | ||||
|  | ||||
|     public JsonString(CharSequence value) { | ||||
|         _value = value == null ? null : value.toString(); | ||||
|     } | ||||
|  | ||||
|     public static JsonString deserialize(Map<String, Object> map) { | ||||
|         return new JsonString(map.get("stringValue").toString()); | ||||
|     } | ||||
|  | ||||
|     @Override public void writeJson(JsonWriter writer) throws IOException { | ||||
|         writer.value(getValue()); | ||||
|     } | ||||
|  | ||||
|     public String getValue() { | ||||
|         return _value; | ||||
|     } | ||||
|  | ||||
|     public Map<String, Object> serialize() { | ||||
|         HashMap<String, Object> theSingleValue = new HashMap<String, Object>(); | ||||
|         theSingleValue.put("stringValue", _value); | ||||
|         return theSingleValue; | ||||
|     } | ||||
|  | ||||
|     @Override public String toString() { | ||||
|         return _value; | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,179 +0,0 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  * | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.chat; | ||||
|  | ||||
| import com.google.common.collect.BiMap; | ||||
| import com.google.common.collect.ImmutableBiMap; | ||||
| import com.google.gson.stream.JsonWriter; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.ChatColor; | ||||
| import org.bukkit.configuration.serialization.ConfigurationSerializable; | ||||
| import org.bukkit.configuration.serialization.ConfigurationSerialization; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.util.ArrayList; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
| import java.util.logging.Level; | ||||
|  | ||||
| /** | ||||
|  * Internal class: Represents a component of a JSON-serializable {@link FancyMessage}. | ||||
|  */ | ||||
| final class MessagePart implements JsonRepresentedObject, ConfigurationSerializable, Cloneable { | ||||
|  | ||||
|     static final BiMap<ChatColor, String> stylesToNames; | ||||
|  | ||||
|     static { | ||||
|         ImmutableBiMap.Builder<ChatColor, String> builder = ImmutableBiMap.builder(); | ||||
|         for (final ChatColor style : ChatColor.values()) { | ||||
|             if (!style.isFormat()) { | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             String styleName; | ||||
|             switch (style) { | ||||
|                 case MAGIC: | ||||
|                     styleName = "obfuscated"; | ||||
|                     break; | ||||
|                 case UNDERLINE: | ||||
|                     styleName = "underlined"; | ||||
|                     break; | ||||
|                 default: | ||||
|                     styleName = style.name().toLowerCase(); | ||||
|                     break; | ||||
|             } | ||||
|  | ||||
|             builder.put(style, styleName); | ||||
|         } | ||||
|         stylesToNames = builder.build(); | ||||
|     } | ||||
|  | ||||
|     static { | ||||
|         ConfigurationSerialization.registerClass(MessagePart.class); | ||||
|     } | ||||
|  | ||||
|     ChatColor color = ChatColor.WHITE; | ||||
|     ArrayList<ChatColor> styles = new ArrayList<>(); | ||||
|     String clickActionName = null; | ||||
|     String clickActionData = null; | ||||
|     String hoverActionName = null; | ||||
|     JsonRepresentedObject hoverActionData = null; | ||||
|     TextualComponent text = null; | ||||
|     String insertionData = null; | ||||
|     ArrayList<JsonRepresentedObject> translationReplacements = new ArrayList<>(); | ||||
|  | ||||
|     MessagePart(final TextualComponent text) { | ||||
|         this.text = text; | ||||
|     } | ||||
|  | ||||
|     MessagePart() { | ||||
|         this.text = null; | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("unchecked") | ||||
|     public static MessagePart deserialize(Map<String, Object> serialized) { | ||||
|         MessagePart part = new MessagePart((TextualComponent) serialized.get("text")); | ||||
|         part.styles = (ArrayList<ChatColor>) serialized.get("styles"); | ||||
|         part.color = ChatColor.getByChar(serialized.get("color").toString()); | ||||
|         part.hoverActionName = (String) serialized.get("hoverActionName"); | ||||
|         part.hoverActionData = (JsonRepresentedObject) serialized.get("hoverActionData"); | ||||
|         part.clickActionName = (String) serialized.get("clickActionName"); | ||||
|         part.clickActionData = (String) serialized.get("clickActionData"); | ||||
|         part.insertionData = (String) serialized.get("insertion"); | ||||
|         part.translationReplacements = | ||||
|             (ArrayList<JsonRepresentedObject>) serialized.get("translationReplacements"); | ||||
|         return part; | ||||
|     } | ||||
|  | ||||
|     boolean hasText() { | ||||
|         return text != null; | ||||
|     } | ||||
|  | ||||
|     @Override @SuppressWarnings("unchecked") public MessagePart clone() | ||||
|         throws CloneNotSupportedException { | ||||
|         MessagePart obj = (MessagePart) super.clone(); | ||||
|         obj.styles = (ArrayList<ChatColor>) styles.clone(); | ||||
|         if (hoverActionData instanceof JsonString) { | ||||
|             obj.hoverActionData = new JsonString(((JsonString) hoverActionData).getValue()); | ||||
|         } else if (hoverActionData instanceof FancyMessage) { | ||||
|             obj.hoverActionData = ((FancyMessage) hoverActionData).clone(); | ||||
|         } | ||||
|         obj.translationReplacements = | ||||
|             (ArrayList<JsonRepresentedObject>) translationReplacements.clone(); | ||||
|         return obj; | ||||
|  | ||||
|     } | ||||
|  | ||||
|     public void writeJson(JsonWriter json) { | ||||
|         try { | ||||
|             json.beginObject(); | ||||
|             text.writeJson(json); | ||||
|             json.name("color").value(color.name().toLowerCase()); | ||||
|             for (final ChatColor style : styles) { | ||||
|                 json.name(stylesToNames.get(style)).value(true); | ||||
|             } | ||||
|             if (clickActionName != null && clickActionData != null) { | ||||
|                 json.name("clickEvent").beginObject().name("action").value(clickActionName) | ||||
|                     .name("value").value(clickActionData).endObject(); | ||||
|             } | ||||
|             if (hoverActionName != null && hoverActionData != null) { | ||||
|                 json.name("hoverEvent").beginObject().name("action").value(hoverActionName) | ||||
|                     .name("value"); | ||||
|                 hoverActionData.writeJson(json); | ||||
|                 json.endObject(); | ||||
|             } | ||||
|             if (insertionData != null) { | ||||
|                 json.name("insertion").value(insertionData); | ||||
|             } | ||||
|             if (translationReplacements.size() > 0 && TextualComponent.isTranslatableText(text)) { | ||||
|                 json.name("with").beginArray(); | ||||
|                 for (JsonRepresentedObject obj : translationReplacements) { | ||||
|                     obj.writeJson(json); | ||||
|                 } | ||||
|                 json.endArray(); | ||||
|             } | ||||
|             json.endObject(); | ||||
|         } catch (IOException e) { | ||||
|             Bukkit.getLogger() | ||||
|                 .log(Level.WARNING, "A problem occurred during writing of JSON string", e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public Map<String, Object> serialize() { | ||||
|         HashMap<String, Object> map = new HashMap<>(); | ||||
|         map.put("text", text); | ||||
|         map.put("styles", styles); | ||||
|         map.put("color", color.getChar()); | ||||
|         map.put("hoverActionName", hoverActionName); | ||||
|         map.put("hoverActionData", hoverActionData); | ||||
|         map.put("clickActionName", clickActionName); | ||||
|         map.put("clickActionData", clickActionData); | ||||
|         map.put("insertion", insertionData); | ||||
|         map.put("translationReplacements", translationReplacements); | ||||
|         return map; | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,227 +0,0 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  * | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.chat; | ||||
|  | ||||
| import org.bukkit.Bukkit; | ||||
|  | ||||
| import java.lang.reflect.Field; | ||||
| import java.lang.reflect.InvocationTargetException; | ||||
| import java.lang.reflect.Method; | ||||
| import java.util.Arrays; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * A class containing static utility methods and caches which are intended as reflective conveniences. | ||||
|  * Unless otherwise noted, upon failure methods will return {@code null}. | ||||
|  */ | ||||
| public final class Reflection { | ||||
|  | ||||
|     /** | ||||
|      * Stores loaded classes from the {@code net.minecraft.server} package. | ||||
|      */ | ||||
|     private static final Map<String, Class<?>> _loadedNMSClasses = new HashMap<>(); | ||||
|     /** | ||||
|      * Stores loaded classes from the {@code org.bukkit.craftbukkit} package (and subpackages). | ||||
|      */ | ||||
|     private static final Map<String, Class<?>> _loadedOBCClasses = new HashMap<>(); | ||||
|     private static final Map<Class<?>, Map<String, Field>> _loadedFields = new HashMap<>(); | ||||
|     /** | ||||
|      * Contains loaded methods in a cache. | ||||
|      * The map maps [types to maps of [method names to maps of [parameter types to method instances]]]. | ||||
|      */ | ||||
|     private static final Map<Class<?>, Map<String, Map<ArrayWrapper<Class<?>>, Method>>> | ||||
|         _loadedMethods = new HashMap<>(); | ||||
|     private static String _versionString; | ||||
|  | ||||
|     private Reflection() { | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets the version string from the package name of the CraftBukkit server implementation. | ||||
|      * This is needed to bypass the JAR package name changing on each update. | ||||
|      * | ||||
|      * @return The version string of the OBC and NMS packages, <em>including the trailing dot</em>. | ||||
|      */ | ||||
|     public synchronized static String getVersion() { | ||||
|         if (_versionString == null) { | ||||
|             String name = Bukkit.getServer().getClass().getPackage().getName(); | ||||
|             _versionString = name.substring(name.lastIndexOf('.') + 1) + "."; | ||||
|         } | ||||
|  | ||||
|         return _versionString; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets a {@link Class} object representing a type contained within the {@code net.minecraft.server} versioned package. | ||||
|      * The class instances returned by this method are cached, such that no lookup will be done twice (unless multiple threads are accessing this method simultaneously). | ||||
|      * | ||||
|      * @param className The name of the class, excluding the package, within NMS. | ||||
|      * @return The class instance representing the specified NMS class, or {@code null} if it could not be loaded. | ||||
|      */ | ||||
|     public synchronized static Class<?> getNMSClass(String className) { | ||||
|         if (_loadedNMSClasses.containsKey(className)) { | ||||
|             return _loadedNMSClasses.get(className); | ||||
|         } | ||||
|  | ||||
|         String fullName = "net.minecraft.server." + getVersion() + className; | ||||
|         Class<?> clazz; | ||||
|         try { | ||||
|             clazz = Class.forName(fullName); | ||||
|         } catch (ClassNotFoundException e) { | ||||
|             _loadedNMSClasses.put(className, null); | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|         _loadedNMSClasses.put(className, clazz); | ||||
|         return clazz; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets a {@link Class} object representing a type contained within the {@code org.bukkit.craftbukkit} versioned package. | ||||
|      * The class instances returned by this method are cached, such that no lookup will be done twice (unless multiple threads are accessing this method simultaneously). | ||||
|      * | ||||
|      * @param className The name of the class, excluding the package, within OBC. This name may contain a subpackage name, such as {@code inventory.CraftItemStack}. | ||||
|      * @return The class instance representing the specified OBC class, or {@code null} if it could not be loaded. | ||||
|      */ | ||||
|     public synchronized static Class<?> getOBCClass(String className) { | ||||
|         if (_loadedOBCClasses.containsKey(className)) { | ||||
|             return _loadedOBCClasses.get(className); | ||||
|         } | ||||
|  | ||||
|         String fullName = "org.bukkit.craftbukkit." + getVersion() + className; | ||||
|         Class<?> clazz; | ||||
|         try { | ||||
|             clazz = Class.forName(fullName); | ||||
|         } catch (ClassNotFoundException e) { | ||||
|             _loadedOBCClasses.put(className, null); | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|         _loadedOBCClasses.put(className, clazz); | ||||
|         return clazz; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Attempts to get the NMS handle of a CraftBukkit object. | ||||
|      * <p> | ||||
|      * The only match currently attempted by this method is a retrieval by using a parameterless {@code getHandle()} method implemented by the runtime type of the specified object. | ||||
|      * </p> | ||||
|      * | ||||
|      * @param obj The object for which to retrieve an NMS handle. | ||||
|      * @return The NMS handle of the specified object, or {@code null} if it could not be retrieved using {@code getHandle()}. | ||||
|      */ | ||||
|     public synchronized static Object getHandle(Object obj) | ||||
|         throws InvocationTargetException, IllegalAccessException, IllegalArgumentException { | ||||
|         return getMethod(obj.getClass(), "getHandle").invoke(obj); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Retrieves a {@link Field} instance declared by the specified class with the specified name. | ||||
|      * Java access modifiers are ignored during this retrieval. No guarantee is made as to whether the field | ||||
|      * returned will be an instance or static field. | ||||
|      * <p> | ||||
|      * A global caching mechanism within this class is used to store fields. Combined with synchronization, this guarantees that | ||||
|      * no field will be reflectively looked up twice. | ||||
|      * </p> | ||||
|      * <p> | ||||
|      * If a field is deemed suitable for return, {@link Field#setAccessible(boolean) setAccessible} will be invoked with an argument of {@code true} before it is returned. | ||||
|      * This ensures that callers do not have to check or worry about Java access modifiers when dealing with the returned instance. | ||||
|      * </p> | ||||
|      * | ||||
|      * @param clazz The class which contains the field to retrieve. | ||||
|      * @param name  The declared name of the field in the class. | ||||
|      * @return A field object with the specified name declared by the specified class. | ||||
|      * @see Class#getDeclaredField(String) | ||||
|      */ | ||||
|     public synchronized static Field getField(Class<?> clazz, String name) { | ||||
|         Map<String, Field> loaded; | ||||
|         if (!_loadedFields.containsKey(clazz)) { | ||||
|             loaded = new HashMap<>(); | ||||
|             _loadedFields.put(clazz, loaded); | ||||
|         } else { | ||||
|             loaded = _loadedFields.get(clazz); | ||||
|         } | ||||
|         if (loaded.containsKey(name)) { | ||||
|             // If the field is loaded (or cached as not existing), return the relevant value, which might be null | ||||
|             return loaded.get(name); | ||||
|         } | ||||
|         try { | ||||
|             Field field = clazz.getDeclaredField(name); | ||||
|             field.setAccessible(true); | ||||
|             loaded.put(name, field); | ||||
|             return field; | ||||
|         } catch (NoSuchFieldException | SecurityException e) { | ||||
|             // Error loading | ||||
|             e.printStackTrace(); | ||||
|             // Cache field as not existing | ||||
|             loaded.put(name, null); | ||||
|             return null; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Retrieves a {@link Method} instance declared by the specified class with the specified name and argument types. | ||||
|      * Java access modifiers are ignored during this retrieval. No guarantee is made as to whether the field | ||||
|      * returned will be an instance or static field. | ||||
|      * <p> | ||||
|      * A global caching mechanism within this class is used to store method. Combined with synchronization, this guarantees that | ||||
|      * no method will be reflectively looked up twice. | ||||
|      * <p> | ||||
|      * If a method is deemed suitable for return, {@link Method#setAccessible(boolean) setAccessible} will be invoked with an argument of {@code true} before it is returned. | ||||
|      * This ensures that callers do not have to check or worry about Java access modifiers when dealing with the returned instance. | ||||
|      * <p> | ||||
|      * This method does <em>not</em> search superclasses of the specified type for methods with the specified signature. | ||||
|      * Callers wishing this behavior should use {@link Class#getDeclaredMethod(String, Class...)}. | ||||
|      * | ||||
|      * @param clazz The class which contains the method to retrieve. | ||||
|      * @param name  The declared name of the method in the class. | ||||
|      * @param args  The formal argument types of the method. | ||||
|      * @return A method object with the specified name declared by the specified class. | ||||
|      */ | ||||
|     public synchronized static Method getMethod(Class<?> clazz, String name, Class<?>... args) { | ||||
|         _loadedMethods.computeIfAbsent(clazz, k -> new HashMap<>()); | ||||
|  | ||||
|         Map<String, Map<ArrayWrapper<Class<?>>, Method>> loadedMethodNames = | ||||
|             _loadedMethods.get(clazz); | ||||
|         loadedMethodNames.computeIfAbsent(name, k -> new HashMap<>()); | ||||
|  | ||||
|         Map<ArrayWrapper<Class<?>>, Method> loadedSignatures = loadedMethodNames.get(name); | ||||
|         ArrayWrapper<Class<?>> wrappedArg = new ArrayWrapper<>(args); | ||||
|         if (loadedSignatures.containsKey(wrappedArg)) { | ||||
|             return loadedSignatures.get(wrappedArg); | ||||
|         } | ||||
|  | ||||
|         for (Method m : clazz.getMethods()) { | ||||
|             if (m.getName().equals(name) && Arrays.equals(args, m.getParameterTypes())) { | ||||
|                 m.setAccessible(true); | ||||
|                 loadedSignatures.put(wrappedArg, m); | ||||
|                 return m; | ||||
|             } | ||||
|         } | ||||
|         loadedSignatures.put(wrappedArg, null); | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,338 +0,0 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  * | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.chat; | ||||
|  | ||||
| import com.google.common.base.Preconditions; | ||||
| import com.google.common.collect.ImmutableMap; | ||||
| import com.google.gson.stream.JsonWriter; | ||||
| import org.bukkit.configuration.serialization.ConfigurationSerializable; | ||||
| import org.bukkit.configuration.serialization.ConfigurationSerialization; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map; | ||||
|  | ||||
| /** | ||||
|  * Represents a textual component of a message part. | ||||
|  * This can be used to not only represent string literals in a JSON message, | ||||
|  * but also to represent localized strings and other text values. | ||||
|  * <p>Different instances of this class can be created with static constructor methods.</p> | ||||
|  */ | ||||
| public abstract class TextualComponent implements Cloneable { | ||||
|  | ||||
|     static { | ||||
|         ConfigurationSerialization.registerClass(TextualComponent.ArbitraryTextTypeComponent.class); | ||||
|         ConfigurationSerialization.registerClass(TextualComponent.ComplexTextTypeComponent.class); | ||||
|     } | ||||
|  | ||||
|     static TextualComponent deserialize(Map<String, Object> map) { | ||||
|         if (map.containsKey("key") && map.size() == 2 && map.containsKey("value")) { | ||||
|             // Arbitrary text component | ||||
|             return ArbitraryTextTypeComponent.deserialize(map); | ||||
|         } else if (map.size() >= 2 && map.containsKey("key") && !map | ||||
|             .containsKey("value") /* It contains keys that START WITH value */) { | ||||
|             // Complex JSON object | ||||
|             return ComplexTextTypeComponent.deserialize(map); | ||||
|         } | ||||
|  | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     static boolean isTextKey(String key) { | ||||
|         return key.equals("translate") || key.equals("text") || key.equals("score") || key | ||||
|             .equals("selector"); | ||||
|     } | ||||
|  | ||||
|     static boolean isTranslatableText(TextualComponent component) { | ||||
|         return component instanceof ComplexTextTypeComponent && component.getKey() | ||||
|             .equals("translate"); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Create a textual component representing a string literal. | ||||
|      * | ||||
|      * <p>This is the default type of textual component when a single string | ||||
|      * literal is given to a method. | ||||
|      * | ||||
|      * @param textValue The text which will be represented. | ||||
|      * @return The text component representing the specified literal text. | ||||
|      */ | ||||
|     public static TextualComponent rawText(String textValue) { | ||||
|         return new ArbitraryTextTypeComponent("text", textValue); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Create a textual component representing a localized string. | ||||
|      * The client will see this text component as their localized version of the specified string <em>key</em>, which can be overridden by a | ||||
|      * resource pack. | ||||
|      * <p> | ||||
|      * If the specified translation key is not present on the client resource pack, the translation key will be displayed as a string literal to | ||||
|      * the client. | ||||
|      * </p> | ||||
|      * | ||||
|      * @param translateKey The string key which maps to localized text. | ||||
|      * @return The text component representing the specified localized text. | ||||
|      */ | ||||
|     public static TextualComponent localizedText(String translateKey) { | ||||
|         return new ArbitraryTextTypeComponent("translate", translateKey); | ||||
|     } | ||||
|  | ||||
|     private static void throwUnsupportedSnapshot() { | ||||
|         throw new UnsupportedOperationException( | ||||
|             "This feature is only supported in snapshot releases."); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Create a textual component representing a scoreboard value. | ||||
|      * The client will see their own score for the specified objective as the text represented by this component. | ||||
|      * <p> | ||||
|      * <b>This method is currently guaranteed to throw an {@code UnsupportedOperationException} as it is only supported on snapshot clients.</b> | ||||
|      * </p> | ||||
|      * | ||||
|      * @param scoreboardObjective The name of the objective for which to display the score. | ||||
|      * @return The text component representing the specified scoreboard score (for the viewing player), or {@code null} if an error occurs during | ||||
|      * JSON serialization. | ||||
|      */ | ||||
|     public static TextualComponent objectiveScore(String scoreboardObjective) { | ||||
|         return objectiveScore("*", scoreboardObjective); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Create a textual component representing a scoreboard value. | ||||
|      * The client will see the score of the specified player for the specified objective as the text represented by this component. | ||||
|      * | ||||
|      * <p><b>This method is currently guaranteed to throw an {@code UnsupportedOperationException} | ||||
|      * as it is only supported on snapshot clients.</b> | ||||
|      * | ||||
|      * @param playerName          The name of the player whos score will be shown. If | ||||
|      *                            this string represents the single-character sequence | ||||
|      *                            "*", the viewing player's score will be displayed. | ||||
|      *                            Standard minecraft selectors (@a, @p, etc) | ||||
|      *                            are <em>not</em> supported. | ||||
|      * @param scoreboardObjective The name of the objective for | ||||
|      *                            which to display the score. | ||||
|      * @return The text component representing the specified scoreboard score | ||||
|      * for the specified player, or {@code null} if an error occurs during JSON serialization. | ||||
|      */ | ||||
|     public static TextualComponent objectiveScore(String playerName, String scoreboardObjective) { | ||||
|         throwUnsupportedSnapshot(); // Remove this line when the feature is released to non-snapshot versions, in addition to updating ALL THE | ||||
|         // OVERLOADS documentation accordingly | ||||
|  | ||||
|         return new ComplexTextTypeComponent("score", | ||||
|             ImmutableMap.<String, String>builder().put("name", playerName) | ||||
|                 .put("objective", scoreboardObjective).build()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Create a textual component representing a player name, retrievable by using a standard minecraft selector. | ||||
|      * The client will see the players or entities captured by the specified selector as the text represented by this component. | ||||
|      * <p> | ||||
|      * <b>This method is currently guaranteed to throw an {@code UnsupportedOperationException} as it is only supported on snapshot clients.</b> | ||||
|      * </p> | ||||
|      * | ||||
|      * @param selector The minecraft player or entity selector which will capture the entities whose string representations will be displayed in | ||||
|      *                 the place of this text component. | ||||
|      * @return The text component representing the name of the entities captured by the selector. | ||||
|      */ | ||||
|     public static TextualComponent selector(String selector) { | ||||
|         throwUnsupportedSnapshot(); // Remove this line when the feature is released to non-snapshot versions, in addition to updating ALL THE | ||||
|         // OVERLOADS documentation accordingly | ||||
|  | ||||
|         return new ArbitraryTextTypeComponent("selector", selector); | ||||
|     } | ||||
|  | ||||
|     @Override public String toString() { | ||||
|         return getReadableString(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return The JSON key used to represent text components of this type. | ||||
|      */ | ||||
|     public abstract String getKey(); | ||||
|  | ||||
|     /** | ||||
|      * @return A readable String | ||||
|      */ | ||||
|     public abstract String getReadableString(); | ||||
|  | ||||
|     /** | ||||
|      * Clones a textual component instance. | ||||
|      * The returned object should not reference this textual component instance, but should maintain the same key and value. | ||||
|      */ | ||||
|     @Override public abstract TextualComponent clone() throws CloneNotSupportedException; | ||||
|  | ||||
|     /** | ||||
|      * Writes the text data represented by this textual component to the specified JSON writer object. | ||||
|      * A new object within the writer is not started. | ||||
|      * | ||||
|      * @param writer The object to which to write the JSON data. | ||||
|      * @throws IOException If an error occurs while writing to the stream. | ||||
|      */ | ||||
|     public abstract void writeJson(JsonWriter writer) throws IOException; | ||||
|  | ||||
|     /** | ||||
|      * Internal class used to represent all types of text components. | ||||
|      * Exception validating done is on keys and values. | ||||
|      */ | ||||
|     private static final class ArbitraryTextTypeComponent extends TextualComponent | ||||
|         implements ConfigurationSerializable { | ||||
|  | ||||
|         private String key; | ||||
|         private String value; | ||||
|  | ||||
|         public ArbitraryTextTypeComponent(String key, String value) { | ||||
|             setKey(key); | ||||
|             setValue(value); | ||||
|         } | ||||
|  | ||||
|         public static ArbitraryTextTypeComponent deserialize(Map<String, Object> map) { | ||||
|             return new ArbitraryTextTypeComponent(map.get("key").toString(), | ||||
|                 map.get("value").toString()); | ||||
|         } | ||||
|  | ||||
|         @Override public String getKey() { | ||||
|             return key; | ||||
|         } | ||||
|  | ||||
|         public void setKey(String key) { | ||||
|             Preconditions | ||||
|                 .checkArgument(key != null && !key.isEmpty(), "The key must be specified."); | ||||
|             this.key = key; | ||||
|         } | ||||
|  | ||||
|         public String getValue() { | ||||
|             return value; | ||||
|         } | ||||
|  | ||||
|         public void setValue(String value) { | ||||
|             Preconditions.checkArgument(value != null, "The value must be specified."); | ||||
|             this.value = value; | ||||
|         } | ||||
|  | ||||
|         @Override public TextualComponent clone() throws CloneNotSupportedException { | ||||
|             // Since this is a private and final class, we can just reinstantiate this class instead of casting super.clone | ||||
|             return new ArbitraryTextTypeComponent(getKey(), getValue()); | ||||
|         } | ||||
|  | ||||
|         @Override public void writeJson(JsonWriter writer) throws IOException { | ||||
|             writer.name(getKey()).value(getValue()); | ||||
|         } | ||||
|  | ||||
|         @Override @SuppressWarnings("serial") public Map<String, Object> serialize() { | ||||
|             return new HashMap<String, Object>() { | ||||
|                 { | ||||
|                     put("key", getKey()); | ||||
|                     put("value", getValue()); | ||||
|                 } | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         @Override public String getReadableString() { | ||||
|             return getValue(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /** | ||||
|      * Internal class used to represent a text component with a nested JSON | ||||
|      * value. | ||||
|      * | ||||
|      * <p>Exception validating done is on keys and values. | ||||
|      */ | ||||
|     private static final class ComplexTextTypeComponent extends TextualComponent | ||||
|         implements ConfigurationSerializable { | ||||
|  | ||||
|         private String key; | ||||
|         private Map<String, String> value; | ||||
|  | ||||
|         public ComplexTextTypeComponent(String key, Map<String, String> values) { | ||||
|             setKey(key); | ||||
|             setValue(values); | ||||
|         } | ||||
|  | ||||
|         public static ComplexTextTypeComponent deserialize(Map<String, Object> map) { | ||||
|             String key = null; | ||||
|             Map<String, String> value = new HashMap<>(); | ||||
|             for (Map.Entry<String, Object> valEntry : map.entrySet()) { | ||||
|                 if (valEntry.getKey().equals("key")) { | ||||
|                     key = (String) valEntry.getValue(); | ||||
|                 } else if (valEntry.getKey().startsWith("value.")) { | ||||
|                     value.put(valEntry.getKey().substring(6) /* Strips out the value prefix */, | ||||
|                         valEntry.getValue().toString()); | ||||
|                 } | ||||
|             } | ||||
|             return new ComplexTextTypeComponent(key, value); | ||||
|         } | ||||
|  | ||||
|         @Override public String getKey() { | ||||
|             return key; | ||||
|         } | ||||
|  | ||||
|         public void setKey(String key) { | ||||
|             Preconditions | ||||
|                 .checkArgument(key != null && !key.isEmpty(), "The key must be specified."); | ||||
|             this.key = key; | ||||
|         } | ||||
|  | ||||
|         public Map<String, String> getValue() { | ||||
|             return value; | ||||
|         } | ||||
|  | ||||
|         public void setValue(Map<String, String> value) { | ||||
|             Preconditions.checkArgument(value != null, "The value must be specified."); | ||||
|             this.value = value; | ||||
|         } | ||||
|  | ||||
|         @Override public TextualComponent clone() { | ||||
|             // Since this is a private and final class, we can just reinstantiate this class instead of casting super.clone | ||||
|             return new ComplexTextTypeComponent(getKey(), getValue()); | ||||
|         } | ||||
|  | ||||
|         @Override public void writeJson(JsonWriter writer) throws IOException { | ||||
|             writer.name(getKey()); | ||||
|             writer.beginObject(); | ||||
|             for (Map.Entry<String, String> jsonPair : value.entrySet()) { | ||||
|                 writer.name(jsonPair.getKey()).value(jsonPair.getValue()); | ||||
|             } | ||||
|             writer.endObject(); | ||||
|         } | ||||
|  | ||||
|         @Override @SuppressWarnings("serial") public Map<String, Object> serialize() { | ||||
|             return new java.util.HashMap<String, Object>() { | ||||
|                 { | ||||
|                     put("key", getKey()); | ||||
|                     for (Map.Entry<String, String> valEntry : getValue().entrySet()) { | ||||
|                         put("value." + valEntry.getKey(), valEntry.getValue()); | ||||
|                     } | ||||
|                 } | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         @Override public String getReadableString() { | ||||
|             return getKey(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.entity; | ||||
|  | ||||
| @@ -30,4 +30,5 @@ class AgeableStats { | ||||
|     int age; | ||||
|     boolean locked; | ||||
|     boolean adult; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.entity; | ||||
|  | ||||
| @@ -37,4 +37,5 @@ class ArmorStandStats { | ||||
|     boolean noPlate; | ||||
|     boolean invisible; | ||||
|     boolean small; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.entity; | ||||
|  | ||||
| @@ -34,4 +34,5 @@ class EntityBaseStats { | ||||
|     double vZ; | ||||
|     double vY; | ||||
|     double vX; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,18 +21,16 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.entity; | ||||
|  | ||||
| import lombok.Getter; | ||||
| import lombok.NonNull; | ||||
| import org.bukkit.Location; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.entity.Entity; | ||||
| import org.bukkit.entity.EntityType; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| @Getter | ||||
| public abstract class EntityWrapper { | ||||
|  | ||||
|     protected final float yaw; | ||||
| @@ -43,7 +41,7 @@ public abstract class EntityWrapper { | ||||
|     public double y; | ||||
|     public double z; | ||||
|  | ||||
|     EntityWrapper(@NonNull final Entity entity) { | ||||
|     EntityWrapper(final @NonNull Entity entity) { | ||||
|         this.entity = entity; | ||||
|         this.type = entity.getType(); | ||||
|  | ||||
| @@ -55,7 +53,9 @@ public abstract class EntityWrapper { | ||||
|         this.pitch = location.getPitch(); | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("deprecation") @Override public String toString() { | ||||
|     @SuppressWarnings("deprecation") | ||||
|     @Override | ||||
|     public String toString() { | ||||
|         return String.format("[%s, x=%s, y=%s, z=%s]", type.getName(), x, y, z); | ||||
|     } | ||||
|  | ||||
| @@ -63,4 +63,32 @@ public abstract class EntityWrapper { | ||||
|  | ||||
|     public abstract void saveEntity(); | ||||
|  | ||||
|     public float getYaw() { | ||||
|         return this.yaw; | ||||
|     } | ||||
|  | ||||
|     public float getPitch() { | ||||
|         return this.pitch; | ||||
|     } | ||||
|  | ||||
|     public Entity getEntity() { | ||||
|         return this.entity; | ||||
|     } | ||||
|  | ||||
|     public EntityType getType() { | ||||
|         return this.type; | ||||
|     } | ||||
|  | ||||
|     public double getX() { | ||||
|         return this.x; | ||||
|     } | ||||
|  | ||||
|     public double getY() { | ||||
|         return this.y; | ||||
|     } | ||||
|  | ||||
|     public double getZ() { | ||||
|         return this.z; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.entity; | ||||
|  | ||||
| @@ -34,4 +34,5 @@ class HorseStats { | ||||
|     Horse.Variant variant; | ||||
|     Horse.Color color; | ||||
|     Horse.Style style; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.entity; | ||||
|  | ||||
| @@ -50,4 +50,5 @@ class LivingEntityStats { | ||||
|     ItemStack chestplate; | ||||
|     Collection<PotionEffect> potions; | ||||
|     ItemStack offHand; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,11 +21,13 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.entity; | ||||
|  | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.bukkit.Art; | ||||
| import org.bukkit.DyeColor; | ||||
| import org.bukkit.Location; | ||||
| @@ -60,6 +62,8 @@ import java.util.List; | ||||
|  | ||||
| public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + ReplicatingEntityWrapper.class.getSimpleName()); | ||||
|  | ||||
|     private final short depth; | ||||
|     private final int hash; | ||||
|     private final EntityBaseStats base = new EntityBaseStats(); | ||||
| @@ -148,17 +152,17 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|                 this.stack = item.getItemStack(); | ||||
|                 return; | ||||
|             case "ITEM_FRAME": | ||||
|                 this.x = Math.floor(this.x); | ||||
|                 this.y = Math.floor(this.y); | ||||
|                 this.z = Math.floor(this.z); | ||||
|                 this.x = Math.floor(this.getX()); | ||||
|                 this.y = Math.floor(this.getY()); | ||||
|                 this.z = Math.floor(this.getZ()); | ||||
|                 ItemFrame itemFrame = (ItemFrame) entity; | ||||
|                 this.dataByte = getOrdinal(Rotation.values(), itemFrame.getRotation()); | ||||
|                 this.stack = itemFrame.getItem().clone(); | ||||
|                 return; | ||||
|             case "PAINTING": | ||||
|                 this.x = Math.floor(this.x); | ||||
|                 this.y = Math.floor(this.y); | ||||
|                 this.z = Math.floor(this.z); | ||||
|                 this.x = Math.floor(this.getX()); | ||||
|                 this.y = Math.floor(this.getY()); | ||||
|                 this.z = Math.floor(this.getZ()); | ||||
|                 Painting painting = (Painting) entity; | ||||
|                 Art art = painting.getArt(); | ||||
|                 this.dataByte = getOrdinal(BlockFace.values(), painting.getFacing()); | ||||
| @@ -185,8 +189,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|                 AbstractHorse horse = (AbstractHorse) entity; | ||||
|                 this.horse = new HorseStats(); | ||||
|                 this.horse.jump = horse.getJumpStrength(); | ||||
|                 if (horse instanceof ChestedHorse) { | ||||
|                     ChestedHorse horse1 = (ChestedHorse) horse; | ||||
|                 if (horse instanceof ChestedHorse horse1) { | ||||
|                     this.horse.chest = horse1.isCarryingChest(); | ||||
|                 } | ||||
|                 //todo these horse features need fixing | ||||
| @@ -237,9 +240,9 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|             case "ARMOR_STAND": | ||||
|                 ArmorStand stand = (ArmorStand) entity; | ||||
|                 this.inventory = | ||||
|                     new ItemStack[] {stand.getItemInHand().clone(), stand.getHelmet().clone(), | ||||
|                         stand.getChestplate().clone(), stand.getLeggings().clone(), | ||||
|                         stand.getBoots().clone()}; | ||||
|                         new ItemStack[]{stand.getItemInHand().clone(), stand.getHelmet().clone(), | ||||
|                                 stand.getChestplate().clone(), stand.getLeggings().clone(), | ||||
|                                 stand.getBoots().clone()}; | ||||
|                 storeLiving(stand); | ||||
|                 this.stand = new ArmorStandStats(); | ||||
|  | ||||
| @@ -332,18 +335,17 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|                     this.dataByte = (byte) 0; | ||||
|                 } | ||||
|                 storeLiving((LivingEntity) entity); | ||||
|                 return; | ||||
|             // END LIVING // | ||||
|             default: | ||||
|                 PlotSquared.debug("&cCOULD NOT IDENTIFY ENTITY: " + entity.getType()); | ||||
|                 // END LIVING // | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public boolean equals(Object obj) { | ||||
|     @Override | ||||
|     public boolean equals(Object obj) { | ||||
|         return this.hash == obj.hashCode(); | ||||
|     } | ||||
|  | ||||
|     @Override public int hashCode() { | ||||
|     @Override | ||||
|     public int hashCode() { | ||||
|         return this.hash; | ||||
|     } | ||||
|  | ||||
| @@ -391,7 +393,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|         try { | ||||
|             entity.getInventory().setContents(this.inventory); | ||||
|         } catch (IllegalArgumentException e) { | ||||
|             PlotSquared.debug("&c[WARN] Failed to restore inventory.\n Reason: " + e.getMessage()); | ||||
|             LOGGER.error("Failed to restore inventory", e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -407,9 +409,9 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|         this.lived.leashed = lived.isLeashed(); | ||||
|         if (this.lived.leashed) { | ||||
|             Location location = lived.getLeashHolder().getLocation(); | ||||
|             this.lived.leashX = (short) (this.x - location.getBlockX()); | ||||
|             this.lived.leashY = (short) (this.y - location.getBlockY()); | ||||
|             this.lived.leashZ = (short) (this.z - location.getBlockZ()); | ||||
|             this.lived.leashX = (short) (this.getX() - location.getBlockX()); | ||||
|             this.lived.leashY = (short) (this.getY() - location.getBlockY()); | ||||
|             this.lived.leashZ = (short) (this.getZ() - location.getBlockZ()); | ||||
|         } | ||||
|         EntityEquipment equipment = lived.getEquipment(); | ||||
|         this.lived.equipped = equipment != null; | ||||
| @@ -459,8 +461,10 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|         this.tamed.tamed = tamed.isTamed(); | ||||
|     } | ||||
|  | ||||
|     @Override public Entity spawn(World world, int xOffset, int zOffset) { | ||||
|         Location location = new Location(world, this.x + xOffset, this.y, this.z + zOffset); | ||||
|     @SuppressWarnings("deprecation") // Paper deprecation | ||||
|     @Override | ||||
|     public Entity spawn(World world, int xOffset, int zOffset) { | ||||
|         Location location = new Location(world, this.getX() + xOffset, this.getY(), this.z + zOffset); | ||||
|         location.setYaw(this.yaw); | ||||
|         location.setPitch(this.pitch); | ||||
|         if (!this.getType().isSpawnable()) { | ||||
| @@ -646,36 +650,40 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|                 } | ||||
|                 if (this.stand.head[0] != 0 || this.stand.head[1] != 0 || this.stand.head[2] != 0) { | ||||
|                     EulerAngle pose = | ||||
|                         new EulerAngle(this.stand.head[0], this.stand.head[1], this.stand.head[2]); | ||||
|                             new EulerAngle(this.stand.head[0], this.stand.head[1], this.stand.head[2]); | ||||
|                     stand.setHeadPose(pose); | ||||
|                 } | ||||
|                 if (this.stand.body[0] != 0 || this.stand.body[1] != 0 || this.stand.body[2] != 0) { | ||||
|                     EulerAngle pose = | ||||
|                         new EulerAngle(this.stand.body[0], this.stand.body[1], this.stand.body[2]); | ||||
|                             new EulerAngle(this.stand.body[0], this.stand.body[1], this.stand.body[2]); | ||||
|                     stand.setBodyPose(pose); | ||||
|                 } | ||||
|                 if (this.stand.leftLeg[0] != 0 || this.stand.leftLeg[1] != 0 | ||||
|                     || this.stand.leftLeg[2] != 0) { | ||||
|                         || this.stand.leftLeg[2] != 0) { | ||||
|                     EulerAngle pose = new EulerAngle(this.stand.leftLeg[0], this.stand.leftLeg[1], | ||||
|                         this.stand.leftLeg[2]); | ||||
|                             this.stand.leftLeg[2] | ||||
|                     ); | ||||
|                     stand.setLeftLegPose(pose); | ||||
|                 } | ||||
|                 if (this.stand.rightLeg[0] != 0 || this.stand.rightLeg[1] != 0 | ||||
|                     || this.stand.rightLeg[2] != 0) { | ||||
|                         || this.stand.rightLeg[2] != 0) { | ||||
|                     EulerAngle pose = new EulerAngle(this.stand.rightLeg[0], this.stand.rightLeg[1], | ||||
|                         this.stand.rightLeg[2]); | ||||
|                             this.stand.rightLeg[2] | ||||
|                     ); | ||||
|                     stand.setRightLegPose(pose); | ||||
|                 } | ||||
|                 if (this.stand.leftArm[0] != 0 || this.stand.leftArm[1] != 0 | ||||
|                     || this.stand.leftArm[2] != 0) { | ||||
|                         || this.stand.leftArm[2] != 0) { | ||||
|                     EulerAngle pose = new EulerAngle(this.stand.leftArm[0], this.stand.leftArm[1], | ||||
|                         this.stand.leftArm[2]); | ||||
|                             this.stand.leftArm[2] | ||||
|                     ); | ||||
|                     stand.setLeftArmPose(pose); | ||||
|                 } | ||||
|                 if (this.stand.rightArm[0] != 0 || this.stand.rightArm[1] != 0 | ||||
|                     || this.stand.rightArm[2] != 0) { | ||||
|                         || this.stand.rightArm[2] != 0) { | ||||
|                     EulerAngle pose = new EulerAngle(this.stand.rightArm[0], this.stand.rightArm[1], | ||||
|                         this.stand.rightArm[2]); | ||||
|                             this.stand.rightArm[2] | ||||
|                     ); | ||||
|                     stand.setRightArmPose(pose); | ||||
|                 } | ||||
|                 if (this.stand.invisible) { | ||||
| @@ -738,7 +746,9 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|                 restoreLiving((LivingEntity) entity); | ||||
|                 return entity; | ||||
|             default: | ||||
|                 PlotSquared.debug("&cCOULD NOT IDENTIFY ENTITY: " + entity.getType()); | ||||
|                 if (Settings.DEBUG) { | ||||
|                     LOGGER.info("Could not identify entity: {}", entity.getType()); | ||||
|                 } | ||||
|                 return entity; | ||||
|             // END LIVING | ||||
|         } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.entity; | ||||
|  | ||||
| @@ -31,4 +31,5 @@ class TameableStats { | ||||
|  | ||||
|     AnimalTamer owner; | ||||
|     boolean tamed; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,11 +21,11 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.entity; | ||||
|  | ||||
| import com.plotsquared.bukkit.BukkitMain; | ||||
| import com.plotsquared.bukkit.BukkitPlatform; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.Location; | ||||
| import org.bukkit.World; | ||||
| @@ -44,7 +44,8 @@ public class TeleportEntityWrapper extends EntityWrapper { | ||||
|         super(entity); | ||||
|     } | ||||
|  | ||||
|     @Override public Entity spawn(final World world, final int xOffset, final int zOffset) { | ||||
|     @Override | ||||
|     public Entity spawn(final World world, final int xOffset, final int zOffset) { | ||||
|         if (!getEntity().getLocation().getChunk().equals(oldLocation.getChunk())) { | ||||
|             final Location oldLocation = this.oldLocation.clone(); | ||||
|             oldLocation.add(xOffset, 0, xOffset); | ||||
| @@ -53,12 +54,13 @@ public class TeleportEntityWrapper extends EntityWrapper { | ||||
|             getEntity().setInvulnerable(invulnerableOld); | ||||
|             getEntity().setFireTicks(fireTicksOld); | ||||
|             getEntity().setTicksLived(livingTicksOld); | ||||
|             getEntity().removeMetadata("ps-tmp-teleport", BukkitMain.getPlugin(BukkitMain.class)); | ||||
|             getEntity().removeMetadata("ps-tmp-teleport", BukkitPlatform.getPlugin(BukkitPlatform.class)); | ||||
|         } | ||||
|         return getEntity(); | ||||
|     } | ||||
|  | ||||
|     @Override public void saveEntity() { | ||||
|     @Override | ||||
|     public void saveEntity() { | ||||
|         if (getEntity().hasMetadata("ps-tmp-teleport")) { | ||||
|             this.oldLocation = (Location) this.getEntity().getMetadata("ps-tmp-teleport").get(0); | ||||
|         } else { | ||||
| @@ -67,9 +69,9 @@ public class TeleportEntityWrapper extends EntityWrapper { | ||||
|  | ||||
|         // To account for offsets in the chunk manager | ||||
|         this.oldLocation = oldLocation.clone(); | ||||
|         this.oldLocation.setX(this.x); | ||||
|         this.oldLocation.setY(this.y); | ||||
|         this.oldLocation.setZ(this.z); | ||||
|         this.oldLocation.setX(this.getX()); | ||||
|         this.oldLocation.setY(this.getY()); | ||||
|         this.oldLocation.setZ(this.getZ()); | ||||
|  | ||||
|         this.gravityOld = this.getEntity().hasGravity(); | ||||
|         this.getEntity().setGravity(false); | ||||
| @@ -77,11 +79,13 @@ public class TeleportEntityWrapper extends EntityWrapper { | ||||
|         this.getEntity().setInvulnerable(true); | ||||
|         this.fireTicksOld = this.getEntity().getFireTicks(); | ||||
|         this.livingTicksOld = this.getEntity().getTicksLived(); | ||||
|         this.getEntity().setMetadata("ps-tmp-teleport", | ||||
|             new FixedMetadataValue(BukkitMain.getPlugin(BukkitMain.class), oldLocation)); | ||||
|         this.getEntity().setMetadata( | ||||
|                 "ps-tmp-teleport", | ||||
|                 new FixedMetadataValue(BukkitPlatform.getPlugin(BukkitPlatform.class), oldLocation) | ||||
|         ); | ||||
|         final Chunk newChunk = getNewChunk(); | ||||
|         this.getEntity().teleport( | ||||
|             new Location(newChunk.getWorld(), newChunk.getX() << 4, 5000, newChunk.getZ() << 4)); | ||||
|                 new Location(newChunk.getWorld(), newChunk.getX() << 4, 5000, newChunk.getZ() << 4)); | ||||
|     } | ||||
|  | ||||
|     private Chunk getNewChunk() { | ||||
| @@ -115,4 +119,5 @@ public class TeleportEntityWrapper extends EntityWrapper { | ||||
|     private Chunk getChunkRelative(final Chunk chunk, final int dx, final int dz) { | ||||
|         return chunk.getWorld().getChunkAt(chunk.getX() + dx, chunk.getZ() + dz); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.generator; | ||||
|  | ||||
| @@ -29,37 +29,45 @@ import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.generator.IndependentPlotGenerator; | ||||
| import com.plotsquared.core.location.ChunkWrapper; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.queue.GlobalBlockQueue; | ||||
| import com.plotsquared.core.queue.LocalBlockQueue; | ||||
| import com.plotsquared.core.queue.ScopedLocalBlockQueue; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.queue.QueueCoordinator; | ||||
| import com.plotsquared.core.queue.ScopedQueueCoordinator; | ||||
| import com.sk89q.worldedit.bukkit.BukkitWorld; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.generator.BlockPopulator; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Random; | ||||
|  | ||||
| final class BlockStatePopulator extends BlockPopulator { | ||||
|  | ||||
|     private final IndependentPlotGenerator plotGenerator; | ||||
|     private LocalBlockQueue queue; | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|  | ||||
|     public BlockStatePopulator(IndependentPlotGenerator plotGenerator) { | ||||
|     private QueueCoordinator queue; | ||||
|  | ||||
|     public BlockStatePopulator( | ||||
|             final @NonNull IndependentPlotGenerator plotGenerator, | ||||
|             final @NonNull PlotAreaManager plotAreaManager | ||||
|     ) { | ||||
|         this.plotGenerator = plotGenerator; | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void populate(@NotNull final World world, @NotNull final Random random, | ||||
|         @NotNull final Chunk source) { | ||||
|     public void populate(final @NonNull World world, final @NonNull Random random, final @NonNull Chunk source) { | ||||
|         if (this.queue == null) { | ||||
|             this.queue = GlobalBlockQueue.IMP.getNewQueue(world.getName(), false); | ||||
|             this.queue = PlotSquared.platform().globalBlockQueue().getNewQueue(new BukkitWorld(world)); | ||||
|         } | ||||
|         final PlotArea area = PlotSquared.get().getPlotArea(world.getName(), null); | ||||
|         final ChunkWrapper wrap = | ||||
|             new ChunkWrapper(area.getWorldName(), source.getX(), source.getZ()); | ||||
|         final ScopedLocalBlockQueue chunk = this.queue.getForChunk(wrap.x, wrap.z); | ||||
|         final PlotArea area = this.plotAreaManager.getPlotArea(world.getName(), null); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         final ChunkWrapper wrap = new ChunkWrapper(area.getWorldName(), source.getX(), source.getZ()); | ||||
|         final ScopedQueueCoordinator chunk = this.queue.getForChunk(wrap.x, wrap.z); | ||||
|         if (this.plotGenerator.populateChunk(chunk, area)) { | ||||
|             this.queue.flush(); | ||||
|             this.queue.enqueue(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.generator; | ||||
|  | ||||
| @@ -29,7 +29,7 @@ import com.plotsquared.core.generator.AugmentedUtils; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.generator.BlockPopulator; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Random; | ||||
|  | ||||
| @@ -51,7 +51,8 @@ public class BukkitAugmentedGenerator extends BlockPopulator { | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void populate(@NotNull World world, @NotNull Random random, @NotNull Chunk source) { | ||||
|     public void populate(@NonNull World world, @NonNull Random random, @NonNull Chunk source) { | ||||
|         AugmentedUtils.generate(source, world.getName(), source.getX(), source.getZ(), null); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.generator; | ||||
|  | ||||
| @@ -33,16 +33,15 @@ import com.plotsquared.core.generator.IndependentPlotGenerator; | ||||
| import com.plotsquared.core.generator.SingleWorldGenerator; | ||||
| import com.plotsquared.core.location.ChunkWrapper; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.queue.ScopedLocalBlockQueue; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.queue.ScopedQueueCoordinator; | ||||
| import com.plotsquared.core.util.ChunkManager; | ||||
| import com.plotsquared.core.util.MainUtil; | ||||
| import com.sk89q.worldedit.math.BlockVector2; | ||||
| import lombok.Getter; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.Biome; | ||||
| import org.bukkit.generator.BlockPopulator; | ||||
| import org.bukkit.generator.ChunkGenerator; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| @@ -50,65 +49,72 @@ import java.util.Random; | ||||
| import java.util.Set; | ||||
|  | ||||
| public class BukkitPlotGenerator extends ChunkGenerator | ||||
|     implements GeneratorWrapper<ChunkGenerator> { | ||||
|         implements GeneratorWrapper<ChunkGenerator> { | ||||
|  | ||||
|     @SuppressWarnings("unused") public final boolean PAPER_ASYNC_SAFE = true; | ||||
|     @SuppressWarnings("unused") | ||||
|     public final boolean PAPER_ASYNC_SAFE = true; | ||||
|  | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|     private final IndependentPlotGenerator plotGenerator; | ||||
|     private final ChunkGenerator platformGenerator; | ||||
|     private final boolean full; | ||||
|     private final String levelName; | ||||
|     private List<BlockPopulator> populators; | ||||
|     private boolean loaded = false; | ||||
|  | ||||
|     @Getter private final String levelName; | ||||
|  | ||||
|     public BukkitPlotGenerator(String name, IndependentPlotGenerator generator) { | ||||
|         if (generator == null) { | ||||
|             throw new IllegalArgumentException("Generator may not be null!"); | ||||
|         } | ||||
|     public BukkitPlotGenerator( | ||||
|             final @NonNull String name, | ||||
|             final @NonNull IndependentPlotGenerator generator, | ||||
|             final @NonNull PlotAreaManager plotAreaManager | ||||
|     ) { | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|         this.levelName = name; | ||||
|         this.plotGenerator = generator; | ||||
|         this.platformGenerator = this; | ||||
|         this.populators = new ArrayList<>(); | ||||
|         this.populators.add(new BlockStatePopulator(this.plotGenerator)); | ||||
|         this.populators.add(new BlockStatePopulator(this.plotGenerator, this.plotAreaManager)); | ||||
|         this.full = true; | ||||
|         MainUtil.initCache(); | ||||
|     } | ||||
|  | ||||
|     public BukkitPlotGenerator(final String world, final ChunkGenerator cg) { | ||||
|     public BukkitPlotGenerator(final String world, final ChunkGenerator cg, final @NonNull PlotAreaManager plotAreaManager) { | ||||
|         if (cg instanceof BukkitPlotGenerator) { | ||||
|             throw new IllegalArgumentException("ChunkGenerator: " + cg.getClass().getName() | ||||
|                 + " is already a BukkitPlotGenerator!"); | ||||
|                     + " is already a BukkitPlotGenerator!"); | ||||
|         } | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|         this.levelName = world; | ||||
|         this.full = false; | ||||
|         this.platformGenerator = cg; | ||||
|         this.plotGenerator = new DelegatePlotGenerator(cg, world); | ||||
|         MainUtil.initCache(); | ||||
|     } | ||||
|  | ||||
|     @Override public void augment(PlotArea area) { | ||||
|     @Override | ||||
|     public void augment(PlotArea area) { | ||||
|         BukkitAugmentedGenerator.get(BukkitUtil.getWorld(area.getWorldName())); | ||||
|     } | ||||
|  | ||||
|     @Override public boolean isFull() { | ||||
|     @Override | ||||
|     public boolean isFull() { | ||||
|         return this.full; | ||||
|     } | ||||
|  | ||||
|     @Override public IndependentPlotGenerator getPlotGenerator() { | ||||
|     @Override | ||||
|     public IndependentPlotGenerator getPlotGenerator() { | ||||
|         return this.plotGenerator; | ||||
|     } | ||||
|  | ||||
|     @Override public ChunkGenerator getPlatformGenerator() { | ||||
|     @Override | ||||
|     public ChunkGenerator getPlatformGenerator() { | ||||
|         return this.platformGenerator; | ||||
|     } | ||||
|  | ||||
|     @Override @NotNull public List<BlockPopulator> getDefaultPopulators(@NotNull World world) { | ||||
|     @Override | ||||
|     public @NonNull List<BlockPopulator> getDefaultPopulators(@NonNull World world) { | ||||
|         try { | ||||
|             if (!this.loaded) { | ||||
|                 String name = world.getName(); | ||||
|                 PlotSquared.get().loadWorld(name, this); | ||||
|                 Set<PlotArea> areas = PlotSquared.get().getPlotAreas(name); | ||||
|                 final Set<PlotArea> areas = this.plotAreaManager.getPlotAreasSet(name); | ||||
|                 if (!areas.isEmpty()) { | ||||
|                     PlotArea area = areas.iterator().next(); | ||||
|                     if (!area.isMobSpawning()) { | ||||
| @@ -147,9 +153,11 @@ public class BukkitPlotGenerator extends ChunkGenerator | ||||
|         return toAdd; | ||||
|     } | ||||
|  | ||||
|     @Override @NotNull | ||||
|     public ChunkData generateChunkData(@NotNull World world, @NotNull Random random, int x, int z, | ||||
|         @NotNull BiomeGrid biome) { | ||||
|     @Override | ||||
|     public @NonNull ChunkData generateChunkData( | ||||
|             @NonNull World world, @NonNull Random random, int x, int z, | ||||
|             @NonNull BiomeGrid biome | ||||
|     ) { | ||||
|  | ||||
|         GenChunk result = new GenChunk(); | ||||
|         if (this.getPlotGenerator() instanceof SingleWorldGenerator) { | ||||
| @@ -187,7 +195,7 @@ public class BukkitPlotGenerator extends ChunkGenerator | ||||
|         return result.getChunkData(); | ||||
|     } | ||||
|  | ||||
|     private void generate(BlockVector2 loc, World world, ScopedLocalBlockQueue result) { | ||||
|     private void generate(BlockVector2 loc, World world, ScopedQueueCoordinator result) { | ||||
|         // Load if improperly loaded | ||||
|         if (!this.loaded) { | ||||
|             String name = world.getName(); | ||||
| @@ -198,11 +206,11 @@ public class BukkitPlotGenerator extends ChunkGenerator | ||||
|         if (ChunkManager.preProcessChunk(loc, result)) { | ||||
|             return; | ||||
|         } | ||||
|         PlotArea area = PlotSquared.get().getPlotArea(world.getName(), null); | ||||
|         if (area == null && (area = PlotSquared.get().getPlotArea(this.levelName, null)) == null) { | ||||
|         PlotArea area = this.plotAreaManager.getPlotArea(world.getName(), null); | ||||
|         if (area == null && (area = this.plotAreaManager.getPlotArea(this.levelName, null)) == null) { | ||||
|             throw new IllegalStateException( | ||||
|                 "Cannot regenerate chunk that does not belong to a plot area." + " Location: " + loc | ||||
|                     + ", world: " + world); | ||||
|                     "Cannot regenerate chunk that does not belong to a plot area." + " Location: " + loc | ||||
|                             + ", world: " + world); | ||||
|         } | ||||
|         try { | ||||
|             this.plotGenerator.generateChunk(result, area); | ||||
| @@ -213,7 +221,8 @@ public class BukkitPlotGenerator extends ChunkGenerator | ||||
|         ChunkManager.postProcessChunk(loc, result); | ||||
|     } | ||||
|  | ||||
|     @Override public boolean canSpawn(@NotNull final World world, final int x, final int z) { | ||||
|     @Override | ||||
|     public boolean canSpawn(final @NonNull World world, final int x, final int z) { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
| @@ -237,7 +246,8 @@ public class BukkitPlotGenerator extends ChunkGenerator | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     @Override public String toString() { | ||||
|     @Override | ||||
|     public String toString() { | ||||
|         if (this.platformGenerator == this) { | ||||
|             return this.plotGenerator.getName(); | ||||
|         } | ||||
| @@ -248,11 +258,16 @@ public class BukkitPlotGenerator extends ChunkGenerator | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public boolean equals(final Object obj) { | ||||
|     @Override | ||||
|     public boolean equals(final Object obj) { | ||||
|         if (obj == null) { | ||||
|             return false; | ||||
|         } | ||||
|         return toString().equals(obj.toString()) || toString().equals(obj.getClass().getName()); | ||||
|     } | ||||
|  | ||||
|     public String getLevelName() { | ||||
|         return this.levelName; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.generator; | ||||
|  | ||||
| @@ -31,15 +31,14 @@ import com.plotsquared.core.generator.IndependentPlotGenerator; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.PlotId; | ||||
| import com.plotsquared.core.queue.ScopedLocalBlockQueue; | ||||
| import com.plotsquared.core.queue.ScopedQueueCoordinator; | ||||
| import com.plotsquared.core.util.MathMan; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.Biome; | ||||
| import org.bukkit.generator.BlockPopulator; | ||||
| import org.bukkit.generator.ChunkGenerator; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Range; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Random; | ||||
|  | ||||
| @@ -53,18 +52,22 @@ final class DelegatePlotGenerator extends IndependentPlotGenerator { | ||||
|         this.world = world; | ||||
|     } | ||||
|  | ||||
|     @Override public void initialize(PlotArea area) { | ||||
|     @Override | ||||
|     public void initialize(PlotArea area) { | ||||
|     } | ||||
|  | ||||
|     @Override public String getName() { | ||||
|     @Override | ||||
|     public String getName() { | ||||
|         return this.chunkGenerator.getClass().getName(); | ||||
|     } | ||||
|  | ||||
|     @Override public PlotArea getNewPlotArea(String world, String id, PlotId min, PlotId max) { | ||||
|         return PlotSquared.get().IMP.getDefaultGenerator().getNewPlotArea(world, id, min, max); | ||||
|     @Override | ||||
|     public PlotArea getNewPlotArea(String world, String id, PlotId min, PlotId max) { | ||||
|         return PlotSquared.platform().defaultGenerator().getNewPlotArea(world, id, min, max); | ||||
|     } | ||||
|  | ||||
|     @Override public void generateChunk(final ScopedLocalBlockQueue result, PlotArea settings) { | ||||
|     @Override | ||||
|     public void generateChunk(final ScopedQueueCoordinator result, PlotArea settings) { | ||||
|         World world = BukkitUtil.getWorld(this.world); | ||||
|         Location min = result.getMin(); | ||||
|         int chunkX = min.getX() >> 4; | ||||
| @@ -73,22 +76,23 @@ final class DelegatePlotGenerator extends IndependentPlotGenerator { | ||||
|         try { | ||||
|             ChunkGenerator.BiomeGrid grid = new ChunkGenerator.BiomeGrid() { | ||||
|                 @Override | ||||
|                 public void setBiome(@Range(from = 0, to = 15) int x, | ||||
|                     @Range(from = 0, to = 15) int z, @NotNull Biome biome) { | ||||
|                 public void setBiome(int x, int z, @NonNull Biome biome) { | ||||
|                     result.setBiome(x, z, BukkitAdapter.adapt(biome)); | ||||
|                 } | ||||
|  | ||||
|                 //do not annotate with Override until we discontinue support for 1.4.4 | ||||
|                 public void setBiome(int x, int y, int z, @NotNull Biome biome) { | ||||
|                 public void setBiome(int x, int y, int z, @NonNull Biome biome) { | ||||
|                     result.setBiome(x, z, BukkitAdapter.adapt(biome)); | ||||
|  | ||||
|                 } | ||||
|  | ||||
|                 @Override @NotNull public Biome getBiome(int x, int z) { | ||||
|                 @Override | ||||
|                 public @NonNull Biome getBiome(int x, int z) { | ||||
|                     return Biome.FOREST; | ||||
|                 } | ||||
|  | ||||
|                 @Override public @NotNull Biome getBiome(int x, int y, int z) { | ||||
|                 @Override | ||||
|                 public @NonNull Biome getBiome(int x, int y, int z) { | ||||
|                     return Biome.FOREST; | ||||
|                 } | ||||
|             }; | ||||
|   | ||||
| @@ -0,0 +1,56 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.inject; | ||||
|  | ||||
| import com.google.inject.AbstractModule; | ||||
| import com.google.inject.assistedinject.FactoryModuleBuilder; | ||||
| import com.plotsquared.core.backup.BackupManager; | ||||
| import com.plotsquared.core.backup.BackupProfile; | ||||
| import com.plotsquared.core.backup.NullBackupManager; | ||||
| import com.plotsquared.core.backup.PlayerBackupProfile; | ||||
| import com.plotsquared.core.backup.SimpleBackupManager; | ||||
| import com.plotsquared.core.inject.factory.PlayerBackupProfileFactory; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
|  | ||||
| public class BackupModule extends AbstractModule { | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + BackupModule.class.getSimpleName()); | ||||
|  | ||||
|     @Override | ||||
|     protected void configure() { | ||||
|         try { | ||||
|             install(new FactoryModuleBuilder() | ||||
|                     .implement(BackupProfile.class, PlayerBackupProfile.class).build(PlayerBackupProfileFactory.class)); | ||||
|             bind(BackupManager.class).to(SimpleBackupManager.class); | ||||
|         } catch (final Exception e) { | ||||
|             LOGGER.error("Failed to initialize backup manager", e); | ||||
|             LOGGER.error("Backup features will be disabled"); | ||||
|             bind(BackupManager.class).to(NullBackupManager.class); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,155 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.inject; | ||||
|  | ||||
| import com.google.inject.AbstractModule; | ||||
| import com.google.inject.Provides; | ||||
| import com.google.inject.Singleton; | ||||
| import com.google.inject.assistedinject.FactoryModuleBuilder; | ||||
| import com.plotsquared.bukkit.BukkitPlatform; | ||||
| import com.plotsquared.bukkit.listener.SingleWorldListener; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayerManager; | ||||
| import com.plotsquared.bukkit.queue.BukkitChunkCoordinator; | ||||
| import com.plotsquared.bukkit.queue.BukkitQueueCoordinator; | ||||
| import com.plotsquared.bukkit.schematic.BukkitSchematicHandler; | ||||
| import com.plotsquared.bukkit.util.BukkitChunkManager; | ||||
| import com.plotsquared.bukkit.util.BukkitEconHandler; | ||||
| import com.plotsquared.bukkit.util.BukkitInventoryUtil; | ||||
| import com.plotsquared.bukkit.util.BukkitRegionManager; | ||||
| import com.plotsquared.bukkit.util.BukkitSetupUtils; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.bukkit.util.fawe.FaweRegionManager; | ||||
| import com.plotsquared.bukkit.util.fawe.FaweSchematicHandler; | ||||
| import com.plotsquared.core.PlotPlatform; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.generator.HybridGen; | ||||
| import com.plotsquared.core.generator.IndependentPlotGenerator; | ||||
| import com.plotsquared.core.inject.annotations.ConsoleActor; | ||||
| import com.plotsquared.core.inject.annotations.DefaultGenerator; | ||||
| import com.plotsquared.core.inject.factory.ChunkCoordinatorBuilderFactory; | ||||
| import com.plotsquared.core.inject.factory.ChunkCoordinatorFactory; | ||||
| import com.plotsquared.core.inject.factory.HybridPlotWorldFactory; | ||||
| import com.plotsquared.core.inject.factory.ProgressSubscriberFactory; | ||||
| import com.plotsquared.core.plot.world.DefaultPlotAreaManager; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.plot.world.SinglePlotAreaManager; | ||||
| import com.plotsquared.core.queue.ChunkCoordinator; | ||||
| import com.plotsquared.core.queue.GlobalBlockQueue; | ||||
| import com.plotsquared.core.queue.QueueProvider; | ||||
| import com.plotsquared.core.queue.subscriber.DefaultProgressSubscriber; | ||||
| import com.plotsquared.core.queue.subscriber.ProgressSubscriber; | ||||
| import com.plotsquared.core.util.ChunkManager; | ||||
| import com.plotsquared.core.util.EconHandler; | ||||
| import com.plotsquared.core.util.InventoryUtil; | ||||
| import com.plotsquared.core.util.PlayerManager; | ||||
| import com.plotsquared.core.util.RegionManager; | ||||
| import com.plotsquared.core.util.SchematicHandler; | ||||
| import com.plotsquared.core.util.SetupUtils; | ||||
| import com.plotsquared.core.util.WorldUtil; | ||||
| import com.sk89q.worldedit.bukkit.WorldEditPlugin; | ||||
| import com.sk89q.worldedit.extension.platform.Actor; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.command.ConsoleCommandSender; | ||||
| import org.bukkit.plugin.java.JavaPlugin; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| public class BukkitModule extends AbstractModule { | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + BukkitModule.class.getSimpleName()); | ||||
|  | ||||
|     private final BukkitPlatform bukkitPlatform; | ||||
|  | ||||
|     public BukkitModule(final @NonNull BukkitPlatform bukkitPlatform) { | ||||
|         this.bukkitPlatform = bukkitPlatform; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void configure() { | ||||
|         bind(PlayerManager.class).to(BukkitPlayerManager.class); | ||||
|         bind(JavaPlugin.class).toInstance(bukkitPlatform); | ||||
|         bind(PlotPlatform.class).toInstance(bukkitPlatform); | ||||
|         bind(BukkitPlatform.class).toInstance(bukkitPlatform); | ||||
|         bind(IndependentPlotGenerator.class).annotatedWith(DefaultGenerator.class).to(HybridGen.class); | ||||
|         // Console actor | ||||
|         @NonNull ConsoleCommandSender console = Bukkit.getServer().getConsoleSender(); | ||||
|         WorldEditPlugin wePlugin = ((WorldEditPlugin) Bukkit.getPluginManager().getPlugin("WorldEdit")); | ||||
|         bind(Actor.class).annotatedWith(ConsoleActor.class).toInstance(wePlugin.wrapCommandSender(console)); | ||||
|         bind(InventoryUtil.class).to(BukkitInventoryUtil.class); | ||||
|         bind(SetupUtils.class).to(BukkitSetupUtils.class); | ||||
|         bind(WorldUtil.class).to(BukkitUtil.class); | ||||
|         install(new FactoryModuleBuilder() | ||||
|                 .implement(ProgressSubscriber.class, DefaultProgressSubscriber.class) | ||||
|                 .build(ProgressSubscriberFactory.class)); | ||||
|         bind(ChunkManager.class).to(BukkitChunkManager.class); | ||||
|         if (PlotSquared.platform().isFaweHooking()) { | ||||
|             bind(SchematicHandler.class).to(FaweSchematicHandler.class); | ||||
|             bind(RegionManager.class).to(FaweRegionManager.class); | ||||
|         } else { | ||||
|             bind(SchematicHandler.class).to(BukkitSchematicHandler.class); | ||||
|             bind(RegionManager.class).to(BukkitRegionManager.class); | ||||
|         } | ||||
|         bind(GlobalBlockQueue.class).toInstance(new GlobalBlockQueue(QueueProvider.of(BukkitQueueCoordinator.class))); | ||||
|         if (Settings.Enabled_Components.WORLDS) { | ||||
|             bind(PlotAreaManager.class).to(SinglePlotAreaManager.class); | ||||
|             try { | ||||
|                 bind(SingleWorldListener.class).toInstance(new SingleWorldListener()); | ||||
|             } catch (Exception e) { | ||||
|                 e.printStackTrace(); | ||||
|             } | ||||
|         } else { | ||||
|             bind(PlotAreaManager.class).to(DefaultPlotAreaManager.class); | ||||
|         } | ||||
|         install(new FactoryModuleBuilder().build(HybridPlotWorldFactory.class)); | ||||
|         install(new FactoryModuleBuilder() | ||||
|                 .implement(ChunkCoordinator.class, BukkitChunkCoordinator.class) | ||||
|                 .build(ChunkCoordinatorFactory.class)); | ||||
|         install(new FactoryModuleBuilder().build(ChunkCoordinatorBuilderFactory.class)); | ||||
|     } | ||||
|  | ||||
|     @Provides | ||||
|     @Singleton | ||||
|     @NonNull EconHandler provideEconHandler() { | ||||
|         if (!Settings.Enabled_Components.ECONOMY) { | ||||
|             return EconHandler.nullEconHandler(); | ||||
|         } | ||||
|         if (Bukkit.getPluginManager().isPluginEnabled("Vault")) { | ||||
|             try { | ||||
|                 BukkitEconHandler econHandler = new BukkitEconHandler(); | ||||
|                 if (!econHandler.init()) { | ||||
|                     LOGGER.warn("Economy is enabled but no plugin is providing an economy service. Falling back..."); | ||||
|                     return EconHandler.nullEconHandler(); | ||||
|                 } | ||||
|                 return econHandler; | ||||
|             } catch (final Exception ignored) { | ||||
|             } | ||||
|         } | ||||
|         return EconHandler.nullEconHandler(); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,50 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.inject; | ||||
|  | ||||
| import com.google.inject.AbstractModule; | ||||
| import com.google.inject.Provides; | ||||
| import com.google.inject.Singleton; | ||||
| import com.plotsquared.bukkit.permissions.BukkitPermissionHandler; | ||||
| import com.plotsquared.bukkit.permissions.VaultPermissionHandler; | ||||
| import com.plotsquared.core.permissions.PermissionHandler; | ||||
| import org.bukkit.Bukkit; | ||||
|  | ||||
| public class PermissionModule extends AbstractModule { | ||||
|  | ||||
|     @Provides | ||||
|     @Singleton | ||||
|     PermissionHandler providePermissionHandler() { | ||||
|         try { | ||||
|             if (Bukkit.getPluginManager().isPluginEnabled("Vault")) { | ||||
|                 return new VaultPermissionHandler(); | ||||
|             } | ||||
|         } catch (final Exception ignored) { | ||||
|         } | ||||
|         return new BukkitPermissionHandler(); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,50 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.inject; | ||||
|  | ||||
| import com.google.inject.AbstractModule; | ||||
| import com.google.inject.Provides; | ||||
| import com.google.inject.Singleton; | ||||
| import com.plotsquared.bukkit.managers.BukkitWorldManager; | ||||
| import com.plotsquared.bukkit.managers.MultiverseWorldManager; | ||||
| import com.plotsquared.core.util.PlatformWorldManager; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.World; | ||||
|  | ||||
| public class WorldManagerModule extends AbstractModule { | ||||
|  | ||||
|     @SuppressWarnings("removal") // Internal use only | ||||
|     @Provides | ||||
|     @Singleton | ||||
|     PlatformWorldManager<World> provideWorldManager() { | ||||
|         if (Bukkit.getPluginManager().getPlugin("Multiverse-Core") != null) { | ||||
|             return new MultiverseWorldManager(); | ||||
|         } else { | ||||
|             return new BukkitWorldManager(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -0,0 +1,182 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.location.Location; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.flag.implementations.CopperOxideFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.MiscInteractFlag; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.entity.Entity; | ||||
| import org.bukkit.entity.Item; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.EventPriority; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.event.block.BlockFertilizeEvent; | ||||
| import org.bukkit.event.block.BlockFormEvent; | ||||
| import org.bukkit.event.block.BlockReceiveGameEvent; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
| import java.util.UUID; | ||||
|  | ||||
| @SuppressWarnings("unused") | ||||
| public class BlockEventListener117 implements Listener { | ||||
|  | ||||
|     @Inject | ||||
|     public BlockEventListener117() { | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onBlockReceiveGame(BlockReceiveGameEvent event) { | ||||
|         Block block = event.getBlock(); | ||||
|         Location location = BukkitUtil.adapt(block.getLocation()); | ||||
|         Entity entity = event.getEntity(); | ||||
|  | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot == null || !plot.getFlag(MiscInteractFlag.class)) { | ||||
|             if (entity instanceof Player player) { | ||||
|                 BukkitPlayer plotPlayer = BukkitUtil.adapt(player); | ||||
|                 if (plot != null) { | ||||
|                     if (!plot.isAdded(plotPlayer.getUUID())) { | ||||
|                         plot.debug(plotPlayer.getName() + " couldn't trigger sculk sensors because misc-interact = false"); | ||||
|                         event.setCancelled(true); | ||||
|                     } | ||||
|                 } | ||||
|                 return; | ||||
|             } | ||||
|             if (entity instanceof Item item) { | ||||
|                 UUID itemThrower = item.getThrower(); | ||||
|                 if (plot != null) { | ||||
|                     if (!plot.isAdded(itemThrower)) { | ||||
|                         if (!plot.isAdded(itemThrower)) { | ||||
|                             plot.debug("A thrown item couldn't trigger sculk sensors because misc-interact = false"); | ||||
|                             event.setCancelled(true); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onBlockFertilize(BlockFertilizeEvent event) { | ||||
|         Block block = event.getBlock(); | ||||
|         List<org.bukkit.block.BlockState> blocks = event.getBlocks(); | ||||
|         Location location = BukkitUtil.adapt(blocks.get(0).getLocation()); | ||||
|  | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             for (int i = blocks.size() - 1; i >= 0; i--) { | ||||
|                 location = BukkitUtil.adapt(blocks.get(i).getLocation()); | ||||
|                 if (location.isPlotArea()) { | ||||
|                     blocks.remove(i); | ||||
|                 } | ||||
|             } | ||||
|             return; | ||||
|         } else { | ||||
|             Plot origin = area.getOwnedPlot(location); | ||||
|             if (origin == null) { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|             for (int i = blocks.size() - 1; i >= 0; i--) { | ||||
|                 location = BukkitUtil.adapt(blocks.get(i).getLocation()); | ||||
|                 if (!area.contains(location.getX(), location.getZ())) { | ||||
|                     blocks.remove(i); | ||||
|                     continue; | ||||
|                 } | ||||
|                 Plot plot = area.getOwnedPlot(location); | ||||
|                 if (!Objects.equals(plot, origin)) { | ||||
|                     event.getBlocks().remove(i); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         Plot origin = area.getPlot(location); | ||||
|         if (origin == null) { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
|         for (int i = blocks.size() - 1; i >= 0; i--) { | ||||
|             location = BukkitUtil.adapt(blocks.get(i).getLocation()); | ||||
|             Plot plot = area.getOwnedPlot(location); | ||||
|             if (!Objects.equals(plot, origin) && (!plot.isMerged() && !origin.isMerged())) { | ||||
|                 event.getBlocks().remove(i); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onBlockForm(BlockFormEvent event) { | ||||
|         Block block = event.getBlock(); | ||||
|         Location location = BukkitUtil.adapt(block.getLocation()); | ||||
|         if (location.isPlotRoad()) { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = area.getOwnedPlot(location); | ||||
|         if (plot == null) { | ||||
|             return; | ||||
|         } | ||||
|         switch (event.getNewState().getType()) { | ||||
|             case COPPER_BLOCK: | ||||
|             case EXPOSED_COPPER: | ||||
|             case WEATHERED_COPPER: | ||||
|             case OXIDIZED_COPPER: | ||||
|             case CUT_COPPER: | ||||
|             case EXPOSED_CUT_COPPER: | ||||
|             case WEATHERED_CUT_COPPER: | ||||
|             case OXIDIZED_CUT_COPPER: | ||||
|             case CUT_COPPER_STAIRS: | ||||
|             case EXPOSED_CUT_COPPER_STAIRS: | ||||
|             case WEATHERED_CUT_COPPER_STAIRS: | ||||
|             case OXIDIZED_CUT_COPPER_STAIRS: | ||||
|             case CUT_COPPER_SLAB: | ||||
|             case EXPOSED_CUT_COPPER_SLAB: | ||||
|             case WEATHERED_CUT_COPPER_SLAB: | ||||
|             case OXIDIZED_CUT_COPPER_SLAB: | ||||
|                 if (!plot.getFlag(CopperOxideFlag.class)) { | ||||
|                     plot.debug("Copper could not oxide because copper-oxide = false"); | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,18 +21,24 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     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.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.plot.world.SinglePlotArea; | ||||
| import com.plotsquared.core.util.ReflectionUtils.RefClass; | ||||
| import com.plotsquared.core.util.ReflectionUtils.RefField; | ||||
| import com.plotsquared.core.util.ReflectionUtils.RefMethod; | ||||
| import com.plotsquared.core.util.task.PlotSquaredTask; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import com.plotsquared.core.util.task.TaskTime; | ||||
| import io.papermc.lib.PaperLib; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.Chunk; | ||||
| @@ -51,65 +57,101 @@ import org.bukkit.event.entity.CreatureSpawnEvent; | ||||
| import org.bukkit.event.entity.ItemSpawnEvent; | ||||
| import org.bukkit.event.world.ChunkLoadEvent; | ||||
| import org.bukkit.event.world.ChunkUnloadEvent; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.lang.reflect.Method; | ||||
| import java.util.HashSet; | ||||
| import java.util.Objects; | ||||
|  | ||||
| import static com.plotsquared.core.util.ReflectionUtils.getRefClass; | ||||
|  | ||||
| @SuppressWarnings("unused") | ||||
| public class ChunkListener implements Listener { | ||||
|  | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|     private final int version; | ||||
|  | ||||
|     private RefMethod methodGetHandleChunk; | ||||
|     private RefMethod methodGetHandleWorld; | ||||
|     private RefField mustSave; | ||||
|     /* | ||||
|     private RefMethod methodGetFullChunk; | ||||
|     private RefMethod methodGetBukkitChunk; | ||||
|     private RefMethod methodGetChunkProvider; | ||||
|     private RefMethod methodGetVisibleMap; | ||||
|     private RefField worldServer; | ||||
|     private RefField playerChunkMap; | ||||
|     private RefField updatingChunks; | ||||
|     private RefField visibleChunks; | ||||
|     */ | ||||
|     private Chunk lastChunk; | ||||
|     private boolean ignoreUnload = false; | ||||
|     private boolean isTrueForNotSave = true; | ||||
|  | ||||
|     public ChunkListener() { | ||||
|         if (Settings.Chunk_Processor.AUTO_TRIM) { | ||||
|             try { | ||||
|                 RefClass classChunk = getRefClass("{nms}.Chunk"); | ||||
|                 RefClass classCraftChunk = getRefClass("{cb}.CraftChunk"); | ||||
|                 this.mustSave = classChunk.getField("mustSave"); | ||||
|                 this.methodGetHandleChunk = classCraftChunk.getMethod("getHandle"); | ||||
|             } catch (Throwable ignored) { | ||||
|                 PlotSquared.debug(PlotSquared.get().IMP.getPluginName() | ||||
|                     + "/Server not compatible for chunk processor trim/gc"); | ||||
|  | ||||
|                 Settings.Chunk_Processor.AUTO_TRIM = false; | ||||
|             } | ||||
|         } | ||||
|     @Inject | ||||
|     public ChunkListener(final @NonNull PlotAreaManager plotAreaManager) { | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|         version = PlotSquared.platform().serverVersion()[1]; | ||||
|         if (!Settings.Chunk_Processor.AUTO_TRIM) { | ||||
|             return; | ||||
|         } | ||||
|         try { | ||||
|             RefClass classCraftWorld = getRefClass("{cb}.CraftWorld"); | ||||
|             this.methodGetHandleWorld = classCraftWorld.getMethod("getHandle"); | ||||
|             RefClass classCraftChunk = getRefClass("{cb}.CraftChunk"); | ||||
|             this.methodGetHandleChunk = classCraftChunk.getMethod("getHandle"); | ||||
|             try { | ||||
|                 if (version < 17) { | ||||
|                     RefClass classChunk = getRefClass("{nms}.Chunk"); | ||||
|                     if (version == 13) { | ||||
|                         this.mustSave = classChunk.getField("mustSave"); | ||||
|                         this.isTrueForNotSave = false; | ||||
|                     } else { | ||||
|                         this.mustSave = classChunk.getField("mustNotSave"); | ||||
|                     } | ||||
|                 } else { | ||||
|                     RefClass classChunk = getRefClass("net.minecraft.world.level.chunk.Chunk"); | ||||
|                     this.mustSave = classChunk.getField("mustNotSave"); | ||||
|  | ||||
|                 } | ||||
|             } catch (NoSuchFieldException e) { | ||||
|                 e.printStackTrace(); | ||||
|             } | ||||
|         } catch (Throwable ignored) { | ||||
|             Settings.Chunk_Processor.AUTO_TRIM = false; | ||||
|         } | ||||
|         for (World world : Bukkit.getWorlds()) { | ||||
|             world.setAutoSave(false); | ||||
|         } | ||||
|         if (version > 13) { | ||||
|             return; | ||||
|         } | ||||
|         TaskManager.runTaskRepeat(() -> { | ||||
|             try { | ||||
|                 HashSet<Chunk> toUnload = new HashSet<>(); | ||||
|                 for (World world : Bukkit.getWorlds()) { | ||||
|                     String worldName = world.getName(); | ||||
|                     if (!PlotSquared.get().hasPlotArea(worldName)) { | ||||
|                     if (!this.plotAreaManager.hasPlotArea(worldName)) { | ||||
|                         continue; | ||||
|                     } | ||||
|                     Object w = world.getClass().getDeclaredMethod("getHandle").invoke(world); | ||||
|                     Object chunkMap = w.getClass().getDeclaredMethod("getPlayerChunkMap").invoke(w); | ||||
|                     Method methodIsChunkInUse = | ||||
|                         chunkMap.getClass().getDeclaredMethod("isChunkInUse", int.class, int.class); | ||||
|                     Chunk[] chunks = world.getLoadedChunks(); | ||||
|                     for (Chunk chunk : chunks) { | ||||
|                         if ((boolean) methodIsChunkInUse | ||||
|                             .invoke(chunkMap, chunk.getX(), chunk.getZ())) { | ||||
|                             continue; | ||||
|                     Object craftWorld = methodGetHandleWorld.of(world).call(); | ||||
|                     if (version == 13) { | ||||
|                         Object chunkMap = craftWorld.getClass().getDeclaredMethod("getPlayerChunkMap").invoke(craftWorld); | ||||
|                         Method methodIsChunkInUse = | ||||
|                                 chunkMap.getClass().getDeclaredMethod("isChunkInUse", int.class, int.class); | ||||
|                         Chunk[] chunks = world.getLoadedChunks(); | ||||
|                         for (Chunk chunk : chunks) { | ||||
|                             if ((boolean) methodIsChunkInUse.invoke(chunkMap, chunk.getX(), chunk.getZ())) { | ||||
|                                 continue; | ||||
|                             } | ||||
|                             int x = chunk.getX(); | ||||
|                             int z = chunk.getZ(); | ||||
|                             if (!shouldSave(worldName, x, z)) { | ||||
|                                 unloadChunk(worldName, chunk, false); | ||||
|                                 continue; | ||||
|                             } | ||||
|                             toUnload.add(chunk); | ||||
|                         } | ||||
|                         int x = chunk.getX(); | ||||
|                         int z = chunk.getZ(); | ||||
|                         if (!shouldSave(worldName, x, z)) { | ||||
|                             unloadChunk(worldName, chunk, false); | ||||
|                             continue; | ||||
|                         } | ||||
|                         toUnload.add(chunk); | ||||
|                     } | ||||
|                 } | ||||
|                 if (toUnload.isEmpty()) { | ||||
| @@ -125,7 +167,7 @@ public class ChunkListener implements Listener { | ||||
|             } catch (Throwable e) { | ||||
|                 e.printStackTrace(); | ||||
|             } | ||||
|         }, 1); | ||||
|         }, TaskTime.ticks(1L)); | ||||
|     } | ||||
|  | ||||
|     public boolean unloadChunk(String world, Chunk chunk, boolean safe) { | ||||
| @@ -134,8 +176,8 @@ public class ChunkListener implements Listener { | ||||
|         } | ||||
|         Object c = this.methodGetHandleChunk.of(chunk).call(); | ||||
|         RefField.RefExecutor field = this.mustSave.of(c); | ||||
|         if ((Boolean) field.get()) { | ||||
|             field.set(false); | ||||
|         if ((Boolean) field.get() != isTrueForNotSave) { | ||||
|             field.set(isTrueForNotSave); | ||||
|             if (chunk.isLoaded()) { | ||||
|                 ignoreUnload = true; | ||||
|                 chunk.unload(false); | ||||
| @@ -150,34 +192,56 @@ public class ChunkListener implements Listener { | ||||
|         int z = chunkZ << 4; | ||||
|         int x2 = x + 15; | ||||
|         int z2 = z + 15; | ||||
|         Plot plot = new Location(world, x, 1, z).getOwnedPlotAbs(); | ||||
|         if (plot != null && plot.hasOwner()) { | ||||
|             return true; | ||||
|         Location loc = Location.at(world, x, 1, z); | ||||
|         PlotArea plotArea = plotAreaManager.getPlotArea(loc); | ||||
|         if (plotArea != null) { | ||||
|             Plot plot = plotArea.getPlot(loc); | ||||
|             if (plot != null && plot.hasOwner()) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         plot = new Location(world, x2, 1, z2).getOwnedPlotAbs(); | ||||
|         if (plot != null && plot.hasOwner()) { | ||||
|             return true; | ||||
|         loc = Location.at(world, x2, 1, z2); | ||||
|         plotArea = plotAreaManager.getPlotArea(loc); | ||||
|         if (plotArea != null) { | ||||
|             Plot plot = plotArea.getPlot(loc); | ||||
|             if (plot != null && plot.hasOwner()) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         plot = new Location(world, x2, 1, z).getOwnedPlotAbs(); | ||||
|         if (plot != null && plot.hasOwner()) { | ||||
|             return true; | ||||
|         loc = Location.at(world, x2, 1, z); | ||||
|         plotArea = plotAreaManager.getPlotArea(loc); | ||||
|         if (plotArea != null) { | ||||
|             Plot plot = plotArea.getPlot(loc); | ||||
|             if (plot != null && plot.hasOwner()) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         plot = new Location(world, x, 1, z2).getOwnedPlotAbs(); | ||||
|         if (plot != null && plot.hasOwner()) { | ||||
|             return true; | ||||
|         loc = Location.at(world, x, 1, z2); | ||||
|         plotArea = plotAreaManager.getPlotArea(loc); | ||||
|         if (plotArea != null) { | ||||
|             Plot plot = plotArea.getPlot(loc); | ||||
|             if (plot != null && plot.hasOwner()) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         plot = new Location(world, x + 7, 1, z + 7).getOwnedPlotAbs(); | ||||
|         loc = Location.at(world, x + 7, 1, z + 7); | ||||
|         plotArea = plotAreaManager.getPlotArea(loc); | ||||
|         if (plotArea == null) { | ||||
|             return false; | ||||
|         } | ||||
|         Plot plot = plotArea.getPlot(loc); | ||||
|         return plot != null && plot.hasOwner(); | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onChunkUnload(ChunkUnloadEvent event) { | ||||
|     @EventHandler | ||||
|     public void onChunkUnload(ChunkUnloadEvent event) { | ||||
|         if (ignoreUnload) { | ||||
|             return; | ||||
|         } | ||||
|         Chunk chunk = event.getChunk(); | ||||
|         if (Settings.Chunk_Processor.AUTO_TRIM) { | ||||
|             String world = chunk.getWorld().getName(); | ||||
|             if (PlotSquared.get().hasPlotArea(world)) { | ||||
|             if ((!Settings.Enabled_Components.WORLDS || !SinglePlotArea.isSinglePlotWorld(world)) && this.plotAreaManager.hasPlotArea(world)) { | ||||
|                 if (unloadChunk(world, chunk, true)) { | ||||
|                     return; | ||||
|                 } | ||||
| @@ -188,11 +252,13 @@ public class ChunkListener implements Listener { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onChunkLoad(ChunkLoadEvent event) { | ||||
|     @EventHandler | ||||
|     public void onChunkLoad(ChunkLoadEvent event) { | ||||
|         processChunk(event.getChunk(), false); | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.LOWEST) public void onItemSpawn(ItemSpawnEvent event) { | ||||
|     @EventHandler(priority = EventPriority.LOWEST) | ||||
|     public void onItemSpawn(ItemSpawnEvent event) { | ||||
|         Item entity = event.getEntity(); | ||||
|         PaperLib.getChunkAtAsync(event.getLocation()).thenAccept(chunk -> { | ||||
|             if (chunk == this.lastChunk) { | ||||
| @@ -200,7 +266,7 @@ public class ChunkListener implements Listener { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|             if (!PlotSquared.get().hasPlotArea(chunk.getWorld().getName())) { | ||||
|             if (!this.plotAreaManager.hasPlotArea(chunk.getWorld().getName())) { | ||||
|                 return; | ||||
|             } | ||||
|             Entity[] entities = chunk.getEntities(); | ||||
| @@ -230,7 +296,7 @@ public class ChunkListener implements Listener { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|             if (!PlotSquared.get().hasPlotArea(chunk.getWorld().getName())) { | ||||
|             if (!this.plotAreaManager.hasPlotArea(chunk.getWorld().getName())) { | ||||
|                 return; | ||||
|             } | ||||
|             Entity[] entities = chunk.getEntities(); | ||||
| @@ -245,21 +311,16 @@ public class ChunkListener implements Listener { | ||||
|     } | ||||
|  | ||||
|     private void cleanChunk(final Chunk chunk) { | ||||
|         TaskManager.index.incrementAndGet(); | ||||
|         final Integer currentIndex = TaskManager.index.get(); | ||||
|         Integer task = TaskManager.runTaskRepeat(() -> { | ||||
|         final int currentIndex = TaskManager.index.incrementAndGet(); | ||||
|         PlotSquaredTask task = TaskManager.runTaskRepeat(() -> { | ||||
|             if (!chunk.isLoaded()) { | ||||
|                 Bukkit.getScheduler().cancelTask(TaskManager.tasks.get(currentIndex)); | ||||
|                 TaskManager.tasks.remove(currentIndex); | ||||
|                 PlotSquared.debug("Successfully processed and unloaded chunk!"); | ||||
|                 Objects.requireNonNull(TaskManager.removeTask(currentIndex)).cancel(); | ||||
|                 chunk.unload(true); | ||||
|                 return; | ||||
|             } | ||||
|             BlockState[] tiles = chunk.getTileEntities(); | ||||
|             if (tiles.length == 0) { | ||||
|                 Bukkit.getScheduler().cancelTask(TaskManager.tasks.get(currentIndex)); | ||||
|                 TaskManager.tasks.remove(currentIndex); | ||||
|                 PlotSquared.debug("Successfully processed and unloaded chunk!"); | ||||
|                 Objects.requireNonNull(TaskManager.removeTask(currentIndex)).cancel(); | ||||
|                 chunk.unload(true); | ||||
|                 return; | ||||
|             } | ||||
| @@ -267,21 +328,19 @@ public class ChunkListener implements Listener { | ||||
|             int i = 0; | ||||
|             while (System.currentTimeMillis() - start < 250) { | ||||
|                 if (i >= tiles.length - Settings.Chunk_Processor.MAX_TILES) { | ||||
|                     Bukkit.getScheduler().cancelTask(TaskManager.tasks.get(currentIndex)); | ||||
|                     TaskManager.tasks.remove(currentIndex); | ||||
|                     PlotSquared.debug("Successfully processed and unloaded chunk!"); | ||||
|                     Objects.requireNonNull(TaskManager.removeTask(currentIndex)).cancel(); | ||||
|                     chunk.unload(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 tiles[i].getBlock().setType(Material.AIR, false); | ||||
|                 i++; | ||||
|             } | ||||
|         }, 5); | ||||
|         TaskManager.tasks.put(currentIndex, task); | ||||
|         }, TaskTime.ticks(5L)); | ||||
|         TaskManager.addTask(task, currentIndex); | ||||
|     } | ||||
|  | ||||
|     public boolean processChunk(Chunk chunk, boolean unload) { | ||||
|         if (!PlotSquared.get().hasPlotArea(chunk.getWorld().getName())) { | ||||
|         if (!this.plotAreaManager.hasPlotArea(chunk.getWorld().getName())) { | ||||
|             return false; | ||||
|         } | ||||
|         Entity[] entities = chunk.getEntities(); | ||||
| @@ -296,24 +355,18 @@ public class ChunkListener implements Listener { | ||||
|                     toRemove--; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             PlotSquared.debug( | ||||
|                 "PlotSquared detected unsafe chunk and processed: " + (chunk.getX() << 4) + "," + ( | ||||
|                     chunk.getX() << 4)); | ||||
|         } | ||||
|         if (tiles.length > Settings.Chunk_Processor.MAX_TILES) { | ||||
|             if (unload) { | ||||
|                 PlotSquared.debug( | ||||
|                     "PlotSquared detected unsafe chunk: " + (chunk.getX() << 4) + "," + ( | ||||
|                         chunk.getX() << 4)); | ||||
|                 cleanChunk(chunk); | ||||
|                 return true; | ||||
|             } | ||||
|  | ||||
|             for (int i = 0 ; i < (tiles.length - Settings.Chunk_Processor.MAX_TILES); i++) { | ||||
|             for (int i = 0; i < (tiles.length - Settings.Chunk_Processor.MAX_TILES); i++) { | ||||
|                 tiles[i].getBlock().setType(Material.AIR, false); | ||||
|             } | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,28 +21,43 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     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.BukkitEntityUtil; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.listener.PlayerBlockEventType; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.PlotHandler; | ||||
| import com.plotsquared.core.plot.flag.implementations.DisablePhysicsFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.EntityChangeBlockFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.ExplosionFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.InvincibleFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.MobPlaceFlag; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.world.block.BlockType; | ||||
| import org.bukkit.Material; | ||||
| import org.bukkit.Particle; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.entity.Ageable; | ||||
| import org.bukkit.entity.Boat; | ||||
| import org.bukkit.entity.Entity; | ||||
| import org.bukkit.entity.EntityType; | ||||
| import org.bukkit.entity.FallingBlock; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.entity.Projectile; | ||||
| import org.bukkit.entity.TNTPrimed; | ||||
| import org.bukkit.entity.Vehicle; | ||||
| import org.bukkit.event.EventHandler; | ||||
| @@ -59,6 +74,9 @@ import org.bukkit.event.vehicle.VehicleCreateEvent; | ||||
| import org.bukkit.metadata.FixedMetadataValue; | ||||
| import org.bukkit.metadata.MetadataValue; | ||||
| import org.bukkit.plugin.Plugin; | ||||
| import org.bukkit.projectiles.BlockProjectileSource; | ||||
| import org.bukkit.projectiles.ProjectileSource; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Iterator; | ||||
| import java.util.List; | ||||
| @@ -66,13 +84,28 @@ import java.util.List; | ||||
| @SuppressWarnings("unused") | ||||
| public class EntityEventListener implements Listener { | ||||
|  | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|     private final EventDispatcher eventDispatcher; | ||||
|     private float lastRadius; | ||||
|  | ||||
|     @Inject | ||||
|     public EntityEventListener( | ||||
|             final @NonNull PlotAreaManager plotAreaManager, | ||||
|             final @NonNull EventDispatcher eventDispatcher | ||||
|     ) { | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|         this.eventDispatcher = eventDispatcher; | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST) | ||||
|     public void onEntityCombustByEntity(EntityCombustByEntityEvent event) { | ||||
|         EntityDamageByEntityEvent eventChange = | ||||
|             new EntityDamageByEntityEvent(event.getCombuster(), event.getEntity(), | ||||
|                 EntityDamageEvent.DamageCause.FIRE_TICK, event.getDuration()); | ||||
|                 new EntityDamageByEntityEvent( | ||||
|                         event.getCombuster(), | ||||
|                         event.getEntity(), | ||||
|                         EntityDamageEvent.DamageCause.FIRE_TICK, | ||||
|                         event.getDuration() | ||||
|                 ); | ||||
|         onEntityDamageByEntityEvent(eventChange); | ||||
|         if (eventChange.isCancelled()) { | ||||
|             event.setCancelled(true); | ||||
| @@ -82,8 +115,8 @@ public class EntityEventListener implements Listener { | ||||
|     @EventHandler(priority = EventPriority.HIGHEST) | ||||
|     public void onEntityDamageByEntityEvent(EntityDamageByEntityEvent event) { | ||||
|         Entity damager = event.getDamager(); | ||||
|         Location location = BukkitUtil.getLocation(damager); | ||||
|         if (!PlotSquared.get().hasPlotArea(location.getWorld())) { | ||||
|         Location location = BukkitUtil.adapt(damager.getLocation()); | ||||
|         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||
|             return; | ||||
|         } | ||||
|         Entity victim = event.getEntity(); | ||||
| @@ -98,8 +131,7 @@ public class EntityEventListener implements Listener { | ||||
| */ | ||||
|         if (!BukkitEntityUtil.entityDamage(damager, victim, event.getCause())) { | ||||
|             if (event.isCancelled()) { | ||||
|                 if (victim instanceof Ageable) { | ||||
|                     Ageable ageable = (Ageable) victim; | ||||
|                 if (victim instanceof Ageable ageable) { | ||||
|                     if (ageable.getAge() == -24000) { | ||||
|                         ageable.setAge(0); | ||||
|                         ageable.setAdult(); | ||||
| @@ -113,7 +145,7 @@ public class EntityEventListener implements Listener { | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void creatureSpawnEvent(CreatureSpawnEvent event) { | ||||
|         Entity entity = event.getEntity(); | ||||
|         Location location = BukkitUtil.getLocation(entity.getLocation()); | ||||
|         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
| @@ -135,7 +167,6 @@ public class EntityEventListener implements Listener { | ||||
|             case "PATROL": | ||||
|             case "RAID": | ||||
|             case "SHEARED": | ||||
|             case "SHOULDER_ENTITY": | ||||
|             case "SILVERFISH_BLOCK": | ||||
|             case "TRAP": | ||||
|             case "VILLAGE_DEFENSE": | ||||
| @@ -189,10 +220,10 @@ public class EntityEventListener implements Listener { | ||||
|         Block block = event.getBlock(); | ||||
|         World world = block.getWorld(); | ||||
|         String worldName = world.getName(); | ||||
|         if (!PlotSquared.get().hasPlotArea(worldName)) { | ||||
|         if (!this.plotAreaManager.hasPlotArea(worldName)) { | ||||
|             return; | ||||
|         } | ||||
|         Location location = BukkitUtil.getLocation(block.getLocation()); | ||||
|         Location location = BukkitUtil.adapt(block.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
| @@ -220,16 +251,16 @@ public class EntityEventListener implements Listener { | ||||
|                 entity.remove(); | ||||
|             } | ||||
|         } else if (event.getTo() == Material.AIR) { | ||||
|             event.getEntity() | ||||
|                 .setMetadata("plot", new FixedMetadataValue((Plugin) PlotSquared.get().IMP, plot)); | ||||
|             event.getEntity().setMetadata("plot", new FixedMetadataValue((Plugin) PlotSquared.platform(), plot)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGH) public void onDamage(EntityDamageEvent event) { | ||||
|     @EventHandler(priority = EventPriority.HIGH) | ||||
|     public void onDamage(EntityDamageEvent event) { | ||||
|         if (event.getEntityType() != EntityType.PLAYER) { | ||||
|             return; | ||||
|         } | ||||
|         Location location = BukkitUtil.getLocation(event.getEntity()); | ||||
|         Location location = BukkitUtil.adapt(event.getEntity().getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
| @@ -242,19 +273,18 @@ public class EntityEventListener implements Listener { | ||||
|             return; | ||||
|         } | ||||
|         if (plot.getFlag(InvincibleFlag.class)) { | ||||
|             plot.debug( | ||||
|                 event.getEntity().getName() + " could not take damage because invincible = true"); | ||||
|             plot.debug(event.getEntity().getName() + " could not take damage because invincible = true"); | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onBigBoom(EntityExplodeEvent event) { | ||||
|         Location location = BukkitUtil.getLocation(event.getLocation()); | ||||
|         Location location = BukkitUtil.adapt(event.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         boolean plotArea = location.isPlotArea(); | ||||
|         if (!plotArea) { | ||||
|             if (!PlotSquared.get().hasPlotArea(location.getWorld())) { | ||||
|             if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||
|                 return; | ||||
|             } | ||||
|             return; | ||||
| @@ -270,14 +300,11 @@ public class EntityEventListener implements Listener { | ||||
|                     origin = (Plot) meta.get(0).value(); | ||||
|                 } | ||||
|                 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) { | ||||
|                         if (near instanceof TNTPrimed || near.getType() | ||||
|                             .equals(EntityType.MINECART_TNT)) { | ||||
|                         if (near instanceof TNTPrimed || near.getType().equals(EntityType.MINECART_TNT)) { | ||||
|                             if (!near.hasMetadata("plot")) { | ||||
|                                 near.setMetadata("plot", | ||||
|                                     new FixedMetadataValue((Plugin) PlotSquared.get().IMP, plot)); | ||||
|                                 near.setMetadata("plot", new FixedMetadataValue((Plugin) PlotSquared.platform(), plot)); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
| @@ -286,9 +313,8 @@ public class EntityEventListener implements Listener { | ||||
|                 Iterator<Block> iterator = event.blockList().iterator(); | ||||
|                 while (iterator.hasNext()) { | ||||
|                     Block block = iterator.next(); | ||||
|                     location = BukkitUtil.getLocation(block.getLocation()); | ||||
|                     if (!area.contains(location.getX(), location.getZ()) || (origin != null | ||||
|                         && !origin.equals(area.getOwnedPlot(location)))) { | ||||
|                     location = BukkitUtil.adapt(block.getLocation()); | ||||
|                     if (!area.contains(location.getX(), location.getZ()) || !origin.equals(area.getOwnedPlot(location))) { | ||||
|                         iterator.remove(); | ||||
|                     } | ||||
|                 } | ||||
| @@ -298,35 +324,111 @@ public class EntityEventListener implements Listener { | ||||
|             } | ||||
|         } | ||||
|         event.setCancelled(true); | ||||
|         //Spawn Explosion Particles when enabled in settings | ||||
|         if (Settings.General.ALWAYS_SHOW_EXPLOSIONS) { | ||||
|             event.getLocation().getWorld().spawnParticle(Particle.EXPLOSION_HUGE, event.getLocation(), 0); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onPeskyMobsChangeTheWorldLikeWTFEvent(EntityChangeBlockEvent event) { | ||||
|         Entity e = event.getEntity(); | ||||
|         if (!(e instanceof FallingBlock)) { | ||||
|             Location location = BukkitUtil.getLocation(event.getBlock().getLocation()); | ||||
|             PlotArea area = location.getPlotArea(); | ||||
|             if (area != null) { | ||||
|                 Plot plot = area.getOwnedPlot(location); | ||||
|                 if (plot != null && plot.getFlag(MobPlaceFlag.class)) { | ||||
|         Material type = event.getBlock().getType(); | ||||
|         Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         if (e instanceof FallingBlock) { | ||||
|             // allow falling blocks converting to blocks and vice versa | ||||
|             return; | ||||
|         } else if (e instanceof Boat) { | ||||
|             // allow boats destroying lily pads | ||||
|             if (type == Material.LILY_PAD) { | ||||
|                 return; | ||||
|             } | ||||
|         } else if (e instanceof Player player) { | ||||
|             BukkitPlayer pp = BukkitUtil.adapt(player); | ||||
|             if (type.toString().equals("POWDER_SNOW")) { | ||||
|                 // Burning player evaporating powder snow. Use same checks as | ||||
|                 // trampling farmland | ||||
|                 BlockType blockType = BukkitAdapter.asBlockType(type); | ||||
|                 if (!this.eventDispatcher.checkPlayerBlockEvent(pp, | ||||
|                         PlayerBlockEventType.TRIGGER_PHYSICAL, location, blockType, true | ||||
|                 )) { | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
|                 return; | ||||
|             } else { | ||||
|                 // already handled by other flags (mainly the 'use' flag): | ||||
|                 // - player tilting big dripleaf by standing on it | ||||
|                 // - player picking glow berries from cave vine | ||||
|                 // - player trampling farmland | ||||
|                 // - player standing on or clicking redstone ore | ||||
|                 return; | ||||
|             } | ||||
|         } else if (e instanceof Projectile entity) { | ||||
|             // Exact same as the ProjectileHitEvent listener, except that we let | ||||
|             // the entity-change-block determine what to do with shooters that | ||||
|             // aren't players and aren't blocks | ||||
|             Plot plot = area.getPlot(location); | ||||
|             ProjectileSource shooter = entity.getShooter(); | ||||
|             if (shooter instanceof Player) { | ||||
|                 PlotPlayer<?> pp = BukkitUtil.adapt((Player) shooter); | ||||
|                 if (plot == null) { | ||||
|                     if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { | ||||
|                         entity.remove(); | ||||
|                         event.setCancelled(true); | ||||
|                     } | ||||
|                     return; | ||||
|                 } | ||||
|                 if (plot != null) { | ||||
|                     plot.debug(e.getType() + " could not change block because mob-place = false"); | ||||
|                 if (plot.isAdded(pp.getUUID()) || Permissions | ||||
|                         .hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|                     return; | ||||
|                 } | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|             if (!(shooter instanceof Entity) && shooter != null) { | ||||
|                 if (plot == null) { | ||||
|                     entity.remove(); | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 Location sLoc = | ||||
|                         BukkitUtil.adapt(((BlockProjectileSource) shooter).getBlock().getLocation()); | ||||
|                 if (!area.contains(sLoc.getX(), sLoc.getZ())) { | ||||
|                     entity.remove(); | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 Plot sPlot = area.getOwnedPlotAbs(sLoc); | ||||
|                 if (sPlot == null || !PlotHandler.sameOwners(plot, sPlot)) { | ||||
|                     entity.remove(); | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
|                 return; | ||||
|             } | ||||
|             // fall back to entity-change-block flag | ||||
|         } | ||||
|  | ||||
|         Plot plot = area.getOwnedPlot(location); | ||||
|         if (plot != null && !plot.getFlag(EntityChangeBlockFlag.class)) { | ||||
|             plot.debug(e.getType() + " could not change block because entity-change-block = false"); | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onPrime(ExplosionPrimeEvent event) { | ||||
|     @EventHandler | ||||
|     public void onPrime(ExplosionPrimeEvent event) { | ||||
|         this.lastRadius = event.getRadius() + 1; | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onVehicleCreate(VehicleCreateEvent event) { | ||||
|         Vehicle entity = event.getVehicle(); | ||||
|         Location location = BukkitUtil.getLocation(entity); | ||||
|         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
| @@ -337,8 +439,8 @@ public class EntityEventListener implements Listener { | ||||
|             return; | ||||
|         } | ||||
|         if (Settings.Enabled_Components.KILL_ROAD_VEHICLES) { | ||||
|             entity | ||||
|                 .setMetadata("plot", new FixedMetadataValue((Plugin) PlotSquared.get().IMP, plot)); | ||||
|             entity.setMetadata("plot", new FixedMetadataValue((Plugin) PlotSquared.platform(), plot)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.listener; | ||||
|  | ||||
| @@ -54,40 +54,43 @@ import org.bukkit.event.world.ChunkLoadEvent; | ||||
| import org.bukkit.metadata.FixedMetadataValue; | ||||
| import org.bukkit.metadata.MetadataValue; | ||||
| import org.bukkit.plugin.Plugin; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| @SuppressWarnings("unused") | ||||
| public class EntitySpawnListener implements Listener { | ||||
|  | ||||
|     private final static String KEY = "P2"; | ||||
|     private static final String KEY = "P2"; | ||||
|     private static boolean ignoreTP = false; | ||||
|     private static boolean hasPlotArea = false; | ||||
|     private static String areaName = null; | ||||
|  | ||||
|     public static void testNether(final Entity entity) { | ||||
|         @NotNull World world = entity.getWorld(); | ||||
|         if (world.getEnvironment() != World.Environment.NETHER | ||||
|             && world.getEnvironment() != World.Environment.THE_END) { | ||||
|         @NonNull World world = entity.getWorld(); | ||||
|         if (world.getEnvironment() != World.Environment.NETHER && world.getEnvironment() != World.Environment.THE_END) { | ||||
|             return; | ||||
|         } | ||||
|         test(entity); | ||||
|     } | ||||
|  | ||||
|     public static void testCreate(final Entity entity) { | ||||
|         @NotNull World world = entity.getWorld(); | ||||
|         if (!PlotSquared.get().hasPlotArea(entity.getWorld().getName())) { | ||||
|         @NonNull World world = entity.getWorld(); | ||||
|         if (!world.getName().equals(areaName)) { | ||||
|             areaName = world.getName(); | ||||
|             hasPlotArea = PlotSquared.get().getPlotAreaManager().hasPlotArea(areaName); | ||||
|         } | ||||
|         if (!hasPlotArea) { | ||||
|             return; | ||||
|         } | ||||
|         test(entity); | ||||
|     } | ||||
|  | ||||
|     public static void test(Entity entity) { | ||||
|         @NotNull World world = entity.getWorld(); | ||||
|         @NonNull World world = entity.getWorld(); | ||||
|         List<MetadataValue> meta = entity.getMetadata(KEY); | ||||
|         if (meta.isEmpty()) { | ||||
|             if (PlotSquared.get().hasPlotArea(world.getName())) { | ||||
|                 entity.setMetadata(KEY, | ||||
|                     new FixedMetadataValue((Plugin) PlotSquared.get().IMP, entity.getLocation())); | ||||
|             if (PlotSquared.get().getPlotAreaManager().hasPlotArea(world.getName())) { | ||||
|                 entity.setMetadata(KEY, new FixedMetadataValue((Plugin) PlotSquared.platform(), entity.getLocation())); | ||||
|             } | ||||
|         } else { | ||||
|             org.bukkit.Location origin = (org.bukkit.Location) meta.get(0).value(); | ||||
| @@ -121,7 +124,7 @@ public class EntitySpawnListener implements Listener { | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void creatureSpawnEvent(EntitySpawnEvent event) { | ||||
|         Entity entity = event.getEntity(); | ||||
|         Location location = BukkitUtil.getLocation(entity.getLocation()); | ||||
|         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (!location.isPlotArea()) { | ||||
|             return; | ||||
| @@ -158,32 +161,36 @@ public class EntitySpawnListener implements Listener { | ||||
|                 } | ||||
|             case SHULKER: | ||||
|                 if (!entity.hasMetadata("shulkerPlot")) { | ||||
|                     entity.setMetadata("shulkerPlot", | ||||
|                         new FixedMetadataValue((Plugin) PlotSquared.get().IMP, plot.getId())); | ||||
|                     entity.setMetadata("shulkerPlot", new FixedMetadataValue((Plugin) PlotSquared.platform(), plot.getId())); | ||||
|                 } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onChunkLoad(ChunkLoadEvent event) { | ||||
|         @NotNull Chunk chunk = event.getChunk(); | ||||
|     @EventHandler | ||||
|     public void onChunkLoad(ChunkLoadEvent event) { | ||||
|         @NonNull Chunk chunk = event.getChunk(); | ||||
|         for (final Entity entity : chunk.getEntities()) { | ||||
|             testCreate(entity); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onVehicle(VehicleUpdateEvent event) { | ||||
|     @EventHandler | ||||
|     public void onVehicle(VehicleUpdateEvent event) { | ||||
|         testNether(event.getVehicle()); | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onVehicle(VehicleCreateEvent event) { | ||||
|     @EventHandler | ||||
|     public void onVehicle(VehicleCreateEvent event) { | ||||
|         testCreate(event.getVehicle()); | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onVehicle(VehicleBlockCollisionEvent event) { | ||||
|     @EventHandler | ||||
|     public void onVehicle(VehicleBlockCollisionEvent event) { | ||||
|         testNether(event.getVehicle()); | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onTeleport(EntityTeleportEvent event) { | ||||
|     @EventHandler | ||||
|     public void onTeleport(EntityTeleportEvent event) { | ||||
|         Entity ent = event.getEntity(); | ||||
|         if (ent instanceof Vehicle || ent instanceof ArmorStand) { | ||||
|             testNether(event.getEntity()); | ||||
| @@ -195,9 +202,11 @@ public class EntitySpawnListener implements Listener { | ||||
|         testNether(event.getVehicle()); | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void spawn(CreatureSpawnEvent event) { | ||||
|     @EventHandler | ||||
|     public void spawn(CreatureSpawnEvent event) { | ||||
|         if (event.getEntityType() == EntityType.ARMOR_STAND) { | ||||
|             testCreate(event.getEntity()); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,15 +21,15 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.listener; | ||||
|  | ||||
| import com.google.common.collect.Iterables; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.configuration.Captions; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.flag.implementations.ForcefieldFlag; | ||||
| @@ -44,13 +44,13 @@ import java.util.UUID; | ||||
| @SuppressWarnings("unused") | ||||
| public class ForceFieldListener { | ||||
|  | ||||
|     private static Set<PlotPlayer> getNearbyPlayers(Player player, Plot plot) { | ||||
|         Set<PlotPlayer> players = new HashSet<>(); | ||||
|     private static Set<PlotPlayer<?>> getNearbyPlayers(Player player, Plot plot) { | ||||
|         Set<PlotPlayer<?>> players = new HashSet<>(); | ||||
|         for (Player nearPlayer : Iterables | ||||
|             .filter(player.getNearbyEntities(5d, 5d, 5d), Player.class)) { | ||||
|             PlotPlayer plotPlayer; | ||||
|             if ((plotPlayer = BukkitUtil.getPlayer(nearPlayer)) == null || !plot | ||||
|                 .equals(plotPlayer.getCurrentPlot())) { | ||||
|                 .filter(player.getNearbyEntities(5d, 5d, 5d), Player.class)) { | ||||
|             PlotPlayer<?> plotPlayer; | ||||
|             if ((plotPlayer = BukkitUtil.adapt(nearPlayer)) == null || !plot | ||||
|                     .equals(plotPlayer.getCurrentPlot())) { | ||||
|                 continue; | ||||
|             } | ||||
|             if (!plot.isAdded(plotPlayer.getUUID())) { | ||||
| @@ -60,12 +60,12 @@ public class ForceFieldListener { | ||||
|         return players; | ||||
|     } | ||||
|  | ||||
|     private static PlotPlayer hasNearbyPermitted(Player player, Plot plot) { | ||||
|     private static PlotPlayer<?> hasNearbyPermitted(Player player, Plot plot) { | ||||
|         for (Player nearPlayer : Iterables | ||||
|             .filter(player.getNearbyEntities(5d, 5d, 5d), Player.class)) { | ||||
|             PlotPlayer plotPlayer; | ||||
|             if ((plotPlayer = BukkitUtil.getPlayer(nearPlayer)) == null || !plot | ||||
|                 .equals(plotPlayer.getCurrentPlot())) { | ||||
|                 .filter(player.getNearbyEntities(5d, 5d, 5d), Player.class)) { | ||||
|             PlotPlayer<?> plotPlayer; | ||||
|             if ((plotPlayer = BukkitUtil.adapt(nearPlayer)) == null || !plot | ||||
|                     .equals(plotPlayer.getCurrentPlot())) { | ||||
|                 continue; | ||||
|             } | ||||
|             if (plot.isAdded(plotPlayer.getUUID())) { | ||||
| @@ -75,7 +75,7 @@ public class ForceFieldListener { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     private static Vector calculateVelocity(PlotPlayer player, PlotPlayer e) { | ||||
|     private static Vector calculateVelocity(PlotPlayer<?> player, PlotPlayer<?> e) { | ||||
|         Location playerLocation = player.getLocationFull(); | ||||
|         Location oPlayerLocation = e.getLocation(); | ||||
|         double playerX = playerLocation.getX(); | ||||
| @@ -105,28 +105,29 @@ public class ForceFieldListener { | ||||
|         return new Vector(x, y, z); | ||||
|     } | ||||
|  | ||||
|     public static void handleForcefield(Player player, PlotPlayer plotPlayer, Plot plot) { | ||||
|     public static void handleForcefield(Player player, PlotPlayer<?> plotPlayer, Plot plot) { | ||||
|         if (plot.getFlag(ForcefieldFlag.class)) { | ||||
|             UUID uuid = plotPlayer.getUUID(); | ||||
|             if (plot.isAdded(uuid)) { | ||||
|                 Set<PlotPlayer> players = getNearbyPlayers(player, plot); | ||||
|                 for (PlotPlayer oPlayer : players) { | ||||
|                 Set<PlotPlayer<?>> players = getNearbyPlayers(player, plot); | ||||
|                 for (PlotPlayer<?> oPlayer : players) { | ||||
|                     if (!Permissions | ||||
|                         .hasPermission(oPlayer, Captions.PERMISSION_ADMIN_ENTRY_FORCEFIELD)) { | ||||
|                             .hasPermission(oPlayer, Permission.PERMISSION_ADMIN_ENTRY_FORCEFIELD)) { | ||||
|                         ((BukkitPlayer) oPlayer).player | ||||
|                             .setVelocity(calculateVelocity(plotPlayer, oPlayer)); | ||||
|                                 .setVelocity(calculateVelocity(plotPlayer, oPlayer)); | ||||
|                     } | ||||
|                 } | ||||
|             } else { | ||||
|                 PlotPlayer oPlayer = hasNearbyPermitted(player, plot); | ||||
|                 PlotPlayer<?> oPlayer = hasNearbyPermitted(player, plot); | ||||
|                 if (oPlayer == null) { | ||||
|                     return; | ||||
|                 } | ||||
|                 if (!Permissions | ||||
|                     .hasPermission(plotPlayer, Captions.PERMISSION_ADMIN_ENTRY_FORCEFIELD)) { | ||||
|                         .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_ENTRY_FORCEFIELD)) { | ||||
|                     player.setVelocity(calculateVelocity(oPlayer, plotPlayer)); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.listener; | ||||
|  | ||||
| @@ -32,17 +32,21 @@ import com.destroystokyo.paper.event.entity.PreSpawnerSpawnEvent; | ||||
| import com.destroystokyo.paper.event.entity.SlimePathfindEvent; | ||||
| import com.destroystokyo.paper.event.player.PlayerLaunchProjectileEvent; | ||||
| import com.destroystokyo.paper.event.server.AsyncTabCompleteEvent; | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.command.Command; | ||||
| import com.plotsquared.core.command.MainCommand; | ||||
| import com.plotsquared.core.configuration.Captions; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| 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.DoneFlag; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import net.kyori.adventure.text.minimessage.Template; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.block.TileState; | ||||
| @@ -51,13 +55,13 @@ import org.bukkit.entity.EntityType; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.entity.Projectile; | ||||
| import org.bukkit.entity.Slime; | ||||
| import org.bukkit.entity.ThrownPotion; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.EventPriority; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.event.block.BlockPlaceEvent; | ||||
| import org.bukkit.event.entity.CreatureSpawnEvent; | ||||
| import org.bukkit.projectiles.ProjectileSource; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collection; | ||||
| @@ -71,14 +75,21 @@ import java.util.regex.Pattern; | ||||
| @SuppressWarnings("unused") | ||||
| public class PaperListener implements Listener { | ||||
|  | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|     private Chunk lastChunk; | ||||
|  | ||||
|     @EventHandler public void onEntityPathfind(EntityPathfindEvent event) { | ||||
|     @Inject | ||||
|     public PaperListener(final @NonNull PlotAreaManager plotAreaManager) { | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onEntityPathfind(EntityPathfindEvent event) { | ||||
|         if (!Settings.Paper_Components.ENTITY_PATHING) { | ||||
|             return; | ||||
|         } | ||||
|         Location toLoc = BukkitUtil.getLocation(event.getLoc()); | ||||
|         Location fromLoc = BukkitUtil.getLocation(event.getEntity().getLocation()); | ||||
|         Location toLoc = BukkitUtil.adapt(event.getLoc()); | ||||
|         Location fromLoc = BukkitUtil.adapt(event.getEntity().getLocation()); | ||||
|         PlotArea tarea = toLoc.getPlotArea(); | ||||
|         if (tarea == null) { | ||||
|             return; | ||||
| @@ -106,7 +117,8 @@ public class PaperListener implements Listener { | ||||
|         event.setCancelled(true); | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onEntityPathfind(SlimePathfindEvent event) { | ||||
|     @EventHandler | ||||
|     public void onEntityPathfind(SlimePathfindEvent event) { | ||||
|         if (!Settings.Paper_Components.ENTITY_PATHING) { | ||||
|             return; | ||||
|         } | ||||
| @@ -117,8 +129,8 @@ public class PaperListener implements Listener { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         Location toLoc = BukkitUtil.getLocation(b.getLocation()); | ||||
|         Location fromLoc = BukkitUtil.getLocation(event.getEntity().getLocation()); | ||||
|         Location toLoc = BukkitUtil.adapt(b.getLocation()); | ||||
|         Location fromLoc = BukkitUtil.adapt(event.getEntity().getLocation()); | ||||
|         PlotArea tarea = toLoc.getPlotArea(); | ||||
|         if (tarea == null) { | ||||
|             return; | ||||
| @@ -147,11 +159,12 @@ public class PaperListener implements Listener { | ||||
|         event.setCancelled(true); | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onPreCreatureSpawnEvent(PreCreatureSpawnEvent event) { | ||||
|     @EventHandler | ||||
|     public void onPreCreatureSpawnEvent(PreCreatureSpawnEvent event) { | ||||
|         if (!Settings.Paper_Components.CREATURE_SPAWN) { | ||||
|             return; | ||||
|         } | ||||
|         Location location = BukkitUtil.getLocation(event.getSpawnLocation()); | ||||
|         Location location = BukkitUtil.adapt(event.getSpawnLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (!location.isPlotArea()) { | ||||
|             return; | ||||
| @@ -181,7 +194,6 @@ public class PaperListener implements Listener { | ||||
|             case "PATROL": | ||||
|             case "RAID": | ||||
|             case "SHEARED": | ||||
|             case "SHOULDER_ENTITY": | ||||
|             case "SILVERFISH_BLOCK": | ||||
|             case "TRAP": | ||||
|             case "VILLAGE_DEFENSE": | ||||
| @@ -253,7 +265,7 @@ public class PaperListener implements Listener { | ||||
|     @EventHandler | ||||
|     public void onPlayerNaturallySpawnCreaturesEvent(PlayerNaturallySpawnCreaturesEvent event) { | ||||
|         if (Settings.Paper_Components.CANCEL_CHUNK_SPAWN) { | ||||
|             Location location = BukkitUtil.getLocation(event.getPlayer().getLocation()); | ||||
|             Location location = BukkitUtil.adapt(event.getPlayer().getLocation()); | ||||
|             PlotArea area = location.getPlotArea(); | ||||
|             if (area != null && !area.isMobSpawning()) { | ||||
|                 event.setCancelled(true); | ||||
| @@ -261,9 +273,10 @@ public class PaperListener implements Listener { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onPreSpawnerSpawnEvent(PreSpawnerSpawnEvent event) { | ||||
|     @EventHandler | ||||
|     public void onPreSpawnerSpawnEvent(PreSpawnerSpawnEvent event) { | ||||
|         if (Settings.Paper_Components.SPAWNER_SPAWN) { | ||||
|             Location location = BukkitUtil.getLocation(event.getSpawnerLocation()); | ||||
|             Location location = BukkitUtil.adapt(event.getSpawnerLocation()); | ||||
|             PlotArea area = location.getPlotArea(); | ||||
|             if (area != null && !area.isMobSpawnerSpawning()) { | ||||
|                 event.setCancelled(true); | ||||
| @@ -272,22 +285,26 @@ public class PaperListener implements Listener { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST) public void onBlockPlace(BlockPlaceEvent event) { | ||||
|     @EventHandler(priority = EventPriority.HIGHEST) | ||||
|     public void onBlockPlace(BlockPlaceEvent event) { | ||||
|         if (!Settings.Paper_Components.TILE_ENTITY_CHECK || !Settings.Enabled_Components.CHUNK_PROCESSOR) { | ||||
|             return; | ||||
|         } | ||||
|         if (!(event.getBlock().getState(false) instanceof TileState)) { | ||||
|             return; | ||||
|         } | ||||
|         final Location location = BukkitUtil.getLocation(event.getBlock().getLocation()); | ||||
|         final Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||
|         final PlotArea plotArea = location.getPlotArea(); | ||||
|         if (plotArea == null) { | ||||
|             return; | ||||
|         } | ||||
|         final int tileEntityCount = event.getBlock().getChunk().getTileEntities(false).length; | ||||
|         if (tileEntityCount >= Settings.Chunk_Processor.MAX_TILES) { | ||||
|             final PlotPlayer<?> plotPlayer = BukkitUtil.getPlayer(event.getPlayer()); | ||||
|             Captions.TILE_ENTITY_CAP_REACHED.send(plotPlayer, Settings.Chunk_Processor.MAX_TILES); | ||||
|             final PlotPlayer<?> plotPlayer = BukkitUtil.adapt(event.getPlayer()); | ||||
|             plotPlayer.sendMessage( | ||||
|                     TranslatableCaption.of("errors.tile_entity_cap_reached"), | ||||
|                     Template.of("amount", String.valueOf(Settings.Chunk_Processor.MAX_TILES)) | ||||
|             ); | ||||
|             event.setCancelled(true); | ||||
|             event.setBuild(false); | ||||
|         } | ||||
| @@ -299,31 +316,55 @@ public class PaperListener implements Listener { | ||||
|      * | ||||
|      * @param event Paper's PlayerLaunchProjectileEvent | ||||
|      */ | ||||
|     @EventHandler public void onProjectileLaunch(PlayerLaunchProjectileEvent event) { | ||||
|     @EventHandler | ||||
|     public void onProjectileLaunch(PlayerLaunchProjectileEvent event) { | ||||
|         if (!Settings.Paper_Components.PLAYER_PROJECTILE) { | ||||
|             return; | ||||
|         } | ||||
|         Projectile entity = event.getProjectile(); | ||||
|         if (!(entity instanceof ThrownPotion)) { | ||||
|             return; | ||||
|         } | ||||
|         ProjectileSource shooter = entity.getShooter(); | ||||
|         if (!(shooter instanceof Player)) { | ||||
|             return; | ||||
|         } | ||||
|         Location location = BukkitUtil.getLocation(entity); | ||||
|         if (!PlotSquared.get().hasPlotArea(location.getWorld())) { | ||||
|         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||
|             return; | ||||
|         } | ||||
|         PlotPlayer<?> pp = BukkitUtil.getPlayer((Player) shooter); | ||||
|         PlotPlayer<Player> pp = BukkitUtil.adapt((Player) shooter); | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot != null && !plot.isAdded(pp.getUUID())) { | ||||
|             entity.remove(); | ||||
|             event.setCancelled(true); | ||||
|  | ||||
|         if (plot == null) { | ||||
|             if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_ROAD)) { | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_PROJECTILE_ROAD)) | ||||
|                 ); | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } else if (!plot.hasOwner()) { | ||||
|             if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) | ||||
|                 ); | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } else if (!plot.isAdded(pp.getUUID())) { | ||||
|             if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) | ||||
|                 ); | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onAsyncTabCompletion(final AsyncTabCompleteEvent event) { | ||||
|     @EventHandler | ||||
|     public void onAsyncTabCompletion(final AsyncTabCompleteEvent event) { | ||||
|         if (!Settings.Paper_Components.ASYNC_TAB_COMPLETION) { | ||||
|             return; | ||||
|         } | ||||
| @@ -341,13 +382,13 @@ public class PaperListener implements Listener { | ||||
|         if (unprocessedArgs.length == 1) { | ||||
|             return; // We don't do anything in this case | ||||
|         } else if (!Settings.Enabled_Components.TAB_COMPLETED_ALIASES | ||||
|             .contains(unprocessedArgs[0].toLowerCase(Locale.ENGLISH))) { | ||||
|                 .contains(unprocessedArgs[0].toLowerCase(Locale.ENGLISH))) { | ||||
|             return; | ||||
|         } | ||||
|         final String[] args = new String[unprocessedArgs.length - 1]; | ||||
|         System.arraycopy(unprocessedArgs, 1, args, 0, args.length); | ||||
|         try { | ||||
|             final PlotPlayer<?> player = BukkitUtil.getPlayer((Player) event.getSender()); | ||||
|             final PlotPlayer<?> player = BukkitUtil.adapt((Player) event.getSender()); | ||||
|             final Collection<Command> objects = MainCommand.getInstance().tab(player, args, buffer.endsWith(" ")); | ||||
|             if (objects == null) { | ||||
|                 return; | ||||
| @@ -358,7 +399,8 @@ public class PaperListener implements Listener { | ||||
|             } | ||||
|             event.setCompletions(result); | ||||
|             event.setHandled(true); | ||||
|         } catch (final Exception ignored) {} | ||||
|         } catch (final Exception ignored) { | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,19 +21,21 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     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.util.BukkitUtil; | ||||
| import com.plotsquared.core.configuration.Captions; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import net.kyori.adventure.text.minimessage.Template; | ||||
| import org.bukkit.block.Banner; | ||||
| import org.bukkit.block.Beacon; | ||||
| import org.bukkit.block.Bed; | ||||
| import org.bukkit.block.BlockState; | ||||
| import org.bukkit.block.CommandBlock; | ||||
| import org.bukkit.block.Comparator; | ||||
| @@ -48,36 +50,45 @@ import org.bukkit.block.Jukebox; | ||||
| import org.bukkit.block.Sign; | ||||
| import org.bukkit.block.Skull; | ||||
| import org.bukkit.block.Structure; | ||||
| import org.bukkit.block.data.type.Bed; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.block.BlockPlaceEvent; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| public class PaperListener113 extends PaperListener { | ||||
|  | ||||
|     @Inject | ||||
|     public PaperListener113(@NonNull PlotAreaManager plotAreaManager) { | ||||
|         super(plotAreaManager); | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onBlockPlace(BlockPlaceEvent event) { | ||||
|         if (!Settings.Paper_Components.TILE_ENTITY_CHECK || !Settings.Enabled_Components.CHUNK_PROCESSOR) { | ||||
|             return; | ||||
|         } | ||||
|         BlockState state = event.getBlock().getState(false); | ||||
|         if (!(state instanceof Banner || state instanceof Beacon || state instanceof Bed | ||||
|                 || state instanceof CommandBlock || state instanceof Comparator || state instanceof Conduit | ||||
|                 || state instanceof Container || state instanceof CreatureSpawner || state instanceof DaylightDetector | ||||
|                 || state instanceof EnchantingTable || state instanceof EnderChest || state instanceof EndGateway | ||||
|                 || state instanceof Jukebox || state instanceof Sign || state instanceof Skull | ||||
|                 || state instanceof Structure)) { | ||||
|         if (!(state instanceof Banner || state instanceof Beacon || state instanceof Bed || state instanceof CommandBlock | ||||
|                 || state instanceof Comparator || state instanceof Conduit || state instanceof Container || state instanceof CreatureSpawner | ||||
|                 || state instanceof DaylightDetector || state instanceof EnchantingTable || state instanceof EnderChest || state instanceof EndGateway | ||||
|                 || state instanceof Jukebox || state instanceof Sign || state instanceof Skull || state instanceof Structure)) { | ||||
|             return; | ||||
|         } | ||||
|         final Location location = BukkitUtil.getLocation(event.getBlock().getLocation()); | ||||
|         final Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||
|         final PlotArea plotArea = location.getPlotArea(); | ||||
|         if (plotArea == null) { | ||||
|             return; | ||||
|         } | ||||
|         final int tileEntityCount = event.getBlock().getChunk().getTileEntities(false).length; | ||||
|         if (tileEntityCount >= Settings.Chunk_Processor.MAX_TILES) { | ||||
|             final PlotPlayer<?> plotPlayer = BukkitUtil.getPlayer(event.getPlayer()); | ||||
|             Captions.TILE_ENTITY_CAP_REACHED.send(plotPlayer, Settings.Chunk_Processor.MAX_TILES); | ||||
|             final PlotPlayer<?> plotPlayer = BukkitUtil.adapt(event.getPlayer()); | ||||
|             plotPlayer.sendMessage( | ||||
|                     TranslatableCaption.of("errors.tile_entity_cap_reached"), | ||||
|                     Template.of("amount", String.valueOf(Settings.Chunk_Processor.MAX_TILES)) | ||||
|             ); | ||||
|             event.setCancelled(true); | ||||
|             event.setBuild(false); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,20 +21,23 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     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.util.BukkitEntityUtil; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Captions; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.PlotHandler; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import net.kyori.adventure.text.minimessage.Template; | ||||
| import org.bukkit.entity.Entity; | ||||
| import org.bukkit.entity.LivingEntity; | ||||
| import org.bukkit.entity.Player; | ||||
| @@ -49,27 +52,30 @@ import org.bukkit.event.entity.ProjectileHitEvent; | ||||
| import org.bukkit.event.entity.ProjectileLaunchEvent; | ||||
| import org.bukkit.projectiles.BlockProjectileSource; | ||||
| import org.bukkit.projectiles.ProjectileSource; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| @SuppressWarnings("unused") | ||||
| public class ProjectileEventListener implements Listener { | ||||
|  | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|  | ||||
|     @Inject | ||||
|     public ProjectileEventListener(final @NonNull PlotAreaManager plotAreaManager) { | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onPotionSplash(LingeringPotionSplashEvent event) { | ||||
|         Projectile entity = event.getEntity(); | ||||
|         Location location = BukkitUtil.getLocation(entity); | ||||
|         if (!PlotSquared.get().hasPlotArea(location.getWorld())) { | ||||
|             return; | ||||
|         } | ||||
|         if (!this.onProjectileHit(event)) { | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|     public void onLingeringPotionSplash(LingeringPotionSplashEvent event) { | ||||
|         // Cancelling projectile hit events still results in area effect clouds. | ||||
|         // We need to cancel the splash events to get rid of those. | ||||
|         onProjectileHit(event); | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onPotionSplash(PotionSplashEvent event) { | ||||
|         ThrownPotion damager = event.getPotion(); | ||||
|         Location location = BukkitUtil.getLocation(damager); | ||||
|         if (!PlotSquared.get().hasPlotArea(location.getWorld())) { | ||||
|         Location location = BukkitUtil.adapt(damager.getLocation()); | ||||
|         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||
|             return; | ||||
|         } | ||||
|         int count = 0; | ||||
| @@ -79,78 +85,110 @@ public class ProjectileEventListener implements Listener { | ||||
|                 count++; | ||||
|             } | ||||
|         } | ||||
|         if ((count > 0 && count == event.getAffectedEntities().size()) || !onProjectileHit(event)) { | ||||
|         if (count > 0 && count == event.getAffectedEntities().size()) { | ||||
|             event.setCancelled(true); | ||||
|         } else { | ||||
|             // Cancelling projectile hit events still results in potions | ||||
|             // splashing in the world. We need to cancel the splash events to | ||||
|             // avoid that. | ||||
|             onProjectileHit(event); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onProjectileLaunch(ProjectileLaunchEvent event) { | ||||
|     @EventHandler | ||||
|     public void onProjectileLaunch(ProjectileLaunchEvent event) { | ||||
|         Projectile entity = event.getEntity(); | ||||
|         if (!(entity instanceof ThrownPotion)) { | ||||
|             return; | ||||
|         } | ||||
|         ProjectileSource shooter = entity.getShooter(); | ||||
|         if (!(shooter instanceof Player)) { | ||||
|             return; | ||||
|         } | ||||
|         Location location = BukkitUtil.getLocation(entity); | ||||
|         if (!PlotSquared.get().hasPlotArea(location.getWorld())) { | ||||
|         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||
|             return; | ||||
|         } | ||||
|         PlotPlayer<Player> pp = BukkitUtil.getPlayer((Player) shooter); | ||||
|         PlotPlayer<Player> pp = BukkitUtil.adapt((Player) shooter); | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot != null && !plot.isAdded(pp.getUUID())) { | ||||
|             entity.remove(); | ||||
|             event.setCancelled(true); | ||||
|  | ||||
|         if (plot == null) { | ||||
|             if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_ROAD)) { | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_PROJECTILE_ROAD)) | ||||
|                 ); | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } else if (!plot.hasOwner()) { | ||||
|             if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) | ||||
|                 ); | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } else if (!plot.isAdded(pp.getUUID())) { | ||||
|             if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) | ||||
|                 ); | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings({"BooleanMethodIsAlwaysInverted", "cos it's not... dum IntelliJ"}) @EventHandler | ||||
|     public boolean onProjectileHit(ProjectileHitEvent event) { | ||||
|     @EventHandler | ||||
|     public void onProjectileHit(ProjectileHitEvent event) { | ||||
|         Projectile entity = event.getEntity(); | ||||
|         Location location = BukkitUtil.getLocation(entity); | ||||
|         if (!PlotSquared.get().hasPlotArea(location.getWorld())) { | ||||
|             return true; | ||||
|         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||
|             return; | ||||
|         } | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return true; | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = area.getPlot(location); | ||||
|         ProjectileSource shooter = entity.getShooter(); | ||||
|         if (shooter instanceof Player) { | ||||
|             PlotPlayer<?> pp = BukkitUtil.getPlayer((Player) shooter); | ||||
|             PlotPlayer<?> pp = BukkitUtil.adapt((Player) shooter); | ||||
|             if (plot == null) { | ||||
|                 if (!Permissions.hasPermission(pp, Captions.PERMISSION_PROJECTILE_UNOWNED)) { | ||||
|                 if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { | ||||
|                     entity.remove(); | ||||
|                     return false; | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
|                 return true; | ||||
|                 return; | ||||
|             } | ||||
|             if (plot.isAdded(pp.getUUID()) || Permissions | ||||
|                 .hasPermission(pp, Captions.PERMISSION_PROJECTILE_OTHER)) { | ||||
|                 return true; | ||||
|                     .hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|                 return; | ||||
|             } | ||||
|             entity.remove(); | ||||
|             return false; | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
|         if (!(shooter instanceof Entity) && shooter != null) { | ||||
|             if (plot == null) { | ||||
|                 entity.remove(); | ||||
|                 return false; | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|             Location sLoc = | ||||
|                 BukkitUtil.getLocation(((BlockProjectileSource) shooter).getBlock().getLocation()); | ||||
|                     BukkitUtil.adapt(((BlockProjectileSource) shooter).getBlock().getLocation()); | ||||
|             if (!area.contains(sLoc.getX(), sLoc.getZ())) { | ||||
|                 entity.remove(); | ||||
|                 return false; | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|             Plot sPlot = area.getOwnedPlotAbs(sLoc); | ||||
|             if (sPlot == null || !PlotHandler.sameOwners(plot, sPlot)) { | ||||
|                 entity.remove(); | ||||
|                 return false; | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,32 +21,37 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.listener; | ||||
|  | ||||
| import com.plotsquared.bukkit.BukkitMain; | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.bukkit.BukkitPlatform; | ||||
| import com.plotsquared.bukkit.placeholder.MVdWPlaceholders; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Captions; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.player.ConsolePlayer; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.event.server.ServerLoadEvent; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| public class ServerListener implements Listener { | ||||
|  | ||||
|     private final BukkitMain plugin; | ||||
|     private final BukkitPlatform plugin; | ||||
|  | ||||
|     public ServerListener(BukkitMain plugin) { | ||||
|     @Inject | ||||
|     public ServerListener(final @NonNull BukkitPlatform plugin) { | ||||
|         this.plugin = plugin; | ||||
|     } | ||||
|  | ||||
|     @EventHandler public void onServerLoad(ServerLoadEvent event) { | ||||
|     @EventHandler | ||||
|     public void onServerLoad(ServerLoadEvent event) { | ||||
|         if (Bukkit.getPluginManager().getPlugin("MVdWPlaceholderAPI") != null && Settings.Enabled_Components.USE_MVDWAPI) { | ||||
|             new MVdWPlaceholders(this.plugin, PlotSquared.get().getPlaceholderRegistry()); | ||||
|             PlotSquared.log(Captions.PREFIX + "&6PlotSquared hooked into MVdWPlaceholderAPI"); | ||||
|             new MVdWPlaceholders(this.plugin, this.plugin.placeholderRegistry()); | ||||
|             ConsolePlayer.getConsole().sendMessage(TranslatableCaption.of("placeholder.hooked")); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,15 +21,15 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.listener; | ||||
|  | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.plot.world.SinglePlotArea; | ||||
| import com.plotsquared.core.plot.world.SinglePlotAreaManager; | ||||
| import com.plotsquared.core.util.ReflectionUtils; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.event.EventHandler; | ||||
| @@ -37,42 +37,42 @@ import org.bukkit.event.EventPriority; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.event.world.ChunkEvent; | ||||
| import org.bukkit.event.world.ChunkLoadEvent; | ||||
| import org.bukkit.plugin.Plugin; | ||||
|  | ||||
| import java.lang.reflect.Field; | ||||
| import java.lang.reflect.Method; | ||||
|  | ||||
| import static com.plotsquared.core.util.ReflectionUtils.getRefClass; | ||||
|  | ||||
| @SuppressWarnings("unused") | ||||
| public class SingleWorldListener implements Listener { | ||||
|  | ||||
|     private Method methodGetHandleChunk; | ||||
|     private Field mustSave; | ||||
|     private boolean isTrueForNotSave = true; | ||||
|     private final Method methodGetHandleChunk; | ||||
|     private Field shouldSave; | ||||
|  | ||||
|     public SingleWorldListener(Plugin plugin) throws Exception { | ||||
|         ReflectionUtils.RefClass classChunk = getRefClass("{nms}.Chunk"); | ||||
|     public SingleWorldListener() throws Exception { | ||||
|         ReflectionUtils.RefClass classCraftChunk = getRefClass("{cb}.CraftChunk"); | ||||
|         this.methodGetHandleChunk = classCraftChunk.getMethod("getHandle").getRealMethod(); | ||||
|         try { | ||||
|             if (PlotSquared.get().IMP.getServerVersion()[1] == 13) { | ||||
|                 this.mustSave = classChunk.getField("mustSave").getRealField(); | ||||
|                 this.isTrueForNotSave = false; | ||||
|             if (PlotSquared.platform().serverVersion()[1] < 17) { | ||||
|                 ReflectionUtils.RefClass classChunk = getRefClass("{nms}.Chunk"); | ||||
|                 if (PlotSquared.platform().serverVersion()[1] == 13) { | ||||
|                     this.shouldSave = classChunk.getField("mustSave").getRealField(); | ||||
|                 } else { | ||||
|                     this.shouldSave = classChunk.getField("s").getRealField(); | ||||
|                 } | ||||
|             } else { | ||||
|                 this.mustSave = classChunk.getField("mustNotSave").getRealField(); | ||||
|                 ReflectionUtils.RefClass classChunk = getRefClass("net.minecraft.world.level.chunk.Chunk"); | ||||
|                 this.shouldSave = classChunk.getField("r").getRealField(); | ||||
|             } | ||||
|         } catch (NoSuchFieldException e) { | ||||
|             e.printStackTrace(); | ||||
|         } | ||||
|         Bukkit.getPluginManager().registerEvents(this, plugin); | ||||
|     } | ||||
|  | ||||
|     public void markChunkAsClean(Chunk chunk) { | ||||
|         try { | ||||
|             Object nmsChunk = methodGetHandleChunk.invoke(chunk); | ||||
|             if (mustSave != null) { | ||||
|                 this.mustSave.set(nmsChunk, false); | ||||
|             if (shouldSave != null) { | ||||
|                 this.shouldSave.set(nmsChunk, false); | ||||
|             } | ||||
|         } catch (Throwable e) { | ||||
|             e.printStackTrace(); | ||||
| @@ -86,7 +86,7 @@ public class SingleWorldListener implements Listener { | ||||
|         if (!(man instanceof SinglePlotAreaManager)) { | ||||
|             return; | ||||
|         } | ||||
|         if (!isPlotId(name)) { | ||||
|         if (!SinglePlotArea.isSinglePlotWorld(name)) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
| @@ -98,35 +98,9 @@ public class SingleWorldListener implements Listener { | ||||
|     //        handle(event); | ||||
|     //    } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.LOWEST) public void onChunkLoad(ChunkLoadEvent event) { | ||||
|     @EventHandler(priority = EventPriority.LOWEST) | ||||
|     public void onChunkLoad(ChunkLoadEvent event) { | ||||
|         handle(event); | ||||
|     } | ||||
|  | ||||
|     private boolean isPlotId(String worldName) { | ||||
|         int len = worldName.length(); | ||||
|         int separator = 0; | ||||
|         for (int i = 0; i < len; i++) { | ||||
|             switch (worldName.charAt(i)) { | ||||
|                 case ',': | ||||
|                 case ';': | ||||
|                     separator++; | ||||
|                     break; | ||||
|                 case '-': | ||||
|                 case '0': | ||||
|                 case '1': | ||||
|                 case '2': | ||||
|                 case '3': | ||||
|                 case '4': | ||||
|                 case '5': | ||||
|                 case '6': | ||||
|                 case '7': | ||||
|                 case '8': | ||||
|                 case '9': | ||||
|                     break; | ||||
|                 default: | ||||
|                     return false; | ||||
|             } | ||||
|         } | ||||
|         return separator == 1; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,10 +21,11 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     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.generator.BukkitPlotGenerator; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.generator.GeneratorWrapper; | ||||
| @@ -36,17 +37,23 @@ import org.bukkit.event.EventPriority; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.event.world.WorldInitEvent; | ||||
| import org.bukkit.generator.ChunkGenerator; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| @SuppressWarnings("unused") | ||||
| public class WorldEvents implements Listener { | ||||
|  | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|  | ||||
|     @Inject | ||||
|     public WorldEvents(final @NonNull PlotAreaManager plotAreaManager) { | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onWorldInit(WorldInitEvent event) { | ||||
|         World world = event.getWorld(); | ||||
|         String name = world.getName(); | ||||
|         PlotAreaManager manager = PlotSquared.get().getPlotAreaManager(); | ||||
|         if (manager instanceof SinglePlotAreaManager) { | ||||
|             SinglePlotAreaManager single = (SinglePlotAreaManager) manager; | ||||
|         if (this.plotAreaManager instanceof final SinglePlotAreaManager single) { | ||||
|             if (single.isWorld(name)) { | ||||
|                 world.setKeepSpawnInMemory(false); | ||||
|                 return; | ||||
| @@ -56,7 +63,8 @@ public class WorldEvents implements Listener { | ||||
|         if (gen instanceof GeneratorWrapper) { | ||||
|             PlotSquared.get().loadWorld(name, (GeneratorWrapper<?>) gen); | ||||
|         } else { | ||||
|             PlotSquared.get().loadWorld(name, new BukkitPlotGenerator(name, gen)); | ||||
|             PlotSquared.get().loadWorld(name, new BukkitPlotGenerator(name, gen, this.plotAreaManager)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,18 +21,19 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.managers; | ||||
|  | ||||
| import com.google.inject.Singleton; | ||||
| import com.plotsquared.core.configuration.file.YamlConfiguration; | ||||
| import com.plotsquared.core.util.PlatformWorldManager; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.WorldCreator; | ||||
| import org.bukkit.WorldType; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| import java.io.File; | ||||
| import java.io.IOException; | ||||
| @@ -44,13 +45,15 @@ import java.util.List; | ||||
|  * Default Bukkit world manager. It will handle world creation by | ||||
|  * registering the generator in bukkit.yml | ||||
|  */ | ||||
| @Singleton | ||||
| public class BukkitWorldManager implements PlatformWorldManager<World> { | ||||
|  | ||||
|     @Override public void initialize() { | ||||
|     @Override | ||||
|     public void initialize() { | ||||
|     } | ||||
|  | ||||
|     @Override @Nullable | ||||
|     public World handleWorldCreation(@NotNull String worldName, @Nullable String generator) { | ||||
|     @Override | ||||
|     public @Nullable World handleWorldCreation(@NonNull String worldName, @Nullable String generator) { | ||||
|         this.setGenerator(worldName, generator); | ||||
|         final WorldCreator wc = new WorldCreator(worldName); | ||||
|         wc.environment(World.Environment.NORMAL); | ||||
| @@ -61,8 +64,8 @@ public class BukkitWorldManager implements PlatformWorldManager<World> { | ||||
|         return Bukkit.createWorld(wc); | ||||
|     } | ||||
|  | ||||
|     protected void setGenerator(@Nullable final String worldName, @Nullable final String generator) { | ||||
|         if (generator == null || worldName != null && worldName.contains(".")) { | ||||
|     protected void setGenerator(final @Nullable String worldName, final @Nullable String generator) { | ||||
|         if (generator == null) { | ||||
|             return; | ||||
|         } | ||||
|         File file = new File("bukkit.yml").getAbsoluteFile(); | ||||
| @@ -75,11 +78,13 @@ public class BukkitWorldManager implements PlatformWorldManager<World> { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public String getName() { | ||||
|     @Override | ||||
|     public String getName() { | ||||
|         return "bukkit"; | ||||
|     } | ||||
|  | ||||
|     @Override public Collection<String> getWorlds() { | ||||
|     @Override | ||||
|     public Collection<String> getWorlds() { | ||||
|         final List<World> worlds = Bukkit.getWorlds(); | ||||
|         final List<String> worldNames = new ArrayList<>(); | ||||
|         for (final World world : worlds) { | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,46 +21,50 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.managers; | ||||
|  | ||||
| /* | ||||
| import com.google.inject.Singleton; | ||||
| import org.bukkit.World; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
| import se.hyperver.hyperverse.Hyperverse; | ||||
| import se.hyperver.hyperverse.world.WorldConfiguration; | ||||
| import se.hyperver.hyperverse.world.WorldConfigurationBuilder; | ||||
| import se.hyperver.hyperverse.world.WorldFeatures; | ||||
| import se.hyperver.hyperverse.world.WorldType; | ||||
|  | ||||
| /** | ||||
|  * Hyperverse specific manager that creates worlds | ||||
|  * using Hyperverse's API | ||||
|  */ | ||||
| Hyperverse implementation is currently put on ice until Hyperverse is released on a stable line and deployed to the central | ||||
| repository. | ||||
|  | ||||
| @Singleton | ||||
| public class HyperverseWorldManager extends BukkitWorldManager { | ||||
|  | ||||
|     @Override @Nullable | ||||
|     public World handleWorldCreation(@NotNull String worldName, @Nullable String generator) { | ||||
|     @Override | ||||
|     public @Nullable World handleWorldCreation(@NonNull String worldName, @Nullable String generator) { | ||||
|         // First let Bukkit register the world | ||||
|         this.setGenerator(worldName, generator); | ||||
|         // Create the world | ||||
|         final WorldConfigurationBuilder worldConfigurationBuilder = WorldConfiguration.builder() | ||||
|             .setName(worldName).setType(WorldType.OVER_WORLD); | ||||
|                 .setName(worldName).setType(WorldType.OVER_WORLD); | ||||
|         if (generator != null) { | ||||
|             worldConfigurationBuilder.setGenerator(generator).setWorldFeatures(WorldFeatures.FLATLAND); | ||||
|         } | ||||
|         try { | ||||
|             return Hyperverse.getApi().createWorld(worldConfigurationBuilder.createWorldConfiguration()) | ||||
|                 .getBukkitWorld(); | ||||
|                     .getBukkitWorld(); | ||||
|         } catch (final Exception e) { | ||||
|             e.printStackTrace(); | ||||
|         } | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     @Override public String getName() { | ||||
|     @Override | ||||
|     public String getName() { | ||||
|         return "bukkit-hyperverse"; | ||||
|     } | ||||
|  | ||||
| } | ||||
|  */ | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,28 +21,34 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.managers; | ||||
|  | ||||
| import com.google.inject.Singleton; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.World; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| /** | ||||
|  * Multiverse specific manager that informs Multiverse of | ||||
|  * world creation by executing a console command | ||||
|  * | ||||
|  * @deprecated Deprecated and scheduled for removal without replacement | ||||
|  *         in favor of the build in setup wizard. | ||||
|  */ | ||||
| @Deprecated(forRemoval = true) | ||||
| @Singleton | ||||
| public class MultiverseWorldManager extends BukkitWorldManager { | ||||
|  | ||||
|     @Override @Nullable | ||||
|     public World handleWorldCreation(@NotNull final String worldName, @Nullable final String generator) { | ||||
|     @Override | ||||
|     public @Nullable World handleWorldCreation(final @NonNull String worldName, final @Nullable String generator) { | ||||
|         // First let Bukkit register the world | ||||
|         this.setGenerator(worldName, generator); | ||||
|         // Then we send the console command | ||||
|         final StringBuilder commandBuilder = new StringBuilder("mv create ") | ||||
|             .append(worldName).append(" normal"); | ||||
|                 .append(worldName).append(" normal"); | ||||
|         if (generator != null) { | ||||
|             commandBuilder.append(" -g ").append(generator); | ||||
|         } | ||||
| @@ -50,7 +56,8 @@ public class MultiverseWorldManager extends BukkitWorldManager { | ||||
|         return Bukkit.getWorld(worldName); | ||||
|     } | ||||
|  | ||||
|     @Override public String getName() { | ||||
|     @Override | ||||
|     public String getName() { | ||||
|         return "bukkit-multiverse"; | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -0,0 +1,107 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.ConsolePermissionProfile; | ||||
| import com.plotsquared.core.permissions.PermissionHandler; | ||||
| import com.plotsquared.core.permissions.PermissionProfile; | ||||
| import com.plotsquared.core.player.ConsolePlayer; | ||||
| import com.plotsquared.core.player.OfflinePlotPlayer; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| import java.lang.ref.WeakReference; | ||||
| import java.util.EnumSet; | ||||
| import java.util.Optional; | ||||
| import java.util.Set; | ||||
|  | ||||
| public class BukkitPermissionHandler implements PermissionHandler { | ||||
|  | ||||
|     @Override | ||||
|     public void initialize() { | ||||
|     } | ||||
|  | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public Optional<PermissionProfile> getPermissionProfile( | ||||
|             @NonNull PlotPlayer<?> playerPlotPlayer | ||||
|     ) { | ||||
|         if (playerPlotPlayer instanceof final BukkitPlayer bukkitPlayer) { | ||||
|             return Optional.of(new BukkitPermissionProfile(bukkitPlayer.getPlatformPlayer())); | ||||
|         } else if (playerPlotPlayer instanceof ConsolePlayer) { | ||||
|             return Optional.of(ConsolePermissionProfile.INSTANCE); | ||||
|         } | ||||
|         return Optional.empty(); | ||||
|     } | ||||
|  | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public Optional<PermissionProfile> getPermissionProfile( | ||||
|             @NonNull OfflinePlotPlayer offlinePlotPlayer | ||||
|     ) { | ||||
|         return Optional.empty(); | ||||
|     } | ||||
|  | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public Set<PermissionHandlerCapability> getCapabilities() { | ||||
|         return EnumSet.of(PermissionHandlerCapability.ONLINE_PERMISSIONS); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     private static final class BukkitPermissionProfile implements PermissionProfile { | ||||
|  | ||||
|         private final WeakReference<Player> playerReference; | ||||
|  | ||||
|         private BukkitPermissionProfile(final @NonNull Player player) { | ||||
|             this.playerReference = new WeakReference<>(player); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public boolean hasPermission( | ||||
|                 final @Nullable String world, | ||||
|                 final @NonNull String permission | ||||
|         ) { | ||||
|             final Player player = this.playerReference.get(); | ||||
|             return player != null && player.hasPermission(permission); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public boolean hasKeyedPermission( | ||||
|                 final @Nullable String world, | ||||
|                 final @NonNull String stub, | ||||
|                 final @NonNull String key | ||||
|         ) { | ||||
|             final Player player = this.playerReference.get(); | ||||
|             return player != null && (player.hasPermission(stub + "." + key) || player.hasPermission(stub + ".*")); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,142 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.BukkitOfflinePlayer; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | ||||
| import com.plotsquared.core.permissions.ConsolePermissionProfile; | ||||
| import com.plotsquared.core.permissions.PermissionHandler; | ||||
| import com.plotsquared.core.permissions.PermissionProfile; | ||||
| import com.plotsquared.core.player.ConsolePlayer; | ||||
| import com.plotsquared.core.player.OfflinePlotPlayer; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import net.milkbowl.vault.permission.Permission; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.OfflinePlayer; | ||||
| import org.bukkit.plugin.RegisteredServiceProvider; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| import java.util.EnumSet; | ||||
| import java.util.Optional; | ||||
| import java.util.Set; | ||||
|  | ||||
| public class VaultPermissionHandler implements PermissionHandler { | ||||
|  | ||||
|     private Permission permissions; | ||||
|  | ||||
|     @Override | ||||
|     public void initialize() { | ||||
|         if (Bukkit.getServer().getPluginManager().getPlugin("Vault") == null) { | ||||
|             throw new IllegalStateException("Vault is not present on the server"); | ||||
|         } | ||||
|         RegisteredServiceProvider<Permission> permissionProvider = | ||||
|                 Bukkit.getServer().getServicesManager().getRegistration(Permission.class); | ||||
|         if (permissionProvider != null) { | ||||
|             this.permissions = permissionProvider.getProvider(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public Optional<PermissionProfile> getPermissionProfile( | ||||
|             @NonNull PlotPlayer<?> playerPlotPlayer | ||||
|     ) { | ||||
|         if (playerPlotPlayer instanceof final BukkitPlayer bukkitPlayer) { | ||||
|             return Optional.of(new VaultPermissionProfile(bukkitPlayer.getPlatformPlayer())); | ||||
|         } else if (playerPlotPlayer instanceof ConsolePlayer) { | ||||
|             return Optional.of(ConsolePermissionProfile.INSTANCE); | ||||
|         } | ||||
|         return Optional.empty(); | ||||
|     } | ||||
|  | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public Optional<PermissionProfile> getPermissionProfile( | ||||
|             @NonNull OfflinePlotPlayer offlinePlotPlayer | ||||
|     ) { | ||||
|         if (offlinePlotPlayer instanceof BukkitOfflinePlayer) { | ||||
|             return Optional.of(new VaultPermissionProfile(((BukkitOfflinePlayer) offlinePlotPlayer).player)); | ||||
|         } | ||||
|         return Optional.empty(); | ||||
|     } | ||||
|  | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public Set<PermissionHandlerCapability> getCapabilities() { | ||||
|         return EnumSet.of( | ||||
|                 PermissionHandlerCapability.PER_WORLD_PERMISSIONS, | ||||
|                 PermissionHandlerCapability.ONLINE_PERMISSIONS, | ||||
|                 PermissionHandlerCapability.OFFLINE_PERMISSIONS | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     private final class VaultPermissionProfile implements PermissionProfile { | ||||
|  | ||||
|         private final OfflinePlayer offlinePlayer; | ||||
|  | ||||
|         private VaultPermissionProfile(final @NonNull OfflinePlayer offlinePlayer) { | ||||
|             this.offlinePlayer = offlinePlayer; | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public boolean hasPermission( | ||||
|                 final @Nullable String world, | ||||
|                 final @NonNull String permission | ||||
|         ) { | ||||
|             if (permissions == null) { | ||||
|                 return false; | ||||
|             } | ||||
|             if (world == null && offlinePlayer instanceof BukkitPlayer) { | ||||
|                 return permissions.playerHas(((BukkitPlayer) offlinePlayer).getPlatformPlayer(), permission); | ||||
|             } | ||||
|             return permissions.playerHas(world, offlinePlayer, permission); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public boolean hasKeyedPermission( | ||||
|                 final @Nullable String world, | ||||
|                 final @NonNull String stub, | ||||
|                 final @NonNull String key | ||||
|         ) { | ||||
|             if (permissions == null) { | ||||
|                 return false; | ||||
|             } | ||||
|             if (world == null && offlinePlayer instanceof BukkitPlayer) { | ||||
|                 return permissions.playerHas( | ||||
|                         ((BukkitPlayer) offlinePlayer).getPlatformPlayer(), | ||||
|                         stub + ".*" | ||||
|                 ) || permissions.playerHas(((BukkitPlayer) offlinePlayer).getPlatformPlayer(), stub + "." + key); | ||||
|             } | ||||
|             return permissions.playerHas(world, offlinePlayer, stub + ".*") || permissions.playerHas(world, offlinePlayer, | ||||
|                     stub + "." + key | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.placeholder; | ||||
|  | ||||
| @@ -34,7 +34,7 @@ import com.plotsquared.core.util.placeholders.Placeholder; | ||||
| import com.plotsquared.core.util.placeholders.PlaceholderRegistry; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.plugin.Plugin; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| /** | ||||
|  * Placeholder support for MVdWPlaceholderAPI | ||||
| @@ -45,8 +45,10 @@ public class MVdWPlaceholders { | ||||
|     private final Plugin plugin; | ||||
|     private final PlaceholderRegistry registry; | ||||
|  | ||||
|     public MVdWPlaceholders(@NotNull final Plugin plugin, | ||||
|                             @NotNull final PlaceholderRegistry registry) { | ||||
|     public MVdWPlaceholders( | ||||
|             final @NonNull Plugin plugin, | ||||
|             final @NonNull PlaceholderRegistry registry | ||||
|     ) { | ||||
|         this.plugin = plugin; | ||||
|         this.registry = registry; | ||||
|         for (final Placeholder placeholder : registry.getPlaceholders()) { | ||||
| @@ -55,24 +57,24 @@ public class MVdWPlaceholders { | ||||
|         PlotSquared.get().getEventDispatcher().registerListener(this); | ||||
|     } | ||||
|  | ||||
|     @Subscribe public void onNewPlaceholder(@NotNull final | ||||
|         PlaceholderRegistry.PlaceholderAddedEvent event) { | ||||
|     @Subscribe | ||||
|     public void onNewPlaceholder(final PlaceholderRegistry.@NonNull PlaceholderAddedEvent event) { | ||||
|         this.addPlaceholder(event.getPlaceholder()); | ||||
|     } | ||||
|  | ||||
|     private void addPlaceholder(@NotNull final Placeholder placeholder) { | ||||
|         PlaceholderAPI.registerPlaceholder(plugin, PREFIX + String.format("%s", placeholder.getKey()), | ||||
|             placeholderReplaceEvent -> { | ||||
|                 if (!placeholderReplaceEvent.isOnline() || placeholderReplaceEvent.getPlayer() == null) { | ||||
|                     return ""; | ||||
|     private void addPlaceholder(final @NonNull Placeholder placeholder) { | ||||
|         PlaceholderAPI.registerPlaceholder( | ||||
|                 plugin, | ||||
|                 PREFIX + String.format("%s", placeholder.getKey()), | ||||
|                 placeholderReplaceEvent -> { | ||||
|                     if (!placeholderReplaceEvent.isOnline() || placeholderReplaceEvent.getPlayer() == null) { | ||||
|                         return ""; | ||||
|                     } | ||||
|                     final PlotPlayer<Player> player = BukkitUtil.adapt(placeholderReplaceEvent.getPlayer()); | ||||
|                     String key = placeholderReplaceEvent.getPlaceholder().substring(PREFIX.length()); | ||||
|                     return registry.getPlaceholderValue(key, player); | ||||
|                 } | ||||
|                 final PlotPlayer<Player> player = BukkitUtil.getPlayer(placeholderReplaceEvent.getPlayer()); | ||||
|                 if (player == null) { | ||||
|                     return ""; | ||||
|                 } | ||||
|                 String key = placeholderReplaceEvent.getPlaceholder().substring(PREFIX.length()); | ||||
|                 return registry.getPlaceholderValue(key, player); | ||||
|             }); | ||||
|         ); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.placeholder; | ||||
|  | ||||
| @@ -36,28 +36,34 @@ public class PAPIPlaceholders extends PlaceholderExpansion { | ||||
|     public PAPIPlaceholders() { | ||||
|     } | ||||
|  | ||||
|     @Override public boolean persist() { | ||||
|     @Override | ||||
|     public boolean persist() { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     @Override public boolean canRegister() { | ||||
|     @Override | ||||
|     public boolean canRegister() { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     @Override public String getAuthor() { | ||||
|     @Override | ||||
|     public String getAuthor() { | ||||
|         return "IntellectualSites"; | ||||
|     } | ||||
|  | ||||
|     @Override public String getIdentifier() { | ||||
|     @Override | ||||
|     public String getIdentifier() { | ||||
|         return "plotsquared"; | ||||
|     } | ||||
|  | ||||
|     @Override public String getVersion() { | ||||
|     @Override | ||||
|     public String getVersion() { | ||||
|         return "3"; | ||||
|     } | ||||
|  | ||||
|     @Override public String onPlaceholderRequest(Player p, String identifier) { | ||||
|         final PlotPlayer<?> pl = PlotSquared.imp().getPlayerManager().getPlayerIfExists(p.getUniqueId()); | ||||
|     @Override | ||||
|     public String onPlaceholderRequest(Player p, String identifier) { | ||||
|         final PlotPlayer<?> pl = PlotSquared.platform().playerManager().getPlayerIfExists(p.getUniqueId()); | ||||
|  | ||||
|         if (pl == null) { | ||||
|             return ""; | ||||
| @@ -66,24 +72,26 @@ public class PAPIPlaceholders extends PlaceholderExpansion { | ||||
|         // PAPI specific ones that don't translate well over into other placeholder APIs | ||||
|         if (identifier.startsWith("has_plot_")) { | ||||
|             identifier = identifier.substring("has_plot_".length()); | ||||
|             if (identifier.isEmpty()) | ||||
|             if (identifier.isEmpty()) { | ||||
|                 return ""; | ||||
|             } | ||||
|  | ||||
|             return pl.getPlotCount(identifier) > 0 ? | ||||
|                 PlaceholderAPIPlugin.booleanTrue() : | ||||
|                 PlaceholderAPIPlugin.booleanFalse(); | ||||
|                     PlaceholderAPIPlugin.booleanTrue() : | ||||
|                     PlaceholderAPIPlugin.booleanFalse(); | ||||
|         } | ||||
|  | ||||
|         if (identifier.startsWith("plot_count_")) { | ||||
|             identifier = identifier.substring("plot_count_".length()); | ||||
|             if (identifier.isEmpty()) | ||||
|             if (identifier.isEmpty()) { | ||||
|                 return ""; | ||||
|             } | ||||
|  | ||||
|             return String.valueOf(pl.getPlotCount(identifier)); | ||||
|         } | ||||
|  | ||||
|         // PlotSquared placeholders | ||||
|         return PlotSquared.get().getPlaceholderRegistry().getPlaceholderValue(identifier, pl); | ||||
|         return PlotSquared.platform().placeholderRegistry().getPlaceholderValue(identifier, pl); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,20 +21,22 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.placeholder; | ||||
|  | ||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | ||||
| import com.plotsquared.core.configuration.ChatFormatter; | ||||
| import com.plotsquared.core.configuration.caption.ChatFormatter; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import me.clip.placeholderapi.PlaceholderAPI; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| public class PlaceholderFormatter implements ChatFormatter { | ||||
|  | ||||
|     @Override public void format(final ChatContext context) { | ||||
|         final PlotPlayer recipient = context.getRecipient(); | ||||
|     @Override | ||||
|     public void format(final @NonNull ChatContext context) { | ||||
|         final PlotPlayer<?> recipient = context.getRecipient(); | ||||
|         if (recipient instanceof BukkitPlayer) { | ||||
|             if (context.isRawOutput()) { | ||||
|                 context.setMessage(context.getMessage().replace('%', '\u2010')); | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,43 +21,77 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.player; | ||||
|  | ||||
| import com.plotsquared.core.permissions.NullPermissionProfile; | ||||
| import com.plotsquared.core.permissions.PermissionHandler; | ||||
| import com.plotsquared.core.permissions.PermissionProfile; | ||||
| import com.plotsquared.core.player.OfflinePlotPlayer; | ||||
| import org.bukkit.OfflinePlayer; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.checkerframework.checker.index.qual.NonNegative; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| import java.util.UUID; | ||||
|  | ||||
| public class BukkitOfflinePlayer implements OfflinePlotPlayer { | ||||
|  | ||||
|     public final OfflinePlayer player; | ||||
|     private final PermissionProfile permissionProfile; | ||||
|  | ||||
|     /** | ||||
|      * Please do not use this method. Instead use BukkitUtil.getPlayer(Player), | ||||
|      * as it caches player objects. | ||||
|      * | ||||
|      * @param player | ||||
|      * @param player            Bukkit OfflinePlayer player to convert | ||||
|      * @param permissionHandler Permission Profile to be used | ||||
|      */ | ||||
|     public BukkitOfflinePlayer(OfflinePlayer player) { | ||||
|     public BukkitOfflinePlayer( | ||||
|             final @NonNull OfflinePlayer player, final @NonNull | ||||
|             PermissionHandler permissionHandler | ||||
|     ) { | ||||
|         this.player = player; | ||||
|         this.permissionProfile = permissionHandler.getPermissionProfile(this) | ||||
|                 .orElse(NullPermissionProfile.INSTANCE); | ||||
|     } | ||||
|  | ||||
|     @NotNull @Override public UUID getUUID() { | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public UUID getUUID() { | ||||
|         return this.player.getUniqueId(); | ||||
|     } | ||||
|  | ||||
|     @Override public long getLastPlayed() { | ||||
|         return this.player.getLastPlayed(); | ||||
|     @Override | ||||
|     @NonNegative | ||||
|     public long getLastPlayed() { | ||||
|         return this.player.getLastSeen(); | ||||
|     } | ||||
|  | ||||
|     @Override public boolean isOnline() { | ||||
|         return this.player.isOnline(); | ||||
|     } | ||||
|  | ||||
|     @Override public String getName() { | ||||
|     @Override | ||||
|     public String getName() { | ||||
|         return this.player.getName(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean hasPermission( | ||||
|             final @Nullable String world, | ||||
|             final @NonNull String permission | ||||
|     ) { | ||||
|         return this.permissionProfile.hasPermission(world, permission); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean hasKeyedPermission( | ||||
|             final @Nullable String world, | ||||
|             final @NonNull String stub, | ||||
|             final @NonNull String key | ||||
|     ) { | ||||
|         return this.permissionProfile.hasPermission(world, stub + "." + key) || this.permissionProfile.hasPermission( | ||||
|                 world, | ||||
|                 stub + ".*" | ||||
|         ); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,28 +21,30 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.player; | ||||
|  | ||||
| import com.google.common.base.Charsets; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Captions; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.events.TeleportCause; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.permissions.PermissionHandler; | ||||
| import com.plotsquared.core.player.ConsolePlayer; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.PlotWeather; | ||||
| import com.plotsquared.core.util.EconHandler; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.MathMan; | ||||
| import com.plotsquared.core.util.StringMan; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.extension.platform.Actor; | ||||
| import com.sk89q.worldedit.world.item.ItemType; | ||||
| import com.sk89q.worldedit.world.item.ItemTypes; | ||||
| import io.papermc.lib.PaperLib; | ||||
| import net.kyori.adventure.audience.Audience; | ||||
| import org.bukkit.GameMode; | ||||
| import org.bukkit.Sound; | ||||
| import org.bukkit.WeatherType; | ||||
| @@ -52,7 +54,8 @@ import org.bukkit.event.EventException; | ||||
| import org.bukkit.event.player.PlayerTeleportEvent; | ||||
| import org.bukkit.permissions.PermissionAttachmentInfo; | ||||
| import org.bukkit.plugin.RegisteredListener; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.checkerframework.checker.index.qual.NonNegative; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Arrays; | ||||
| import java.util.Set; | ||||
| @@ -68,63 +71,72 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|  | ||||
|     private static boolean CHECK_EFFECTIVE = true; | ||||
|     public final Player player; | ||||
|     private boolean offline; | ||||
|     private String name; | ||||
|  | ||||
|     /** | ||||
|      * <p>Please do not use this method. Instead use | ||||
|      * BukkitUtil.getPlayer(Player), as it caches player objects.</p> | ||||
|      * | ||||
|      * @param player Bukkit player instance | ||||
|      * @param plotAreaManager   PlotAreaManager instance | ||||
|      * @param eventDispatcher   EventDispatcher instance | ||||
|      * @param player            Bukkit player instance | ||||
|      * @param permissionHandler PermissionHandler instance | ||||
|      */ | ||||
|     public BukkitPlayer(@NotNull final Player player) { | ||||
|         this(player, false); | ||||
|     public BukkitPlayer( | ||||
|             final @NonNull PlotAreaManager plotAreaManager, final @NonNull EventDispatcher eventDispatcher, | ||||
|             final @NonNull Player player, final @NonNull PermissionHandler permissionHandler | ||||
|     ) { | ||||
|         this(plotAreaManager, eventDispatcher, player, false, permissionHandler); | ||||
|     } | ||||
|  | ||||
|     public BukkitPlayer(@NotNull final Player player, final boolean offline) { | ||||
|         this(player, offline, true); | ||||
|     } | ||||
|  | ||||
|     public BukkitPlayer(@NotNull final Player player, final boolean offline, final boolean realPlayer) { | ||||
|     public BukkitPlayer( | ||||
|             final @NonNull PlotAreaManager plotAreaManager, final @NonNull | ||||
|             EventDispatcher eventDispatcher, final @NonNull Player player, | ||||
|             final boolean realPlayer, | ||||
|             final @NonNull PermissionHandler permissionHandler | ||||
|     ) { | ||||
|         super(plotAreaManager, eventDispatcher, permissionHandler); | ||||
|         this.player = player; | ||||
|         this.offline = offline; | ||||
|         this.setupPermissionProfile(); | ||||
|         if (realPlayer) { | ||||
|             super.populatePersistentMetaMap(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public Actor toActor() { | ||||
|     @Override | ||||
|     public Actor toActor() { | ||||
|         return BukkitAdapter.adapt(player); | ||||
|     } | ||||
|  | ||||
|     @Override public Player getPlatformPlayer() { | ||||
|     @Override | ||||
|     public Player getPlatformPlayer() { | ||||
|         return this.player; | ||||
|     } | ||||
|  | ||||
|     @NotNull @Override public Location getLocation() { | ||||
|         final Location location = super.getLocation(); | ||||
|         return location == null ? BukkitUtil.getLocation(this.player) : location; | ||||
|     } | ||||
|  | ||||
|     @NotNull @Override public UUID getUUID() { | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public UUID getUUID() { | ||||
|         if (Settings.UUID.OFFLINE) { | ||||
|             if (Settings.UUID.FORCE_LOWERCASE) { | ||||
|                 return UUID.nameUUIDFromBytes(("OfflinePlayer:" + | ||||
|                     getName().toLowerCase()).getBytes(Charsets.UTF_8)); | ||||
|                         getName().toLowerCase()).getBytes(Charsets.UTF_8)); | ||||
|             } else { | ||||
|                 return UUID.nameUUIDFromBytes(("OfflinePlayer:" + | ||||
|                     getName()).getBytes(Charsets.UTF_8)); | ||||
|                         getName()).getBytes(Charsets.UTF_8)); | ||||
|             } | ||||
|         } | ||||
|         return player.getUniqueId(); | ||||
|     } | ||||
|  | ||||
|     @Override public long getLastPlayed() { | ||||
|         return this.player.getLastPlayed(); | ||||
|     @Override | ||||
|     @NonNegative | ||||
|     public long getLastPlayed() { | ||||
|         return this.player.getLastSeen(); | ||||
|     } | ||||
|  | ||||
|     @Override public boolean canTeleport(@NotNull final Location location) { | ||||
|         final org.bukkit.Location to = BukkitUtil.getLocation(location); | ||||
|     @Override | ||||
|     public boolean canTeleport(final @NonNull Location location) { | ||||
|         final org.bukkit.Location to = BukkitUtil.adapt(location); | ||||
|         final org.bukkit.Location from = player.getLocation(); | ||||
|         PlayerTeleportEvent event = new PlayerTeleportEvent(player, from, to); | ||||
|         callEvent(event); | ||||
| @@ -136,15 +148,10 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void sendTitle(String title, String subtitle, int fadeIn, int stay, int fadeOut) { | ||||
|         player.sendTitle(title, subtitle, fadeIn, stay, fadeOut); | ||||
|     } | ||||
|  | ||||
|     private void callEvent(@NotNull final Event event) { | ||||
|     private void callEvent(final @NonNull Event event) { | ||||
|         final RegisteredListener[] listeners = event.getHandlers().getRegisteredListeners(); | ||||
|         for (final RegisteredListener listener : listeners) { | ||||
|             if (listener.getPlugin().getName().equals(PlotSquared.imp().getPluginName())) { | ||||
|             if (listener.getPlugin().getName().equals(PlotSquared.platform().pluginName())) { | ||||
|                 continue; | ||||
|             } | ||||
|             try { | ||||
| @@ -155,23 +162,22 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public boolean hasPermission(final String permission) { | ||||
|         if (this.offline && EconHandler.getEconHandler() != null) { | ||||
|             return EconHandler.getEconHandler().hasPermission(getName(), permission); | ||||
|         } | ||||
|         return this.player.hasPermission(permission); | ||||
|     } | ||||
|  | ||||
|     @Override public int hasPermissionRange(final String stub, final int range) { | ||||
|         if (hasPermission(Captions.PERMISSION_ADMIN.getTranslated())) { | ||||
|     @SuppressWarnings("StringSplitter") | ||||
|     @Override | ||||
|     @NonNegative | ||||
|     public int hasPermissionRange( | ||||
|             final @NonNull String stub, | ||||
|             @NonNegative final int range | ||||
|     ) { | ||||
|         if (hasPermission(Permission.PERMISSION_ADMIN.toString())) { | ||||
|             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 + Captions.PERMISSION_STAR.getTranslated())) { | ||||
|                 if (hasPermission(n + Captions.PERMISSION_STAR.getTranslated())) { | ||||
|             if (!stub.equals(n + Permission.PERMISSION_STAR.toString())) { | ||||
|                 if (hasPermission(n + Permission.PERMISSION_STAR.toString())) { | ||||
|                     return Integer.MAX_VALUE; | ||||
|                 } | ||||
|             } | ||||
| @@ -222,85 +228,63 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|         return max; | ||||
|     } | ||||
|  | ||||
|     @Override public boolean isPermissionSet(final String permission) { | ||||
|         return this.player.isPermissionSet(permission); | ||||
|     } | ||||
|  | ||||
|     @Override public void sendMessage(String message) { | ||||
|         message = message.replace('\u2010', '%').replace('\u2021', '&').replace('\u2030', '&'); | ||||
|         if (!StringMan.isEqual(this.getMeta("lastMessage"), message) || ( | ||||
|             System.currentTimeMillis() - this.<Long>getMeta("lastMessageTime") > 5000)) { | ||||
|             setMeta("lastMessage", message); | ||||
|             setMeta("lastMessageTime", System.currentTimeMillis()); | ||||
|             this.player.sendMessage(message); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void teleport(@NotNull final Location location, @NotNull final TeleportCause cause) { | ||||
|     public void teleport(final @NonNull Location location, final @NonNull TeleportCause cause) { | ||||
|         if (Math.abs(location.getX()) >= 30000000 || Math.abs(location.getZ()) >= 30000000) { | ||||
|             return; | ||||
|         } | ||||
|         final org.bukkit.Location bukkitLocation = | ||||
|             new org.bukkit.Location(BukkitUtil.getWorld(location.getWorld()), location.getX() + 0.5, | ||||
|                 location.getY(), location.getZ() + 0.5, location.getYaw(), location.getPitch()); | ||||
|                 new org.bukkit.Location(BukkitUtil.getWorld(location.getWorldName()), location.getX() + 0.5, | ||||
|                         location.getY(), location.getZ() + 0.5, location.getYaw(), location.getPitch() | ||||
|                 ); | ||||
|         PaperLib.teleportAsync(player, bukkitLocation, getTeleportCause(cause)); | ||||
|     } | ||||
|  | ||||
|     @Override public String getName() { | ||||
|     @Override | ||||
|     public String getName() { | ||||
|         if (this.name == null) { | ||||
|             this.name = this.player.getName(); | ||||
|         } | ||||
|         return this.name; | ||||
|     } | ||||
|  | ||||
|     @Override public boolean isOnline() { | ||||
|         return !this.offline && this.player.isOnline(); | ||||
|     } | ||||
|  | ||||
|     @Override public void setCompassTarget(Location location) { | ||||
|     @Override | ||||
|     public void setCompassTarget(Location location) { | ||||
|         this.player.setCompassTarget( | ||||
|             new org.bukkit.Location(BukkitUtil.getWorld(location.getWorld()), location.getX(), | ||||
|                 location.getY(), location.getZ())); | ||||
|                 new org.bukkit.Location(BukkitUtil.getWorld(location.getWorldName()), location.getX(), | ||||
|                         location.getY(), location.getZ() | ||||
|                 )); | ||||
|     } | ||||
|  | ||||
|     @Override public Location getLocationFull() { | ||||
|         return BukkitUtil.getLocationFull(this.player); | ||||
|     @Override | ||||
|     public Location getLocationFull() { | ||||
|         return BukkitUtil.adaptComplete(this.player.getLocation()); | ||||
|     } | ||||
|  | ||||
|     @Override public void setWeather(@NotNull final PlotWeather weather) { | ||||
|     @Override | ||||
|     public void setWeather(final @NonNull PlotWeather weather) { | ||||
|         switch (weather) { | ||||
|             case CLEAR: | ||||
|                 this.player.setPlayerWeather(WeatherType.CLEAR); | ||||
|                 break; | ||||
|             case RAIN: | ||||
|                 this.player.setPlayerWeather(WeatherType.DOWNFALL); | ||||
|                 break; | ||||
|             case WORLD: | ||||
|                 this.player.resetPlayerWeather(); | ||||
|                 break; | ||||
|             default: | ||||
|             case CLEAR -> this.player.setPlayerWeather(WeatherType.CLEAR); | ||||
|             case RAIN -> this.player.setPlayerWeather(WeatherType.DOWNFALL); | ||||
|             case WORLD -> this.player.resetPlayerWeather(); | ||||
|             default -> { | ||||
|                 //do nothing as this is PlotWeather.OFF | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @NotNull @Override public com.sk89q.worldedit.world.gamemode.GameMode getGameMode() { | ||||
|         switch (this.player.getGameMode()) { | ||||
|             case ADVENTURE: | ||||
|                 return ADVENTURE; | ||||
|             case CREATIVE: | ||||
|                 return CREATIVE; | ||||
|             case SPECTATOR: | ||||
|                 return SPECTATOR; | ||||
|             case SURVIVAL: | ||||
|             default: | ||||
|                 return SURVIVAL; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void setGameMode(@NotNull final com.sk89q.worldedit.world.gamemode.GameMode gameMode) { | ||||
|     public com.sk89q.worldedit.world.gamemode.GameMode getGameMode() { | ||||
|         return switch (this.player.getGameMode()) { | ||||
|             case ADVENTURE -> ADVENTURE; | ||||
|             case CREATIVE -> CREATIVE; | ||||
|             case SPECTATOR -> SPECTATOR; | ||||
|             default -> SURVIVAL; | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void setGameMode(final com.sk89q.worldedit.world.gamemode.GameMode gameMode) { | ||||
|         if (ADVENTURE.equals(gameMode)) { | ||||
|             this.player.setGameMode(GameMode.ADVENTURE); | ||||
|         } else if (CREATIVE.equals(gameMode)) { | ||||
| @@ -312,7 +296,8 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public void setTime(final long time) { | ||||
|     @Override | ||||
|     public void setTime(final long time) { | ||||
|         if (time != Long.MAX_VALUE) { | ||||
|             this.player.setPlayerTime(time, false); | ||||
|         } else { | ||||
| @@ -320,44 +305,58 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public boolean getFlight() { | ||||
|     @Override | ||||
|     public boolean getFlight() { | ||||
|         return player.getAllowFlight(); | ||||
|     } | ||||
|  | ||||
|     @Override public void setFlight(boolean fly) { | ||||
|     @Override | ||||
|     public void setFlight(boolean fly) { | ||||
|         this.player.setAllowFlight(fly); | ||||
|     } | ||||
|  | ||||
|     @Override public void playMusic(@NotNull final Location location, @NotNull final ItemType id) { | ||||
|     @Override | ||||
|     public void playMusic(final @NonNull Location location, final @NonNull ItemType id) { | ||||
|         if (id == ItemTypes.AIR) { | ||||
|             // Let's just stop all the discs because why not? | ||||
|             for (final Sound sound : Arrays.stream(Sound.values()) | ||||
|                 .filter(sound -> sound.name().contains("DISC")).collect(Collectors.toList())) { | ||||
|                     .filter(sound -> sound.name().contains("DISC")).toList()) { | ||||
|                 player.stopSound(sound); | ||||
|             } | ||||
|             // this.player.playEffect(BukkitUtil.getLocation(location), Effect.RECORD_PLAY, Material.AIR); | ||||
|         } else { | ||||
|             // this.player.playEffect(BukkitUtil.getLocation(location), Effect.RECORD_PLAY, id.to(Material.class)); | ||||
|             this.player.playSound(BukkitUtil.getLocation(location), | ||||
|                 Sound.valueOf(BukkitAdapter.adapt(id).name()), Float.MAX_VALUE, 1f); | ||||
|             this.player.playSound(BukkitUtil.adapt(location), | ||||
|                     Sound.valueOf(BukkitAdapter.adapt(id).name()), Float.MAX_VALUE, 1f | ||||
|             ); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public void kick(final String message) { | ||||
|     @SuppressWarnings("deprecation") // Needed for Spigot compatibility | ||||
|     @Override | ||||
|     public void kick(final String message) { | ||||
|         this.player.kickPlayer(message); | ||||
|     } | ||||
|  | ||||
|     @Override public void stopSpectating() { | ||||
|     @Override | ||||
|     public void stopSpectating() { | ||||
|         if (getGameMode() == SPECTATOR) { | ||||
|             this.player.setSpectatorTarget(null); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public boolean isBanned() { | ||||
|     @Override | ||||
|     public boolean isBanned() { | ||||
|         return this.player.isBanned(); | ||||
|     } | ||||
|  | ||||
|     @Override public boolean canSee(final PlotPlayer<?> other) { | ||||
|     @Override | ||||
|     public @NonNull Audience getAudience() { | ||||
|         return BukkitUtil.BUKKIT_AUDIENCES.player(this.player); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean canSee(final PlotPlayer<?> other) { | ||||
|         if (other instanceof ConsolePlayer) { | ||||
|             return true; | ||||
|         } else { | ||||
| @@ -365,14 +364,19 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public PlayerTeleportEvent.TeleportCause getTeleportCause(@NotNull final TeleportCause cause) { | ||||
|         switch (cause) { | ||||
|             case COMMAND: | ||||
|                 return PlayerTeleportEvent.TeleportCause.COMMAND; | ||||
|             case PLUGIN: | ||||
|                 return PlayerTeleportEvent.TeleportCause.PLUGIN; | ||||
|             default: | ||||
|                 return PlayerTeleportEvent.TeleportCause.UNKNOWN; | ||||
|     /** | ||||
|      * Convert from PlotSquared's {@link TeleportCause} to Bukkit's {@link PlayerTeleportEvent.TeleportCause} | ||||
|      * | ||||
|      * @param cause PlotSquared teleport cause to convert | ||||
|      * @return Bukkit's equivalent teleport cause | ||||
|      */ | ||||
|     public PlayerTeleportEvent.TeleportCause getTeleportCause(final @NonNull TeleportCause cause) { | ||||
|         if (TeleportCause.CauseSets.COMMAND.contains(cause)) { | ||||
|             return PlayerTeleportEvent.TeleportCause.COMMAND; | ||||
|         } else if (cause == TeleportCause.UNKNOWN) { | ||||
|             return PlayerTeleportEvent.TeleportCause.UNKNOWN; | ||||
|         } | ||||
|         return PlayerTeleportEvent.TeleportCause.PLUGIN; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,48 +21,78 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.player; | ||||
|  | ||||
| import com.google.inject.Inject; | ||||
| import com.google.inject.Singleton; | ||||
| import com.plotsquared.core.permissions.PermissionHandler; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.PlayerManager; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| import java.util.UUID; | ||||
|  | ||||
| /** | ||||
|  * Player manager providing {@link BukkitPlayer Bukkit players} | ||||
|  */ | ||||
| @Singleton | ||||
| public class BukkitPlayerManager extends PlayerManager<BukkitPlayer, Player> { | ||||
|  | ||||
|     @NotNull @Override public BukkitPlayer getPlayer(@NotNull final Player object) { | ||||
|         try { | ||||
|             return getPlayer(object.getUniqueId()); | ||||
|         } catch (final NoSuchPlayerException exception) { | ||||
|             return new BukkitPlayer(object, object.isOnline(), false); | ||||
|         } | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|     private final EventDispatcher eventDispatcher; | ||||
|     private final PermissionHandler permissionHandler; | ||||
|  | ||||
|     @Inject | ||||
|     public BukkitPlayerManager( | ||||
|             final @NonNull PlotAreaManager plotAreaManager, | ||||
|             final @NonNull EventDispatcher eventDispatcher, | ||||
|             final @NonNull PermissionHandler permissionHandler | ||||
|     ) { | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|         this.eventDispatcher = eventDispatcher; | ||||
|         this.permissionHandler = permissionHandler; | ||||
|     } | ||||
|  | ||||
|     @Override @NotNull public BukkitPlayer createPlayer(@NotNull final UUID uuid) { | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public BukkitPlayer getPlayer(final @NonNull Player object) { | ||||
|         if (object.getUniqueId().version() == 2) { // not a real player | ||||
|             return new BukkitPlayer(this.plotAreaManager, this.eventDispatcher, object, false, this.permissionHandler); | ||||
|         } | ||||
|         if (!object.isOnline()) { | ||||
|             throw new NoSuchPlayerException(object.getUniqueId()); | ||||
|         } | ||||
|         return getPlayer(object.getUniqueId()); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public @NonNull BukkitPlayer createPlayer(final @NonNull UUID uuid) { | ||||
|         final Player player = Bukkit.getPlayer(uuid); | ||||
|         if (player == null || !player.isOnline()) { | ||||
|             throw new NoSuchPlayerException(uuid); | ||||
|         } | ||||
|         return new BukkitPlayer(player); | ||||
|         return new BukkitPlayer(this.plotAreaManager, this.eventDispatcher, player, this.permissionHandler); | ||||
|     } | ||||
|  | ||||
|     @Nullable @Override public BukkitOfflinePlayer getOfflinePlayer(@Nullable final UUID uuid) { | ||||
|     @Nullable | ||||
|     @Override | ||||
|     public BukkitOfflinePlayer getOfflinePlayer(final @Nullable UUID uuid) { | ||||
|         if (uuid == null) { | ||||
|             return null; | ||||
|         } | ||||
|         return new BukkitOfflinePlayer(Bukkit.getOfflinePlayer(uuid)); | ||||
|         return new BukkitOfflinePlayer(Bukkit.getOfflinePlayer(uuid), this.permissionHandler); | ||||
|     } | ||||
|  | ||||
|     @NotNull @Override public BukkitOfflinePlayer getOfflinePlayer(@NotNull final String username) { | ||||
|         return new BukkitOfflinePlayer(Bukkit.getOfflinePlayer(username)); | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public BukkitOfflinePlayer getOfflinePlayer(final @NonNull String username) { | ||||
|         return new BukkitOfflinePlayer(Bukkit.getOfflinePlayer(username), this.permissionHandler); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,261 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.queue; | ||||
|  | ||||
| import com.google.inject.Inject; | ||||
| import com.google.inject.assistedinject.Assisted; | ||||
| import com.plotsquared.bukkit.BukkitPlatform; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.queue.ChunkCoordinator; | ||||
| import com.plotsquared.core.queue.subscriber.ProgressSubscriber; | ||||
| import com.plotsquared.core.util.task.PlotSquaredTask; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import com.plotsquared.core.util.task.TaskTime; | ||||
| import com.sk89q.worldedit.math.BlockVector2; | ||||
| import com.sk89q.worldedit.world.World; | ||||
| import io.papermc.lib.PaperLib; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.plugin.Plugin; | ||||
| import org.bukkit.plugin.java.JavaPlugin; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Collection; | ||||
| import java.util.LinkedList; | ||||
| import java.util.List; | ||||
| import java.util.Queue; | ||||
| import java.util.concurrent.LinkedBlockingQueue; | ||||
| import java.util.concurrent.atomic.AtomicInteger; | ||||
| import java.util.function.Consumer; | ||||
|  | ||||
| /** | ||||
|  * Utility that allows for the loading and coordination of chunk actions | ||||
|  * <p> | ||||
|  * The coordinator takes in collection of chunk coordinates, loads them | ||||
|  * and allows the caller to specify a sink for the loaded chunks. The | ||||
|  * coordinator will prevent the chunks from being unloaded until the sink | ||||
|  * has fully consumed the chunk | ||||
|  * </p> | ||||
|  **/ | ||||
| public final class BukkitChunkCoordinator extends ChunkCoordinator { | ||||
|  | ||||
|     private final List<ProgressSubscriber> progressSubscribers = new LinkedList<>(); | ||||
|  | ||||
|     private final Queue<BlockVector2> requestedChunks; | ||||
|     private final Queue<Chunk> availableChunks; | ||||
|     private final long maxIterationTime; | ||||
|     private final Plugin plugin; | ||||
|     private final Consumer<BlockVector2> chunkConsumer; | ||||
|     private final org.bukkit.World bukkitWorld; | ||||
|     private final Runnable whenDone; | ||||
|     private final Consumer<Throwable> throwableConsumer; | ||||
|     private final boolean unloadAfter; | ||||
|     private final int totalSize; | ||||
|  | ||||
|     private final AtomicInteger expectedSize; | ||||
|     private int batchSize; | ||||
|     private PlotSquaredTask task; | ||||
|     private boolean shouldCancel; | ||||
|     private boolean finished; | ||||
|  | ||||
|     @Inject | ||||
|     private BukkitChunkCoordinator( | ||||
|             @Assisted final long maxIterationTime, | ||||
|             @Assisted final int initialBatchSize, | ||||
|             @Assisted final @NonNull Consumer<BlockVector2> chunkConsumer, | ||||
|             @Assisted final @NonNull World world, | ||||
|             @Assisted final @NonNull Collection<BlockVector2> requestedChunks, | ||||
|             @Assisted final @NonNull Runnable whenDone, | ||||
|             @Assisted final @NonNull Consumer<Throwable> throwableConsumer, | ||||
|             @Assisted final boolean unloadAfter, | ||||
|             @Assisted final @NonNull Collection<ProgressSubscriber> progressSubscribers | ||||
|     ) { | ||||
|         this.requestedChunks = new LinkedBlockingQueue<>(requestedChunks); | ||||
|         this.availableChunks = new LinkedBlockingQueue<>(); | ||||
|         this.totalSize = requestedChunks.size(); | ||||
|         this.expectedSize = new AtomicInteger(this.totalSize); | ||||
|         this.batchSize = initialBatchSize; | ||||
|         this.chunkConsumer = chunkConsumer; | ||||
|         this.maxIterationTime = maxIterationTime; | ||||
|         this.whenDone = whenDone; | ||||
|         this.throwableConsumer = throwableConsumer; | ||||
|         this.unloadAfter = unloadAfter; | ||||
|         this.plugin = JavaPlugin.getPlugin(BukkitPlatform.class); | ||||
|         this.bukkitWorld = Bukkit.getWorld(world.getName()); | ||||
|         this.progressSubscribers.addAll(progressSubscribers); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void start() { | ||||
|         // Request initial batch | ||||
|         this.requestBatch(); | ||||
|         // Wait until next tick to give the chunks a chance to be loaded | ||||
|         TaskManager.runTaskLater(() -> task = TaskManager.runTaskRepeat(this, TaskTime.ticks(1)), TaskTime.ticks(1)); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void cancel() { | ||||
|         shouldCancel = true; | ||||
|     } | ||||
|  | ||||
|     private void finish() { | ||||
|         try { | ||||
|             this.whenDone.run(); | ||||
|         } catch (final Throwable throwable) { | ||||
|             this.throwableConsumer.accept(throwable); | ||||
|         } finally { | ||||
|             for (final ProgressSubscriber subscriber : this.progressSubscribers) { | ||||
|                 subscriber.notifyEnd(); | ||||
|             } | ||||
|             task.cancel(); | ||||
|             finished = true; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void run() { | ||||
|         if (shouldCancel) { | ||||
|             if (unloadAfter) { | ||||
|                 Chunk chunk; | ||||
|                 while ((chunk = availableChunks.poll()) != null) { | ||||
|                     freeChunk(chunk); | ||||
|                 } | ||||
|             } | ||||
|             finish(); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         Chunk chunk = this.availableChunks.poll(); | ||||
|         if (chunk == null) { | ||||
|             return; | ||||
|         } | ||||
|         long[] iterationTime = new long[2]; | ||||
|         int processedChunks = 0; | ||||
|         do { | ||||
|             final long start = System.currentTimeMillis(); | ||||
|             try { | ||||
|                 this.chunkConsumer.accept(BlockVector2.at(chunk.getX(), chunk.getZ())); | ||||
|             } catch (final Throwable throwable) { | ||||
|                 this.throwableConsumer.accept(throwable); | ||||
|             } | ||||
|             if (unloadAfter) { | ||||
|                 this.freeChunk(chunk); | ||||
|             } | ||||
|             processedChunks++; | ||||
|             final long end = System.currentTimeMillis(); | ||||
|             // Update iteration time | ||||
|             iterationTime[0] = iterationTime[1]; | ||||
|             iterationTime[1] = end - start; | ||||
|         } while (iterationTime[0] + iterationTime[1] < this.maxIterationTime * 2 && (chunk = availableChunks.poll()) != null); | ||||
|         if (processedChunks < this.batchSize) { | ||||
|             // Adjust batch size based on the amount of processed chunks per tick | ||||
|             this.batchSize = processedChunks; | ||||
|         } | ||||
|  | ||||
|         final int expected = this.expectedSize.addAndGet(-processedChunks); | ||||
|  | ||||
|         if (expected <= 0) { | ||||
|             finish(); | ||||
|         } else { | ||||
|             if (this.availableChunks.size() < processedChunks) { | ||||
|                 final double progress = ((double) totalSize - (double) expected) / (double) totalSize; | ||||
|                 for (final ProgressSubscriber subscriber : this.progressSubscribers) { | ||||
|                     subscriber.notifyProgress(this, progress); | ||||
|                 } | ||||
|                 this.requestBatch(); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Requests a batch of chunks to be loaded | ||||
|      */ | ||||
|     private void requestBatch() { | ||||
|         BlockVector2 chunk; | ||||
|         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 | ||||
|             PaperLib | ||||
|                     .getChunkAtAsync(this.bukkitWorld, chunk.getX(), chunk.getZ(), true, true) | ||||
|                     .whenComplete((chunkObject, throwable) -> { | ||||
|                         if (throwable != null) { | ||||
|                             throwable.printStackTrace(); | ||||
|                             // We want one less because this couldn't be processed | ||||
|                             this.expectedSize.decrementAndGet(); | ||||
|                         } else if (PlotSquared.get().isMainThread(Thread.currentThread())) { | ||||
|                             this.processChunk(chunkObject); | ||||
|                         } else { | ||||
|                             TaskManager.runTask(() -> this.processChunk(chunkObject)); | ||||
|                         } | ||||
|                     }); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Once a chunk has been loaded, process it (add a plugin ticket and add to | ||||
|      * available chunks list). It is important that this gets executed on the | ||||
|      * server's main thread. | ||||
|      */ | ||||
|     private void processChunk(final @NonNull Chunk chunk) { | ||||
|         if (!chunk.isLoaded()) { | ||||
|             throw new IllegalArgumentException(String.format("Chunk %d;%d is is not loaded", chunk.getX(), chunk.getZ())); | ||||
|         } | ||||
|         if (finished) { | ||||
|             return; | ||||
|         } | ||||
|         chunk.addPluginChunkTicket(this.plugin); | ||||
|         this.availableChunks.add(chunk); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Once a chunk has been used, free it up for unload by removing the plugin ticket | ||||
|      */ | ||||
|     private void freeChunk(final @NonNull Chunk chunk) { | ||||
|         if (!chunk.isLoaded()) { | ||||
|             throw new IllegalArgumentException(String.format("Chunk %d;%d is is not loaded", chunk.getX(), chunk.getZ())); | ||||
|         } | ||||
|         chunk.removePluginChunkTicket(this.plugin); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public int getRemainingChunks() { | ||||
|         return this.expectedSize.get(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public int getTotalChunks() { | ||||
|         return this.totalSize; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Subscribe to coordinator progress updates | ||||
|      * | ||||
|      * @param subscriber Subscriber | ||||
|      */ | ||||
|     public void subscribeToProgress(final @NonNull ProgressSubscriber subscriber) { | ||||
|         this.progressSubscribers.add(subscriber); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,245 +0,0 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  * | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.queue; | ||||
|  | ||||
| import com.plotsquared.bukkit.schematic.StateWrapper; | ||||
| import com.plotsquared.bukkit.util.BukkitBlockUtil; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.queue.BasicLocalBlockQueue; | ||||
| import com.plotsquared.core.util.BlockUtil; | ||||
| import com.plotsquared.core.util.MainUtil; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import com.sk89q.jnbt.CompoundTag; | ||||
| import com.sk89q.worldedit.EditSession; | ||||
| import com.sk89q.worldedit.WorldEdit; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.math.BlockVector3; | ||||
| import com.sk89q.worldedit.regions.CuboidRegion; | ||||
| import com.sk89q.worldedit.world.biome.BiomeType; | ||||
| import com.sk89q.worldedit.world.block.BaseBlock; | ||||
| import com.sk89q.worldedit.world.block.BlockState; | ||||
| import io.papermc.lib.PaperLib; | ||||
| import lombok.NonNull; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.Material; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.Biome; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.block.Container; | ||||
| import org.bukkit.block.data.BlockData; | ||||
|  | ||||
| import java.util.concurrent.ExecutionException; | ||||
| import java.util.function.Consumer; | ||||
|  | ||||
| public class BukkitLocalQueue extends BasicLocalBlockQueue { | ||||
|  | ||||
|     public BukkitLocalQueue(String world) { | ||||
|         super(world); | ||||
|     } | ||||
|  | ||||
|     @Override public LocalChunk getLocalChunk(int x, int z) { | ||||
|         return new BasicLocalChunk(this, x, z) { | ||||
|             // Custom stuff? | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     @Override public void optimize() { | ||||
|  | ||||
|     } | ||||
|  | ||||
|     @Override public BlockState getBlock(int x, int y, int z) { | ||||
|         World worldObj = Bukkit.getWorld(getWorld()); | ||||
|         if (worldObj != null) { | ||||
|             Block block = worldObj.getBlockAt(x, y, z); | ||||
|             return BukkitBlockUtil.get(block); | ||||
|         } else { | ||||
|             return BlockUtil.get(0, 0); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public void refreshChunk(int x, int z) { | ||||
|         World worldObj = Bukkit.getWorld(getWorld()); | ||||
|         if (worldObj != null) { | ||||
|             worldObj.refreshChunk(x, z); | ||||
|         } else { | ||||
|             PlotSquared.debug("Error Refreshing Chunk"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public void fixChunkLighting(int x, int z) { | ||||
|         // Do nothing | ||||
|     } | ||||
|  | ||||
|     @Override public final void regenChunk(int x, int z) { | ||||
|         World worldObj = Bukkit.getWorld(getWorld()); | ||||
|         if (worldObj != null) { | ||||
|             try { | ||||
|                 worldObj.regenerateChunk(x, z); | ||||
|             } catch (UnsupportedOperationException e) { | ||||
|                 com.sk89q.worldedit.world.World world = BukkitAdapter.adapt(worldObj); | ||||
|                 try (EditSession editSession = WorldEdit.getInstance().getEditSessionFactory() | ||||
|                     .getEditSession(world, -1);) { | ||||
|                     CuboidRegion region = | ||||
|                         new CuboidRegion(world, BlockVector3.at((x << 4), 0, (z << 4)), | ||||
|                             BlockVector3.at((x << 4) + 15, 255, (z << 4) + 15)); | ||||
|                     world.regenerate(region, editSession); | ||||
|                 } | ||||
|             } | ||||
|         } else { | ||||
|             PlotSquared.debug("Error Regenerating Chunk"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public final void setComponents(LocalChunk lc) | ||||
|         throws ExecutionException, InterruptedException { | ||||
|         setBaseBlocks(lc); | ||||
|     } | ||||
|  | ||||
|     public void setBaseBlocks(LocalChunk localChunk) { | ||||
|         World worldObj = Bukkit.getWorld(getWorld()); | ||||
|         if (worldObj == null) { | ||||
|             throw new NullPointerException("World cannot be null."); | ||||
|         } | ||||
|         final Consumer<Chunk> chunkConsumer = chunk -> { | ||||
|             for (int layer = 0; layer < localChunk.baseblocks.length; layer++) { | ||||
|                 BaseBlock[] blocksLayer = localChunk.baseblocks[layer]; | ||||
|                 if (blocksLayer != null) { | ||||
|                     for (int j = 0; j < blocksLayer.length; j++) { | ||||
|                         if (blocksLayer[j] != null) { | ||||
|                             BaseBlock block = blocksLayer[j]; | ||||
|                             int x = MainUtil.x_loc[layer][j]; | ||||
|                             int y = MainUtil.y_loc[layer][j]; | ||||
|                             int z = MainUtil.z_loc[layer][j]; | ||||
|  | ||||
|                             BlockData blockData = BukkitAdapter.adapt(block); | ||||
|  | ||||
|                             Block existing = chunk.getBlock(x, y, z); | ||||
|                             final BlockState existingBaseBlock = | ||||
|                                 BukkitAdapter.adapt(existing.getBlockData()); | ||||
|                             if (BukkitBlockUtil.get(existing).equals(existingBaseBlock) && existing | ||||
|                                 .getBlockData().matches(blockData)) { | ||||
|                                 continue; | ||||
|                             } | ||||
|  | ||||
|                             if (existing.getState() instanceof Container) { | ||||
|                                 ((Container) existing.getState()).getInventory().clear(); | ||||
|                             } | ||||
|  | ||||
|                             existing.setType(BukkitAdapter.adapt(block.getBlockType()), false); | ||||
|                             existing.setBlockData(blockData, false); | ||||
|                             if (block.hasNbtData()) { | ||||
|                                 CompoundTag tag = block.getNbtData(); | ||||
|                                 StateWrapper sw = new StateWrapper(tag); | ||||
|  | ||||
|                                 sw.restoreTag(worldObj.getName(), existing.getX(), existing.getY(), | ||||
|                                     existing.getZ()); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             if (setBiome() && localChunk.biomes != null) { | ||||
|                 for (int x = 0; x < localChunk.biomes.length; x++) { | ||||
|                     BiomeType[] biomeZ = localChunk.biomes[x]; | ||||
|                     if (biomeZ != null) { | ||||
|                         for (int z = 0; z < biomeZ.length; z++) { | ||||
|                             if (biomeZ[z] != null) { | ||||
|                                 BiomeType biomeType = biomeZ[z]; | ||||
|  | ||||
|                                 Biome biome = BukkitAdapter.adapt(biomeType); | ||||
|                                 worldObj.setBiome((chunk.getX() << 4) + x, (chunk.getZ() << 4) + z, | ||||
|                                     biome); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }; | ||||
|         if (isForceSync()) { | ||||
|             chunkConsumer.accept(getChunk(worldObj, localChunk)); | ||||
|         } else { | ||||
|             PaperLib.getChunkAtAsync(worldObj, localChunk.getX(), localChunk.getZ(), true) | ||||
|                 .thenAccept(chunkConsumer); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private Chunk getChunk(final World world, final LocalChunk localChunk) { | ||||
|         Chunk chunk = null; | ||||
|         if (this.getChunkObject() != null && this.getChunkObject() instanceof Chunk) { | ||||
|             chunk = (Chunk) this.getChunkObject(); | ||||
|         } | ||||
|         if (chunk == null) { | ||||
|             chunk = world.getChunkAt(localChunk.getX(), localChunk.getZ()); | ||||
|         } | ||||
|         if (!chunk.isLoaded()) { | ||||
|             chunk.load(true); | ||||
|         } | ||||
|         return chunk; | ||||
|     } | ||||
|  | ||||
|     private void setMaterial(@NonNull final BlockState plotBlock, @NonNull final Block block) { | ||||
|         Material material = BukkitAdapter.adapt(plotBlock.getBlockType()); | ||||
|         block.setType(material, false); | ||||
|     } | ||||
|  | ||||
|     private boolean equals(@NonNull final BlockState plotBlock, @NonNull final Block block) { | ||||
|         return plotBlock.equals(BukkitBlockUtil.get(block)); | ||||
|     } | ||||
|  | ||||
|     public void setBiomes(LocalChunk lc) { | ||||
|         World worldObj = Bukkit.getWorld(getWorld()); | ||||
|         if (worldObj == null) { | ||||
|             throw new NullPointerException("World cannot be null."); | ||||
|         } | ||||
|         if (lc.biomes == null) { | ||||
|             throw new NullPointerException("Biomes cannot be null."); | ||||
|         } | ||||
|         final Consumer<Chunk> chunkConsumer = chunk -> { | ||||
|             for (int x = 0; x < lc.biomes.length; x++) { | ||||
|                 BiomeType[] biomeZ = lc.biomes[x]; | ||||
|                 if (biomeZ != null) { | ||||
|                     for (int z = 0; z < biomeZ.length; z++) { | ||||
|                         if (biomeZ[z] != null) { | ||||
|                             BiomeType biomeType = biomeZ[z]; | ||||
|  | ||||
|                             Biome biome = BukkitAdapter.adapt(biomeType); | ||||
|                             worldObj | ||||
|                                 .setBiome((chunk.getX() << 4) + x, (chunk.getZ() << 4) + z, biome); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }; | ||||
|         if (this.isForceSync()) { | ||||
|             chunkConsumer.accept(getChunk(worldObj, lc)); | ||||
|         } else { | ||||
|             PaperLib.getChunkAtAsync(worldObj, lc.getX(), lc.getZ(), true) | ||||
|                 .thenAccept(chunkConsumer); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,377 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.queue; | ||||
|  | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.bukkit.schematic.StateWrapper; | ||||
| import com.plotsquared.bukkit.util.BukkitBlockUtil; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.inject.factory.ChunkCoordinatorBuilderFactory; | ||||
| import com.plotsquared.core.inject.factory.ChunkCoordinatorFactory; | ||||
| import com.plotsquared.core.queue.BasicQueueCoordinator; | ||||
| import com.plotsquared.core.queue.ChunkCoordinator; | ||||
| import com.plotsquared.core.queue.LocalChunk; | ||||
| import com.plotsquared.core.util.ChunkUtil; | ||||
| import com.sk89q.jnbt.CompoundTag; | ||||
| import com.sk89q.worldedit.WorldEditException; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.extent.clipboard.BlockArrayClipboard; | ||||
| import com.sk89q.worldedit.extent.clipboard.Clipboard; | ||||
| import com.sk89q.worldedit.math.BlockVector2; | ||||
| import com.sk89q.worldedit.math.BlockVector3; | ||||
| import com.sk89q.worldedit.regions.CuboidRegion; | ||||
| import com.sk89q.worldedit.regions.Region; | ||||
| import com.sk89q.worldedit.util.SideEffect; | ||||
| import com.sk89q.worldedit.util.SideEffectSet; | ||||
| import com.sk89q.worldedit.world.World; | ||||
| import com.sk89q.worldedit.world.biome.BiomeType; | ||||
| import com.sk89q.worldedit.world.block.BaseBlock; | ||||
| import com.sk89q.worldedit.world.block.BlockState; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.block.Container; | ||||
| import org.bukkit.block.data.BlockData; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collection; | ||||
| import java.util.function.Consumer; | ||||
|  | ||||
| public class BukkitQueueCoordinator extends BasicQueueCoordinator { | ||||
|  | ||||
|     private final SideEffectSet noSideEffectSet; | ||||
|     private final SideEffectSet lightingSideEffectSet; | ||||
|     private final SideEffectSet edgeSideEffectSet; | ||||
|     private final SideEffectSet edgeLightingSideEffectSet; | ||||
|     private org.bukkit.World bukkitWorld; | ||||
|     @Inject | ||||
|     private ChunkCoordinatorBuilderFactory chunkCoordinatorBuilderFactory; | ||||
|     @Inject | ||||
|     private ChunkCoordinatorFactory chunkCoordinatorFactory; | ||||
|     private ChunkCoordinator chunkCoordinator; | ||||
|  | ||||
|     @Inject | ||||
|     public BukkitQueueCoordinator(@NonNull World world) { | ||||
|         super(world); | ||||
|         noSideEffectSet = SideEffectSet.none().with(SideEffect.LIGHTING, SideEffect.State.OFF).with( | ||||
|                 SideEffect.NEIGHBORS, | ||||
|                 SideEffect.State.OFF | ||||
|         ); | ||||
|         lightingSideEffectSet = SideEffectSet.none().with(SideEffect.NEIGHBORS, SideEffect.State.OFF); | ||||
|         edgeSideEffectSet = noSideEffectSet.with(SideEffect.UPDATE, SideEffect.State.ON).with( | ||||
|                 SideEffect.NEIGHBORS, | ||||
|                 SideEffect.State.ON | ||||
|         ); | ||||
|         edgeLightingSideEffectSet = noSideEffectSet.with(SideEffect.UPDATE, SideEffect.State.ON).with( | ||||
|                 SideEffect.NEIGHBORS, | ||||
|                 SideEffect.State.ON | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public BlockState getBlock(int x, int y, int z) { | ||||
|         Block block = getBukkitWorld().getBlockAt(x, y, z); | ||||
|         return BukkitBlockUtil.get(block); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void start() { | ||||
|         chunkCoordinator.start(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void cancel() { | ||||
|         chunkCoordinator.cancel(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean enqueue() { | ||||
|         final Clipboard regenClipboard; | ||||
|         if (isRegen()) { | ||||
|             BlockVector3 start = BlockVector3.at(getRegenStart()[0] << 4, 0, getRegenStart()[1] << 4); | ||||
|             BlockVector3 end = BlockVector3.at((getRegenEnd()[0] << 4) + 15, 255, (getRegenEnd()[1] << 4) + 15); | ||||
|             Region region = new CuboidRegion(start, end); | ||||
|             regenClipboard = new BlockArrayClipboard(region); | ||||
|             regenClipboard.setOrigin(start); | ||||
|             getWorld().regenerate(region, regenClipboard); | ||||
|         } else if (getRegenRegion() != null) { | ||||
|             regenClipboard = new BlockArrayClipboard(getRegenRegion()); | ||||
|             regenClipboard.setOrigin(getRegenRegion().getMinimumPoint()); | ||||
|             getWorld().regenerate(getRegenRegion(), regenClipboard); | ||||
|         } else { | ||||
|             regenClipboard = null; | ||||
|         } | ||||
|         Consumer<BlockVector2> consumer = getChunkConsumer(); | ||||
|         if (consumer == null) { | ||||
|             consumer = blockVector2 -> { | ||||
|                 LocalChunk localChunk = getBlockChunks().get(blockVector2); | ||||
|                 boolean isRegenChunk = | ||||
|                         regenClipboard != null && blockVector2.getBlockX() > getRegenStart()[0] && blockVector2.getBlockZ() > getRegenStart()[1] | ||||
|                                 && blockVector2.getBlockX() < getRegenEnd()[0] && blockVector2.getBlockZ() < getRegenEnd()[1]; | ||||
|                 int sx = blockVector2.getX() << 4; | ||||
|                 int sz = blockVector2.getZ() << 4; | ||||
|                 if (isRegenChunk) { | ||||
|                     for (int layer = 0; layer < 16; layer++) { | ||||
|                         for (int y = 0; y < 16; y++) { | ||||
|                             for (int x = 0; x < 16; x++) { | ||||
|                                 for (int z = 0; z < 16; z++) { | ||||
|                                     x += sx; | ||||
|                                     y += layer << 4; | ||||
|                                     z += sz; | ||||
|                                     BaseBlock block = regenClipboard.getFullBlock(BlockVector3.at(x, y, z)); | ||||
|                                     if (block != null) { | ||||
|                                         boolean edge = Settings.QUEUE.UPDATE_EDGES && isEdgeRegen(x & 15, z & 15, blockVector2); | ||||
|                                         setWorldBlock(x, y, z, block, blockVector2, edge); | ||||
|                                     } | ||||
|                                 } | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|                 // Allow regen and then blocks to be placed (plot schematic etc) | ||||
|                 if (localChunk == null) { | ||||
|                     return; | ||||
|                 } | ||||
|                 for (int layer = 0; layer < localChunk.getBaseblocks().length; layer++) { | ||||
|                     BaseBlock[] blocksLayer = localChunk.getBaseblocks()[layer]; | ||||
|                     if (blocksLayer == null) { | ||||
|                         continue; | ||||
|                     } | ||||
|                     for (int j = 0; j < blocksLayer.length; j++) { | ||||
|                         if (blocksLayer[j] == null) { | ||||
|                             continue; | ||||
|                         } | ||||
|                         BaseBlock block = blocksLayer[j]; | ||||
|  | ||||
|                         if (block != null) { | ||||
|                             int lx = ChunkUtil.getX(j); | ||||
|                             int lz = ChunkUtil.getZ(j); | ||||
|                             int x = sx + lx; | ||||
|                             int y = ChunkUtil.getY(layer, j); | ||||
|                             int z = sz + lz; | ||||
|                             boolean edge = Settings.QUEUE.UPDATE_EDGES && isEdge(y >> 4, lx, y & 15, lz, blockVector2, | ||||
|                                     localChunk | ||||
|                             ); | ||||
|                             setWorldBlock(x, y, z, block, blockVector2, edge); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|                 for (int layer = 0; layer < localChunk.getBaseblocks().length; layer++) { | ||||
|                     BiomeType[] biomesLayer = localChunk.getBiomes()[layer]; | ||||
|                     if (biomesLayer == null) { | ||||
|                         continue; | ||||
|                     } | ||||
|                     for (int j = 0; j < biomesLayer.length; j++) { | ||||
|                         if (biomesLayer[j] == null) { | ||||
|                             continue; | ||||
|                         } | ||||
|                         BiomeType biome = biomesLayer[j]; | ||||
|                         if (biome != null) { | ||||
|                             int x = sx + ChunkUtil.getX(j); | ||||
|                             int y = ChunkUtil.getY(layer, j); | ||||
|                             int z = sz + ChunkUtil.getZ(j); | ||||
|                             getWorld().setBiome(BlockVector3.at(x, y, z), biome); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|                 if (localChunk.getTiles().size() > 0) { | ||||
|                     localChunk.getTiles().forEach((blockVector3, tag) -> { | ||||
|                         try { | ||||
|                             BaseBlock block = getWorld().getBlock(blockVector3).toBaseBlock(tag); | ||||
|                             getWorld().setBlock(blockVector3, block, noSideEffectSet); | ||||
|                         } catch (WorldEditException ignored) { | ||||
|                             StateWrapper sw = new StateWrapper(tag); | ||||
|                             sw.restoreTag(getWorld().getName(), blockVector3.getX(), blockVector3.getY(), blockVector3.getZ()); | ||||
|                         } | ||||
|                     }); | ||||
|                 } | ||||
|                 if (localChunk.getEntities().size() > 0) { | ||||
|                     localChunk.getEntities().forEach((location, entity) -> getWorld().createEntity(location, entity)); | ||||
|                 } | ||||
|             }; | ||||
|         } | ||||
|         Collection<BlockVector2> read = new ArrayList<>(); | ||||
|         if (getReadChunks().size() > 0) { | ||||
|             read.addAll(getReadChunks()); | ||||
|         } | ||||
|         chunkCoordinator = | ||||
|                 chunkCoordinatorBuilderFactory | ||||
|                         .create(chunkCoordinatorFactory) | ||||
|                         .inWorld(getWorld()) | ||||
|                         .withChunks(getBlockChunks().keySet()) | ||||
|                         .withChunks(read) | ||||
|                         .withInitialBatchSize(3) | ||||
|                         .withMaxIterationTime(40) | ||||
|                         .withThrowableConsumer(Throwable::printStackTrace) | ||||
|                         .withFinalAction(getCompleteTask()) | ||||
|                         .withConsumer(consumer) | ||||
|                         .unloadAfter(isUnloadAfter()) | ||||
|                         .withProgressSubscribers(getProgressSubscribers()) | ||||
|                         .build(); | ||||
|         return super.enqueue(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set a block to the world. First tries WNA but defaults to normal block setting methods if that fails | ||||
|      */ | ||||
|     @SuppressWarnings("unused") | ||||
|     private void setWorldBlock(int x, int y, int z, @NonNull BaseBlock block, @NonNull BlockVector2 blockVector2, boolean edge) { | ||||
|         try { | ||||
|             BlockVector3 loc = BlockVector3.at(x, y, z); | ||||
|             boolean lighting = false; | ||||
|             switch (getLightingMode()) { | ||||
|                 case NONE: | ||||
|                     break; | ||||
|                 case PLACEMENT: | ||||
|                     lighting = block.getBlockType().getMaterial().getLightValue() > 0; | ||||
|                     break; | ||||
|                 case REPLACEMENT: | ||||
|                     lighting = block.getBlockType().getMaterial().getLightValue() > 0 | ||||
|                             || getWorld().getBlock(loc).getBlockType().getMaterial().getLightValue() > 0; | ||||
|                     break; | ||||
|                 default: | ||||
|                     // Can only be "all" | ||||
|                     lighting = true; | ||||
|             } | ||||
|             SideEffectSet sideEffectSet; | ||||
|             if (lighting) { | ||||
|                 sideEffectSet = edge ? edgeLightingSideEffectSet : lightingSideEffectSet; | ||||
|             } else { | ||||
|                 sideEffectSet = edge ? edgeSideEffectSet : noSideEffectSet; | ||||
|             } | ||||
|             getWorld().setBlock(loc, block, sideEffectSet); | ||||
|         } catch (WorldEditException ignored) { | ||||
|             // Fallback to not so nice method | ||||
|             BlockData blockData = BukkitAdapter.adapt(block); | ||||
|             Block existing = getBukkitWorld().getBlockAt(x, y, z); | ||||
|             final BlockState existingBaseBlock = BukkitAdapter.adapt(existing.getBlockData()); | ||||
|             if (BukkitBlockUtil.get(existing).equals(existingBaseBlock) && existing.getBlockData().matches(blockData)) { | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             if (existing.getState() instanceof Container) { | ||||
|                 ((Container) existing.getState()).getInventory().clear(); | ||||
|             } | ||||
|  | ||||
|             existing.setType(BukkitAdapter.adapt(block.getBlockType()), false); | ||||
|             existing.setBlockData(blockData, false); | ||||
|             if (block.hasNbtData()) { | ||||
|                 CompoundTag tag = block.getNbtData(); | ||||
|                 StateWrapper sw = new StateWrapper(tag); | ||||
|  | ||||
|                 sw.restoreTag(getWorld().getName(), existing.getX(), existing.getY(), existing.getZ()); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private org.bukkit.World getBukkitWorld() { | ||||
|         if (bukkitWorld == null) { | ||||
|             bukkitWorld = Bukkit.getWorld(getWorld().getName()); | ||||
|         } | ||||
|         return bukkitWorld; | ||||
|     } | ||||
|  | ||||
|     private boolean isEdge(int layer, int x, int y, int z, BlockVector2 blockVector2, LocalChunk localChunk) { | ||||
|         if (layer == 0 || layer == localChunk.getBaseblocks().length - 1) { | ||||
|             return false; | ||||
|         } | ||||
|         if (x == 0) { | ||||
|             LocalChunk localChunkX = getBlockChunks().get(blockVector2.withX(blockVector2.getX() - 1)); | ||||
|             if (localChunkX == null || localChunkX.getBaseblocks()[layer] == null || | ||||
|                     localChunkX.getBaseblocks()[layer][ChunkUtil.getJ(15, y, z)] != null) { | ||||
|                 return true; | ||||
|             } | ||||
|         } else if (x == 15) { | ||||
|             LocalChunk localChunkX = getBlockChunks().get(blockVector2.withX(blockVector2.getX() + 1)); | ||||
|             if (localChunkX == null || localChunkX.getBaseblocks()[layer] == null || | ||||
|                     localChunkX.getBaseblocks()[layer][ChunkUtil.getJ(0, y, z)] != null) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         if (z == 0) { | ||||
|             LocalChunk localChunkZ = getBlockChunks().get(blockVector2.withZ(blockVector2.getZ() - 1)); | ||||
|             if (localChunkZ == null || localChunkZ.getBaseblocks()[layer] == null || | ||||
|                     localChunkZ.getBaseblocks()[layer][ChunkUtil.getJ(x, y, 15)] != null) { | ||||
|                 return true; | ||||
|             } | ||||
|         } else if (z == 15) { | ||||
|             LocalChunk localChunkZ = getBlockChunks().get(blockVector2.withZ(blockVector2.getZ() + 1)); | ||||
|             if (localChunkZ == null || localChunkZ.getBaseblocks()[layer] == null || | ||||
|                     localChunkZ.getBaseblocks()[layer][ChunkUtil.getJ(x, y, 0)] != null) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         if (y == 0) { | ||||
|             if (localChunk.getBaseblocks()[layer - 1] == null || | ||||
|                     localChunk.getBaseblocks()[layer][ChunkUtil.getJ(x, 15, z)] != null) { | ||||
|                 return true; | ||||
|             } | ||||
|         } else if (y == 15) { | ||||
|             if (localChunk.getBaseblocks()[layer + 1] == null || | ||||
|                     localChunk.getBaseblocks()[layer][ChunkUtil.getJ(x, 0, z)] != null) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         BaseBlock[] baseBlocks = localChunk.getBaseblocks()[layer]; | ||||
|         if (x > 0 && baseBlocks[ChunkUtil.getJ(x - 1, y, z)] == null) { | ||||
|             return true; | ||||
|         } | ||||
|         if (x < 15 && baseBlocks[ChunkUtil.getJ(x + 1, y, z)] == null) { | ||||
|             return true; | ||||
|         } | ||||
|         if (y > 0 && baseBlocks[ChunkUtil.getJ(x, y - 1, z)] == null) { | ||||
|             return true; | ||||
|         } | ||||
|         if (y < 15 && baseBlocks[ChunkUtil.getJ(x, y + 1, z)] == null) { | ||||
|             return true; | ||||
|         } | ||||
|         if (z > 0 && baseBlocks[ChunkUtil.getJ(x, y, z - 1)] == null) { | ||||
|             return true; | ||||
|         } | ||||
|         return z < 15 && baseBlocks[ChunkUtil.getJ(x, y, z + 1)] == null; | ||||
|     } | ||||
|  | ||||
|     private boolean isEdgeRegen(int x, int z, BlockVector2 blockVector2) { | ||||
|         if (x == 0) { | ||||
|             LocalChunk localChunkX = getBlockChunks().get(blockVector2.withX(blockVector2.getX() - 1)); | ||||
|             if (localChunkX == null) { | ||||
|                 return true; | ||||
|             } | ||||
|         } else if (x == 15) { | ||||
|             LocalChunk localChunkX = getBlockChunks().get(blockVector2.withX(blockVector2.getX() + 1)); | ||||
|             if (localChunkX == null) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         if (z == 0) { | ||||
|             return getBlockChunks().get(blockVector2.withZ(blockVector2.getZ() - 1)) == null; | ||||
|         } else if (z == 15) { | ||||
|             return getBlockChunks().get(blockVector2.withZ(blockVector2.getZ() + 1)) == null; | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,326 +0,0 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  * | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.queue; | ||||
|  | ||||
| import com.google.common.base.Preconditions; | ||||
| import com.plotsquared.bukkit.BukkitMain; | ||||
| import com.sk89q.worldedit.math.BlockVector2; | ||||
| import io.papermc.lib.PaperLib; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.plugin.Plugin; | ||||
| import org.bukkit.plugin.java.JavaPlugin; | ||||
| import org.bukkit.scheduler.BukkitRunnable; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
|  | ||||
| import java.util.Collection; | ||||
| import java.util.LinkedList; | ||||
| import java.util.List; | ||||
| import java.util.Queue; | ||||
| import java.util.concurrent.LinkedBlockingQueue; | ||||
| import java.util.concurrent.atomic.AtomicInteger; | ||||
| import java.util.function.Consumer; | ||||
|  | ||||
| /** | ||||
|  * Utility that allows for the loading and coordination of chunk actions | ||||
|  * <p> | ||||
|  * The coordinator takes in collection of chunk coordinates, loads them | ||||
|  * and allows the caller to specify a sink for the loaded chunks. The | ||||
|  * coordinator will prevent the chunks from being unloaded until the sink | ||||
|  * has fully consumed the chunk | ||||
|  * <p> | ||||
|  * Usage: | ||||
|  * <pre>{@code | ||||
|  * final ChunkCoordinator chunkCoordinator = ChunkCoordinator.builder() | ||||
|  *     .inWorld(Objects.requireNonNull(Bukkit.getWorld("world"))).withChunk(BlockVector2.at(0, 0)) | ||||
|  *     .withConsumer(chunk -> System.out.printf("Got chunk %d;%d", chunk.getX(), chunk.getZ())) | ||||
|  *     .withFinalAction(() -> System.out.println("All chunks have been loaded")) | ||||
|  *     .withThrowableConsumer(throwable -> System.err.println("Something went wrong... =(")) | ||||
|  *     .withMaxIterationTime(25L) | ||||
|  *     .build(); | ||||
|  * chunkCoordinator.subscribeToProgress((coordinator, progress) -> | ||||
|  *     System.out.printf("Progress: %.1f", progress * 100.0f)); | ||||
|  * chunkCoordinator.start(); | ||||
|  * }</pre> | ||||
|  * | ||||
|  * @author Alexander Söderberg | ||||
|  * @see #builder() To create a new coordinator instance | ||||
|  */ | ||||
| public final class ChunkCoordinator extends BukkitRunnable { | ||||
|  | ||||
|     private final List<ProgressSubscriber> progressSubscribers = new LinkedList<>(); | ||||
|  | ||||
|     private final Queue<BlockVector2> requestedChunks; | ||||
|     private final Queue<Chunk> availableChunks; | ||||
|     private final long maxIterationTime; | ||||
|     private final Plugin plugin; | ||||
|     private final Consumer<Chunk> chunkConsumer; | ||||
|     private final World world; | ||||
|     private final Runnable whenDone; | ||||
|     private final Consumer<Throwable> throwableConsumer; | ||||
|     private final int totalSize; | ||||
|  | ||||
|     private AtomicInteger expectedSize; | ||||
|     private int batchSize; | ||||
|  | ||||
|     private ChunkCoordinator(final long maxIterationTime, final int initialBatchSize, | ||||
|         @NotNull final Consumer<Chunk> chunkConsumer, @NotNull final World world, | ||||
|         @NotNull final Collection<BlockVector2> requestedChunks, @NotNull final Runnable whenDone, | ||||
|         @NotNull final Consumer<Throwable> throwableConsumer) { | ||||
|         this.requestedChunks = new LinkedBlockingQueue<>(requestedChunks); | ||||
|         this.availableChunks = new LinkedBlockingQueue<>(); | ||||
|         this.totalSize = requestedChunks.size(); | ||||
|         this.expectedSize = new AtomicInteger(this.totalSize); | ||||
|         this.world = world; | ||||
|         this.batchSize = initialBatchSize; | ||||
|         this.chunkConsumer = chunkConsumer; | ||||
|         this.maxIterationTime = maxIterationTime; | ||||
|         this.whenDone = whenDone; | ||||
|         this.throwableConsumer = throwableConsumer; | ||||
|         this.plugin = JavaPlugin.getPlugin(BukkitMain.class); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Create a new {@link ChunkCoordinator} instance | ||||
|      * | ||||
|      * @return Coordinator builder instance | ||||
|      */ | ||||
|     @NotNull public static ChunkCoordinatorBuilder builder() { | ||||
|         return new ChunkCoordinatorBuilder(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Start the coordinator instance | ||||
|      */ | ||||
|     public void start() { | ||||
|         // Request initial batch | ||||
|         this.requestBatch(); | ||||
|         // Wait until next tick to give the chunks a chance to be loaded | ||||
|         this.runTaskTimer(this.plugin, 1L, 1L); | ||||
|     } | ||||
|  | ||||
|     @Override public void run() { | ||||
|         Chunk chunk = this.availableChunks.poll(); | ||||
|         if (chunk == null) { | ||||
|             return; | ||||
|         } | ||||
|         long iterationTime; | ||||
|         int processedChunks = 0; | ||||
|         do { | ||||
|             final long start = System.currentTimeMillis(); | ||||
|             try { | ||||
|                 this.chunkConsumer.accept(chunk); | ||||
|             } catch (final Throwable throwable) { | ||||
|                 this.throwableConsumer.accept(throwable); | ||||
|             } | ||||
|             this.freeChunk(chunk); | ||||
|             processedChunks++; | ||||
|             final long end = System.currentTimeMillis(); | ||||
|             // Update iteration time | ||||
|             iterationTime = end - start; | ||||
|         } while (2 * iterationTime /* last chunk + next chunk */ < this.maxIterationTime | ||||
|             && (chunk = availableChunks.poll()) != null); | ||||
|         if (processedChunks < this.batchSize) { | ||||
|             // Adjust batch size based on the amount of processed chunks per tick | ||||
|             this.batchSize = processedChunks; | ||||
|         } | ||||
|  | ||||
|         final int expected = this.expectedSize.addAndGet(-processedChunks); | ||||
|  | ||||
|         final float progress = ((float) totalSize - (float) expected) / (float) totalSize; | ||||
|         for (final ProgressSubscriber subscriber : this.progressSubscribers) { | ||||
|             subscriber.notifyProgress(this, progress); | ||||
|         } | ||||
|  | ||||
|         if (expected <= 0) { | ||||
|             try { | ||||
|                 this.whenDone.run(); | ||||
|             } catch (final Throwable throwable) { | ||||
|                 this.throwableConsumer.accept(throwable); | ||||
|             } | ||||
|             this.cancel(); | ||||
|         } else { | ||||
|             if (this.availableChunks.size() < processedChunks) { | ||||
|                 this.requestBatch(); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void requestBatch() { | ||||
|         BlockVector2 chunk; | ||||
|         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 | ||||
|             PaperLib.getChunkAtAsync(this.world, chunk.getX(), chunk.getZ(), true, true) | ||||
|                 .whenComplete((chunkObject, throwable) -> { | ||||
|                     if (throwable != null) { | ||||
|                         throwable.printStackTrace(); | ||||
|                         // We want one less because this couldn't be processed | ||||
|                         this.expectedSize.decrementAndGet(); | ||||
|                     } else { | ||||
|                         this.processChunk(chunkObject); | ||||
|                     } | ||||
|                 }); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void processChunk(@NotNull final Chunk chunk) { | ||||
|         if (!chunk.isLoaded()) { | ||||
|             throw new IllegalArgumentException( | ||||
|                 String.format("Chunk %d;%d is is not loaded", chunk.getX(), chunk.getZ())); | ||||
|         } | ||||
|         chunk.addPluginChunkTicket(this.plugin); | ||||
|         this.availableChunks.add(chunk); | ||||
|     } | ||||
|  | ||||
|     private void freeChunk(@NotNull final Chunk chunk) { | ||||
|         if (!chunk.isLoaded()) { | ||||
|             throw new IllegalArgumentException( | ||||
|                 String.format("Chunk %d;%d is is not loaded", chunk.getX(), chunk.getZ())); | ||||
|         } | ||||
|         chunk.removePluginChunkTicket(this.plugin); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get the amount of remaining chunks (at the time of the method call) | ||||
|      * | ||||
|      * @return Snapshot view of remaining chunk count | ||||
|      */ | ||||
|     public int getRemainingChunks() { | ||||
|         return this.expectedSize.get(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get the amount of requested chunks | ||||
|      * | ||||
|      * @return Requested chunk count | ||||
|      */ | ||||
|     public int getTotalChunks() { | ||||
|         return this.totalSize; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Subscribe to coordinator progress updates | ||||
|      * | ||||
|      * @param subscriber Subscriber | ||||
|      */ | ||||
|     public void subscribeToProgress(@NotNull final ChunkCoordinator.ProgressSubscriber subscriber) { | ||||
|         this.progressSubscribers.add(subscriber); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     @FunctionalInterface | ||||
|     public interface ProgressSubscriber { | ||||
|  | ||||
|         /** | ||||
|          * Notify about a progress update in the coordinator | ||||
|          * | ||||
|          * @param coordinator Coordinator instance that triggered the notification | ||||
|          * @param progress    Progress in the range [0, 1] | ||||
|          */ | ||||
|         void notifyProgress(@NotNull final ChunkCoordinator coordinator, final float progress); | ||||
|  | ||||
|     } | ||||
|  | ||||
|  | ||||
|     public static final class ChunkCoordinatorBuilder { | ||||
|  | ||||
|         private final List<BlockVector2> requestedChunks = new LinkedList<>(); | ||||
|         private Consumer<Throwable> throwableConsumer = Throwable::printStackTrace; | ||||
|         private World world; | ||||
|         private Consumer<Chunk> chunkConsumer; | ||||
|         private Runnable whenDone = () -> { | ||||
|         }; | ||||
|         private long maxIterationTime = 60; // A little over 1 tick; | ||||
|         private int initialBatchSize = 4; | ||||
|  | ||||
|         private ChunkCoordinatorBuilder() { | ||||
|         } | ||||
|  | ||||
|         @NotNull public ChunkCoordinatorBuilder inWorld(@NotNull final World world) { | ||||
|             this.world = Preconditions.checkNotNull(world, "World may not be null"); | ||||
|             return this; | ||||
|         } | ||||
|  | ||||
|         @NotNull | ||||
|         public ChunkCoordinatorBuilder withChunk(@NotNull final BlockVector2 chunkLocation) { | ||||
|             this.requestedChunks | ||||
|                 .add(Preconditions.checkNotNull(chunkLocation, "Chunk location may not be null")); | ||||
|             return this; | ||||
|         } | ||||
|  | ||||
|         @NotNull public ChunkCoordinatorBuilder withChunks( | ||||
|             @NotNull final Collection<BlockVector2> chunkLocations) { | ||||
|             chunkLocations.forEach(this::withChunk); | ||||
|             return this; | ||||
|         } | ||||
|  | ||||
|         @NotNull | ||||
|         public ChunkCoordinatorBuilder withConsumer(@NotNull final Consumer<Chunk> chunkConsumer) { | ||||
|             this.chunkConsumer = | ||||
|                 Preconditions.checkNotNull(chunkConsumer, "Chunk consumer may not be null"); | ||||
|             return this; | ||||
|         } | ||||
|  | ||||
|         @NotNull public ChunkCoordinatorBuilder withFinalAction(@NotNull final Runnable whenDone) { | ||||
|             this.whenDone = Preconditions.checkNotNull(whenDone, "Final action may not be null"); | ||||
|             return this; | ||||
|         } | ||||
|  | ||||
|         @NotNull public ChunkCoordinatorBuilder withMaxIterationTime(final long maxIterationTime) { | ||||
|             Preconditions | ||||
|                 .checkArgument(maxIterationTime > 0, "Max iteration time must be positive"); | ||||
|             this.maxIterationTime = maxIterationTime; | ||||
|             return this; | ||||
|         } | ||||
|  | ||||
|         @NotNull public ChunkCoordinatorBuilder withInitialBatchSize(final int initialBatchSize) { | ||||
|             Preconditions | ||||
|                 .checkArgument(initialBatchSize > 0, "Initial batch size must be positive"); | ||||
|             this.initialBatchSize = initialBatchSize; | ||||
|             return this; | ||||
|         } | ||||
|  | ||||
|         @NotNull public ChunkCoordinatorBuilder withThrowableConsumer( | ||||
|             @NotNull final Consumer<Throwable> throwableConsumer) { | ||||
|             this.throwableConsumer = | ||||
|                 Preconditions.checkNotNull(throwableConsumer, "Throwable consumer may not be null"); | ||||
|             return this; | ||||
|         } | ||||
|  | ||||
|         @NotNull public ChunkCoordinator build() { | ||||
|             Preconditions.checkNotNull(this.world, "No world was supplied"); | ||||
|             Preconditions.checkNotNull(this.chunkConsumer, "No chunk consumer was supplied"); | ||||
|             Preconditions.checkNotNull(this.whenDone, "No final action was supplied"); | ||||
|             Preconditions | ||||
|                 .checkNotNull(this.throwableConsumer, "No throwable consumer was supplied"); | ||||
|             return new ChunkCoordinator(this.maxIterationTime, this.initialBatchSize, | ||||
|                 this.chunkConsumer, this.world, this.requestedChunks, this.whenDone, | ||||
|                 this.throwableConsumer); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.queue; | ||||
|  | ||||
| @@ -30,8 +30,8 @@ import com.plotsquared.bukkit.util.BukkitBlockUtil; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.location.ChunkWrapper; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.queue.ScopedLocalBlockQueue; | ||||
| import com.plotsquared.core.util.MainUtil; | ||||
| import com.plotsquared.core.queue.ScopedQueueCoordinator; | ||||
| import com.plotsquared.core.util.ChunkUtil; | ||||
| import com.plotsquared.core.util.PatternUtil; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.function.pattern.Pattern; | ||||
| @@ -39,18 +39,18 @@ import com.sk89q.worldedit.world.biome.BiomeType; | ||||
| import com.sk89q.worldedit.world.block.BaseBlock; | ||||
| import com.sk89q.worldedit.world.block.BlockState; | ||||
| import com.sk89q.worldedit.world.block.BlockTypes; | ||||
| import lombok.Getter; | ||||
| import lombok.Setter; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.Biome; | ||||
| import org.bukkit.generator.ChunkGenerator.BiomeGrid; | ||||
| import org.bukkit.generator.ChunkGenerator.ChunkData; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| import java.util.Arrays; | ||||
|  | ||||
| public class GenChunk extends ScopedLocalBlockQueue { | ||||
| public class GenChunk extends ScopedQueueCoordinator { | ||||
|  | ||||
|     public final Biome[] biomes; | ||||
|     public BlockState[][] result; | ||||
| @@ -59,14 +59,27 @@ public class GenChunk extends ScopedLocalBlockQueue { | ||||
|     public String world; | ||||
|     public int chunkX; | ||||
|     public int chunkZ; | ||||
|     @Getter @Setter private ChunkData chunkData = null; | ||||
|     private ChunkData chunkData = null; | ||||
|  | ||||
|     public GenChunk() { | ||||
|         super(null, new Location(null, 0, 0, 0), new Location(null, 15, 255, 15)); | ||||
|         super(null, Location.at("", 0, 0, 0), Location.at("", 15, 255, 15)); | ||||
|         this.biomes = Biome.values(); | ||||
|     } | ||||
|  | ||||
|     public Chunk getChunk() { | ||||
|     public @Nullable ChunkData getChunkData() { | ||||
|         return this.chunkData; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Set the internal Bukkit chunk data | ||||
|      * | ||||
|      * @param chunkData Bukkit ChunkData | ||||
|      */ | ||||
|     public void setChunkData(@NonNull ChunkData chunkData) { | ||||
|         this.chunkData = chunkData; | ||||
|     } | ||||
|  | ||||
|     public @NonNull Chunk getChunk() { | ||||
|         if (chunk == null) { | ||||
|             World worldObj = BukkitUtil.getWorld(world); | ||||
|             if (worldObj != null) { | ||||
| @@ -76,32 +89,46 @@ public class GenChunk extends ScopedLocalBlockQueue { | ||||
|         return chunk; | ||||
|     } | ||||
|  | ||||
|     public void setChunk(Chunk chunk) { | ||||
|     /** | ||||
|      * Set the chunk being represented | ||||
|      * | ||||
|      * @param chunk Bukkit Chunk | ||||
|      */ | ||||
|     public void setChunk(@NonNull Chunk chunk) { | ||||
|         this.chunk = chunk; | ||||
|     } | ||||
|  | ||||
|     public void setChunk(ChunkWrapper wrap) { | ||||
|  | ||||
|     /** | ||||
|      * Set the world and XZ of the chunk being represented via {@link ChunkWrapper} | ||||
|      * | ||||
|      * @param wrap P2 ChunkWrapper | ||||
|      */ | ||||
|     public void setChunk(@NonNull ChunkWrapper wrap) { | ||||
|         chunk = null; | ||||
|         world = wrap.world; | ||||
|         chunkX = wrap.x; | ||||
|         chunkZ = wrap.z; | ||||
|     } | ||||
|  | ||||
|     @Override public void fillBiome(BiomeType biomeType) { | ||||
|     @Override | ||||
|     public void fillBiome(@NonNull BiomeType biomeType) { | ||||
|         if (biomeGrid == null) { | ||||
|             return; | ||||
|         } | ||||
|         Biome biome = BukkitAdapter.adapt(biomeType); | ||||
|         for (int x = 0; x < 16; x++) { | ||||
|             for (int z = 0; z < 16; z++) { | ||||
|                 this.biomeGrid.setBiome(x, z, biome); | ||||
|         for (int y = 0; y < 256; y++) { | ||||
|             for (int x = 0; x < 16; x++) { | ||||
|                 for (int z = 0; z < 16; z++) { | ||||
|                     this.biomeGrid.setBiome(x, y, z, biome); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public void setCuboid(Location pos1, Location pos2, BlockState block) { | ||||
|         if (result != null && pos1.getX() == 0 && pos1.getZ() == 0 && pos2.getX() == 15 | ||||
|             && pos2.getZ() == 15) { | ||||
|     @Override | ||||
|     public void setCuboid(@NonNull Location pos1, @NonNull Location pos2, @NonNull BlockState block) { | ||||
|         if (result != null && pos1.getX() == 0 && pos1.getZ() == 0 && pos2.getX() == 15 && pos2.getZ() == 15) { | ||||
|             for (int y = pos1.getY(); y <= pos2.getY(); y++) { | ||||
|                 int layer = y >> 4; | ||||
|                 BlockState[] data = result[layer]; | ||||
| @@ -119,28 +146,47 @@ public class GenChunk extends ScopedLocalBlockQueue { | ||||
|         int maxX = Math.max(pos1.getX(), pos2.getX()); | ||||
|         int maxY = Math.max(pos1.getY(), pos2.getY()); | ||||
|         int maxZ = Math.max(pos1.getZ(), pos2.getZ()); | ||||
|         chunkData | ||||
|             .setRegion(minX, minY, minZ, maxX + 1, maxY + 1, maxZ + 1, BukkitAdapter.adapt(block)); | ||||
|         chunkData.setRegion(minX, minY, minZ, maxX + 1, maxY + 1, maxZ + 1, BukkitAdapter.adapt(block)); | ||||
|     } | ||||
|  | ||||
|     @Override public boolean setBiome(int x, int z, BiomeType biomeType) { | ||||
|     @Override | ||||
|     public boolean setBiome(int x, int z, @NonNull BiomeType biomeType) { | ||||
|         return setBiome(x, z, BukkitAdapter.adapt(biomeType)); | ||||
|     } | ||||
|  | ||||
|     public boolean setBiome(int x, int z, Biome biome) { | ||||
|     /** | ||||
|      * Set the in the whole column of XZ | ||||
|      * | ||||
|      * @param x     Relative x location within the chunk (0 - 15) | ||||
|      * @param z     Relative z location within the chunk (0 - 15) | ||||
|      * @param biome Bukkit biome to set | ||||
|      * @return if successful | ||||
|      */ | ||||
|     public boolean setBiome(int x, int z, @NonNull Biome biome) { | ||||
|         if (this.biomeGrid != null) { | ||||
|             this.biomeGrid.setBiome(x, z, biome); | ||||
|             for (int y = 0; y < 256; y++) { | ||||
|                 this.setBiome(x, y, z, biome); | ||||
|             } | ||||
|             return true; | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     @Override public boolean setBlock(int x, int y, int z, @NotNull Pattern pattern) { | ||||
|         return setBlock(x, y, z, PatternUtil | ||||
|             .apply(Preconditions.checkNotNull(pattern, "Pattern may not be null"), x, y, z)); | ||||
|     public boolean setBiome(int x, int y, int z, @NonNull Biome biome) { | ||||
|         if (this.biomeGrid != null) { | ||||
|             this.biomeGrid.setBiome(x, y, z, biome); | ||||
|             return true; | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     @Override public boolean setBlock(int x, int y, int z, BlockState id) { | ||||
|     @Override | ||||
|     public boolean setBlock(int x, int y, int z, @NonNull Pattern pattern) { | ||||
|         return setBlock(x, y, z, PatternUtil.apply(Preconditions.checkNotNull(pattern, "Pattern may not be null"), x, y, z)); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean setBlock(int x, int y, int z, @NonNull BlockState id) { | ||||
|         if (this.result == null) { | ||||
|             this.chunkData.setBlock(x, y, z, BukkitAdapter.adapt(id)); | ||||
|             return true; | ||||
| @@ -150,17 +196,18 @@ public class GenChunk extends ScopedLocalBlockQueue { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     private void storeCache(final int x, final int y, final int z, final BlockState id) { | ||||
|         int i = MainUtil.CACHE_I[y][x][z]; | ||||
|     private void storeCache(final int x, final int y, final int z, final @NonNull BlockState id) { | ||||
|         int i = y >> 4; | ||||
|         BlockState[] v = this.result[i]; | ||||
|         if (v == null) { | ||||
|             this.result[i] = v = new BlockState[4096]; | ||||
|         } | ||||
|         int j = MainUtil.CACHE_J[y][x][z]; | ||||
|         int j = ChunkUtil.getJ(x, y, z); | ||||
|         v[j] = id; | ||||
|     } | ||||
|  | ||||
|     @Override public boolean setBlock(int x, int y, int z, BaseBlock id) { | ||||
|     @Override | ||||
|     public boolean setBlock(int x, int y, int z, @NonNull BaseBlock id) { | ||||
|         if (this.result == null) { | ||||
|             this.chunkData.setBlock(x, y, z, BukkitAdapter.adapt(id)); | ||||
|             return true; | ||||
| @@ -170,8 +217,9 @@ public class GenChunk extends ScopedLocalBlockQueue { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     @Override public BlockState getBlock(int x, int y, int z) { | ||||
|         int i = MainUtil.CACHE_I[y][x][z]; | ||||
|     @Override | ||||
|     public @Nullable BlockState getBlock(int x, int y, int z) { | ||||
|         int i = y >> 4; | ||||
|         if (result == null) { | ||||
|             return BukkitBlockUtil.get(chunkData.getType(x, y, z)); | ||||
|         } | ||||
| @@ -179,7 +227,7 @@ public class GenChunk extends ScopedLocalBlockQueue { | ||||
|         if (array == null) { | ||||
|             return BlockTypes.AIR.getDefaultState(); | ||||
|         } | ||||
|         int j = MainUtil.CACHE_J[y][x][z]; | ||||
|         int j = ChunkUtil.getJ(x, y, z); | ||||
|         return array[j]; | ||||
|     } | ||||
|  | ||||
| @@ -191,19 +239,22 @@ public class GenChunk extends ScopedLocalBlockQueue { | ||||
|         return chunk == null ? chunkZ : chunk.getZ(); | ||||
|     } | ||||
|  | ||||
|     @Override public String getWorld() { | ||||
|         return chunk == null ? world : chunk.getWorld().getName(); | ||||
|     @Override | ||||
|     public com.sk89q.worldedit.world.@NonNull World getWorld() { | ||||
|         return chunk == null ? BukkitAdapter.adapt(Bukkit.getWorld(world)) : BukkitAdapter.adapt(chunk.getWorld()); | ||||
|     } | ||||
|  | ||||
|     @Override public Location getMax() { | ||||
|         return new Location(getWorld(), 15 + (getX() << 4), 255, 15 + (getZ() << 4)); | ||||
|     @Override | ||||
|     public @NonNull Location getMax() { | ||||
|         return Location.at(getWorld().getName(), 15 + (getX() << 4), 255, 15 + (getZ() << 4)); | ||||
|     } | ||||
|  | ||||
|     @Override public Location getMin() { | ||||
|         return new Location(getWorld(), getX() << 4, 0, getZ() << 4); | ||||
|     @Override | ||||
|     public @NonNull Location getMin() { | ||||
|         return Location.at(getWorld().getName(), getX() << 4, 0, getZ() << 4); | ||||
|     } | ||||
|  | ||||
|     public GenChunk clone() { | ||||
|     public @NonNull GenChunk clone() { | ||||
|         GenChunk toReturn = new GenChunk(); | ||||
|         if (this.result != null) { | ||||
|             for (int i = 0; i < this.result.length; i++) { | ||||
| @@ -217,4 +268,5 @@ public class GenChunk extends ScopedLocalBlockQueue { | ||||
|         toReturn.chunkData = this.chunkData; | ||||
|         return toReturn; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,21 +21,33 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.schematic; | ||||
|  | ||||
| import com.plotsquared.core.queue.LocalBlockQueue; | ||||
| import com.google.inject.Inject; | ||||
| import com.google.inject.Singleton; | ||||
| import com.plotsquared.core.inject.factory.ProgressSubscriberFactory; | ||||
| import com.plotsquared.core.queue.QueueCoordinator; | ||||
| import com.plotsquared.core.util.SchematicHandler; | ||||
| import com.plotsquared.core.util.WorldUtil; | ||||
| import com.sk89q.jnbt.CompoundTag; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| /** | ||||
|  * Schematic Handler. | ||||
|  */ | ||||
| @Singleton | ||||
| public class BukkitSchematicHandler extends SchematicHandler { | ||||
|  | ||||
|     @Override | ||||
|     public boolean restoreTile(LocalBlockQueue queue, CompoundTag ct, int x, int y, int z) { | ||||
|         return new StateWrapper(ct).restoreTag(queue.getWorld(), x, y, z); | ||||
|     @Inject | ||||
|     public BukkitSchematicHandler(final @NonNull WorldUtil worldUtil, @NonNull ProgressSubscriberFactory subscriberFactory) { | ||||
|         super(worldUtil, subscriberFactory); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean restoreTile(QueueCoordinator queue, CompoundTag ct, int x, int y, int z) { | ||||
|         return new StateWrapper(ct).restoreTag(queue.getWorld().getName(), x, y, z); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,12 +21,11 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.schematic; | ||||
|  | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.configuration.Captions; | ||||
| import com.sk89q.jnbt.ByteTag; | ||||
| import com.sk89q.jnbt.CompoundTag; | ||||
| import com.sk89q.jnbt.ListTag; | ||||
| @@ -36,6 +35,7 @@ import com.sk89q.jnbt.Tag; | ||||
| import com.sk89q.worldedit.blocks.BaseItemStack; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.world.item.ItemType; | ||||
| import org.bukkit.ChatColor; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.block.Container; | ||||
| @@ -66,108 +66,107 @@ public class StateWrapper { | ||||
|  | ||||
|     public static String jsonToColourCode(String str) { | ||||
|         str = str.replace("{\"extra\":", "").replace("],\"text\":\"\"}", "]") | ||||
|             .replace("[{\"color\":\"black\",\"text\":\"", "&0") | ||||
|             .replace("[{\"color\":\"dark_blue\",\"text\":\"", "&1") | ||||
|             .replace("[{\"color\":\"dark_green\",\"text\":\"", "&2") | ||||
|             .replace("[{\"color\":\"dark_aqua\",\"text\":\"", "&3") | ||||
|             .replace("[{\"color\":\"dark_red\",\"text\":\"", "&4") | ||||
|             .replace("[{\"color\":\"dark_purple\",\"text\":\"", "&5") | ||||
|             .replace("[{\"color\":\"gold\",\"text\":\"", "&6") | ||||
|             .replace("[{\"color\":\"gray\",\"text\":\"", "&7") | ||||
|             .replace("[{\"color\":\"dark_gray\",\"text\":\"", "&8") | ||||
|             .replace("[{\"color\":\"blue\",\"text\":\"", "&9") | ||||
|             .replace("[{\"color\":\"green\",\"text\":\"", "&a") | ||||
|             .replace("[{\"color\":\"aqua\",\"text\":\"", "&b") | ||||
|             .replace("[{\"color\":\"red\",\"text\":\"", "&c") | ||||
|             .replace("[{\"color\":\"light_purple\",\"text\":\"", "&d") | ||||
|             .replace("[{\"color\":\"yellow\",\"text\":\"", "&e") | ||||
|             .replace("[{\"color\":\"white\",\"text\":\"", "&f") | ||||
|             .replace("[{\"obfuscated\":true,\"text\":\"", "&k") | ||||
|             .replace("[{\"bold\":true,\"text\":\"", "&l") | ||||
|             .replace("[{\"strikethrough\":true,\"text\":\"", "&m") | ||||
|             .replace("[{\"underlined\":true,\"text\":\"", "&n") | ||||
|             .replace("[{\"italic\":true,\"text\":\"", "&o").replace("[{\"color\":\"black\",", "&0") | ||||
|             .replace("[{\"color\":\"dark_blue\",", "&1") | ||||
|             .replace("[{\"color\":\"dark_green\",", "&2") | ||||
|             .replace("[{\"color\":\"dark_aqua\",", "&3").replace("[{\"color\":\"dark_red\",", "&4") | ||||
|             .replace("[{\"color\":\"dark_purple\",", "&5").replace("[{\"color\":\"gold\",", "&6") | ||||
|             .replace("[{\"color\":\"gray\",", "&7").replace("[{\"color\":\"dark_gray\",", "&8") | ||||
|             .replace("[{\"color\":\"blue\",", "&9").replace("[{\"color\":\"green\",", "&a") | ||||
|             .replace("[{\"color\":\"aqua\",", "&b").replace("[{\"color\":\"red\",", "&c") | ||||
|             .replace("[{\"color\":\"light_purple\",", "&d").replace("[{\"color\":\"yellow\",", "&e") | ||||
|             .replace("[{\"color\":\"white\",", "&f").replace("[{\"obfuscated\":true,", "&k") | ||||
|             .replace("[{\"bold\":true,", "&l").replace("[{\"strikethrough\":true,", "&m") | ||||
|             .replace("[{\"underlined\":true,", "&n").replace("[{\"italic\":true,", "&o") | ||||
|             .replace("{\"color\":\"black\",\"text\":\"", "&0") | ||||
|             .replace("{\"color\":\"dark_blue\",\"text\":\"", "&1") | ||||
|             .replace("{\"color\":\"dark_green\",\"text\":\"", "&2") | ||||
|             .replace("{\"color\":\"dark_aqua\",\"text\":\"", "&3") | ||||
|             .replace("{\"color\":\"dark_red\",\"text\":\"", "&4") | ||||
|             .replace("{\"color\":\"dark_purple\",\"text\":\"", "&5") | ||||
|             .replace("{\"color\":\"gold\",\"text\":\"", "&6") | ||||
|             .replace("{\"color\":\"gray\",\"text\":\"", "&7") | ||||
|             .replace("{\"color\":\"dark_gray\",\"text\":\"", "&8") | ||||
|             .replace("{\"color\":\"blue\",\"text\":\"", "&9") | ||||
|             .replace("{\"color\":\"green\",\"text\":\"", "&a") | ||||
|             .replace("{\"color\":\"aqua\",\"text\":\"", "&b") | ||||
|             .replace("{\"color\":\"red\",\"text\":\"", "&c") | ||||
|             .replace("{\"color\":\"light_purple\",\"text\":\"", "&d") | ||||
|             .replace("{\"color\":\"yellow\",\"text\":\"", "&e") | ||||
|             .replace("{\"color\":\"white\",\"text\":\"", "&f") | ||||
|             .replace("{\"obfuscated\":true,\"text\":\"", "&k") | ||||
|             .replace("{\"bold\":true,\"text\":\"", "&l") | ||||
|             .replace("{\"strikethrough\":true,\"text\":\"", "&m") | ||||
|             .replace("{\"underlined\":true,\"text\":\"", "&n") | ||||
|             .replace("{\"italic\":true,\"text\":\"", "&o").replace("{\"color\":\"black\",", "&0") | ||||
|             .replace("{\"color\":\"dark_blue\",", "&1").replace("{\"color\":\"dark_green\",", "&2") | ||||
|             .replace("{\"color\":\"dark_aqua\",", "&3").replace("{\"color\":\"dark_red\",", "&4") | ||||
|             .replace("{\"color\":\"dark_purple\",", "&5").replace("{\"color\":\"gold\",", "&6") | ||||
|             .replace("{\"color\":\"gray\",", "&7").replace("{\"color\":\"dark_gray\",", "&8") | ||||
|             .replace("{\"color\":\"blue\",", "&9").replace("{\"color\":\"green\",", "&a") | ||||
|             .replace("{\"color\":\"aqua\",", "&b").replace("{\"color\":\"red\",", "&c") | ||||
|             .replace("{\"color\":\"light_purple\",", "&d").replace("{\"color\":\"yellow\",", "&e") | ||||
|             .replace("{\"color\":\"white\",", "&f").replace("{\"obfuscated\":true,", "&k") | ||||
|             .replace("{\"bold\":true,", "&l").replace("{\"strikethrough\":true,", "&m") | ||||
|             .replace("{\"underlined\":true,", "&n").replace("{\"italic\":true,", "&o") | ||||
|             .replace("\"color\":\"black\",\"text\":\"", "&0") | ||||
|             .replace("\"color\":\"dark_blue\",\"text\":\"", "&1") | ||||
|             .replace("\"color\":\"dark_green\",\"text\":\"", "&2") | ||||
|             .replace("\"color\":\"dark_aqua\",\"text\":\"", "&3") | ||||
|             .replace("\"color\":\"dark_red\",\"text\":\"", "&4") | ||||
|             .replace("\"color\":\"dark_purple\",\"text\":\"", "&5") | ||||
|             .replace("\"color\":\"gold\",\"text\":\"", "&6") | ||||
|             .replace("\"color\":\"gray\",\"text\":\"", "&7") | ||||
|             .replace("\"color\":\"dark_gray\",\"text\":\"", "&8") | ||||
|             .replace("\"color\":\"blue\",\"text\":\"", "&9") | ||||
|             .replace("\"color\":\"green\",\"text\":\"", "&a") | ||||
|             .replace("\"color\":\"aqua\",\"text\":\"", "&b") | ||||
|             .replace("\"color\":\"red\",\"text\":\"", "&c") | ||||
|             .replace("\"color\":\"light_purple\",\"text\":\"", "&d") | ||||
|             .replace("\"color\":\"yellow\",\"text\":\"", "&e") | ||||
|             .replace("\"color\":\"white\",\"text\":\"", "&f") | ||||
|             .replace("\"obfuscated\":true,\"text\":\"", "&k") | ||||
|             .replace("\"bold\":true,\"text\":\"", "&l") | ||||
|             .replace("\"strikethrough\":true,\"text\":\"", "&m") | ||||
|             .replace("\"underlined\":true,\"text\":\"", "&n") | ||||
|             .replace("\"italic\":true,\"text\":\"", "&o").replace("\"color\":\"black\",", "&0") | ||||
|             .replace("\"color\":\"dark_blue\",", "&1").replace("\"color\":\"dark_green\",", "&2") | ||||
|             .replace("\"color\":\"dark_aqua\",", "&3").replace("\"color\":\"dark_red\",", "&4") | ||||
|             .replace("\"color\":\"dark_purple\",", "&5").replace("\"color\":\"gold\",", "&6") | ||||
|             .replace("\"color\":\"gray\",", "&7").replace("\"color\":\"dark_gray\",", "&8") | ||||
|             .replace("\"color\":\"blue\",", "&9").replace("\"color\":\"green\",", "&a") | ||||
|             .replace("\"color\":\"aqua\",", "&b").replace("\"color\":\"red\",", "&c") | ||||
|             .replace("\"color\":\"light_purple\",", "&d").replace("\"color\":\"yellow\",", "&e") | ||||
|             .replace("\"color\":\"white\",", "&f").replace("\"obfuscated\":true,", "&k") | ||||
|             .replace("\"bold\":true,", "&l").replace("\"strikethrough\":true,", "&m") | ||||
|             .replace("\"underlined\":true,", "&n").replace("\"italic\":true,", "&o") | ||||
|             .replace("[{\"text\":\"", "&0").replace("{\"text\":\"", "&0").replace("\"},", "") | ||||
|             .replace("\"}]", "").replace("\"}", ""); | ||||
|         for (Entry<String, String> entry : Captions.replacements.entrySet()) { | ||||
|             str = str.replace(entry.getKey(), entry.getValue()); | ||||
|         } | ||||
|                 .replace("[{\"color\":\"black\",\"text\":\"", "&0") | ||||
|                 .replace("[{\"color\":\"dark_blue\",\"text\":\"", "&1") | ||||
|                 .replace("[{\"color\":\"dark_green\",\"text\":\"", "&2") | ||||
|                 .replace("[{\"color\":\"dark_aqua\",\"text\":\"", "&3") | ||||
|                 .replace("[{\"color\":\"dark_red\",\"text\":\"", "&4") | ||||
|                 .replace("[{\"color\":\"dark_purple\",\"text\":\"", "&5") | ||||
|                 .replace("[{\"color\":\"gold\",\"text\":\"", "&6") | ||||
|                 .replace("[{\"color\":\"gray\",\"text\":\"", "&7") | ||||
|                 .replace("[{\"color\":\"dark_gray\",\"text\":\"", "&8") | ||||
|                 .replace("[{\"color\":\"blue\",\"text\":\"", "&9") | ||||
|                 .replace("[{\"color\":\"green\",\"text\":\"", "&a") | ||||
|                 .replace("[{\"color\":\"aqua\",\"text\":\"", "&b") | ||||
|                 .replace("[{\"color\":\"red\",\"text\":\"", "&c") | ||||
|                 .replace("[{\"color\":\"light_purple\",\"text\":\"", "&d") | ||||
|                 .replace("[{\"color\":\"yellow\",\"text\":\"", "&e") | ||||
|                 .replace("[{\"color\":\"white\",\"text\":\"", "&f") | ||||
|                 .replace("[{\"obfuscated\":true,\"text\":\"", "&k") | ||||
|                 .replace("[{\"bold\":true,\"text\":\"", "&l") | ||||
|                 .replace("[{\"strikethrough\":true,\"text\":\"", "&m") | ||||
|                 .replace("[{\"underlined\":true,\"text\":\"", "&n") | ||||
|                 .replace("[{\"italic\":true,\"text\":\"", "&o").replace("[{\"color\":\"black\",", "&0") | ||||
|                 .replace("[{\"color\":\"dark_blue\",", "&1") | ||||
|                 .replace("[{\"color\":\"dark_green\",", "&2") | ||||
|                 .replace("[{\"color\":\"dark_aqua\",", "&3").replace("[{\"color\":\"dark_red\",", "&4") | ||||
|                 .replace("[{\"color\":\"dark_purple\",", "&5").replace("[{\"color\":\"gold\",", "&6") | ||||
|                 .replace("[{\"color\":\"gray\",", "&7").replace("[{\"color\":\"dark_gray\",", "&8") | ||||
|                 .replace("[{\"color\":\"blue\",", "&9").replace("[{\"color\":\"green\",", "&a") | ||||
|                 .replace("[{\"color\":\"aqua\",", "&b").replace("[{\"color\":\"red\",", "&c") | ||||
|                 .replace("[{\"color\":\"light_purple\",", "&d").replace("[{\"color\":\"yellow\",", "&e") | ||||
|                 .replace("[{\"color\":\"white\",", "&f").replace("[{\"obfuscated\":true,", "&k") | ||||
|                 .replace("[{\"bold\":true,", "&l").replace("[{\"strikethrough\":true,", "&m") | ||||
|                 .replace("[{\"underlined\":true,", "&n").replace("[{\"italic\":true,", "&o") | ||||
|                 .replace("{\"color\":\"black\",\"text\":\"", "&0") | ||||
|                 .replace("{\"color\":\"dark_blue\",\"text\":\"", "&1") | ||||
|                 .replace("{\"color\":\"dark_green\",\"text\":\"", "&2") | ||||
|                 .replace("{\"color\":\"dark_aqua\",\"text\":\"", "&3") | ||||
|                 .replace("{\"color\":\"dark_red\",\"text\":\"", "&4") | ||||
|                 .replace("{\"color\":\"dark_purple\",\"text\":\"", "&5") | ||||
|                 .replace("{\"color\":\"gold\",\"text\":\"", "&6") | ||||
|                 .replace("{\"color\":\"gray\",\"text\":\"", "&7") | ||||
|                 .replace("{\"color\":\"dark_gray\",\"text\":\"", "&8") | ||||
|                 .replace("{\"color\":\"blue\",\"text\":\"", "&9") | ||||
|                 .replace("{\"color\":\"green\",\"text\":\"", "&a") | ||||
|                 .replace("{\"color\":\"aqua\",\"text\":\"", "&b") | ||||
|                 .replace("{\"color\":\"red\",\"text\":\"", "&c") | ||||
|                 .replace("{\"color\":\"light_purple\",\"text\":\"", "&d") | ||||
|                 .replace("{\"color\":\"yellow\",\"text\":\"", "&e") | ||||
|                 .replace("{\"color\":\"white\",\"text\":\"", "&f") | ||||
|                 .replace("{\"obfuscated\":true,\"text\":\"", "&k") | ||||
|                 .replace("{\"bold\":true,\"text\":\"", "&l") | ||||
|                 .replace("{\"strikethrough\":true,\"text\":\"", "&m") | ||||
|                 .replace("{\"underlined\":true,\"text\":\"", "&n") | ||||
|                 .replace("{\"italic\":true,\"text\":\"", "&o").replace("{\"color\":\"black\",", "&0") | ||||
|                 .replace("{\"color\":\"dark_blue\",", "&1").replace("{\"color\":\"dark_green\",", "&2") | ||||
|                 .replace("{\"color\":\"dark_aqua\",", "&3").replace("{\"color\":\"dark_red\",", "&4") | ||||
|                 .replace("{\"color\":\"dark_purple\",", "&5").replace("{\"color\":\"gold\",", "&6") | ||||
|                 .replace("{\"color\":\"gray\",", "&7").replace("{\"color\":\"dark_gray\",", "&8") | ||||
|                 .replace("{\"color\":\"blue\",", "&9").replace("{\"color\":\"green\",", "&a") | ||||
|                 .replace("{\"color\":\"aqua\",", "&b").replace("{\"color\":\"red\",", "&c") | ||||
|                 .replace("{\"color\":\"light_purple\",", "&d").replace("{\"color\":\"yellow\",", "&e") | ||||
|                 .replace("{\"color\":\"white\",", "&f").replace("{\"obfuscated\":true,", "&k") | ||||
|                 .replace("{\"bold\":true,", "&l").replace("{\"strikethrough\":true,", "&m") | ||||
|                 .replace("{\"underlined\":true,", "&n").replace("{\"italic\":true,", "&o") | ||||
|                 .replace("\"color\":\"black\",\"text\":\"", "&0") | ||||
|                 .replace("\"color\":\"dark_blue\",\"text\":\"", "&1") | ||||
|                 .replace("\"color\":\"dark_green\",\"text\":\"", "&2") | ||||
|                 .replace("\"color\":\"dark_aqua\",\"text\":\"", "&3") | ||||
|                 .replace("\"color\":\"dark_red\",\"text\":\"", "&4") | ||||
|                 .replace("\"color\":\"dark_purple\",\"text\":\"", "&5") | ||||
|                 .replace("\"color\":\"gold\",\"text\":\"", "&6") | ||||
|                 .replace("\"color\":\"gray\",\"text\":\"", "&7") | ||||
|                 .replace("\"color\":\"dark_gray\",\"text\":\"", "&8") | ||||
|                 .replace("\"color\":\"blue\",\"text\":\"", "&9") | ||||
|                 .replace("\"color\":\"green\",\"text\":\"", "&a") | ||||
|                 .replace("\"color\":\"aqua\",\"text\":\"", "&b") | ||||
|                 .replace("\"color\":\"red\",\"text\":\"", "&c") | ||||
|                 .replace("\"color\":\"light_purple\",\"text\":\"", "&d") | ||||
|                 .replace("\"color\":\"yellow\",\"text\":\"", "&e") | ||||
|                 .replace("\"color\":\"white\",\"text\":\"", "&f") | ||||
|                 .replace("\"obfuscated\":true,\"text\":\"", "&k") | ||||
|                 .replace("\"bold\":true,\"text\":\"", "&l") | ||||
|                 .replace("\"strikethrough\":true,\"text\":\"", "&m") | ||||
|                 .replace("\"underlined\":true,\"text\":\"", "&n") | ||||
|                 .replace("\"italic\":true,\"text\":\"", "&o").replace("\"color\":\"black\",", "&0") | ||||
|                 .replace("\"color\":\"dark_blue\",", "&1").replace("\"color\":\"dark_green\",", "&2") | ||||
|                 .replace("\"color\":\"dark_aqua\",", "&3").replace("\"color\":\"dark_red\",", "&4") | ||||
|                 .replace("\"color\":\"dark_purple\",", "&5").replace("\"color\":\"gold\",", "&6") | ||||
|                 .replace("\"color\":\"gray\",", "&7").replace("\"color\":\"dark_gray\",", "&8") | ||||
|                 .replace("\"color\":\"blue\",", "&9").replace("\"color\":\"green\",", "&a") | ||||
|                 .replace("\"color\":\"aqua\",", "&b").replace("\"color\":\"red\",", "&c") | ||||
|                 .replace("\"color\":\"light_purple\",", "&d").replace("\"color\":\"yellow\",", "&e") | ||||
|                 .replace("\"color\":\"white\",", "&f").replace("\"obfuscated\":true,", "&k") | ||||
|                 .replace("\"bold\":true,", "&l").replace("\"strikethrough\":true,", "&m") | ||||
|                 .replace("\"underlined\":true,", "&n").replace("\"italic\":true,", "&o") | ||||
|                 .replace("[{\"text\":\"", "&0").replace("{\"text\":\"", "&0").replace("\"},", "") | ||||
|                 .replace("\"}]", "").replace("\"}", ""); | ||||
|         str = ChatColor.translateAlternateColorCodes('&', str); | ||||
|         return str; | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("deprecation") // #setLine is needed for Spigot compatibility | ||||
|     public boolean restoreTag(String worldName, int x, int y, int z) { | ||||
|         if (this.tag == null) { | ||||
|             return false; | ||||
| @@ -179,19 +178,11 @@ public class StateWrapper { | ||||
|         } | ||||
|         org.bukkit.block.BlockState state = block.getState(); | ||||
|         switch (getId()) { | ||||
|             case "chest": | ||||
|             case "beacon": | ||||
|             case "brewingstand": | ||||
|             case "dispenser": | ||||
|             case "dropper": | ||||
|             case "furnace": | ||||
|             case "hopper": | ||||
|             case "shulkerbox": | ||||
|                 if (!(state instanceof Container)) { | ||||
|             case "chest", "beacon", "brewingstand", "dispenser", "dropper", "furnace", "hopper", "shulkerbox" -> { | ||||
|                 if (!(state instanceof Container container)) { | ||||
|                     return false; | ||||
|                 } | ||||
|                 List<Tag> itemsTag = this.tag.getListTag("Items").getValue(); | ||||
|                 Container container = (Container) state; | ||||
|                 Inventory inv = container.getSnapshotInventory(); | ||||
|                 for (Tag itemTag : itemsTag) { | ||||
|                     CompoundTag itemComp = (CompoundTag) itemTag; | ||||
| @@ -208,9 +199,9 @@ public class StateWrapper { | ||||
|                 } | ||||
|                 container.update(true, false); | ||||
|                 return true; | ||||
|             case "sign": | ||||
|                 if (state instanceof Sign) { | ||||
|                     Sign sign = (Sign) state; | ||||
|             } | ||||
|             case "sign" -> { | ||||
|                 if (state instanceof Sign sign) { | ||||
|                     sign.setLine(0, jsonToColourCode(tag.getString("Text1"))); | ||||
|                     sign.setLine(1, jsonToColourCode(tag.getString("Text2"))); | ||||
|                     sign.setLine(2, jsonToColourCode(tag.getString("Text3"))); | ||||
| @@ -219,6 +210,7 @@ public class StateWrapper { | ||||
|                     return true; | ||||
|                 } | ||||
|                 return false; | ||||
|             } | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
| @@ -227,8 +219,7 @@ public class StateWrapper { | ||||
|         if (this.tag != null) { | ||||
|             return this.tag; | ||||
|         } | ||||
|         if (this.state instanceof InventoryHolder) { | ||||
|             InventoryHolder inv = (InventoryHolder) this.state; | ||||
|         if (this.state instanceof InventoryHolder inv) { | ||||
|             ItemStack[] contents = inv.getInventory().getContents(); | ||||
|             Map<String, Tag> values = new HashMap<>(); | ||||
|             values.put("Items", new ListTag(CompoundTag.class, serializeInventory(contents))); | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,30 +21,16 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.world.block.BlockState; | ||||
| import com.sk89q.worldedit.world.item.ItemType; | ||||
| import org.bukkit.Material; | ||||
| import org.bukkit.block.Block; | ||||
|  | ||||
| import java.util.function.Supplier; | ||||
|  | ||||
| public class BukkitBlockUtil { | ||||
|     public static Supplier<ItemType> supplyItem(Block block) { | ||||
|         return new Supplier<ItemType>() { | ||||
|             @Override public ItemType get() { | ||||
|                 return BukkitAdapter.asItemType(block.getType()); | ||||
|             } | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     public static Supplier<ItemType> supplyItem(Material type) { | ||||
|         return () -> BukkitAdapter.asItemType(type); | ||||
|     } | ||||
|  | ||||
|     public static BlockState get(Block block) { | ||||
|         return get(block.getType()); | ||||
| @@ -53,4 +39,5 @@ public class BukkitBlockUtil { | ||||
|     public static BlockState get(Material material) { | ||||
|         return BukkitAdapter.asBlockType(material).getDefaultState(); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,78 +0,0 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  * | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import com.plotsquared.bukkit.chat.FancyMessage; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | ||||
| import com.plotsquared.core.configuration.Captions; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.player.ConsolePlayer; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.message.PlotMessage; | ||||
| import com.plotsquared.core.util.ChatManager; | ||||
| import org.bukkit.ChatColor; | ||||
|  | ||||
| import java.util.Arrays; | ||||
| import java.util.List; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| public class BukkitChatManager extends ChatManager<FancyMessage> { | ||||
|  | ||||
|     @Override public FancyMessage builder() { | ||||
|         return new FancyMessage(""); | ||||
|     } | ||||
|  | ||||
|     @Override public void color(PlotMessage message, String color) { | ||||
|         message.$(this).color(ChatColor.getByChar(Captions.color(color).substring(1))); | ||||
|     } | ||||
|  | ||||
|     @Override public void tooltip(PlotMessage message, PlotMessage... tooltips) { | ||||
|         List<FancyMessage> lines = | ||||
|             Arrays.stream(tooltips).map(tooltip -> tooltip.$(this)).collect(Collectors.toList()); | ||||
|         message.$(this).formattedTooltip(lines); | ||||
|     } | ||||
|  | ||||
|     @Override public void command(PlotMessage message, String command) { | ||||
|         message.$(this).command(command); | ||||
|     } | ||||
|  | ||||
|     @Override public void text(PlotMessage message, String text) { | ||||
|         message.$(this).then(ChatColor.stripColor(text)); | ||||
|     } | ||||
|  | ||||
|     @Override public void send(PlotMessage plotMessage, PlotPlayer player) { | ||||
|         if (player instanceof ConsolePlayer || !Settings.Chat.INTERACTIVE) { | ||||
|             player.sendMessage(plotMessage.$(this).toOldMessageFormat()); | ||||
|         } else { | ||||
|             plotMessage.$(this).send(((BukkitPlayer) player).player); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public void suggest(PlotMessage plotMessage, String command) { | ||||
|         plotMessage.$(this).suggest(command); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,145 +21,31 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.listener.WEExtent; | ||||
| import com.plotsquared.core.queue.GlobalBlockQueue; | ||||
| import com.plotsquared.core.queue.LocalBlockQueue; | ||||
| import com.google.inject.Singleton; | ||||
| import com.plotsquared.core.util.ChunkManager; | ||||
| import com.plotsquared.core.util.entity.EntityCategories; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.bukkit.BukkitWorld; | ||||
| import com.sk89q.worldedit.math.BlockVector2; | ||||
| import com.sk89q.worldedit.math.BlockVector3; | ||||
| import com.sk89q.worldedit.regions.CuboidRegion; | ||||
| import com.sk89q.worldedit.world.block.BaseBlock; | ||||
| import io.papermc.lib.PaperLib; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.block.data.BlockData; | ||||
| import org.bukkit.entity.Entity; | ||||
|  | ||||
| import java.util.concurrent.CompletableFuture; | ||||
|  | ||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_ANIMAL; | ||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_ENTITY; | ||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_MISC; | ||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_MOB; | ||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_MONSTER; | ||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_VEHICLE; | ||||
|  | ||||
| @Singleton | ||||
| public class BukkitChunkManager extends ChunkManager { | ||||
|  | ||||
|     public static boolean isIn(CuboidRegion region, int x, int z) { | ||||
|         return x >= region.getMinimumPoint().getX() && x <= region.getMaximumPoint().getX() | ||||
|             && z >= region.getMinimumPoint().getZ() && z <= region.getMaximumPoint().getZ(); | ||||
|     } | ||||
|  | ||||
|     public static ContentMap swapChunk(World world1, World world2, Chunk pos1, Chunk pos2, | ||||
|         CuboidRegion r1, CuboidRegion r2) { | ||||
|         ContentMap map = new ContentMap(); | ||||
|         int relX = r2.getMinimumPoint().getX() - r1.getMinimumPoint().getX(); | ||||
|         int relZ = r2.getMinimumPoint().getZ() - r1.getMinimumPoint().getZ(); | ||||
|  | ||||
|         map.saveEntitiesIn(pos1, r1, relX, relZ, true); | ||||
|         map.saveEntitiesIn(pos2, r2, -relX, -relZ, true); | ||||
|  | ||||
|         int sx = pos1.getX() << 4; | ||||
|         int sz = pos1.getZ() << 4; | ||||
|  | ||||
|         String worldName1 = world1.getName(); | ||||
|         String worldName2 = world2.getName(); | ||||
|  | ||||
|         BukkitWorld bukkitWorld1 = new BukkitWorld(world1); | ||||
|         BukkitWorld bukkitWorld2 = new BukkitWorld(world2); | ||||
|  | ||||
|         LocalBlockQueue queue1 = GlobalBlockQueue.IMP.getNewQueue(worldName1, false); | ||||
|         LocalBlockQueue queue2 = GlobalBlockQueue.IMP.getNewQueue(worldName2, false); | ||||
|  | ||||
|         for (int x = Math.max(r1.getMinimumPoint().getX(), sx); | ||||
|              x <= Math.min(r1.getMaximumPoint().getX(), sx + 15); x++) { | ||||
|             for (int z = Math.max(r1.getMinimumPoint().getZ(), sz); | ||||
|                  z <= Math.min(r1.getMaximumPoint().getZ(), sz + 15); z++) { | ||||
|                 for (int y = 0; y < 256; y++) { | ||||
|                     Block block1 = world1.getBlockAt(x, y, z); | ||||
|                     BaseBlock baseBlock1 = bukkitWorld1.getFullBlock(BlockVector3.at(x, y, z)); | ||||
|                     BlockData data1 = block1.getBlockData(); | ||||
|  | ||||
|                     int xx = x + relX; | ||||
|                     int zz = z + relZ; | ||||
|  | ||||
|                     Block block2 = world2.getBlockAt(xx, y, zz); | ||||
|                     BaseBlock baseBlock2 = bukkitWorld2.getFullBlock(BlockVector3.at(xx, y, zz)); | ||||
|                     BlockData data2 = block2.getBlockData(); | ||||
|  | ||||
|                     if (block1.isEmpty()) { | ||||
|                         if (!block2.isEmpty()) { | ||||
|                             queue1.setBlock(x, y, z, baseBlock2); | ||||
|                             queue2.setBlock(xx, y, zz, WEExtent.AIRBASE); | ||||
|                         } | ||||
|                     } else if (block2.isEmpty()) { | ||||
|                         queue1.setBlock(x, y, z, WEExtent.AIRBASE); | ||||
|                         queue2.setBlock(xx, y, zz, baseBlock1); | ||||
|                     } else if (block1.equals(block2)) { | ||||
|                         if (!data1.matches(data2)) { | ||||
|                             block1.setBlockData(data2); | ||||
|                             block2.setBlockData(data1); | ||||
|                         } | ||||
|                     } else { | ||||
|                         queue1.setBlock(x, y, z, baseBlock2); | ||||
|                         queue2.setBlock(xx, y, zz, baseBlock1); | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         queue1.enqueue(); | ||||
|         queue2.enqueue(); | ||||
|         return map; | ||||
|         return x >= region.getMinimumPoint().getX() && x <= region.getMaximumPoint().getX() && z >= region | ||||
|                 .getMinimumPoint() | ||||
|                 .getZ() && z <= region | ||||
|                 .getMaximumPoint().getZ(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public CompletableFuture<?> loadChunk(String world, BlockVector2 chunkLoc, boolean force) { | ||||
|         return PaperLib | ||||
|             .getChunkAtAsync(BukkitUtil.getWorld(world), chunkLoc.getX(), chunkLoc.getZ(), force); | ||||
|         return PaperLib.getChunkAtAsync(BukkitUtil.getWorld(world), chunkLoc.getX(), chunkLoc.getZ(), force); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void unloadChunk(final String world, final BlockVector2 chunkLoc, final boolean save) { | ||||
|         if (!PlotSquared.get().isMainThread(Thread.currentThread())) { | ||||
|             TaskManager.runTask(() -> BukkitUtil.getWorld(world) | ||||
|                 .unloadChunk(chunkLoc.getX(), chunkLoc.getZ(), save)); | ||||
|         } else { | ||||
|             BukkitUtil.getWorld(world).unloadChunk(chunkLoc.getX(), chunkLoc.getZ(), save); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void count(int[] count, Entity entity) { | ||||
|         final com.sk89q.worldedit.world.entity.EntityType entityType = | ||||
|             BukkitAdapter.adapt(entity.getType()); | ||||
|  | ||||
|         if (EntityCategories.PLAYER.contains(entityType)) { | ||||
|             return; | ||||
|         } else if (EntityCategories.PROJECTILE.contains(entityType) || EntityCategories.OTHER | ||||
|             .contains(entityType) || EntityCategories.HANGING.contains(entityType)) { | ||||
|             count[CAP_MISC]++; | ||||
|         } else if (EntityCategories.ANIMAL.contains(entityType) || EntityCategories.VILLAGER | ||||
|             .contains(entityType) || EntityCategories.TAMEABLE.contains(entityType)) { | ||||
|             count[CAP_MOB]++; | ||||
|             count[CAP_ANIMAL]++; | ||||
|         } else if (EntityCategories.VEHICLE.contains(entityType)) { | ||||
|             count[CAP_VEHICLE]++; | ||||
|         } else if (EntityCategories.HOSTILE.contains(entityType)) { | ||||
|             count[CAP_MOB]++; | ||||
|             count[CAP_MONSTER]++; | ||||
|         } | ||||
|         count[CAP_ENTITY]++; | ||||
|     } | ||||
|  | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,25 +21,32 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import com.google.inject.Singleton; | ||||
| import com.plotsquared.bukkit.player.BukkitOfflinePlayer; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.player.OfflinePlotPlayer; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.util.EconHandler; | ||||
| import com.plotsquared.core.util.PermHandler; | ||||
| import net.milkbowl.vault.economy.Economy; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.OfflinePlayer; | ||||
| import org.bukkit.plugin.RegisteredServiceProvider; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| @Singleton | ||||
| public class BukkitEconHandler extends EconHandler { | ||||
|  | ||||
|     private Economy econ; | ||||
|  | ||||
|     private static OfflinePlayer getBukkitOfflinePlayer(PlotPlayer<?> plotPlayer) { | ||||
|         return ((BukkitPlayer) plotPlayer).player; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean init() { | ||||
|         if (this.econ == null) { | ||||
| @@ -53,45 +60,54 @@ public class BukkitEconHandler extends EconHandler { | ||||
|             return; | ||||
|         } | ||||
|         RegisteredServiceProvider<Economy> economyProvider = | ||||
|             Bukkit.getServer().getServicesManager().getRegistration(Economy.class); | ||||
|                 Bukkit.getServer().getServicesManager().getRegistration(Economy.class); | ||||
|         if (economyProvider != null) { | ||||
|             this.econ = economyProvider.getProvider(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public double getMoney(PlotPlayer<?> player) { | ||||
|     @Override | ||||
|     public double getMoney(PlotPlayer<?> player) { | ||||
|         double bal = super.getMoney(player); | ||||
|         if (Double.isNaN(bal)) { | ||||
|             return this.econ.getBalance(((BukkitPlayer) player).player); | ||||
|             return this.econ.getBalance(getBukkitOfflinePlayer(player)); | ||||
|         } | ||||
|         return bal; | ||||
|     } | ||||
|  | ||||
|     @Override public void withdrawMoney(PlotPlayer<?> player, double amount) { | ||||
|         this.econ.withdrawPlayer(((BukkitPlayer) player).player, amount); | ||||
|     @Override | ||||
|     public void withdrawMoney(PlotPlayer<?> player, double amount) { | ||||
|         this.econ.withdrawPlayer(getBukkitOfflinePlayer(player), amount); | ||||
|     } | ||||
|  | ||||
|     @Override public void depositMoney(PlotPlayer<?> player, double amount) { | ||||
|         this.econ.depositPlayer(((BukkitPlayer) player).player, amount); | ||||
|     @Override | ||||
|     public void depositMoney(PlotPlayer<?> player, double amount) { | ||||
|         this.econ.depositPlayer(getBukkitOfflinePlayer(player), amount); | ||||
|     } | ||||
|  | ||||
|     @Override public void depositMoney(OfflinePlotPlayer player, double amount) { | ||||
|     @Override | ||||
|     public void depositMoney(OfflinePlotPlayer player, double amount) { | ||||
|         this.econ.depositPlayer(((BukkitOfflinePlayer) player).player, amount); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @deprecated Use {@link PermHandler#hasPermission(String, String, String)} instead | ||||
|      */ | ||||
|     @Deprecated @Override public boolean hasPermission(String world, String player, String perm) { | ||||
|         if (PlotSquared.imp().getPermissionHandler() != null) { | ||||
|             return PlotSquared.imp().getPermissionHandler().hasPermission(world, player, perm); | ||||
|         } else { | ||||
|             return false; | ||||
|         } | ||||
|     @Override | ||||
|     public boolean isEnabled(PlotArea plotArea) { | ||||
|         return plotArea.useEconomy(); | ||||
|     } | ||||
|  | ||||
|     @Override public double getBalance(PlotPlayer<?> player) { | ||||
|         return this.econ.getBalance(player.getName()); | ||||
|     @Override | ||||
|     public @NonNull String format(double balance) { | ||||
|         return this.econ.format(balance); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean isSupported() { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public double getBalance(PlotPlayer<?> player) { | ||||
|         return this.econ.getBalance(getBukkitOfflinePlayer(player)); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,14 +21,15 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | ||||
| import com.plotsquared.core.configuration.Captions; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| 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.AnimalAttackFlag; | ||||
| @@ -46,10 +47,10 @@ import com.plotsquared.core.plot.flag.implementations.PvpFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.TamedAttackFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.VehicleCapFlag; | ||||
| import com.plotsquared.core.util.EntityUtil; | ||||
| import com.plotsquared.core.util.MainUtil; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.entity.EntityCategories; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import net.kyori.adventure.text.minimessage.Template; | ||||
| import org.bukkit.entity.Arrow; | ||||
| import org.bukkit.entity.Creature; | ||||
| import org.bukkit.entity.Entity; | ||||
| @@ -65,16 +66,15 @@ import java.util.Objects; | ||||
| public class BukkitEntityUtil { | ||||
|  | ||||
|     public static final com.sk89q.worldedit.world.entity.EntityType FAKE_ENTITY_TYPE = | ||||
|         new com.sk89q.worldedit.world.entity.EntityType("plotsquared:fake"); | ||||
|             new com.sk89q.worldedit.world.entity.EntityType("plotsquared:fake"); | ||||
|  | ||||
|     public static boolean entityDamage(Entity damager, Entity victim) { | ||||
|         return entityDamage(damager, victim, null); | ||||
|     } | ||||
|  | ||||
|     public static boolean entityDamage(Entity damager, Entity victim, | ||||
|         EntityDamageEvent.DamageCause cause) { | ||||
|         Location dloc = BukkitUtil.getLocation(damager); | ||||
|         Location vloc = BukkitUtil.getLocation(victim); | ||||
|     public static boolean entityDamage(Entity damager, Entity victim, EntityDamageEvent.DamageCause cause) { | ||||
|         Location dloc = BukkitUtil.adapt(damager.getLocation()); | ||||
|         Location vloc = BukkitUtil.adapt(victim.getLocation()); | ||||
|         PlotArea dArea = dloc.getPlotArea(); | ||||
|         PlotArea vArea; | ||||
|         if (dArea != null && dArea.contains(vloc.getX(), vloc.getZ())) { | ||||
| @@ -144,15 +144,14 @@ public class BukkitEntityUtil { | ||||
|         Player player; | ||||
|         if (damager instanceof Player) { // attacker is player | ||||
|             player = (Player) damager; | ||||
|         } else if (damager instanceof Projectile) { | ||||
|             Projectile projectile = (Projectile) damager; | ||||
|         } else if (damager instanceof Projectile projectile) { | ||||
|             ProjectileSource shooter = projectile.getShooter(); | ||||
|             if (shooter instanceof Player) { // shooter is player | ||||
|                 player = (Player) shooter; | ||||
|             } else { // shooter is not player | ||||
|                 if (shooter instanceof BlockProjectileSource) { | ||||
|                     Location sLoc = BukkitUtil | ||||
|                         .getLocation(((BlockProjectileSource) shooter).getBlock().getLocation()); | ||||
|                             .adapt(((BlockProjectileSource) shooter).getBlock().getLocation()); | ||||
|                     dplot = dArea.getPlot(sLoc); | ||||
|                 } | ||||
|                 player = null; | ||||
| @@ -161,7 +160,7 @@ public class BukkitEntityUtil { | ||||
|             player = null; | ||||
|         } | ||||
|         if (player != null) { | ||||
|             BukkitPlayer plotPlayer = BukkitUtil.getPlayer(player); | ||||
|             BukkitPlayer plotPlayer = BukkitUtil.adapt(player); | ||||
|  | ||||
|             final com.sk89q.worldedit.world.entity.EntityType entityType; | ||||
|  | ||||
| @@ -174,82 +173,93 @@ public class BukkitEntityUtil { | ||||
|  | ||||
|             if (EntityCategories.HANGING.contains(entityType)) { // hanging | ||||
|                 if (plot != null && (plot.getFlag(HangingBreakFlag.class) || plot | ||||
|                     .isAdded(plotPlayer.getUUID()))) { | ||||
|                         .isAdded(plotPlayer.getUUID()))) { | ||||
|                     if (Settings.Done.RESTRICT_BUILDING && DoneFlag.isDone(plot)) { | ||||
|                         if (!Permissions | ||||
|                             .hasPermission(plotPlayer, Captions.PERMISSION_ADMIN_BUILD_OTHER)) { | ||||
|                             MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, | ||||
|                                 Captions.PERMISSION_ADMIN_BUILD_OTHER); | ||||
|                                 .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_BUILD_OTHER)) { | ||||
|                             plotPlayer.sendMessage( | ||||
|                                     TranslatableCaption.of("done.building_restricted") | ||||
|                             ); | ||||
|                             return false; | ||||
|                         } | ||||
|                     } | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.destroy." + stub)) { | ||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, | ||||
|                         "plots.admin.destroy." + stub); | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_DESTROY + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_DESTROY + "." + stub) | ||||
|                     ); | ||||
|                     return false; | ||||
|                 } | ||||
|             } else if (victim.getType() == EntityType.ARMOR_STAND) { | ||||
|                 if (plot != null && (plot.getFlag(MiscBreakFlag.class) || plot | ||||
|                     .isAdded(plotPlayer.getUUID()))) { | ||||
|                         .isAdded(plotPlayer.getUUID()))) { | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.destroy." + stub)) { | ||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, | ||||
|                         "plots.admin.destroy." + stub); | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_DESTROY + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_DESTROY + "." + stub) | ||||
|                     ); | ||||
|                     if (plot != null) { | ||||
|                         plot.debug(player.getName() | ||||
|                             + " could not break armor stand because misc-break = false"); | ||||
|                                 + " could not break armor stand because misc-break = false"); | ||||
|                     } | ||||
|                     return false; | ||||
|                 } | ||||
|             } else if (EntityCategories.HOSTILE.contains(entityType)) { | ||||
|                 if (isPlot) { | ||||
|                     if (plot.getFlag(HostileAttackFlag.class) || plot.getFlag(PveFlag.class) || plot | ||||
|                         .isAdded(plotPlayer.getUUID())) { | ||||
|                             .isAdded(plotPlayer.getUUID())) { | ||||
|                         return true; | ||||
|                     } | ||||
|                 } else if (roadFlags && (area.getRoadFlag(HostileAttackFlag.class) || area | ||||
|                     .getFlag(PveFlag.class))) { | ||||
|                         .getFlag(PveFlag.class))) { | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.pve." + stub)) { | ||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, | ||||
|                         "plots.admin.pve." + stub); | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_PVE + "." + stub) | ||||
|                     ); | ||||
|                     if (plot != null) { | ||||
|                         plot.debug(player.getName() + " could not attack " + entityType | ||||
|                             + " because pve = false OR hostile-attack = false"); | ||||
|                                 + " because pve = false OR hostile-attack = false"); | ||||
|                     } | ||||
|                     return false; | ||||
|                 } | ||||
|             } else if (EntityCategories.TAMEABLE.contains(entityType)) { // victim is tameable | ||||
|                 if (isPlot) { | ||||
|                     if (plot.getFlag(TamedAttackFlag.class) || plot.getFlag(PveFlag.class) || plot | ||||
|                         .isAdded(plotPlayer.getUUID())) { | ||||
|                             .isAdded(plotPlayer.getUUID())) { | ||||
|                         return true; | ||||
|                     } | ||||
|                 } else if (roadFlags && (area.getRoadFlag(TamedAttackFlag.class) || area | ||||
|                     .getFlag(PveFlag.class))) { | ||||
|                         .getFlag(PveFlag.class))) { | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.pve." + stub)) { | ||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, | ||||
|                         "plots.admin.pve." + stub); | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_PVE + "." + stub) | ||||
|                     ); | ||||
|                     if (plot != null) { | ||||
|                         plot.debug(player.getName() + " could not attack " + entityType | ||||
|                             + " because pve = false OR tamned-attack = false"); | ||||
|                                 + " because pve = false OR tamed-attack = false"); | ||||
|                     } | ||||
|                     return false; | ||||
|                 } | ||||
|             } else if (EntityCategories.PLAYER.contains(entityType)) { | ||||
|                 if (isPlot) { | ||||
|                     if (!plot.getFlag(PvpFlag.class) && !Permissions | ||||
|                         .hasPermission(plotPlayer, "plots.admin.pvp." + stub)) { | ||||
|                         MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, | ||||
|                             "plots.admin.pvp." + stub); | ||||
|                             .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVP + "." + stub)) { | ||||
|                         plotPlayer.sendMessage( | ||||
|                                 TranslatableCaption.of("permission.no_permission_event"), | ||||
|                                 Template.of("node", Permission.PERMISSION_ADMIN_PVP + "." + stub) | ||||
|                         ); | ||||
|                         plot.debug(player.getName() + " could not attack " + entityType | ||||
|                             + " because pve = false"); | ||||
|                                 + " because pve = false"); | ||||
|                         return false; | ||||
|                     } else { | ||||
|                         return true; | ||||
| @@ -257,30 +267,36 @@ public class BukkitEntityUtil { | ||||
|                 } else if (roadFlags && area.getRoadFlag(PvpFlag.class)) { | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.pvp." + stub)) { | ||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, | ||||
|                         "plots.admin.pvp." + stub); | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVP + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_PVP + "." + stub) | ||||
|                     ); | ||||
|                     return false; | ||||
|                 } | ||||
|             } else if (EntityCategories.ANIMAL.contains(entityType)) { // victim is animal | ||||
|                 if (isPlot) { | ||||
|                     if (plot.getFlag(AnimalAttackFlag.class) || plot.getFlag(PveFlag.class) || plot | ||||
|                         .isAdded(plotPlayer.getUUID())) { | ||||
|                         plot.debug(player.getName() + " could not attack " + entityType | ||||
|                             + " because pve = false OR animal-attack = false"); | ||||
|                             .isAdded(plotPlayer.getUUID())) { | ||||
|                         return true; | ||||
|                     } | ||||
|                 } else if (roadFlags && (area.getRoadFlag(AnimalAttackFlag.class) || area | ||||
|                     .getFlag(PveFlag.class))) { | ||||
|                         .getFlag(PveFlag.class))) { | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.pve." + stub)) { | ||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, | ||||
|                         "plots.admin.pve." + stub); | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_PVE + "." + stub) | ||||
|                     ); | ||||
|                     if (plot != null) { | ||||
|                         plot.debug(player.getName() + " could not attack " + entityType | ||||
|                                 + " because pve = false OR animal-attack = false"); | ||||
|                     } | ||||
|                     return false; | ||||
|                 } | ||||
|             } else if (EntityCategories.VEHICLE | ||||
|                 .contains(entityType)) { // Vehicles are managed in vehicle destroy event | ||||
|                     .contains(entityType)) { // Vehicles are managed in vehicle destroy event | ||||
|                 return true; | ||||
|             } else { // victim is something else | ||||
|                 if (isPlot) { | ||||
| @@ -290,25 +306,27 @@ public class BukkitEntityUtil { | ||||
|                 } else if (roadFlags && area.getRoadFlag(PveFlag.class)) { | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.pve." + stub)) { | ||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, | ||||
|                         "plots.admin.pve." + stub); | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_PVE + "." + stub) | ||||
|                     ); | ||||
|                     if (plot != null) { | ||||
|                         plot.debug(player.getName() + " could not attack " + entityType | ||||
|                             + " because pve = false"); | ||||
|                                 + " because pve = false"); | ||||
|                     } | ||||
|                     return false; | ||||
|                 } | ||||
|             } | ||||
|             return true; | ||||
|         } else if (dplot != null && (!dplot.equals(vplot) || Objects | ||||
|             .equals(dplot.getOwnerAbs(), vplot.getOwnerAbs()))) { | ||||
|                 .equals(dplot.getOwnerAbs(), vplot.getOwnerAbs()))) { | ||||
|             return vplot != null && vplot.getFlag(PveFlag.class); | ||||
|         } | ||||
|         //disable the firework damage. too much of a headache to support at the moment. | ||||
|         if (vplot != null) { | ||||
|             if (EntityDamageEvent.DamageCause.ENTITY_EXPLOSION == cause | ||||
|                 && damager.getType() == EntityType.FIREWORK) { | ||||
|                     && damager.getType() == EntityType.FIREWORK) { | ||||
|                 return false; | ||||
|             } | ||||
|         } | ||||
| @@ -316,46 +334,50 @@ public class BukkitEntityUtil { | ||||
|             return true; | ||||
|         } | ||||
|         return ((vplot != null && vplot.getFlag(PveFlag.class)) || !(damager instanceof Arrow | ||||
|             && !(victim instanceof Creature))); | ||||
|                 && !(victim instanceof Creature))); | ||||
|     } | ||||
|  | ||||
|     public static boolean checkEntity(Entity entity, Plot plot) { | ||||
|         if (plot == null || !plot.hasOwner() || plot.getFlags().isEmpty() && plot.getArea() | ||||
|             .getFlagContainer().getFlagMap().isEmpty()) { | ||||
|                 .getFlagContainer().getFlagMap().isEmpty()) { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         final com.sk89q.worldedit.world.entity.EntityType entityType = | ||||
|             BukkitAdapter.adapt(entity.getType()); | ||||
|                 BukkitAdapter.adapt(entity.getType()); | ||||
|  | ||||
|         if (EntityCategories.PLAYER.contains(entityType)) { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         if (EntityCategories.PROJECTILE.contains(entityType) || EntityCategories.OTHER | ||||
|             .contains(entityType) || EntityCategories.HANGING.contains(entityType)) { | ||||
|                 .contains(entityType) || EntityCategories.HANGING.contains(entityType)) { | ||||
|             return EntityUtil.checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED, | ||||
|                 MiscCapFlag.MISC_CAP_UNLIMITED); | ||||
|                     MiscCapFlag.MISC_CAP_UNLIMITED | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         // Has to go go before vehicle as horses are both | ||||
|         // animals and vehicles | ||||
|         if (EntityCategories.ANIMAL.contains(entityType) || EntityCategories.VILLAGER | ||||
|             .contains(entityType) || EntityCategories.TAMEABLE.contains(entityType)) { | ||||
|                 .contains(entityType) || EntityCategories.TAMEABLE.contains(entityType)) { | ||||
|             return EntityUtil | ||||
|                 .checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED, MobCapFlag.MOB_CAP_UNLIMITED, | ||||
|                     AnimalCapFlag.ANIMAL_CAP_UNLIMITED); | ||||
|                     .checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED, MobCapFlag.MOB_CAP_UNLIMITED, | ||||
|                             AnimalCapFlag.ANIMAL_CAP_UNLIMITED | ||||
|                     ); | ||||
|         } | ||||
|  | ||||
|         if (EntityCategories.HOSTILE.contains(entityType)) { | ||||
|             return EntityUtil | ||||
|                 .checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED, MobCapFlag.MOB_CAP_UNLIMITED, | ||||
|                     HostileCapFlag.HOSTILE_CAP_UNLIMITED); | ||||
|                     .checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED, MobCapFlag.MOB_CAP_UNLIMITED, | ||||
|                             HostileCapFlag.HOSTILE_CAP_UNLIMITED | ||||
|                     ); | ||||
|         } | ||||
|  | ||||
|         if (EntityCategories.VEHICLE.contains(entityType)) { | ||||
|             return EntityUtil.checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED, | ||||
|                 VehicleCapFlag.VEHICLE_CAP_UNLIMITED); | ||||
|                     VehicleCapFlag.VEHICLE_CAP_UNLIMITED | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         return EntityUtil.checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED); | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,16 +21,18 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import com.google.inject.Singleton; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.PlotInventory; | ||||
| import com.plotsquared.core.plot.PlotItemStack; | ||||
| import com.plotsquared.core.util.InventoryUtil; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.ChatColor; | ||||
| import org.bukkit.Material; | ||||
| @@ -45,57 +47,27 @@ import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import java.util.stream.IntStream; | ||||
|  | ||||
| @Singleton | ||||
| public class BukkitInventoryUtil extends InventoryUtil { | ||||
|  | ||||
|     @Override public void open(PlotInventory inv) { | ||||
|         BukkitPlayer bp = (BukkitPlayer) inv.player; | ||||
|         Inventory inventory = Bukkit.createInventory(null, inv.size * 9, | ||||
|             ChatColor.translateAlternateColorCodes('&', inv.getTitle())); | ||||
|         PlotItemStack[] items = inv.getItems(); | ||||
|         for (int i = 0; i < inv.size * 9; i++) { | ||||
|             PlotItemStack item = items[i]; | ||||
|             if (item != null) { | ||||
|                 inventory.setItem(i, getItem(item)); | ||||
|             } | ||||
|         } | ||||
|         bp.player.openInventory(inventory); | ||||
|     } | ||||
|  | ||||
|     @Override public void close(PlotInventory inv) { | ||||
|         if (!inv.isOpen()) { | ||||
|             return; | ||||
|         } | ||||
|         BukkitPlayer bp = (BukkitPlayer) inv.player; | ||||
|         bp.player.closeInventory(); | ||||
|     } | ||||
|  | ||||
|     @Override public void setItem(PlotInventory inv, int index, PlotItemStack item) { | ||||
|         BukkitPlayer bp = (BukkitPlayer) inv.player; | ||||
|         InventoryView opened = bp.player.getOpenInventory(); | ||||
|         if (!inv.isOpen()) { | ||||
|             return; | ||||
|         } | ||||
|         opened.setItem(index, getItem(item)); | ||||
|         bp.player.updateInventory(); | ||||
|     } | ||||
|  | ||||
|     private static ItemStack getItem(PlotItemStack item) { | ||||
|         if (item == null) { | ||||
|             return null; | ||||
|         } | ||||
|         ItemStack stack = new ItemStack(BukkitAdapter.adapt(item.getType()), item.amount); | ||||
|         ItemStack stack = new ItemStack(BukkitAdapter.adapt(item.getType()), item.getAmount()); | ||||
|         ItemMeta meta = null; | ||||
|         if (item.name != null) { | ||||
|         if (item.getName() != null) { | ||||
|             meta = stack.getItemMeta(); | ||||
|             meta.setDisplayName(ChatColor.translateAlternateColorCodes('&', item.name)); | ||||
|             Component nameComponent = BukkitUtil.MINI_MESSAGE.parse(item.getName()); | ||||
|             meta.setDisplayName(BukkitUtil.LEGACY_COMPONENT_SERIALIZER.serialize(nameComponent)); | ||||
|         } | ||||
|         if (item.lore != null) { | ||||
|         if (item.getLore() != null) { | ||||
|             if (meta == null) { | ||||
|                 meta = stack.getItemMeta(); | ||||
|             } | ||||
|             List<String> lore = new ArrayList<>(); | ||||
|             for (String entry : item.lore) { | ||||
|                 lore.add(ChatColor.translateAlternateColorCodes('&', entry)); | ||||
|             for (String entry : item.getLore()) { | ||||
|                 lore.add(BukkitUtil.LEGACY_COMPONENT_SERIALIZER.serialize(BukkitUtil.MINI_MESSAGE.deserialize(entry))); | ||||
|             } | ||||
|             meta.setLore(lore); | ||||
|         } | ||||
| @@ -105,6 +77,44 @@ public class BukkitInventoryUtil extends InventoryUtil { | ||||
|         return stack; | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("deprecation") // Paper deprecation | ||||
|     @Override | ||||
|     public void open(PlotInventory inv) { | ||||
|         BukkitPlayer bp = (BukkitPlayer) inv.getPlayer(); | ||||
|         Inventory inventory = Bukkit.createInventory(null, inv.getLines() * 9, | ||||
|                 ChatColor.translateAlternateColorCodes('&', inv.getTitle()) | ||||
|         ); | ||||
|         PlotItemStack[] items = inv.getItems(); | ||||
|         for (int i = 0; i < inv.getLines() * 9; i++) { | ||||
|             PlotItemStack item = items[i]; | ||||
|             if (item != null) { | ||||
|                 inventory.setItem(i, getItem(item)); | ||||
|             } | ||||
|         } | ||||
|         bp.player.openInventory(inventory); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void close(PlotInventory inv) { | ||||
|         if (!inv.isOpen()) { | ||||
|             return; | ||||
|         } | ||||
|         BukkitPlayer bp = (BukkitPlayer) inv.getPlayer(); | ||||
|         bp.player.closeInventory(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void setItem(PlotInventory inv, int index, PlotItemStack item) { | ||||
|         BukkitPlayer bp = (BukkitPlayer) inv.getPlayer(); | ||||
|         InventoryView opened = bp.player.getOpenInventory(); | ||||
|         if (!inv.isOpen()) { | ||||
|             return; | ||||
|         } | ||||
|         opened.setItem(index, getItem(item)); | ||||
|         bp.player.updateInventory(); | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("deprecation") // Paper deprecation | ||||
|     public PlotItemStack getItem(ItemStack item) { | ||||
|         if (item == null) { | ||||
|             return null; | ||||
| @@ -129,18 +139,21 @@ public class BukkitInventoryUtil extends InventoryUtil { | ||||
|         return new PlotItemStack(id.name(), amount, name, lore); | ||||
|     } | ||||
|  | ||||
|     @Override public PlotItemStack[] getItems(PlotPlayer player) { | ||||
|     @Override | ||||
|     public PlotItemStack[] getItems(PlotPlayer<?> player) { | ||||
|         BukkitPlayer bp = (BukkitPlayer) player; | ||||
|         PlayerInventory inv = bp.player.getInventory(); | ||||
|         return IntStream.range(0, 36).mapToObj(i -> getItem(inv.getItem(i))) | ||||
|             .toArray(PlotItemStack[]::new); | ||||
|                 .toArray(PlotItemStack[]::new); | ||||
|     } | ||||
|  | ||||
|     @Override public boolean isOpen(PlotInventory plotInventory) { | ||||
|     @SuppressWarnings("deprecation") // #getTitle is needed for Spigot compatibility | ||||
|     @Override | ||||
|     public boolean isOpen(PlotInventory plotInventory) { | ||||
|         if (!plotInventory.isOpen()) { | ||||
|             return false; | ||||
|         } | ||||
|         BukkitPlayer bp = (BukkitPlayer) plotInventory.player; | ||||
|         BukkitPlayer bp = (BukkitPlayer) plotInventory.getPlayer(); | ||||
|         InventoryView opened = bp.player.getOpenInventory(); | ||||
|         if (plotInventory.isOpen()) { | ||||
|             if (opened.getType() == InventoryType.CRAFTING) { | ||||
| @@ -149,4 +162,5 @@ public class BukkitInventoryUtil extends InventoryUtil { | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,33 +21,32 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import com.plotsquared.bukkit.BukkitMain; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.google.inject.Inject; | ||||
| import com.google.inject.Singleton; | ||||
| import com.plotsquared.core.generator.AugmentedUtils; | ||||
| import com.plotsquared.core.inject.factory.ProgressSubscriberFactory; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.location.PlotLoc; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.PlotManager; | ||||
| import com.plotsquared.core.queue.GlobalBlockQueue; | ||||
| import com.plotsquared.core.queue.LocalBlockQueue; | ||||
| import com.plotsquared.core.queue.ScopedLocalBlockQueue; | ||||
| import com.plotsquared.core.queue.QueueCoordinator; | ||||
| import com.plotsquared.core.queue.ScopedQueueCoordinator; | ||||
| import com.plotsquared.core.util.ChunkManager; | ||||
| import com.plotsquared.core.util.MainUtil; | ||||
| import com.plotsquared.core.util.RegionManager; | ||||
| import com.plotsquared.core.util.RegionUtil; | ||||
| import com.plotsquared.core.util.WorldUtil; | ||||
| import com.plotsquared.core.util.entity.EntityCategories; | ||||
| import com.plotsquared.core.util.task.RunnableVal; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.bukkit.BukkitWorld; | ||||
| import com.sk89q.worldedit.math.BlockVector2; | ||||
| import com.sk89q.worldedit.regions.CuboidRegion; | ||||
| import com.sk89q.worldedit.world.biome.BiomeType; | ||||
| import com.sk89q.worldedit.world.block.BaseBlock; | ||||
| import com.sk89q.worldedit.world.block.BlockTypes; | ||||
| import io.papermc.lib.PaperLib; | ||||
| @@ -56,17 +55,14 @@ import org.bukkit.Chunk; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.entity.Entity; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| import java.util.ArrayDeque; | ||||
| import java.util.ArrayList; | ||||
| import java.util.HashSet; | ||||
| import java.util.List; | ||||
| import java.util.Map.Entry; | ||||
| import java.util.Objects; | ||||
| import java.util.Set; | ||||
| import java.util.concurrent.Semaphore; | ||||
|  | ||||
| import static com.google.common.base.Preconditions.checkNotNull; | ||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_ANIMAL; | ||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_ENTITY; | ||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_MISC; | ||||
| @@ -74,55 +70,38 @@ import static com.plotsquared.core.util.entity.EntityCategories.CAP_MOB; | ||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_MONSTER; | ||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_VEHICLE; | ||||
|  | ||||
| @Singleton | ||||
| public class BukkitRegionManager extends RegionManager { | ||||
|  | ||||
|     public static boolean isIn(CuboidRegion region, int x, int z) { | ||||
|         return x >= region.getMinimumPoint().getX() && x <= region.getMaximumPoint().getX() | ||||
|             && z >= region.getMinimumPoint().getZ() && z <= region.getMaximumPoint().getZ(); | ||||
|     private final GlobalBlockQueue blockQueue; | ||||
|  | ||||
|     @Inject | ||||
|     public BukkitRegionManager( | ||||
|             @NonNull WorldUtil worldUtil, @NonNull GlobalBlockQueue blockQueue, @NonNull | ||||
|             ProgressSubscriberFactory subscriberFactory | ||||
|     ) { | ||||
|         super(worldUtil, blockQueue, subscriberFactory); | ||||
|         this.blockQueue = blockQueue; | ||||
|     } | ||||
|  | ||||
|     @Override public Set<BlockVector2> getChunkChunks(String world) { | ||||
|         Set<BlockVector2> chunks = super.getChunkChunks(world); | ||||
|         if (Bukkit.isPrimaryThread()) { | ||||
|             for (Chunk chunk : Objects.requireNonNull(Bukkit.getWorld(world)).getLoadedChunks()) { | ||||
|                 BlockVector2 loc = BlockVector2.at(chunk.getX() >> 5, chunk.getZ() >> 5); | ||||
|                 chunks.add(loc); | ||||
|             } | ||||
|         } else { | ||||
|             final Semaphore semaphore = new Semaphore(1); | ||||
|             try { | ||||
|                 PlotSquared.debug("Attempting to make an asynchronous call to getLoadedChunks." | ||||
|                     + " Will halt the calling thread until completed."); | ||||
|                 semaphore.acquire(); | ||||
|                 Bukkit.getScheduler().runTask(BukkitMain.getPlugin(BukkitMain.class), () -> { | ||||
|                     for (Chunk chunk : Objects.requireNonNull(Bukkit.getWorld(world)) | ||||
|                         .getLoadedChunks()) { | ||||
|                         BlockVector2 loc = BlockVector2.at(chunk.getX() >> 5, chunk.getZ() >> 5); | ||||
|                         chunks.add(loc); | ||||
|                     } | ||||
|                     semaphore.release(); | ||||
|                 }); | ||||
|                 semaphore.acquireUninterruptibly(); | ||||
|             } catch (final Exception e) { | ||||
|                 e.printStackTrace(); | ||||
|             } | ||||
|         } | ||||
|         return chunks; | ||||
|     } | ||||
|  | ||||
|     @Override public boolean handleClear(Plot plot, Runnable whenDone, PlotManager manager) { | ||||
|     @Override | ||||
|     public boolean handleClear( | ||||
|             @NonNull Plot plot, | ||||
|             @Nullable Runnable whenDone, | ||||
|             @NonNull PlotManager manager, | ||||
|             @Nullable PlotPlayer<?> player | ||||
|     ) { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     @Override public int[] countEntities(Plot plot) { | ||||
|     @Override | ||||
|     public int[] countEntities(@NonNull Plot plot) { | ||||
|         int[] existing = (int[]) plot.getMeta("EntityCount"); | ||||
|         if (existing != null && (System.currentTimeMillis() - (long) plot.getMeta("EntityCountTime") | ||||
|             < 1000)) { | ||||
|         if (existing != null && (System.currentTimeMillis() - (long) plot.getMeta("EntityCountTime") < 1000)) { | ||||
|             return existing; | ||||
|         } | ||||
|         PlotArea area = plot.getArea(); | ||||
|         World world = BukkitUtil.getWorld(area.getWorldName()); | ||||
|  | ||||
|         Location bot = plot.getBottomAbs(); | ||||
|         Location top = plot.getTopAbs(); | ||||
|         int bx = bot.getX() >> 4; | ||||
| @@ -162,7 +141,7 @@ public class BukkitRegionManager extends RegionManager { | ||||
|                         if (X > bx && X < tx && Z > bz && Z < tz) { | ||||
|                             count(count, entity); | ||||
|                         } else { | ||||
|                             Plot other = area.getPlot(BukkitUtil.getLocation(location)); | ||||
|                             Plot other = area.getPlot(BukkitUtil.adapt(location)); | ||||
|                             if (plot.equals(other)) { | ||||
|                                 count(count, entity); | ||||
|                             } | ||||
| @@ -177,7 +156,7 @@ public class BukkitRegionManager extends RegionManager { | ||||
|                 Entity[] entities1 = chunk.getEntities(); | ||||
|                 for (Entity entity : entities1) { | ||||
|                     if (X == bx || X == tx || Z == bz || Z == tz) { | ||||
|                         Plot other = area.getPlot(BukkitUtil.getLocation(entity)); | ||||
|                         Plot other = area.getPlot(BukkitUtil.adapt(entity.getLocation())); | ||||
|                         if (plot.equals(other)) { | ||||
|                             count(count, entity); | ||||
|                         } | ||||
| @@ -191,64 +170,13 @@ public class BukkitRegionManager extends RegionManager { | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean copyRegion(Location pos1, Location pos2, Location newPos, | ||||
|         final Runnable whenDone) { | ||||
|         final int relX = newPos.getX() - pos1.getX(); | ||||
|         final int relZ = newPos.getZ() - pos1.getZ(); | ||||
|  | ||||
|         final CuboidRegion region = | ||||
|             RegionUtil.createRegion(pos1.getX(), pos2.getX(), pos1.getZ(), pos2.getZ()); | ||||
|         final World oldWorld = Bukkit.getWorld(pos1.getWorld()); | ||||
|         final BukkitWorld oldBukkitWorld = new BukkitWorld(oldWorld); | ||||
|         final World newWorld = Bukkit.getWorld(newPos.getWorld()); | ||||
|         assert newWorld != null; | ||||
|         assert oldWorld != null; | ||||
|         final String newWorldName = newWorld.getName(); | ||||
|         final ContentMap map = new ContentMap(); | ||||
|         final LocalBlockQueue queue = GlobalBlockQueue.IMP.getNewQueue(newWorldName, false); | ||||
|         ChunkManager.chunkTask(pos1, pos2, new RunnableVal<int[]>() { | ||||
|             @Override public void run(int[] value) { | ||||
|                 int bx = value[2]; | ||||
|                 int bz = value[3]; | ||||
|                 int tx = value[4]; | ||||
|                 int tz = value[5]; | ||||
|                 BlockVector2 loc = BlockVector2.at(value[0], value[1]); | ||||
|                 int cxx = loc.getX() << 4; | ||||
|                 int czz = loc.getZ() << 4; | ||||
|                 PaperLib.getChunkAtAsync(oldWorld, loc.getX(), loc.getZ()) | ||||
|                     .thenAccept(chunk1 -> map.saveEntitiesIn(chunk1, region)).thenRun(() -> { | ||||
|                     for (int x = bx & 15; x <= (tx & 15); x++) { | ||||
|                         for (int z = bz & 15; z <= (tz & 15); z++) { | ||||
|                             map.saveBlocks(oldBukkitWorld, 256, cxx + x, czz + z, relX, relZ); | ||||
|                         } | ||||
|                     } | ||||
|                 }); | ||||
|             } | ||||
|         }, () -> { | ||||
|             for (Entry<PlotLoc, BaseBlock[]> entry : map.allBlocks.entrySet()) { | ||||
|                 PlotLoc loc = entry.getKey(); | ||||
|                 BaseBlock[] blocks = entry.getValue(); | ||||
|                 for (int y = 0; y < blocks.length; y++) { | ||||
|                     if (blocks[y] != null) { | ||||
|                         BaseBlock block = blocks[y]; | ||||
|                         queue.setBlock(loc.getX(), y, loc.getZ(), block); | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             queue.enqueue(); | ||||
|             GlobalBlockQueue.IMP.addEmptyTask(() -> { | ||||
|                 //map.restoreBlocks(newWorld, 0, 0); | ||||
|                 map.restoreEntities(newWorld, relX, relZ); | ||||
|                 TaskManager.runTask(whenDone); | ||||
|             }); | ||||
|         }, 5); | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean regenerateRegion(final Location pos1, final Location pos2, | ||||
|         final boolean ignoreAugment, final Runnable whenDone) { | ||||
|         final String world = pos1.getWorld(); | ||||
|     public boolean regenerateRegion( | ||||
|             final @NonNull Location pos1, | ||||
|             final @NonNull Location pos2, | ||||
|             final boolean ignoreAugment, | ||||
|             final @Nullable Runnable whenDone | ||||
|     ) { | ||||
|         final BukkitWorld world = (BukkitWorld) worldUtil.getWeWorld(pos1.getWorldName()); | ||||
|  | ||||
|         final int p1x = pos1.getX(); | ||||
|         final int p1z = pos1.getZ(); | ||||
| @@ -259,150 +187,136 @@ public class BukkitRegionManager extends RegionManager { | ||||
|         final int tcx = p2x >> 4; | ||||
|         final int tcz = p2z >> 4; | ||||
|  | ||||
|         final List<BlockVector2> chunks = new ArrayList<>(); | ||||
|         final QueueCoordinator queue = blockQueue.getNewQueue(world); | ||||
|         final QueueCoordinator regenQueue = blockQueue.getNewQueue(world); | ||||
|         queue.addReadChunks(new CuboidRegion(pos1.getBlockVector3(), pos2.getBlockVector3()).getChunks()); | ||||
|         queue.setChunkConsumer(chunk -> { | ||||
|  | ||||
|         for (int x = bcx; x <= tcx; x++) { | ||||
|             for (int z = bcz; z <= tcz; z++) { | ||||
|                 chunks.add(BlockVector2.at(x, z)); | ||||
|             int x = chunk.getX(); | ||||
|             int z = chunk.getZ(); | ||||
|             int xxb = x << 4; | ||||
|             int zzb = z << 4; | ||||
|             int xxt = xxb + 15; | ||||
|             int zzt = zzb + 15; | ||||
|             if (xxb >= p1x && xxt <= p2x && zzb >= p1z && zzt <= p2z) { | ||||
|                 AugmentedUtils.bypass(ignoreAugment, () -> regenQueue.regenChunk(chunk.getX(), chunk.getZ())); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|         final World worldObj = Bukkit.getWorld(world); | ||||
|         checkNotNull(worldObj, "Critical error during regeneration."); | ||||
|         final BukkitWorld bukkitWorldObj = new BukkitWorld(worldObj); | ||||
|         TaskManager.runTask(new Runnable() { | ||||
|             @Override public void run() { | ||||
|                 long start = System.currentTimeMillis(); | ||||
|                 while (!chunks.isEmpty() && System.currentTimeMillis() - start < 5) { | ||||
|                     final BlockVector2 chunk = chunks.remove(0); | ||||
|                     int x = chunk.getX(); | ||||
|                     int z = chunk.getZ(); | ||||
|                     int xxb = x << 4; | ||||
|                     int zzb = z << 4; | ||||
|                     int xxt = xxb + 15; | ||||
|                     int zzt = zzb + 15; | ||||
|                     PaperLib.getChunkAtAsync(worldObj, x, z, false).thenAccept(chunkObj -> { | ||||
|                         if (chunkObj == null) { | ||||
|                             return; | ||||
|                         } | ||||
|                         final LocalBlockQueue queue = | ||||
|                             GlobalBlockQueue.IMP.getNewQueue(world, false); | ||||
|                         if (xxb >= p1x && xxt <= p2x && zzb >= p1z && zzt <= p2z) { | ||||
|                             AugmentedUtils.bypass(ignoreAugment, | ||||
|                                 () -> queue.regenChunkSafe(chunk.getX(), chunk.getZ())); | ||||
|                             return; | ||||
|                         } | ||||
|                         boolean checkX1 = false; | ||||
|             boolean checkX1 = false; | ||||
|  | ||||
|                         int xxb2; | ||||
|             int xxb2; | ||||
|  | ||||
|                         if (x == bcx) { | ||||
|                             xxb2 = p1x - 1; | ||||
|                             checkX1 = true; | ||||
|                         } else { | ||||
|                             xxb2 = xxb; | ||||
|                         } | ||||
|                         boolean checkX2 = false; | ||||
|                         int xxt2; | ||||
|                         if (x == tcx) { | ||||
|                             xxt2 = p2x + 1; | ||||
|                             checkX2 = true; | ||||
|                         } else { | ||||
|                             xxt2 = xxt; | ||||
|                         } | ||||
|                         boolean checkZ1 = false; | ||||
|                         int zzb2; | ||||
|                         if (z == bcz) { | ||||
|                             zzb2 = p1z - 1; | ||||
|                             checkZ1 = true; | ||||
|                         } else { | ||||
|                             zzb2 = zzb; | ||||
|                         } | ||||
|                         boolean checkZ2 = false; | ||||
|                         int zzt2; | ||||
|                         if (z == tcz) { | ||||
|                             zzt2 = p2z + 1; | ||||
|                             checkZ2 = true; | ||||
|                         } else { | ||||
|                             zzt2 = zzt; | ||||
|                         } | ||||
|                         final ContentMap map = new ContentMap(); | ||||
|                         if (checkX1) { | ||||
|                             map.saveRegion(bukkitWorldObj, xxb, xxb2, zzb2, zzt2); // | ||||
|                         } | ||||
|                         if (checkX2) { | ||||
|                             map.saveRegion(bukkitWorldObj, xxt2, xxt, zzb2, zzt2); // | ||||
|                         } | ||||
|                         if (checkZ1) { | ||||
|                             map.saveRegion(bukkitWorldObj, xxb2, xxt2, zzb, zzb2); // | ||||
|                         } | ||||
|                         if (checkZ2) { | ||||
|                             map.saveRegion(bukkitWorldObj, xxb2, xxt2, zzt2, zzt); // | ||||
|                         } | ||||
|                         if (checkX1 && checkZ1) { | ||||
|                             map.saveRegion(bukkitWorldObj, xxb, xxb2, zzb, zzb2); // | ||||
|                         } | ||||
|                         if (checkX2 && checkZ1) { | ||||
|                             map.saveRegion(bukkitWorldObj, xxt2, xxt, zzb, zzb2); // ? | ||||
|                         } | ||||
|                         if (checkX1 && checkZ2) { | ||||
|                             map.saveRegion(bukkitWorldObj, xxb, xxb2, zzt2, zzt); // ? | ||||
|                         } | ||||
|                         if (checkX2 && checkZ2) { | ||||
|                             map.saveRegion(bukkitWorldObj, xxt2, xxt, zzt2, zzt); // | ||||
|                         } | ||||
|                         CuboidRegion currentPlotClear = RegionUtil | ||||
|                             .createRegion(pos1.getX(), pos2.getX(), pos1.getZ(), pos2.getZ()); | ||||
|                         map.saveEntitiesOut(chunkObj, currentPlotClear); | ||||
|                         AugmentedUtils.bypass(ignoreAugment, () -> ChunkManager | ||||
|                             .setChunkInPlotArea(null, new RunnableVal<ScopedLocalBlockQueue>() { | ||||
|                                 @Override public void run(ScopedLocalBlockQueue value) { | ||||
|                                     Location min = value.getMin(); | ||||
|                                     int bx = min.getX(); | ||||
|                                     int bz = min.getZ(); | ||||
|                                     for (int x1 = 0; x1 < 16; x1++) { | ||||
|                                         for (int z1 = 0; z1 < 16; z1++) { | ||||
|                                             PlotLoc plotLoc = new PlotLoc(bx + x1, bz + z1); | ||||
|                                             BaseBlock[] ids = map.allBlocks.get(plotLoc); | ||||
|                                             if (ids != null) { | ||||
|                                                 for (int y = 0; | ||||
|                                                      y < Math.min(128, ids.length); y++) { | ||||
|                                                     BaseBlock id = ids[y]; | ||||
|                                                     if (id != null) { | ||||
|                                                         value.setBlock(x1, y, z1, id); | ||||
|                                                     } else { | ||||
|                                                         value.setBlock(x1, y, z1, | ||||
|                                                             BlockTypes.AIR.getDefaultState()); | ||||
|                                                     } | ||||
|                                                 } | ||||
|                                                 for (int y = Math.min(128, ids.length); | ||||
|                                                      y < ids.length; y++) { | ||||
|                                                     BaseBlock id = ids[y]; | ||||
|                                                     if (id != null) { | ||||
|                                                         value.setBlock(x1, y, z1, id); | ||||
|                                                     } | ||||
|                                                 } | ||||
|             if (x == bcx) { | ||||
|                 xxb2 = p1x - 1; | ||||
|                 checkX1 = true; | ||||
|             } else { | ||||
|                 xxb2 = xxb; | ||||
|             } | ||||
|             boolean checkX2 = false; | ||||
|             int xxt2; | ||||
|             if (x == tcx) { | ||||
|                 xxt2 = p2x + 1; | ||||
|                 checkX2 = true; | ||||
|             } else { | ||||
|                 xxt2 = xxt; | ||||
|             } | ||||
|             boolean checkZ1 = false; | ||||
|             int zzb2; | ||||
|             if (z == bcz) { | ||||
|                 zzb2 = p1z - 1; | ||||
|                 checkZ1 = true; | ||||
|             } else { | ||||
|                 zzb2 = zzb; | ||||
|             } | ||||
|             boolean checkZ2 = false; | ||||
|             int zzt2; | ||||
|             if (z == tcz) { | ||||
|                 zzt2 = p2z + 1; | ||||
|                 checkZ2 = true; | ||||
|             } else { | ||||
|                 zzt2 = zzt; | ||||
|             } | ||||
|             final ContentMap map = new ContentMap(); | ||||
|             if (checkX1) { | ||||
|                 map.saveRegion(world, xxb, xxb2, zzb2, zzt2); // | ||||
|             } | ||||
|             if (checkX2) { | ||||
|                 map.saveRegion(world, xxt2, xxt, zzb2, zzt2); // | ||||
|             } | ||||
|             if (checkZ1) { | ||||
|                 map.saveRegion(world, xxb2, xxt2, zzb, zzb2); // | ||||
|             } | ||||
|             if (checkZ2) { | ||||
|                 map.saveRegion(world, xxb2, xxt2, zzt2, zzt); // | ||||
|             } | ||||
|             if (checkX1 && checkZ1) { | ||||
|                 map.saveRegion(world, xxb, xxb2, zzb, zzb2); // | ||||
|             } | ||||
|             if (checkX2 && checkZ1) { | ||||
|                 map.saveRegion(world, xxt2, xxt, zzb, zzb2); // ? | ||||
|             } | ||||
|             if (checkX1 && checkZ2) { | ||||
|                 map.saveRegion(world, xxb, xxb2, zzt2, zzt); // ? | ||||
|             } | ||||
|             if (checkX2 && checkZ2) { | ||||
|                 map.saveRegion(world, xxt2, xxt, zzt2, zzt); // | ||||
|             } | ||||
|             CuboidRegion currentPlotClear = RegionUtil.createRegion(pos1.getX(), pos2.getX(), pos1.getZ(), pos2.getZ()); | ||||
|             map.saveEntitiesOut(Bukkit.getWorld(world.getName()).getChunkAt(x, z), currentPlotClear); | ||||
|             AugmentedUtils.bypass( | ||||
|                     ignoreAugment, | ||||
|                     () -> ChunkManager.setChunkInPlotArea(null, new RunnableVal<ScopedQueueCoordinator>() { | ||||
|                         @Override | ||||
|                         public void run(ScopedQueueCoordinator value) { | ||||
|                             Location min = value.getMin(); | ||||
|                             int bx = min.getX(); | ||||
|                             int bz = min.getZ(); | ||||
|                             for (int x1 = 0; x1 < 16; x1++) { | ||||
|                                 for (int z1 = 0; z1 < 16; z1++) { | ||||
|                                     PlotLoc plotLoc = new PlotLoc(bx + x1, bz + z1); | ||||
|                                     BaseBlock[] ids = map.allBlocks.get(plotLoc); | ||||
|                                     if (ids != null) { | ||||
|                                         for (int y = 0; y < Math.min(128, ids.length); y++) { | ||||
|                                             BaseBlock id = ids[y]; | ||||
|                                             if (id != null) { | ||||
|                                                 value.setBlock(x1, y, z1, id); | ||||
|                                             } else { | ||||
|                                                 value.setBlock(x1, y, z1, BlockTypes.AIR.getDefaultState()); | ||||
|                                             } | ||||
|                                         } | ||||
|                                         for (int y = Math.min(128, ids.length); y < ids.length; y++) { | ||||
|                                             BaseBlock id = ids[y]; | ||||
|                                             if (id != null) { | ||||
|                                                 value.setBlock(x1, y, z1, id); | ||||
|                                             } | ||||
|                                         } | ||||
|                                     } | ||||
|                                 } | ||||
|                             }, world, chunk)); | ||||
|                         //map.restoreBlocks(worldObj, 0, 0); | ||||
|                         map.restoreEntities(worldObj, 0, 0); | ||||
|                     }); | ||||
|                 } | ||||
|                 if (!chunks.isEmpty()) { | ||||
|                     TaskManager.runTaskLater(this, 1); | ||||
|                 } else { | ||||
|                     TaskManager.runTaskLater(whenDone, 1); | ||||
|                 } | ||||
|             } | ||||
|                             } | ||||
|                         } | ||||
|                     }, world.getName(), chunk) | ||||
|             ); | ||||
|             //map.restoreBlocks(worldObj, 0, 0); | ||||
|             map.restoreEntities(Bukkit.getWorld(world.getName()), 0, 0); | ||||
|         }); | ||||
|         regenQueue.setCompleteTask(whenDone); | ||||
|         queue.setCompleteTask(regenQueue::enqueue); | ||||
|         queue.enqueue(); | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     @Override public void clearAllEntities(Location pos1, Location pos2) { | ||||
|         String world = pos1.getWorld(); | ||||
|         List<Entity> entities = BukkitUtil.getEntities(world); | ||||
|     @Override | ||||
|     public void clearAllEntities(@NonNull Location pos1, @NonNull Location pos2) { | ||||
|         String world = pos1.getWorldName(); | ||||
|  | ||||
|         final World bukkitWorld = BukkitUtil.getWorld(world); | ||||
|         final List<Entity> entities; | ||||
|         if (bukkitWorld != null) { | ||||
|             entities = new ArrayList<>(bukkitWorld.getEntities()); | ||||
|         } else { | ||||
|             entities = new ArrayList<>(); | ||||
|         } | ||||
|  | ||||
|         int bx = pos1.getX(); | ||||
|         int bz = pos1.getZ(); | ||||
|         int tx = pos2.getX(); | ||||
| @@ -410,8 +324,7 @@ public class BukkitRegionManager extends RegionManager { | ||||
|         for (Entity entity : entities) { | ||||
|             if (!(entity instanceof Player)) { | ||||
|                 org.bukkit.Location location = entity.getLocation(); | ||||
|                 if (location.getX() >= bx && location.getX() <= tx && location.getZ() >= bz | ||||
|                     && location.getZ() <= tz) { | ||||
|                 if (location.getX() >= bx && location.getX() <= tx && location.getZ() >= bz && location.getZ() <= tz) { | ||||
|                     if (entity.hasMetadata("ps-tmp-teleport")) { | ||||
|                         continue; | ||||
|                     } | ||||
| @@ -421,69 +334,16 @@ public class BukkitRegionManager extends RegionManager { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void swap(Location bot1, Location top1, Location bot2, Location top2, | ||||
|         final Runnable whenDone) { | ||||
|         CuboidRegion region1 = | ||||
|             RegionUtil.createRegion(bot1.getX(), top1.getX(), bot1.getZ(), top1.getZ()); | ||||
|         CuboidRegion region2 = | ||||
|             RegionUtil.createRegion(bot2.getX(), top2.getX(), bot2.getZ(), top2.getZ()); | ||||
|         final World world1 = Bukkit.getWorld(bot1.getWorld()); | ||||
|         final World world2 = Bukkit.getWorld(bot2.getWorld()); | ||||
|         checkNotNull(world1, "Critical error during swap."); | ||||
|         checkNotNull(world2, "Critical error during swap."); | ||||
|         int relX = bot2.getX() - bot1.getX(); | ||||
|         int relZ = bot2.getZ() - bot1.getZ(); | ||||
|  | ||||
|         final ArrayDeque<ContentMap> maps = new ArrayDeque<>(); | ||||
|  | ||||
|         for (int x = bot1.getX() >> 4; x <= top1.getX() >> 4; x++) { | ||||
|             for (int z = bot1.getZ() >> 4; z <= top1.getZ() >> 4; z++) { | ||||
|                 Chunk chunk1 = world1.getChunkAt(x, z); | ||||
|                 Chunk chunk2 = world2.getChunkAt(x + (relX >> 4), z + (relZ >> 4)); | ||||
|                 maps.add( | ||||
|                     BukkitChunkManager.swapChunk(world1, world2, chunk1, chunk2, region1, region2)); | ||||
|             } | ||||
|         } | ||||
|         GlobalBlockQueue.IMP.addEmptyTask(() -> { | ||||
|             for (ContentMap map : maps) { | ||||
|                 map.restoreEntities(world1, 0, 0); | ||||
|                 TaskManager.runTaskLater(whenDone, 1); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void setBiome(final CuboidRegion region, final int extendBiome, final BiomeType biome, | ||||
|         final String world, final Runnable whenDone) { | ||||
|         Location pos1 = new Location(world, region.getMinimumPoint().getX() - extendBiome, | ||||
|             region.getMinimumPoint().getY(), region.getMinimumPoint().getZ() - extendBiome); | ||||
|         Location pos2 = new Location(world, region.getMaximumPoint().getX() + extendBiome, | ||||
|             region.getMaximumPoint().getY(), region.getMaximumPoint().getZ() + extendBiome); | ||||
|         final LocalBlockQueue queue = GlobalBlockQueue.IMP.getNewQueue(world, false); | ||||
|  | ||||
|         ChunkManager.chunkTask(pos1, pos2, new RunnableVal<int[]>() { | ||||
|             @Override public void run(int[] value) { | ||||
|                 BlockVector2 loc = BlockVector2.at(value[0], value[1]); | ||||
|                 ChunkManager.manager.loadChunk(world, loc, false).thenRun(() -> { | ||||
|                     MainUtil.setBiome(world, value[2], value[3], value[4], value[5], biome); | ||||
|                     queue.refreshChunk(value[0], value[1]); | ||||
|                 }); | ||||
|             } | ||||
|         }, whenDone, 5); | ||||
|     } | ||||
|  | ||||
|     private void count(int[] count, Entity entity) { | ||||
|         final com.sk89q.worldedit.world.entity.EntityType entityType = | ||||
|             BukkitAdapter.adapt(entity.getType()); | ||||
|     private void count(int[] count, @NonNull Entity entity) { | ||||
|         final com.sk89q.worldedit.world.entity.EntityType entityType = BukkitAdapter.adapt(entity.getType()); | ||||
|  | ||||
|         if (EntityCategories.PLAYER.contains(entityType)) { | ||||
|             return; | ||||
|         } else if (EntityCategories.PROJECTILE.contains(entityType) || EntityCategories.OTHER | ||||
|             .contains(entityType) || EntityCategories.HANGING.contains(entityType)) { | ||||
|         } else if (EntityCategories.PROJECTILE.contains(entityType) || EntityCategories.OTHER.contains(entityType) || EntityCategories.HANGING | ||||
|                 .contains(entityType)) { | ||||
|             count[CAP_MISC]++; | ||||
|         } else if (EntityCategories.ANIMAL.contains(entityType) || EntityCategories.VILLAGER | ||||
|             .contains(entityType) || EntityCategories.TAMEABLE.contains(entityType)) { | ||||
|         } else if (EntityCategories.ANIMAL.contains(entityType) || EntityCategories.VILLAGER.contains(entityType) || EntityCategories.TAMEABLE | ||||
|                 .contains(entityType)) { | ||||
|             count[CAP_MOB]++; | ||||
|             count[CAP_ANIMAL]++; | ||||
|         } else if (EntityCategories.VEHICLE.contains(entityType)) { | ||||
| @@ -494,4 +354,5 @@ public class BukkitRegionManager extends RegionManager { | ||||
|         } | ||||
|         count[CAP_ENTITY]++; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,37 +21,62 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import com.google.inject.Inject; | ||||
| import com.google.inject.Singleton; | ||||
| import com.plotsquared.bukkit.generator.BukkitPlotGenerator; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.ConfigurationNode; | ||||
| import com.plotsquared.core.configuration.ConfigurationSection; | ||||
| import com.plotsquared.core.configuration.file.YamlConfiguration; | ||||
| import com.plotsquared.core.generator.GeneratorWrapper; | ||||
| import com.plotsquared.core.inject.annotations.WorldConfig; | ||||
| import com.plotsquared.core.inject.annotations.WorldFile; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.PlotAreaType; | ||||
| import com.plotsquared.core.plot.SetupObject; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.setup.PlotAreaBuilder; | ||||
| import com.plotsquared.core.util.SetupUtils; | ||||
| import io.papermc.lib.PaperLib; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.Location; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.generator.ChunkGenerator; | ||||
| import org.bukkit.plugin.Plugin; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.io.File; | ||||
| import java.io.IOException; | ||||
| import java.util.HashMap; | ||||
| import java.util.Map.Entry; | ||||
| import java.util.Objects; | ||||
|  | ||||
| @Singleton | ||||
| public class BukkitSetupUtils extends SetupUtils { | ||||
|  | ||||
|     @Override public void updateGenerators() { | ||||
|         if (!SetupUtils.generators.isEmpty()) { | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|     private final YamlConfiguration worldConfiguration; | ||||
|     private final File worldFile; | ||||
|  | ||||
|     @Inject | ||||
|     public BukkitSetupUtils( | ||||
|             final @NonNull PlotAreaManager plotAreaManager, | ||||
|             @WorldConfig final @NonNull YamlConfiguration worldConfiguration, | ||||
|             @WorldFile final @NonNull File worldFile | ||||
|     ) { | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|         this.worldConfiguration = worldConfiguration; | ||||
|         this.worldFile = worldFile; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void updateGenerators(final boolean force) { | ||||
|         if (!SetupUtils.generators.isEmpty() && !force) { | ||||
|             return; | ||||
|         } | ||||
|         String testWorld = "CheckingPlotSquaredGenerator"; | ||||
| @@ -66,7 +91,7 @@ public class BukkitSetupUtils extends SetupUtils { | ||||
|                         if (generator instanceof GeneratorWrapper<?>) { | ||||
|                             wrapped = (GeneratorWrapper<?>) generator; | ||||
|                         } else { | ||||
|                             wrapped = new BukkitPlotGenerator(testWorld, generator); | ||||
|                             wrapped = new BukkitPlotGenerator(testWorld, generator, this.plotAreaManager); | ||||
|                         } | ||||
|                         SetupUtils.generators.put(name, wrapped); | ||||
|                     } | ||||
| @@ -77,154 +102,46 @@ public class BukkitSetupUtils extends SetupUtils { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public void unload(String worldName, boolean save) { | ||||
|         World world = Bukkit.getWorld(worldName); | ||||
|         if (world == null) { | ||||
|             return; | ||||
|         } | ||||
|         World dw = Bukkit.getWorlds().get(0); | ||||
|         for (Player player : world.getPlayers()) { | ||||
|             PaperLib.teleportAsync(player, dw.getSpawnLocation()); | ||||
|         } | ||||
|         if (save) { | ||||
|             for (Chunk chunk : world.getLoadedChunks()) { | ||||
|                 chunk.unload(true); | ||||
|     @Override | ||||
|     public void unload(String worldName, boolean save) { | ||||
|         TaskManager.runTask(() -> { | ||||
|             World world = Bukkit.getWorld(worldName); | ||||
|             if (world == null) { | ||||
|                 return; | ||||
|             } | ||||
|         } else { | ||||
|             for (Chunk chunk : world.getLoadedChunks()) { | ||||
|                 chunk.unload(false); | ||||
|             Location location = Bukkit.getWorlds().get(0).getSpawnLocation(); | ||||
|             for (Player player : world.getPlayers()) { | ||||
|                 player.teleport(location); | ||||
|             } | ||||
|         } | ||||
|         Bukkit.unloadWorld(world, false); | ||||
|             if (save) { | ||||
|                 for (Chunk chunk : world.getLoadedChunks()) { | ||||
|                     chunk.unload(true); | ||||
|                 } | ||||
|             } else { | ||||
|                 for (Chunk chunk : world.getLoadedChunks()) { | ||||
|                     chunk.unload(false); | ||||
|                 } | ||||
|             } | ||||
|             Bukkit.unloadWorld(world, false); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     @Deprecated @Override public String setupWorld(SetupObject object) { | ||||
|         SetupUtils.manager.updateGenerators(); | ||||
|         ConfigurationNode[] steps = object.step == null ? new ConfigurationNode[0] : object.step; | ||||
|         String world = object.world; | ||||
|         PlotAreaType type = object.type; | ||||
|         String worldPath = "worlds." + object.world; | ||||
|         switch (type) { | ||||
|             case PARTIAL: { | ||||
|                 if (object.id != null) { | ||||
|                     if (!PlotSquared.get().worlds.contains(worldPath)) { | ||||
|                         PlotSquared.get().worlds.createSection(worldPath); | ||||
|                     } | ||||
|                     ConfigurationSection worldSection = | ||||
|                         PlotSquared.get().worlds.getConfigurationSection(worldPath); | ||||
|                     String areaName = object.id + "-" + object.min + "-" + object.max; | ||||
|                     String areaPath = "areas." + areaName; | ||||
|                     if (!worldSection.contains(areaPath)) { | ||||
|                         worldSection.createSection(areaPath); | ||||
|                     } | ||||
|                     ConfigurationSection areaSection = | ||||
|                         worldSection.getConfigurationSection(areaPath); | ||||
|                     HashMap<String, Object> options = new HashMap<>(); | ||||
|                     for (ConfigurationNode step : steps) { | ||||
|                         options.put(step.getConstant(), step.getValue()); | ||||
|                     } | ||||
|                     options.put("generator.type", object.type.toString()); | ||||
|                     options.put("generator.terrain", object.terrain.toString()); | ||||
|                     options.put("generator.plugin", object.plotManager); | ||||
|                     if (object.setupGenerator != null && !object.setupGenerator | ||||
|                         .equals(object.plotManager)) { | ||||
|                         options.put("generator.init", object.setupGenerator); | ||||
|                     } | ||||
|                     for (Entry<String, Object> entry : options.entrySet()) { | ||||
|                         String key = entry.getKey(); | ||||
|                         Object value = entry.getValue(); | ||||
|                         if (worldSection.contains(key)) { | ||||
|                             Object current = worldSection.get(key); | ||||
|                             if (!Objects.equals(value, current)) { | ||||
|                                 areaSection.set(key, value); | ||||
|                             } | ||||
|                         } else { | ||||
|                             worldSection.set(key, value); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|                 GeneratorWrapper<?> gen = SetupUtils.generators.get(object.setupGenerator); | ||||
|                 if (gen != null && gen.isFull()) { | ||||
|                     object.setupGenerator = null; | ||||
|                 } | ||||
|                 break; | ||||
|             } | ||||
|             case AUGMENTED: { | ||||
|                 if (!object.plotManager.endsWith(":single")) { | ||||
|                     if (!PlotSquared.get().worlds.contains(worldPath)) { | ||||
|                         PlotSquared.get().worlds.createSection(worldPath); | ||||
|                     } | ||||
|                     if (steps.length != 0) { | ||||
|                         ConfigurationSection worldSection = | ||||
|                             PlotSquared.get().worlds.getConfigurationSection(worldPath); | ||||
|                         for (ConfigurationNode step : steps) { | ||||
|                             worldSection.set(step.getConstant(), step.getValue()); | ||||
|                         } | ||||
|                     } | ||||
|                     PlotSquared.get().worlds | ||||
|                         .set("worlds." + world + ".generator.type", object.type.toString()); | ||||
|                     PlotSquared.get().worlds | ||||
|                         .set("worlds." + world + ".generator.terrain", object.terrain.toString()); | ||||
|                     PlotSquared.get().worlds | ||||
|                         .set("worlds." + world + ".generator.plugin", object.plotManager); | ||||
|                     if (object.setupGenerator != null && !object.setupGenerator | ||||
|                         .equals(object.plotManager)) { | ||||
|                         PlotSquared.get().worlds | ||||
|                             .set("worlds." + world + ".generator.init", object.setupGenerator); | ||||
|                     } | ||||
|                 } | ||||
|                 GeneratorWrapper<?> gen = SetupUtils.generators.get(object.setupGenerator); | ||||
|                 if (gen != null && gen.isFull()) { | ||||
|                     object.setupGenerator = null; | ||||
|                 } | ||||
|                 break; | ||||
|             } | ||||
|             case NORMAL: { | ||||
|                 if (steps.length != 0) { | ||||
|                     if (!PlotSquared.get().worlds.contains(worldPath)) { | ||||
|                         PlotSquared.get().worlds.createSection(worldPath); | ||||
|                     } | ||||
|                     ConfigurationSection worldSection = | ||||
|                         PlotSquared.get().worlds.getConfigurationSection(worldPath); | ||||
|                     for (ConfigurationNode step : steps) { | ||||
|                         worldSection.set(step.getConstant(), step.getValue()); | ||||
|                     } | ||||
|                 } | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         try { | ||||
|             PlotSquared.get().worlds.save(PlotSquared.get().worldsFile); | ||||
|         } catch (IOException e) { | ||||
|             e.printStackTrace(); | ||||
|         } | ||||
|  | ||||
|         Objects.requireNonNull(PlotSquared.imp()).getWorldManager() | ||||
|             .handleWorldCreation(object.world, object.setupGenerator); | ||||
|  | ||||
|         if (Bukkit.getWorld(world) != null) { | ||||
|             return world; | ||||
|         } | ||||
|  | ||||
|         return object.world; | ||||
|     } | ||||
|  | ||||
|     @Override public String setupWorld(PlotAreaBuilder builder) { | ||||
|         SetupUtils.manager.updateGenerators(); | ||||
|     @Override | ||||
|     public String setupWorld(PlotAreaBuilder builder) { | ||||
|         this.updateGenerators(false); | ||||
|         ConfigurationNode[] steps = builder.settingsNodesWrapper() == null ? | ||||
|                 new ConfigurationNode[0] : builder.settingsNodesWrapper().getSettingsNodes(); | ||||
|         String world = builder.worldName(); | ||||
|         PlotAreaType type = builder.plotAreaType(); | ||||
|         String worldPath = "worlds." + builder.worldName(); | ||||
|         switch (type) { | ||||
|             case PARTIAL: { | ||||
|             case PARTIAL -> { | ||||
|                 if (builder.areaName() != null) { | ||||
|                     if (!PlotSquared.get().worlds.contains(worldPath)) { | ||||
|                         PlotSquared.get().worlds.createSection(worldPath); | ||||
|                     if (!this.worldConfiguration.contains(worldPath)) { | ||||
|                         this.worldConfiguration.createSection(worldPath); | ||||
|                     } | ||||
|                     ConfigurationSection worldSection = | ||||
|                             PlotSquared.get().worlds.getConfigurationSection(worldPath); | ||||
|                             this.worldConfiguration.getConfigurationSection(worldPath); | ||||
|                     String areaName = builder.areaName() + "-" + builder.minimumId() + "-" + builder.maximumId(); | ||||
|                     String areaPath = "areas." + areaName; | ||||
|                     if (!worldSection.contains(areaPath)) { | ||||
| @@ -260,29 +177,28 @@ public class BukkitSetupUtils extends SetupUtils { | ||||
|                 if (gen != null && gen.isFull()) { | ||||
|                     builder.generatorName(null); | ||||
|                 } | ||||
|                 break; | ||||
|             } | ||||
|             case AUGMENTED: { | ||||
|             case AUGMENTED -> { | ||||
|                 if (!builder.plotManager().endsWith(":single")) { | ||||
|                     if (!PlotSquared.get().worlds.contains(worldPath)) { | ||||
|                         PlotSquared.get().worlds.createSection(worldPath); | ||||
|                     if (!this.worldConfiguration.contains(worldPath)) { | ||||
|                         this.worldConfiguration.createSection(worldPath); | ||||
|                     } | ||||
|                     if (steps.length != 0) { | ||||
|                         ConfigurationSection worldSection = | ||||
|                                 PlotSquared.get().worlds.getConfigurationSection(worldPath); | ||||
|                                 this.worldConfiguration.getConfigurationSection(worldPath); | ||||
|                         for (ConfigurationNode step : steps) { | ||||
|                             worldSection.set(step.getConstant(), step.getValue()); | ||||
|                         } | ||||
|                     } | ||||
|                     PlotSquared.get().worlds | ||||
|                     this.worldConfiguration | ||||
|                             .set("worlds." + world + ".generator.type", builder.plotAreaType().toString()); | ||||
|                     PlotSquared.get().worlds | ||||
|                     this.worldConfiguration | ||||
|                             .set("worlds." + world + ".generator.terrain", builder.terrainType().toString()); | ||||
|                     PlotSquared.get().worlds | ||||
|                     this.worldConfiguration | ||||
|                             .set("worlds." + world + ".generator.plugin", builder.plotManager()); | ||||
|                     if (builder.generatorName() != null && !builder.generatorName() | ||||
|                             .equals(builder.plotManager())) { | ||||
|                         PlotSquared.get().worlds | ||||
|                         this.worldConfiguration | ||||
|                                 .set("worlds." + world + ".generator.init", builder.generatorName()); | ||||
|                     } | ||||
|                 } | ||||
| @@ -290,30 +206,28 @@ public class BukkitSetupUtils extends SetupUtils { | ||||
|                 if (gen != null && gen.isFull()) { | ||||
|                     builder.generatorName(null); | ||||
|                 } | ||||
|                 break; | ||||
|             } | ||||
|             case NORMAL: { | ||||
|             case NORMAL -> { | ||||
|                 if (steps.length != 0) { | ||||
|                     if (!PlotSquared.get().worlds.contains(worldPath)) { | ||||
|                         PlotSquared.get().worlds.createSection(worldPath); | ||||
|                     if (!this.worldConfiguration.contains(worldPath)) { | ||||
|                         this.worldConfiguration.createSection(worldPath); | ||||
|                     } | ||||
|                     ConfigurationSection worldSection = | ||||
|                             PlotSquared.get().worlds.getConfigurationSection(worldPath); | ||||
|                             this.worldConfiguration.getConfigurationSection(worldPath); | ||||
|                     for (ConfigurationNode step : steps) { | ||||
|                         worldSection.set(step.getConstant(), step.getValue()); | ||||
|                     } | ||||
|                 } | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         try { | ||||
|             PlotSquared.get().worlds.save(PlotSquared.get().worldsFile); | ||||
|             this.worldConfiguration.save(this.worldFile); | ||||
|         } catch (IOException e) { | ||||
|             e.printStackTrace(); | ||||
|         } | ||||
|  | ||||
|         Objects.requireNonNull(PlotSquared.imp()).getWorldManager() | ||||
|         Objects.requireNonNull(PlotSquared.platform()).worldManager() | ||||
|                 .handleWorldCreation(builder.worldName(), builder.generatorName()); | ||||
|  | ||||
|         if (Bukkit.getWorld(world) != null) { | ||||
| @@ -323,9 +237,10 @@ public class BukkitSetupUtils extends SetupUtils { | ||||
|         return builder.worldName(); | ||||
|     } | ||||
|  | ||||
|     @Override public String getGenerator(PlotArea plotArea) { | ||||
|     @Override | ||||
|     public String getGenerator(PlotArea plotArea) { | ||||
|         if (SetupUtils.generators.isEmpty()) { | ||||
|             updateGenerators(); | ||||
|             updateGenerators(false); | ||||
|         } | ||||
|         World world = Bukkit.getWorld(plotArea.getWorldName()); | ||||
|         if (world == null) { | ||||
| @@ -343,4 +258,5 @@ public class BukkitSetupUtils extends SetupUtils { | ||||
|         } | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,79 +0,0 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  * | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import com.plotsquared.bukkit.BukkitMain; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import org.bukkit.Bukkit; | ||||
|  | ||||
| public class BukkitTaskManager extends TaskManager { | ||||
|  | ||||
|     private final BukkitMain bukkitMain; | ||||
|  | ||||
|     public BukkitTaskManager(BukkitMain bukkitMain) { | ||||
|         this.bukkitMain = bukkitMain; | ||||
|     } | ||||
|  | ||||
|     @Override public int taskRepeat(Runnable runnable, int interval) { | ||||
|         return this.bukkitMain.getServer().getScheduler() | ||||
|             .scheduleSyncRepeatingTask(this.bukkitMain, runnable, interval, interval); | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("deprecation") @Override | ||||
|     public int taskRepeatAsync(Runnable runnable, int interval) { | ||||
|         return this.bukkitMain.getServer().getScheduler() | ||||
|             .scheduleAsyncRepeatingTask(this.bukkitMain, runnable, interval, interval); | ||||
|     } | ||||
|  | ||||
|     @Override public void taskAsync(Runnable runnable) { | ||||
|         if (this.bukkitMain.isEnabled()) { | ||||
|             this.bukkitMain.getServer().getScheduler() | ||||
|                 .runTaskAsynchronously(this.bukkitMain, runnable); | ||||
|         } else { | ||||
|             runnable.run(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public void task(Runnable runnable) { | ||||
|         this.bukkitMain.getServer().getScheduler().runTask(this.bukkitMain, runnable).getTaskId(); | ||||
|     } | ||||
|  | ||||
|     @Override public void taskLater(Runnable runnable, int delay) { | ||||
|         this.bukkitMain.getServer().getScheduler().runTaskLater(this.bukkitMain, runnable, delay) | ||||
|             .getTaskId(); | ||||
|     } | ||||
|  | ||||
|     @Override public void taskLaterAsync(Runnable runnable, int delay) { | ||||
|         this.bukkitMain.getServer().getScheduler() | ||||
|             .runTaskLaterAsynchronously(this.bukkitMain, runnable, delay); | ||||
|     } | ||||
|  | ||||
|     @Override public void cancelTask(int task) { | ||||
|         if (task != -1) { | ||||
|             Bukkit.getScheduler().cancelTask(task); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,25 +21,25 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import com.plotsquared.bukkit.BukkitMain; | ||||
| import com.google.inject.Singleton; | ||||
| import com.plotsquared.bukkit.BukkitPlatform; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayerManager; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Captions; | ||||
| import com.plotsquared.core.configuration.caption.Caption; | ||||
| import com.plotsquared.core.configuration.caption.LocaleHolder; | ||||
| 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.util.BlockUtil; | ||||
| import com.plotsquared.core.util.MainUtil; | ||||
| import com.plotsquared.core.util.MathMan; | ||||
| import com.plotsquared.core.util.PlayerManager; | ||||
| import com.plotsquared.core.util.StringComparison; | ||||
| import com.plotsquared.core.util.WorldUtil; | ||||
| import com.plotsquared.core.util.task.RunnableVal; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.bukkit.BukkitWorld; | ||||
| @@ -51,11 +51,15 @@ import com.sk89q.worldedit.world.block.BlockState; | ||||
| import com.sk89q.worldedit.world.block.BlockType; | ||||
| import com.sk89q.worldedit.world.block.BlockTypes; | ||||
| import io.papermc.lib.PaperLib; | ||||
| import lombok.NonNull; | ||||
| import net.kyori.adventure.platform.bukkit.BukkitAudiences; | ||||
| import net.kyori.adventure.text.minimessage.MiniMessage; | ||||
| import net.kyori.adventure.text.minimessage.Template; | ||||
| import net.kyori.adventure.text.serializer.legacy.LegacyComponentSerializer; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.Material; | ||||
| import org.bukkit.OfflinePlayer; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.Biome; | ||||
| import org.bukkit.block.Block; | ||||
| @@ -93,265 +97,153 @@ import org.bukkit.entity.Snowman; | ||||
| import org.bukkit.entity.Tameable; | ||||
| import org.bukkit.entity.Vehicle; | ||||
| import org.bukkit.entity.WaterMob; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
| import org.checkerframework.checker.index.qual.NonNegative; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collection; | ||||
| import java.util.HashSet; | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
| import java.util.Set; | ||||
| import java.util.UUID; | ||||
| import java.util.concurrent.Semaphore; | ||||
| import java.util.function.Consumer; | ||||
| import java.util.function.IntConsumer; | ||||
| import java.util.stream.Stream; | ||||
|  | ||||
| @SuppressWarnings({"unused", "WeakerAccess"}) | ||||
| @Singleton | ||||
| public class BukkitUtil extends WorldUtil { | ||||
|  | ||||
|     private static String lastString = null; | ||||
|     private static World lastWorld = null; | ||||
|  | ||||
|     private static Player lastPlayer = null; | ||||
|     private static BukkitPlayer lastPlotPlayer = null; | ||||
|  | ||||
|     public static void removePlayer(UUID uuid) { | ||||
|         lastPlayer = null; | ||||
|         lastPlotPlayer = null; | ||||
|         // Make sure that it's removed internally | ||||
|         PlotSquared.imp().getPlayerManager().removePlayer(uuid); | ||||
|     } | ||||
|  | ||||
|     public static PlotPlayer<Player> getPlayer(@NonNull final OfflinePlayer op) { | ||||
|         if (op.isOnline()) { | ||||
|             return getPlayer(op.getPlayer()); | ||||
|         } | ||||
|         final Player player = OfflinePlayerUtil.loadPlayer(op); | ||||
|         player.loadData(); | ||||
|         return new BukkitPlayer(player, true); | ||||
|     } | ||||
|     public static final BukkitAudiences BUKKIT_AUDIENCES = BukkitAudiences.create(BukkitPlatform.getPlugin(BukkitPlatform.class)); | ||||
|     public static final LegacyComponentSerializer LEGACY_COMPONENT_SERIALIZER = LegacyComponentSerializer.legacySection(); | ||||
|     public static final MiniMessage MINI_MESSAGE = MiniMessage.builder().build(); | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + BukkitUtil.class.getSimpleName()); | ||||
|     private final Collection<BlockType> tileEntityTypes = new HashSet<>(); | ||||
|  | ||||
|     /** | ||||
|      * Get a plot based on the location. | ||||
|      * Turn a Bukkit {@link Player} into a PlotSquared {@link PlotPlayer} | ||||
|      * | ||||
|      * @param location the location to check | ||||
|      * @return plot if found, otherwise it creates a temporary plot | ||||
|      * @see Plot | ||||
|      * @param player Bukkit player | ||||
|      * @return PlotSquared player | ||||
|      */ | ||||
|     public static Plot getPlot(org.bukkit.Location location) { | ||||
|         if (location == null) { | ||||
|             return null; | ||||
|         } | ||||
|         return getLocation(location).getPlot(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get a plot based on the player location. | ||||
|      * | ||||
|      * @param player the player to check | ||||
|      * @return plot if found, otherwise it creates a temporary plot | ||||
|      * @see #getPlot(org.bukkit.Location) | ||||
|      * @see Plot | ||||
|      */ | ||||
|     public static Plot getPlot(Player player) { | ||||
|         return getPlot(player.getLocation()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get the PlotPlayer for an offline player. | ||||
|      * | ||||
|      * <p>Note that this will work if the player is offline, however not all | ||||
|      * functionality will work. | ||||
|      * | ||||
|      * @param player the player to wrap | ||||
|      * @return a {@code PlotPlayer} | ||||
|      * @see PlotPlayer#wrap(Object) | ||||
|      */ | ||||
|     public static PlotPlayer<?> wrapPlayer(OfflinePlayer player) { | ||||
|         return PlotPlayer.wrap(player); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets the PlotPlayer for a player. The PlotPlayer is usually cached and | ||||
|      * will provide useful functions relating to players. | ||||
|      * | ||||
|      * @param player the player to wrap | ||||
|      * @return a {@code PlotPlayer} | ||||
|      * @see PlotPlayer#wrap(Object) | ||||
|      */ | ||||
|     public static PlotPlayer<?> wrapPlayer(Player player) { | ||||
|         return PlotPlayer.wrap(player); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets the PlotPlayer for a UUID. The PlotPlayer is usually cached and | ||||
|      * will provide useful functions relating to players. | ||||
|      * | ||||
|      * @param uuid the uuid to wrap | ||||
|      * @return a {@code PlotPlayer} | ||||
|      * @see PlotPlayer#wrap(Object) | ||||
|      */ | ||||
|     @Override public PlotPlayer<?> wrapPlayer(UUID uuid) { | ||||
|         return PlotPlayer.wrap(Bukkit.getOfflinePlayer(uuid)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets the number of plots, which the player is able to build in. | ||||
|      * | ||||
|      * @param player player, for whom we're getting the plots | ||||
|      * @return the number of allowed plots | ||||
|      */ | ||||
|     public static int getAllowedPlots(Player player) { | ||||
|         PlotPlayer<?> plotPlayer = PlotPlayer.wrap(player); | ||||
|         return plotPlayer.getAllowedPlots(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Check whether or not a player is in a plot. | ||||
|      * | ||||
|      * @param player who we're checking for | ||||
|      * @return true if the player is in a plot, false if not- | ||||
|      */ | ||||
|     public static boolean isInPlot(Player player) { | ||||
|         return getPlot(player) != null; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets a collection containing the players plots. | ||||
|      * | ||||
|      * @param world  Specify the world we want to select the plots from | ||||
|      * @param player Player, for whom we're getting the plots | ||||
|      * @return a set containing the players plots | ||||
|      * @see Plot | ||||
|      */ | ||||
|     public static Set<Plot> getPlayerPlots(String world, Player player) { | ||||
|         if (world == null) { | ||||
|             return new HashSet<>(); | ||||
|         } | ||||
|         return BukkitPlayer.wrap(player).getPlots(world); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Send a message to a player. The message supports color codes. | ||||
|      * | ||||
|      * @param player the recipient of the message | ||||
|      * @param string the message | ||||
|      * @see MainUtil#sendMessage(PlotPlayer, String) | ||||
|      */ | ||||
|     public static void sendMessage(Player player, String string) { | ||||
|         MainUtil.sendMessage(BukkitUtil.getPlayer(player), string); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets the player plot count. | ||||
|      * | ||||
|      * @param world  Specify the world we want to select the plots from | ||||
|      * @param player Player, for whom we're getting the plot count | ||||
|      * @return the number of plots the player has | ||||
|      */ | ||||
|     public static int getPlayerPlotCount(String world, Player player) { | ||||
|         if (world == null) { | ||||
|             return 0; | ||||
|         } | ||||
|         return BukkitUtil.getPlayer(player).getPlotCount(world); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Send a message to a player. | ||||
|      * | ||||
|      * @param player  the recipient of the message | ||||
|      * @param caption the message | ||||
|      */ | ||||
|     public static void sendMessage(Player player, Captions caption) { | ||||
|         MainUtil.sendMessage(BukkitUtil.getPlayer(player), caption); | ||||
|     } | ||||
|  | ||||
|     public static BukkitPlayer getPlayer(@NonNull final Player player) { | ||||
|         if (player == lastPlayer) { | ||||
|             return lastPlotPlayer; | ||||
|         } | ||||
|         final PlayerManager<?, ?> playerManager = PlotSquared.imp().getPlayerManager(); | ||||
|     public static @NonNull BukkitPlayer adapt(final @NonNull Player player) { | ||||
|         final PlayerManager<?, ?> playerManager = PlotSquared.platform().playerManager(); | ||||
|         return ((BukkitPlayerManager) playerManager).getPlayer(player); | ||||
|     } | ||||
|  | ||||
|     public static Location getLocation(@NonNull final org.bukkit.Location location) { | ||||
|         return new Location(location.getWorld().getName(), MathMan.roundInt(location.getX()), | ||||
|             MathMan.roundInt(location.getY()), MathMan.roundInt(location.getZ())); | ||||
|     /** | ||||
|      * Turn a Bukkit {@link org.bukkit.Location} into a PlotSquared {@link Location}. | ||||
|      * This only copies the 4-tuple (world,x,y,z) and does not include the yaw and the pitch | ||||
|      * | ||||
|      * @param location Bukkit location | ||||
|      * @return PlotSquared location | ||||
|      */ | ||||
|     public static @NonNull Location adapt(final org.bukkit.@NonNull Location location) { | ||||
|         return Location | ||||
|                 .at( | ||||
|                         com.plotsquared.bukkit.util.BukkitWorld.of(location.getWorld()), | ||||
|                         MathMan.roundInt(location.getX()), | ||||
|                         MathMan.roundInt(location.getY()), | ||||
|                         MathMan.roundInt(location.getZ()) | ||||
|                 ); | ||||
|     } | ||||
|  | ||||
|     public static Location getLocationFull(@NonNull final org.bukkit.Location location) { | ||||
|         return new Location(location.getWorld().getName(), MathMan.roundInt(location.getX()), | ||||
|             MathMan.roundInt(location.getY()), MathMan.roundInt(location.getZ()), location.getYaw(), | ||||
|             location.getPitch()); | ||||
|     /** | ||||
|      * Turn a Bukkit {@link org.bukkit.Location} into a PlotSquared {@link Location}. | ||||
|      * This copies the entire 6-tuple (world,x,y,z,yaw,pitch). | ||||
|      * | ||||
|      * @param location Bukkit location | ||||
|      * @return PlotSquared location | ||||
|      */ | ||||
|     public static @NonNull Location adaptComplete(final org.bukkit.@NonNull Location location) { | ||||
|         return Location | ||||
|                 .at( | ||||
|                         com.plotsquared.bukkit.util.BukkitWorld.of(location.getWorld()), | ||||
|                         MathMan.roundInt(location.getX()), | ||||
|                         MathMan.roundInt(location.getY()), | ||||
|                         MathMan.roundInt(location.getZ()), | ||||
|                         location.getYaw(), | ||||
|                         location.getPitch() | ||||
|                 ); | ||||
|     } | ||||
|  | ||||
|     public static org.bukkit.Location getLocation(@NonNull final Location location) { | ||||
|         return new org.bukkit.Location(getWorld(location.getWorld()), location.getX(), | ||||
|             location.getY(), location.getZ()); | ||||
|     /** | ||||
|      * Turn a PlotSquared {@link Location} into a Bukkit {@link org.bukkit.Location}. | ||||
|      * This only copies the 4-tuple (world,x,y,z) and does not include the yaw and the pitch | ||||
|      * | ||||
|      * @param location PlotSquared location | ||||
|      * @return Bukkit location | ||||
|      */ | ||||
|     public static org.bukkit.@NonNull Location adapt(final @NonNull Location location) { | ||||
|         return new org.bukkit.Location( | ||||
|                 (World) location.getWorld().getPlatformWorld(), | ||||
|                 location.getX(), | ||||
|                 location.getY(), | ||||
|                 location.getZ() | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     public static World getWorld(@NonNull final String string) { | ||||
|     /** | ||||
|      * Get a Bukkit {@link World} from its name | ||||
|      * | ||||
|      * @param string World name | ||||
|      * @return World if it exists, or {@code null} | ||||
|      */ | ||||
|     public static @Nullable World getWorld(final @NonNull String string) { | ||||
|         return Bukkit.getWorld(string); | ||||
|     } | ||||
|  | ||||
|     public static String getWorld(@NonNull final Entity entity) { | ||||
|         return entity.getWorld().getName(); | ||||
|     private static void ensureLoaded( | ||||
|             final @NonNull String world, | ||||
|             final int x, | ||||
|             final int z, | ||||
|             final @NonNull Consumer<Chunk> chunkConsumer | ||||
|     ) { | ||||
|         PaperLib.getChunkAtAsync(Objects.requireNonNull(getWorld(world)), x >> 4, z >> 4, true) | ||||
|                 .thenAccept(chunk -> ensureMainThread(chunkConsumer, chunk)); | ||||
|     } | ||||
|  | ||||
|     public static List<Entity> getEntities(@NonNull final String worldName) { | ||||
|         World world = getWorld(worldName); | ||||
|         if (world != null) { | ||||
|             return world.getEntities(); | ||||
|     private static void ensureLoaded(final @NonNull Location location, final @NonNull Consumer<Chunk> chunkConsumer) { | ||||
|         PaperLib.getChunkAtAsync(adapt(location), true).thenAccept(chunk -> ensureMainThread(chunkConsumer, chunk)); | ||||
|     } | ||||
|  | ||||
|     private static <T> void ensureMainThread(final @NonNull Consumer<T> consumer, final @NonNull T value) { | ||||
|         if (Bukkit.isPrimaryThread()) { | ||||
|             consumer.accept(value); | ||||
|         } else { | ||||
|             return new ArrayList<>(); | ||||
|             Bukkit.getScheduler().runTask(BukkitPlatform.getPlugin(BukkitPlatform.class), () -> consumer.accept(value)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public static Location getLocation(@NonNull final Entity entity) { | ||||
|         final org.bukkit.Location location = entity.getLocation(); | ||||
|         String world = location.getWorld().getName(); | ||||
|         return new Location(world, location.getBlockX(), location.getBlockY(), | ||||
|             location.getBlockZ()); | ||||
|     } | ||||
|  | ||||
|     @NotNull public static Location getLocationFull(@NonNull final Entity entity) { | ||||
|         final org.bukkit.Location location = entity.getLocation(); | ||||
|         return new Location(location.getWorld().getName(), MathMan.roundInt(location.getX()), | ||||
|             MathMan.roundInt(location.getY()), MathMan.roundInt(location.getZ()), location.getYaw(), | ||||
|             location.getPitch()); | ||||
|     } | ||||
|  | ||||
|     public static Material getMaterial(@NonNull final BlockState plotBlock) { | ||||
|         return BukkitAdapter.adapt(plotBlock.getBlockType()); | ||||
|     } | ||||
|  | ||||
|     @Override public boolean isBlockSame(BlockState block1, BlockState block2) { | ||||
|         if (block1.equals(block2)) { | ||||
|             return true; | ||||
|         } | ||||
|         Material mat1 = getMaterial(block1), mat2 = getMaterial(block2); | ||||
|         return mat1 == mat2; | ||||
|     } | ||||
|  | ||||
|     @Override public boolean isWorld(@NonNull final String worldName) { | ||||
|         return getWorld(worldName) != null; | ||||
|     } | ||||
|  | ||||
|     @Override public void getBiome(String world, int x, int z, final Consumer<BiomeType> result) { | ||||
|         ensureLoaded(world, x, z, | ||||
|             chunk -> result.accept(BukkitAdapter.adapt(getWorld(world).getBiome(x, z)))); | ||||
|     } | ||||
|  | ||||
|     @Override public BiomeType getBiomeSynchronous(String world, int x, int z) { | ||||
|         return BukkitAdapter.adapt(getWorld(world).getBiome(x, z)); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void getHighestBlock(@NonNull final String world, final int x, final int z, | ||||
|         final IntConsumer result) { | ||||
|     public boolean isBlockSame(final @NonNull BlockState block1, final @NonNull BlockState block2) { | ||||
|         if (block1.equals(block2)) { | ||||
|             return true; | ||||
|         } | ||||
|         final Material mat1 = BukkitAdapter.adapt(block1.getBlockType()); | ||||
|         final Material mat2 = BukkitAdapter.adapt(block2.getBlockType()); | ||||
|         return mat1 == mat2; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean isWorld(final @NonNull String worldName) { | ||||
|         return getWorld(worldName) != null; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void getBiome(final @NonNull String world, final int x, final int z, final @NonNull Consumer<BiomeType> result) { | ||||
|         ensureLoaded(world, x, z, chunk -> result.accept(BukkitAdapter.adapt(getWorld(world).getBiome(x, z)))); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public @NonNull BiomeType getBiomeSynchronous(final @NonNull String world, final int x, final int z) { | ||||
|         return BukkitAdapter.adapt(Objects.requireNonNull(getWorld(world)).getBiome(x, z)); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void getHighestBlock(final @NonNull String world, final int x, final int z, final @NonNull IntConsumer result) { | ||||
|         ensureLoaded(world, x, z, chunk -> { | ||||
|             final World bukkitWorld = getWorld(world); | ||||
|             final World bukkitWorld = Objects.requireNonNull(getWorld(world)); | ||||
|             // Skip top and bottom block | ||||
|             int air = 1; | ||||
|             for (int y = bukkitWorld.getMaxHeight() - 1; y >= 0; y--) { | ||||
| @@ -375,8 +267,10 @@ public class BukkitUtil extends WorldUtil { | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     @Override public int getHighestBlockSynchronous(String world, int x, int z) { | ||||
|         final World bukkitWorld = getWorld(world); | ||||
|     @Override | ||||
|     @NonNegative | ||||
|     public int getHighestBlockSynchronous(final @NonNull String world, final int x, final int z) { | ||||
|         final World bukkitWorld = Objects.requireNonNull(getWorld(world)); | ||||
|         // Skip top and bottom block | ||||
|         int air = 1; | ||||
|         for (int y = bukkitWorld.getMaxHeight() - 1; y >= 0; y--) { | ||||
| @@ -398,73 +292,73 @@ public class BukkitUtil extends WorldUtil { | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void getSign(@NonNull final Location location, final Consumer<String[]> result) { | ||||
|         ensureLoaded(location, chunk -> { | ||||
|             final Block block = chunk.getWorld().getBlockAt(getLocation(location)); | ||||
|             if (block.getState() instanceof Sign) { | ||||
|                 Sign sign = (Sign) block.getState(); | ||||
|                 result.accept(sign.getLines()); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     @Override @Nullable public String[] getSignSynchronous(@NonNull final Location location) { | ||||
|         Block block = getWorld(location.getWorld()) | ||||
|             .getBlockAt(location.getX(), location.getY(), location.getZ()); | ||||
|         return TaskManager.IMP.sync(new RunnableVal<String[]>() { | ||||
|             @Override public void run(String[] value) { | ||||
|                 if (block.getState() instanceof Sign) { | ||||
|                     Sign sign = (Sign) block.getState(); | ||||
|                     this.value = sign.getLines(); | ||||
|     public @NonNull String[] getSignSynchronous(final @NonNull Location location) { | ||||
|         Block block = Objects.requireNonNull(getWorld(location.getWorldName())).getBlockAt( | ||||
|                 location.getX(), | ||||
|                 location.getY(), | ||||
|                 location.getZ() | ||||
|         ); | ||||
|         try { | ||||
|             return TaskManager.getPlatformImplementation().sync(() -> { | ||||
|                 if (block.getState() instanceof Sign sign) { | ||||
|                     return sign.getLines(); | ||||
|                 } | ||||
|             } | ||||
|         }); | ||||
|                 return new String[0]; | ||||
|             }); | ||||
|         } catch (final Exception e) { | ||||
|             e.printStackTrace(); | ||||
|         } | ||||
|         return new String[0]; | ||||
|     } | ||||
|  | ||||
|     @Override public Location getSpawn(@NonNull final String world) { | ||||
|     @Override | ||||
|     public @NonNull Location getSpawn(final @NonNull String world) { | ||||
|         final org.bukkit.Location temp = getWorld(world).getSpawnLocation(); | ||||
|         return new Location(world, temp.getBlockX(), temp.getBlockY(), temp.getBlockZ(), | ||||
|             temp.getYaw(), temp.getPitch()); | ||||
|         return Location.at(world, temp.getBlockX(), temp.getBlockY(), temp.getBlockZ(), temp.getYaw(), temp.getPitch()); | ||||
|     } | ||||
|  | ||||
|     @Override public void setSpawn(@NonNull final Location location) { | ||||
|         final World world = getWorld(location.getWorld()); | ||||
|     @Override | ||||
|     public void setSpawn(final @NonNull Location location) { | ||||
|         final World world = getWorld(location.getWorldName()); | ||||
|         if (world != null) { | ||||
|             world.setSpawnLocation(location.getX(), location.getY(), location.getZ()); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override public void saveWorld(@NonNull final String worldName) { | ||||
|     @Override | ||||
|     public void saveWorld(final @NonNull String worldName) { | ||||
|         final World world = getWorld(worldName); | ||||
|         if (world != null) { | ||||
|             world.save(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override @SuppressWarnings("deprecation") | ||||
|     public void setSign(@NonNull final String worldName, final int x, final int y, final int z, | ||||
|         @NonNull final String[] lines) { | ||||
|         ensureLoaded(worldName, x, z, chunk -> { | ||||
|             final World world = getWorld(worldName); | ||||
|             final Block block = world.getBlockAt(x, y, z); | ||||
|             //        block.setType(Material.AIR); | ||||
|     @Override | ||||
|     @SuppressWarnings("deprecation") | ||||
|     public void setSign( | ||||
|             final @NonNull Location location, final @NonNull Caption[] lines, | ||||
|             final @NonNull Template... replacements | ||||
|     ) { | ||||
|         ensureLoaded(location.getWorldName(), location.getX(), location.getZ(), chunk -> { | ||||
|             PlotArea area = location.getPlotArea(); | ||||
|             final World world = getWorld(location.getWorldName()); | ||||
|             final Block block = world.getBlockAt(location.getX(), location.getY(), location.getZ()); | ||||
|             final Material type = block.getType(); | ||||
|             if (type != Material.LEGACY_SIGN && type != Material.LEGACY_WALL_SIGN) { | ||||
|                 BlockFace facing = BlockFace.EAST; | ||||
|                 if (world.getBlockAt(x, y, z + 1).getType().isSolid()) { | ||||
|                 if (world.getBlockAt(location.getX(), location.getY(), location.getZ() + 1).getType().isSolid()) { | ||||
|                     facing = BlockFace.NORTH; | ||||
|                 } else if (world.getBlockAt(x + 1, y, z).getType().isSolid()) { | ||||
|                 } else if (world.getBlockAt(location.getX() + 1, location.getY(), location.getZ()).getType().isSolid()) { | ||||
|                     facing = BlockFace.WEST; | ||||
|                 } else if (world.getBlockAt(x, y, z - 1).getType().isSolid()) { | ||||
|                 } else if (world.getBlockAt(location.getX(), location.getY(), location.getZ() - 1).getType().isSolid()) { | ||||
|                     facing = BlockFace.SOUTH; | ||||
|                 } | ||||
|                 if (PlotSquared.get().IMP.getServerVersion()[1] == 13) { | ||||
|                     block.setType(Material.valueOf("WALL_SIGN"), false); | ||||
|                 if (PlotSquared.platform().serverVersion()[1] == 13) { | ||||
|                     block.setType(Material.valueOf(area.legacySignMaterial()), false); | ||||
|                 } else { | ||||
|                     block.setType(Material.valueOf("OAK_WALL_SIGN"), false); | ||||
|                     block.setType(Material.valueOf(area.signMaterial()), false); | ||||
|                 } | ||||
|                 if (!(block.getBlockData() instanceof WallSign)) { | ||||
|                     PlotSquared.debug(block.getBlockData().getAsString()); | ||||
|                     throw new RuntimeException("Something went wrong generating a sign"); | ||||
|                 } | ||||
|                 final Directional sign = (Directional) block.getBlockData(); | ||||
| @@ -472,46 +366,36 @@ public class BukkitUtil extends WorldUtil { | ||||
|                 block.setBlockData(sign, false); | ||||
|             } | ||||
|             final org.bukkit.block.BlockState blockstate = block.getState(); | ||||
|             if (blockstate instanceof Sign) { | ||||
|                 final Sign sign = (Sign) blockstate; | ||||
|             if (blockstate instanceof final Sign sign) { | ||||
|                 for (int i = 0; i < lines.length; i++) { | ||||
|                     sign.setLine(i, lines[i]); | ||||
|                     sign.setLine(i, LEGACY_COMPONENT_SERIALIZER | ||||
|                             .serialize(MINI_MESSAGE.parse(lines[i].getComponent(LocaleHolder.console()), replacements))); | ||||
|                 } | ||||
|                 sign.update(true); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     @Override public boolean isBlockSolid(@NonNull final BlockState block) { | ||||
|         return block.getBlockType().getMaterial().isSolid(); | ||||
|     } | ||||
|  | ||||
|     @Override public String getClosestMatchingName(@NonNull final BlockState block) { | ||||
|         try { | ||||
|             return getMaterial(block).name(); | ||||
|         } catch (Exception ignored) { | ||||
|             return null; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override @Nullable | ||||
|     public StringComparison<BlockState>.ComparisonResult getClosestBlock(String name) { | ||||
|     @Override | ||||
|     public @NonNull StringComparison<BlockState>.ComparisonResult getClosestBlock(@NonNull String name) { | ||||
|         BlockState state = BlockUtil.get(name); | ||||
|         return new StringComparison<BlockState>().new ComparisonResult(1, state); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void setBiomes(@NonNull final String worldName, @NonNull final CuboidRegion region, | ||||
|         @NonNull final BiomeType biomeType) { | ||||
|     public void setBiomes( | ||||
|             final @NonNull String worldName, | ||||
|             final @NonNull CuboidRegion region, | ||||
|             final @NonNull BiomeType biomeType | ||||
|     ) { | ||||
|         final World world = getWorld(worldName); | ||||
|         if (world == null) { | ||||
|             PlotSquared.log("An error occurred setting the biome because the world was null."); | ||||
|             LOGGER.warn("An error occurred while setting the biome because the world was null", new RuntimeException()); | ||||
|             return; | ||||
|         } | ||||
|         final Biome biome = BukkitAdapter.adapt(biomeType); | ||||
|         for (int x = region.getMinimumPoint().getX(); x <= region.getMaximumPoint().getX(); x++) { | ||||
|             for (int z = region.getMinimumPoint().getZ(); | ||||
|                  z <= region.getMaximumPoint().getZ(); z++) { | ||||
|             for (int z = region.getMinimumPoint().getZ(); z <= region.getMaximumPoint().getZ(); z++) { | ||||
|                 if (world.getBiome(x, z) != biome) { | ||||
|                     world.setBiome(x, z, biome); | ||||
|                 } | ||||
| @@ -519,67 +403,68 @@ public class BukkitUtil extends WorldUtil { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public com.sk89q.worldedit.world.World getWeWorld(String world) { | ||||
|     @Override | ||||
|     public com.sk89q.worldedit.world.@NonNull World getWeWorld(final @NonNull String world) { | ||||
|         return new BukkitWorld(Bukkit.getWorld(world)); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void getBlock(@NonNull final Location location, final Consumer<BlockState> result) { | ||||
|     public void refreshChunk(int x, int z, String world) { | ||||
|         Bukkit.getWorld(world).refreshChunk(x, z); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void getBlock(final @NonNull Location location, final @NonNull Consumer<BlockState> result) { | ||||
|         ensureLoaded(location, chunk -> { | ||||
|             final World world = getWorld(location.getWorld()); | ||||
|             final Block block = world.getBlockAt(location.getX(), location.getY(), location.getZ()); | ||||
|             result.accept(BukkitAdapter.asBlockType(block.getType()).getDefaultState()); | ||||
|             final World world = getWorld(location.getWorldName()); | ||||
|             final Block block = Objects.requireNonNull(world).getBlockAt(location.getX(), location.getY(), location.getZ()); | ||||
|             result.accept(Objects.requireNonNull(BukkitAdapter.asBlockType(block.getType())).getDefaultState()); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     @Override public BlockState getBlockSynchronous(@NonNull final Location location) { | ||||
|         final World world = getWorld(location.getWorld()); | ||||
|         final Block block = world.getBlockAt(location.getX(), location.getY(), location.getZ()); | ||||
|         return BukkitAdapter.asBlockType(block.getType()).getDefaultState(); | ||||
|     @Override | ||||
|     public @NonNull BlockState getBlockSynchronous(final @NonNull Location location) { | ||||
|         final World world = getWorld(location.getWorldName()); | ||||
|         final Block block = Objects.requireNonNull(world).getBlockAt(location.getX(), location.getY(), location.getZ()); | ||||
|         return Objects.requireNonNull(BukkitAdapter.asBlockType(block.getType())).getDefaultState(); | ||||
|     } | ||||
|  | ||||
|     @Override public String getMainWorld() { | ||||
|         return Bukkit.getWorlds().get(0).getName(); | ||||
|     @Override | ||||
|     @NonNegative | ||||
|     public double getHealth(final @NonNull PlotPlayer<?> player) { | ||||
|         return Objects.requireNonNull(Bukkit.getPlayer(player.getUUID())).getHealth(); | ||||
|     } | ||||
|  | ||||
|     @Override public double getHealth(PlotPlayer player) { | ||||
|         return Bukkit.getPlayer(player.getUUID()).getHealth(); | ||||
|     @Override | ||||
|     @NonNegative | ||||
|     public int getFoodLevel(final @NonNull PlotPlayer<?> player) { | ||||
|         return Objects.requireNonNull(Bukkit.getPlayer(player.getUUID())).getFoodLevel(); | ||||
|     } | ||||
|  | ||||
|     @Override public int getFoodLevel(PlotPlayer player) { | ||||
|         return Bukkit.getPlayer(player.getUUID()).getFoodLevel(); | ||||
|     @Override | ||||
|     public void setHealth(final @NonNull PlotPlayer<?> player, @NonNegative final double health) { | ||||
|         Objects.requireNonNull(Bukkit.getPlayer(player.getUUID())).setHealth(health); | ||||
|     } | ||||
|  | ||||
|     @Override public void setHealth(PlotPlayer player, double health) { | ||||
|         Bukkit.getPlayer(player.getUUID()).setHealth(health); | ||||
|     } | ||||
|  | ||||
|     @Override public void setFoodLevel(PlotPlayer player, int foodLevel) { | ||||
|     @Override | ||||
|     public void setFoodLevel(final @NonNull PlotPlayer<?> player, @NonNegative final int foodLevel) { | ||||
|         Bukkit.getPlayer(player.getUUID()).setFoodLevel(foodLevel); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public Set<com.sk89q.worldedit.world.entity.EntityType> getTypesInCategory( | ||||
|         final String category) { | ||||
|     public @NonNull Set<com.sk89q.worldedit.world.entity.EntityType> getTypesInCategory(final @NonNull String category) { | ||||
|         final Collection<Class<?>> allowedInterfaces = new HashSet<>(); | ||||
|         switch (category) { | ||||
|             case "animal": { | ||||
|             case "animal" -> { | ||||
|                 allowedInterfaces.add(IronGolem.class); | ||||
|                 allowedInterfaces.add(Snowman.class); | ||||
|                 allowedInterfaces.add(Animals.class); | ||||
|                 allowedInterfaces.add(WaterMob.class); | ||||
|                 allowedInterfaces.add(Ambient.class); | ||||
|             } | ||||
|             break; | ||||
|             case "tameable": { | ||||
|                 allowedInterfaces.add(Tameable.class); | ||||
|             } | ||||
|             break; | ||||
|             case "vehicle": { | ||||
|                 allowedInterfaces.add(Vehicle.class); | ||||
|             } | ||||
|             break; | ||||
|             case "hostile": { | ||||
|             case "tameable" -> allowedInterfaces.add(Tameable.class); | ||||
|             case "vehicle" -> allowedInterfaces.add(Vehicle.class); | ||||
|             case "hostile" -> { | ||||
|                 allowedInterfaces.add(Shulker.class); | ||||
|                 allowedInterfaces.add(Monster.class); | ||||
|                 allowedInterfaces.add(Boss.class); | ||||
| @@ -588,20 +473,10 @@ public class BukkitUtil extends WorldUtil { | ||||
|                 allowedInterfaces.add(Phantom.class); | ||||
|                 allowedInterfaces.add(EnderCrystal.class); | ||||
|             } | ||||
|             break; | ||||
|             case "hanging": { | ||||
|                 allowedInterfaces.add(Hanging.class); | ||||
|             } | ||||
|             break; | ||||
|             case "villager": { | ||||
|                 allowedInterfaces.add(NPC.class); | ||||
|             } | ||||
|             break; | ||||
|             case "projectile": { | ||||
|                 allowedInterfaces.add(Projectile.class); | ||||
|             } | ||||
|             break; | ||||
|             case "other": { | ||||
|             case "hanging" -> allowedInterfaces.add(Hanging.class); | ||||
|             case "villager" -> allowedInterfaces.add(NPC.class); | ||||
|             case "projectile" -> allowedInterfaces.add(Projectile.class); | ||||
|             case "other" -> { | ||||
|                 allowedInterfaces.add(ArmorStand.class); | ||||
|                 allowedInterfaces.add(FallingBlock.class); | ||||
|                 allowedInterfaces.add(Item.class); | ||||
| @@ -613,15 +488,8 @@ public class BukkitUtil extends WorldUtil { | ||||
|                 allowedInterfaces.add(EnderSignal.class); | ||||
|                 allowedInterfaces.add(Firework.class); | ||||
|             } | ||||
|             break; | ||||
|             case "player": { | ||||
|                 allowedInterfaces.add(Player.class); | ||||
|             } | ||||
|             break; | ||||
|             default: { | ||||
|                 PlotSquared.log(Captions.PREFIX + "Unknown entity category requested: " + category); | ||||
|             } | ||||
|             break; | ||||
|             case "player" -> allowedInterfaces.add(Player.class); | ||||
|             default -> LOGGER.error("Unknown entity category requested: {}", category); | ||||
|         } | ||||
|         final Set<com.sk89q.worldedit.world.entity.EntityType> types = new HashSet<>(); | ||||
|         outer: | ||||
| @@ -640,8 +508,8 @@ public class BukkitUtil extends WorldUtil { | ||||
|         return types; | ||||
|     } | ||||
|  | ||||
|     private final Collection<BlockType> tileEntityTypes = new HashSet<>(); | ||||
|     @Override public Collection<BlockType> getTileEntityTypes() { | ||||
|     @Override | ||||
|     public @NonNull Collection<BlockType> getTileEntityTypes() { | ||||
|         if (this.tileEntityTypes.isEmpty()) { | ||||
|             // Categories | ||||
|             tileEntityTypes.addAll(BlockCategories.BANNERS.getAll()); | ||||
| @@ -650,46 +518,83 @@ public class BukkitUtil extends WorldUtil { | ||||
|             tileEntityTypes.addAll(BlockCategories.FLOWER_POTS.getAll()); | ||||
|             // Individual Types | ||||
|             // Add these from strings | ||||
|             Stream.of("barrel", "beacon", "beehive", "bee_nest", "bell", "blast_furnace", | ||||
|                 "brewing_stand", "campfire", "chest", "ender_chest", "trapped_chest", | ||||
|                 "command_block", "end_gateway", "hopper", "jigsaw", "jubekox", | ||||
|                 "lectern", "note_block", "black_shulker_box", "blue_shulker_box", | ||||
|                 "brown_shulker_box", "cyan_shulker_box", "gray_shulker_box", "green_shulker_box", | ||||
|                 "light_blue_shulker_box", "light_gray_shulker_box", "lime_shulker_box", | ||||
|                 "magenta_shulker_box", "orange_shulker_box", "pink_shulker_box", | ||||
|                 "purple_shulker_box", "red_shulker_box", "shulker_box", "white_shulker_box", | ||||
|                 "yellow_shulker_box", "smoker", "structure_block", "structure_void") | ||||
|                 .map(BlockTypes::get) | ||||
|                 .filter(Objects::nonNull) | ||||
|                 .forEach(tileEntityTypes::add); | ||||
|             Stream.of( | ||||
|                             "barrel", | ||||
|                             "beacon", | ||||
|                             "beehive", | ||||
|                             "bee_nest", | ||||
|                             "bell", | ||||
|                             "blast_furnace", | ||||
|                             "brewing_stand", | ||||
|                             "campfire", | ||||
|                             "chest", | ||||
|                             "ender_chest", | ||||
|                             "trapped_chest", | ||||
|                             "command_block", | ||||
|                             "end_gateway", | ||||
|                             "hopper", | ||||
|                             "jigsaw", | ||||
|                             "jubekox", | ||||
|                             "lectern", | ||||
|                             "note_block", | ||||
|                             "black_shulker_box", | ||||
|                             "blue_shulker_box", | ||||
|                             "brown_shulker_box", | ||||
|                             "cyan_shulker_box", | ||||
|                             "gray_shulker_box", | ||||
|                             "green_shulker_box", | ||||
|                             "light_blue_shulker_box", | ||||
|                             "light_gray_shulker_box", | ||||
|                             "lime_shulker_box", | ||||
|                             "magenta_shulker_box", | ||||
|                             "orange_shulker_box", | ||||
|                             "pink_shulker_box", | ||||
|                             "purple_shulker_box", | ||||
|                             "red_shulker_box", | ||||
|                             "shulker_box", | ||||
|                             "white_shulker_box", | ||||
|                             "yellow_shulker_box", | ||||
|                             "smoker", | ||||
|                             "structure_block", | ||||
|                             "structure_void" | ||||
|                     ) | ||||
|                     .map(BlockTypes::get).filter(Objects::nonNull).forEach(tileEntityTypes::add); | ||||
|         } | ||||
|         return this.tileEntityTypes; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public int getTileEntityCount(String world, BlockVector2 chunk) { | ||||
|         return Bukkit.getWorld(world).getChunkAt(chunk.getBlockX(), chunk.getBlockZ()) | ||||
|             .getTileEntities().length; | ||||
|     @NonNegative | ||||
|     public int getTileEntityCount(final @NonNull String world, final @NonNull BlockVector2 chunk) { | ||||
|         return Objects.requireNonNull(getWorld(world)). | ||||
|                 getChunkAt(chunk.getBlockX(), chunk.getBlockZ()).getTileEntities().length; | ||||
|     } | ||||
|  | ||||
|     private static void ensureLoaded(final String world, final int x, final int z, | ||||
|         final Consumer<Chunk> chunkConsumer) { | ||||
|         PaperLib.getChunkAtAsync(getWorld(world), x >> 4, z >> 4, true) | ||||
|             .thenAccept(chunk -> ensureMainThread(chunkConsumer, chunk)); | ||||
|     } | ||||
|  | ||||
|     private static void ensureLoaded(final Location location, final Consumer<Chunk> chunkConsumer) { | ||||
|         PaperLib.getChunkAtAsync(getLocation(location), true) | ||||
|             .thenAccept(chunk -> ensureMainThread(chunkConsumer, chunk)); | ||||
|     } | ||||
|  | ||||
|     private static <T> void ensureMainThread(final Consumer<T> consumer, final T value) { | ||||
|     @Override | ||||
|     public Set<BlockVector2> getChunkChunks(String world) { | ||||
|         Set<BlockVector2> chunks = super.getChunkChunks(world); | ||||
|         if (Bukkit.isPrimaryThread()) { | ||||
|             consumer.accept(value); | ||||
|             for (Chunk chunk : Objects.requireNonNull(Bukkit.getWorld(world)).getLoadedChunks()) { | ||||
|                 BlockVector2 loc = BlockVector2.at(chunk.getX() >> 5, chunk.getZ() >> 5); | ||||
|                 chunks.add(loc); | ||||
|             } | ||||
|         } else { | ||||
|             Bukkit.getScheduler() | ||||
|                 .runTask(BukkitMain.getPlugin(BukkitMain.class), () -> consumer.accept(value)); | ||||
|             final Semaphore semaphore = new Semaphore(1); | ||||
|             try { | ||||
|                 semaphore.acquire(); | ||||
|                 Bukkit.getScheduler().runTask(BukkitPlatform.getPlugin(BukkitPlatform.class), () -> { | ||||
|                     for (Chunk chunk : Objects.requireNonNull(Bukkit.getWorld(world)).getLoadedChunks()) { | ||||
|                         BlockVector2 loc = BlockVector2.at(chunk.getX() >> 5, chunk.getZ() >> 5); | ||||
|                         chunks.add(loc); | ||||
|                     } | ||||
|                     semaphore.release(); | ||||
|                 }); | ||||
|                 semaphore.acquireUninterruptibly(); | ||||
|             } catch (final Exception e) { | ||||
|                 e.printStackTrace(); | ||||
|             } | ||||
|         } | ||||
|         return chunks; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,118 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.util; | ||||
|  | ||||
| import com.google.common.collect.Maps; | ||||
| import com.plotsquared.core.location.World; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Map; | ||||
| import java.util.Objects; | ||||
|  | ||||
| public class BukkitWorld implements World<org.bukkit.World> { | ||||
|  | ||||
|     private static final Map<String, BukkitWorld> worldMap = Maps.newHashMap(); | ||||
|  | ||||
|     private final org.bukkit.World world; | ||||
|  | ||||
|     private BukkitWorld(final org.bukkit.World world) { | ||||
|         this.world = world; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get a new {@link BukkitWorld} from a world name | ||||
|      * | ||||
|      * @param worldName World name | ||||
|      * @return World instance | ||||
|      */ | ||||
|     public static @NonNull BukkitWorld of(final @NonNull String worldName) { | ||||
|         final org.bukkit.World bukkitWorld = Bukkit.getWorld(worldName); | ||||
|         if (bukkitWorld == null) { | ||||
|             throw new IllegalArgumentException(String.format("There is no world with the name '%s'", worldName)); | ||||
|         } | ||||
|         return of(bukkitWorld); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get a new {@link BukkitWorld} from a Bukkit world | ||||
|      * | ||||
|      * @param world Bukkit world | ||||
|      * @return World instance | ||||
|      */ | ||||
|     public static @NonNull BukkitWorld of(final org.bukkit.World world) { | ||||
|         BukkitWorld bukkitWorld = worldMap.get(world.getName()); | ||||
|         if (bukkitWorld != null && bukkitWorld.getPlatformWorld().equals(world)) { | ||||
|             return bukkitWorld; | ||||
|         } | ||||
|         bukkitWorld = new BukkitWorld(world); | ||||
|         worldMap.put(world.getName(), bukkitWorld); | ||||
|         return bukkitWorld; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public org.bukkit.World getPlatformWorld() { | ||||
|         return this.world; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public @NonNull String getName() { | ||||
|         return this.world.getName(); | ||||
|     } | ||||
|  | ||||
|     public boolean equals(final Object o) { | ||||
|         if (o == this) { | ||||
|             return true; | ||||
|         } | ||||
|         if (!(o instanceof final BukkitWorld other)) { | ||||
|             return false; | ||||
|         } | ||||
|         if (!other.canEqual(this)) { | ||||
|             return false; | ||||
|         } | ||||
|         if (!Objects.equals(this.world, other.world)) { | ||||
|             return false; | ||||
|         } | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     protected boolean canEqual(final Object other) { | ||||
|         return other instanceof BukkitWorld; | ||||
|     } | ||||
|  | ||||
|     public int hashCode() { | ||||
|         final int PRIME = 59; | ||||
|         int result = 1; | ||||
|         final Object $world = this.world; | ||||
|         result = result * PRIME + ($world == null ? 43 : $world.hashCode()); | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     public String toString() { | ||||
|         return "BukkitWorld(world=" + this.world + ")"; | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,19 +21,20 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import com.plotsquared.bukkit.entity.EntityWrapper; | ||||
| import com.plotsquared.bukkit.entity.ReplicatingEntityWrapper; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.location.PlotLoc; | ||||
| import com.sk89q.worldedit.bukkit.BukkitWorld; | ||||
| import com.sk89q.worldedit.math.BlockVector3; | ||||
| import com.sk89q.worldedit.regions.CuboidRegion; | ||||
| import com.sk89q.worldedit.world.block.BaseBlock; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.entity.Entity; | ||||
| @@ -46,6 +47,8 @@ import java.util.Set; | ||||
|  | ||||
| public class ContentMap { | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + ContentMap.class.getSimpleName()); | ||||
|  | ||||
|     final Set<EntityWrapper> entities; | ||||
|     final Map<PlotLoc, BaseBlock[]> allBlocks; | ||||
|  | ||||
| @@ -74,7 +77,7 @@ public class ContentMap { | ||||
|  | ||||
|     void saveEntitiesOut(Chunk chunk, CuboidRegion region) { | ||||
|         for (Entity entity : chunk.getEntities()) { | ||||
|             Location location = BukkitUtil.getLocation(entity); | ||||
|             Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|             int x = location.getX(); | ||||
|             int z = location.getZ(); | ||||
|             if (BukkitChunkManager.isIn(region, x, z)) { | ||||
| @@ -93,10 +96,12 @@ public class ContentMap { | ||||
|         saveEntitiesIn(chunk, region, 0, 0, false); | ||||
|     } | ||||
|  | ||||
|     void saveEntitiesIn(Chunk chunk, CuboidRegion region, int offsetX, int offsetZ, | ||||
|         boolean delete) { | ||||
|     void saveEntitiesIn( | ||||
|             Chunk chunk, CuboidRegion region, int offsetX, int offsetZ, | ||||
|             boolean delete | ||||
|     ) { | ||||
|         for (Entity entity : chunk.getEntities()) { | ||||
|             Location location = BukkitUtil.getLocation(entity); | ||||
|             Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|             int x = location.getX(); | ||||
|             int z = location.getZ(); | ||||
|             if (!BukkitChunkManager.isIn(region, x, z)) { | ||||
| @@ -123,8 +128,7 @@ public class ContentMap { | ||||
|             try { | ||||
|                 entity.spawn(world, xOffset, zOffset); | ||||
|             } catch (Exception e) { | ||||
|                 PlotSquared.debug("Failed to restore entity (e): " + e.toString()); | ||||
|                 e.printStackTrace(); | ||||
|                 LOGGER.error("Failed to restore entity", e); | ||||
|             } | ||||
|         } | ||||
|         this.entities.clear(); | ||||
| @@ -141,4 +145,5 @@ public class ContentMap { | ||||
|         PlotLoc loc = new PlotLoc(x + offsetX, z + offsetZ); | ||||
|         this.allBlocks.put(loc, ids); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,157 +0,0 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  * | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.OfflinePlayer; | ||||
| import org.bukkit.entity.Entity; | ||||
| import org.bukkit.entity.Player; | ||||
|  | ||||
| import java.lang.reflect.Constructor; | ||||
| import java.lang.reflect.Method; | ||||
| import java.util.UUID; | ||||
|  | ||||
| import static com.plotsquared.core.util.ReflectionUtils.callConstructor; | ||||
| import static com.plotsquared.core.util.ReflectionUtils.callMethod; | ||||
| import static com.plotsquared.core.util.ReflectionUtils.getCbClass; | ||||
| import static com.plotsquared.core.util.ReflectionUtils.getField; | ||||
| import static com.plotsquared.core.util.ReflectionUtils.getNmsClass; | ||||
| import static com.plotsquared.core.util.ReflectionUtils.getUtilClass; | ||||
| import static com.plotsquared.core.util.ReflectionUtils.makeConstructor; | ||||
| import static com.plotsquared.core.util.ReflectionUtils.makeField; | ||||
| import static com.plotsquared.core.util.ReflectionUtils.makeMethod; | ||||
|  | ||||
| public class OfflinePlayerUtil { | ||||
|  | ||||
|     public static Player loadPlayer(OfflinePlayer player) { | ||||
|         if (player == null) { | ||||
|             return null; | ||||
|         } | ||||
|         if (player instanceof Player) { | ||||
|             return (Player) player; | ||||
|         } | ||||
|         return loadPlayer(player.getUniqueId(), player.getName()); | ||||
|     } | ||||
|  | ||||
|     private static Player loadPlayer(UUID id, String name) { | ||||
|         Object server = getMinecraftServer(); | ||||
|         Object interactManager = newPlayerInteractManager(); | ||||
|         Object worldServer = getWorldServer(); | ||||
|         Object profile = newGameProfile(id, name); | ||||
|         Class<?> entityPlayerClass = getNmsClass("EntityPlayer"); | ||||
|         Constructor entityPlayerConstructor = | ||||
|                 makeConstructor(entityPlayerClass, getNmsClass("MinecraftServer"), | ||||
|                         getNmsClass("WorldServer"), getUtilClass("com.mojang.authlib.GameProfile"), | ||||
|                         getNmsClass("PlayerInteractManager")); | ||||
|         Object entityPlayer = | ||||
|                 callConstructor(entityPlayerConstructor, server, worldServer, profile, interactManager); | ||||
|         return (Player) getBukkitEntity(entityPlayer); | ||||
|     } | ||||
|  | ||||
|     private static Object newGameProfile(UUID id, String name) { | ||||
|         Class<?> gameProfileClass = getUtilClass("com.mojang.authlib.GameProfile"); | ||||
|         if (gameProfileClass == null) { //Before uuids | ||||
|             return name; | ||||
|         } | ||||
|         Constructor gameProfileConstructor = | ||||
|             makeConstructor(gameProfileClass, UUID.class, String.class); | ||||
|         if (gameProfileConstructor == null) { //Version has string constructor | ||||
|             gameProfileConstructor = makeConstructor(gameProfileClass, String.class, String.class); | ||||
|             return callConstructor(gameProfileConstructor, id.toString(), name); | ||||
|         } else { //Version has uuid constructor | ||||
|             return callConstructor(gameProfileConstructor, id, name); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private static Object newPlayerInteractManager() { | ||||
|         Object worldServer = getWorldServer(); | ||||
|         Class<?> playerInteractClass = getNmsClass("PlayerInteractManager"); | ||||
|         Class<?> worldClass = getNmsClass("World"); | ||||
|         Constructor<?> c = makeConstructor(playerInteractClass, worldClass); | ||||
|         if (c == null) { | ||||
|             c = makeConstructor(playerInteractClass, getNmsClass("WorldServer")); | ||||
|         } | ||||
|         return callConstructor(c, worldServer); | ||||
|     } | ||||
|  | ||||
|     public static Object getWorldServer116() { | ||||
|         Object server = getMinecraftServer(); | ||||
|         Class<?> minecraftServerClass = getNmsClass("MinecraftServer"); | ||||
|         Class<?> genericResourceKey = getNmsClass("ResourceKey"); | ||||
|         Class<?> minecraftKey = getNmsClass("MinecraftKey"); | ||||
|         // MinecraftKey + MinecraftKey -> ResourceKey | ||||
|         Method constructResourceKey = makeMethod(genericResourceKey, "a", minecraftKey, minecraftKey); | ||||
|         // MinecraftKey(String) | ||||
|         Constructor<Object> minecraftKeyConstructor = makeConstructor(minecraftKey, String.class); | ||||
|         // minecraft:overworld | ||||
|         Object overworldKey = callConstructor(minecraftKeyConstructor, "overworld"); | ||||
|         // minecraft:dimension | ||||
|         Object dimensionKey = callConstructor(minecraftKeyConstructor, "dimension"); | ||||
|         // ResourceKey[minecraft:dimension / minecraft:overworld] | ||||
|         Object resourceKey = callMethod(constructResourceKey, null, dimensionKey, overworldKey); | ||||
|         Method getWorldServer = | ||||
|                 makeMethod(minecraftServerClass, "getWorldServer", genericResourceKey); | ||||
|         return callMethod(getWorldServer, server, resourceKey); | ||||
|     } | ||||
|  | ||||
|     public static Object getWorldServerNew() { | ||||
|         Object server = getMinecraftServer(); | ||||
|         Class<?> minecraftServerClass = getNmsClass("MinecraftServer"); | ||||
|         Class<?> dimensionManager = getNmsClass("DimensionManager"); | ||||
|         Object overworld = getField(makeField(dimensionManager, "OVERWORLD"), null); | ||||
|         Method getWorldServer = | ||||
|             makeMethod(minecraftServerClass, "getWorldServer", dimensionManager); | ||||
|         return callMethod(getWorldServer, server, overworld); | ||||
|     } | ||||
|  | ||||
|     private static Object getWorldServer() { | ||||
|         Object server = getMinecraftServer(); | ||||
|         Class<?> minecraftServerClass = getNmsClass("MinecraftServer"); | ||||
|         Method getWorldServer = makeMethod(minecraftServerClass, "getWorldServer", int.class); | ||||
|         Object o; | ||||
|         try { | ||||
|             o = callMethod(getWorldServer, server, 0); | ||||
|         } catch (final RuntimeException e) { | ||||
|             try { | ||||
|                 o = getWorldServerNew(); | ||||
|             } catch (final RuntimeException f) { | ||||
|                 o = getWorldServer116(); | ||||
|             } | ||||
|         } | ||||
|         return o; | ||||
|     } | ||||
|  | ||||
|     //NMS Utils | ||||
|  | ||||
|     private static Object getMinecraftServer() { | ||||
|         return callMethod(makeMethod(getCbClass("CraftServer"), "getServer"), Bukkit.getServer()); | ||||
|     } | ||||
|  | ||||
|     private static Entity getBukkitEntity(Object o) { | ||||
|         Method getBukkitEntity = makeMethod(o.getClass(), "getBukkitEntity"); | ||||
|         return callMethod(getBukkitEntity, o); | ||||
|     } | ||||
| } | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| @@ -38,7 +38,7 @@ import java.util.ArrayList; | ||||
| public class SetGenCB { | ||||
|  | ||||
|     public static void setGenerator(World world) throws Exception { | ||||
|         SetupUtils.manager.updateGenerators(); | ||||
|         PlotSquared.platform().setupUtils().updateGenerators(false); | ||||
|         PlotSquared.get().removePlotAreas(world.getName()); | ||||
|         ChunkGenerator gen = world.getGenerator(); | ||||
|         if (gen == null) { | ||||
| @@ -69,9 +69,10 @@ public class SetGenCB { | ||||
|         } | ||||
|         if (!set) { | ||||
|             world.getPopulators() | ||||
|                 .removeIf(blockPopulator -> blockPopulator instanceof BukkitAugmentedGenerator); | ||||
|                     .removeIf(blockPopulator -> blockPopulator instanceof BukkitAugmentedGenerator); | ||||
|         } | ||||
|         PlotSquared.get() | ||||
|             .loadWorld(world.getName(), PlotSquared.get().IMP.getGenerator(world.getName(), null)); | ||||
|                 .loadWorld(world.getName(), PlotSquared.platform().getGenerator(world.getName(), null)); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,17 +21,19 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import com.google.gson.JsonObject; | ||||
| import com.google.gson.JsonParser; | ||||
| import com.google.gson.stream.JsonReader; | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.PlotVersion; | ||||
| import com.plotsquared.core.configuration.Captions; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.plugin.java.JavaPlugin; | ||||
| @@ -44,6 +46,8 @@ import java.net.URL; | ||||
|  | ||||
| public class UpdateUtility implements Listener { | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + UpdateUtility.class.getSimpleName()); | ||||
|  | ||||
|     public static PlotVersion internalVersion; | ||||
|     public static String spigotVersion; | ||||
|     public static boolean hasUpdate; | ||||
| @@ -51,48 +55,48 @@ public class UpdateUtility implements Listener { | ||||
|     public final JavaPlugin javaPlugin; | ||||
|     private boolean notify = true; | ||||
|  | ||||
|     @Inject | ||||
|     public UpdateUtility(final JavaPlugin javaPlugin) { | ||||
|         this.javaPlugin = javaPlugin; | ||||
|         internalVersion = PlotSquared.get().getVersion(); | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings({"deprecation", "DefaultCharset"}) // Suppress Json deprecation, we can't use features from gson 2.8.1 and newer yet | ||||
|     public void updateChecker() { | ||||
|         task = Bukkit.getScheduler().runTaskTimerAsynchronously(this.javaPlugin, () -> { | ||||
|             try { | ||||
|                 HttpsURLConnection connection = (HttpsURLConnection) new URL( | ||||
|                     "https://api.spigotmc.org/simple/0.1/index.php?action=getResource&id=77506") | ||||
|                     .openConnection(); | ||||
|                         "https://api.spigotmc.org/simple/0.1/index.php?action=getResource&id=77506") | ||||
|                         .openConnection(); | ||||
|                 connection.setRequestMethod("GET"); | ||||
|                 JsonObject result = (new JsonParser()) | ||||
|                     .parse(new JsonReader(new InputStreamReader(connection.getInputStream()))) | ||||
|                     .getAsJsonObject(); | ||||
|                 JsonObject result = new JsonParser() | ||||
|                         .parse(new JsonReader(new InputStreamReader(connection.getInputStream()))) | ||||
|                         .getAsJsonObject(); | ||||
|                 spigotVersion = result.get("current_version").getAsString(); | ||||
|             } catch (IOException e) { | ||||
|                 PlotSquared.log(Captions.PREFIX + "&cUnable to check for updates because: " + e); | ||||
|                 LOGGER.error("Unable to check for updates. Error: {}", e.getMessage()); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             if (internalVersion.isLaterVersion(spigotVersion)) { | ||||
|                 PlotSquared | ||||
|                     .log(Captions.PREFIX + "&6There appears to be a PlotSquared update available!"); | ||||
|                 PlotSquared.log( | ||||
|                     Captions.PREFIX + "&6You are running version " + internalVersion.versionString() | ||||
|                         + ", &6latest version is " + spigotVersion); | ||||
|                 PlotSquared | ||||
|                     .log(Captions.PREFIX + "&6https://www.spigotmc.org/resources/77506/updates"); | ||||
|                 LOGGER.info("There appears to be a PlotSquared update available!"); | ||||
|                 LOGGER.info("You are running version {}, the latest version is {}", | ||||
|                         internalVersion.versionString(), spigotVersion | ||||
|                 ); | ||||
|                 LOGGER.info("https://www.spigotmc.org/resources/77506/updates"); | ||||
|                 hasUpdate = true; | ||||
|                 if (Settings.UpdateChecker.NOTIFY_ONCE) { | ||||
|                     cancelTask(); | ||||
|                 } | ||||
|             } else if (notify) { | ||||
|                 notify = false; | ||||
|                 PlotSquared.log(Captions.PREFIX | ||||
|                     + "Congratulations! You are running the latest PlotSquared version."); | ||||
|                 LOGGER.info("Congratulations! You are running the latest PlotSquared version"); | ||||
|             } | ||||
|         }, 0L, Settings.UpdateChecker.POLL_RATE * 60 * 20); | ||||
|         }, 0L, (long) Settings.UpdateChecker.POLL_RATE * 60 * 20); | ||||
|     } | ||||
|  | ||||
|     private void cancelTask() { | ||||
|         Bukkit.getScheduler().runTaskLater(javaPlugin, () -> task.cancel(), 20L); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,129 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.util.fawe; | ||||
|  | ||||
| import com.fastasyncworldedit.bukkit.regions.plotsquared.FaweDelegateRegionManager; | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.bukkit.util.BukkitRegionManager; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.generator.HybridPlotManager; | ||||
| import com.plotsquared.core.inject.factory.ProgressSubscriberFactory; | ||||
| 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.PlotManager; | ||||
| import com.plotsquared.core.queue.GlobalBlockQueue; | ||||
| import com.plotsquared.core.queue.QueueCoordinator; | ||||
| import com.plotsquared.core.util.WorldUtil; | ||||
| import com.sk89q.worldedit.function.pattern.Pattern; | ||||
| import com.sk89q.worldedit.regions.CuboidRegion; | ||||
| import com.sk89q.worldedit.world.biome.BiomeType; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
|  | ||||
| import java.util.Set; | ||||
|  | ||||
| public class FaweRegionManager extends BukkitRegionManager { | ||||
|  | ||||
|     private final FaweDelegateRegionManager delegate = new FaweDelegateRegionManager(); | ||||
|  | ||||
|     @Inject | ||||
|     public FaweRegionManager( | ||||
|             @NonNull WorldUtil worldUtil, @NonNull GlobalBlockQueue blockQueue, @NonNull | ||||
|             ProgressSubscriberFactory subscriberFactory | ||||
|     ) { | ||||
|         super(worldUtil, blockQueue, subscriberFactory); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean setCuboids( | ||||
|             final @NonNull PlotArea area, | ||||
|             final @NonNull Set<CuboidRegion> regions, | ||||
|             final @NonNull Pattern blocks, | ||||
|             int minY, | ||||
|             int maxY, | ||||
|             @Nullable PlotPlayer<?> actor, | ||||
|             @Nullable QueueCoordinator queue | ||||
|     ) { | ||||
|         return delegate.setCuboids(area, regions, blocks, minY, maxY, queue.getCompleteTask()); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean notifyClear(PlotManager manager) { | ||||
|         if (!Settings.FAWE_Components.CLEAR || !(manager instanceof HybridPlotManager)) { | ||||
|             return false; | ||||
|         } | ||||
|         return delegate.notifyClear(manager); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean handleClear( | ||||
|             @NotNull Plot plot, | ||||
|             @Nullable Runnable whenDone, | ||||
|             @NotNull PlotManager manager, | ||||
|             final @Nullable PlotPlayer<?> player | ||||
|     ) { | ||||
|         if (!Settings.FAWE_Components.CLEAR || !(manager instanceof HybridPlotManager)) { | ||||
|             return false; | ||||
|         } | ||||
|         return delegate.handleClear(plot, whenDone, manager); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void swap( | ||||
|             Location pos1, | ||||
|             Location pos2, | ||||
|             Location swapPos, | ||||
|             final @Nullable PlotPlayer<?> player, | ||||
|             final Runnable whenDone | ||||
|     ) { | ||||
|         delegate.swap(pos1, pos2, swapPos, whenDone); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void setBiome(CuboidRegion region, int extendBiome, BiomeType biome, String world, Runnable whenDone) { | ||||
|         delegate.setBiome(region, extendBiome, biome, world, whenDone); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean copyRegion( | ||||
|             final @NonNull Location pos1, | ||||
|             final @NonNull Location pos2, | ||||
|             final @NonNull Location pos3, | ||||
|             final @Nullable PlotPlayer<?> player, | ||||
|             final @NonNull Runnable whenDone | ||||
|     ) { | ||||
|         return delegate.copyRegion(pos1, pos2, pos3, whenDone); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean regenerateRegion(final Location pos1, final Location pos2, boolean ignore, final Runnable whenDone) { | ||||
|         return delegate.regenerateRegion(pos1, pos2, ignore, whenDone); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,90 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.util.fawe; | ||||
|  | ||||
| import com.fastasyncworldedit.bukkit.regions.plotsquared.FaweDelegateSchematicHandler; | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.core.inject.factory.ProgressSubscriberFactory; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.schematic.Schematic; | ||||
| import com.plotsquared.core.queue.QueueCoordinator; | ||||
| import com.plotsquared.core.util.SchematicHandler; | ||||
| import com.plotsquared.core.util.WorldUtil; | ||||
| import com.plotsquared.core.util.task.RunnableVal; | ||||
| import com.sk89q.jnbt.CompoundTag; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
|  | ||||
| import java.io.InputStream; | ||||
| import java.net.URL; | ||||
| import java.util.UUID; | ||||
|  | ||||
| public class FaweSchematicHandler extends SchematicHandler { | ||||
|  | ||||
|     private final FaweDelegateSchematicHandler delegate = new FaweDelegateSchematicHandler(); | ||||
|  | ||||
|     @Inject | ||||
|     public FaweSchematicHandler(@NotNull WorldUtil worldUtil, @NotNull ProgressSubscriberFactory subscriberFactory) { | ||||
|         super(worldUtil, subscriberFactory); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean restoreTile(QueueCoordinator queue, CompoundTag tag, int x, int y, int z) { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void paste( | ||||
|             final Schematic schematic, | ||||
|             final Plot plot, | ||||
|             final int xOffset, | ||||
|             final int yOffset, | ||||
|             final int zOffset, | ||||
|             final boolean autoHeight, | ||||
|             final PlotPlayer<?> actor, | ||||
|             final RunnableVal<Boolean> whenDone | ||||
|     ) { | ||||
|         delegate.paste(schematic, plot, xOffset, yOffset, zOffset, autoHeight, whenDone); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean save(CompoundTag tag, String path) { | ||||
|         return delegate.save(tag, path); | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("removal") // Just the override | ||||
|     @Override | ||||
|     public void upload(final CompoundTag tag, final UUID uuid, final String file, final RunnableVal<URL> whenDone) { | ||||
|         delegate.upload(tag, uuid, file, whenDone); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public Schematic getSchematic(@NotNull InputStream is) { | ||||
|         return delegate.getSchematic(is); | ||||
|     } | ||||
|  | ||||
| } | ||||
|  | ||||
| @@ -0,0 +1,49 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.util.task; | ||||
|  | ||||
| import com.plotsquared.core.util.task.PlotSquaredTask; | ||||
| import org.bukkit.scheduler.BukkitRunnable; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| /** | ||||
|  * Bukkit implementation of {@link PlotSquaredTask} | ||||
|  */ | ||||
| public final class BukkitPlotSquaredTask extends BukkitRunnable implements PlotSquaredTask { | ||||
|  | ||||
|     @NonNull | ||||
|     private final Runnable runnable; | ||||
|  | ||||
|     public BukkitPlotSquaredTask(final @NonNull Runnable runnable) { | ||||
|         this.runnable = runnable; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void runTask() { | ||||
|         this.runnable.run(); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,128 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.util.task; | ||||
|  | ||||
| import com.google.inject.Inject; | ||||
| import com.google.inject.Singleton; | ||||
| import com.plotsquared.bukkit.BukkitPlatform; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.util.task.PlotSquaredTask; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import com.plotsquared.core.util.task.TaskTime; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.concurrent.Callable; | ||||
| import java.util.concurrent.Future; | ||||
| import java.util.concurrent.TimeUnit; | ||||
|  | ||||
| /** | ||||
|  * Bukkit implementation of {@link TaskManager} using | ||||
|  * by {@link org.bukkit.scheduler.BukkitScheduler} and {@link BukkitPlotSquaredTask} | ||||
|  */ | ||||
| @Singleton | ||||
| public class BukkitTaskManager extends TaskManager { | ||||
|  | ||||
|     private final BukkitPlatform bukkitMain; | ||||
|     private final TaskTime.TimeConverter timeConverter; | ||||
|  | ||||
|     @Inject | ||||
|     public BukkitTaskManager( | ||||
|             final @NonNull BukkitPlatform bukkitMain, | ||||
|             final TaskTime.@NonNull TimeConverter timeConverter | ||||
|     ) { | ||||
|         this.bukkitMain = bukkitMain; | ||||
|         this.timeConverter = timeConverter; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public PlotSquaredTask taskRepeat( | ||||
|             final @NonNull Runnable runnable, | ||||
|             final @NonNull TaskTime taskTime | ||||
|     ) { | ||||
|         final long ticks = this.timeConverter.toTicks(taskTime); | ||||
|         final BukkitPlotSquaredTask bukkitPlotSquaredTask = new BukkitPlotSquaredTask(runnable); | ||||
|         bukkitPlotSquaredTask.runTaskTimer(this.bukkitMain, ticks, ticks); | ||||
|         return bukkitPlotSquaredTask; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public PlotSquaredTask taskRepeatAsync( | ||||
|             final @NonNull Runnable runnable, | ||||
|             final @NonNull TaskTime taskTime | ||||
|     ) { | ||||
|         final long ticks = this.timeConverter.toTicks(taskTime); | ||||
|         final BukkitPlotSquaredTask bukkitPlotSquaredTask = new BukkitPlotSquaredTask(runnable); | ||||
|         bukkitPlotSquaredTask.runTaskTimerAsynchronously(this.bukkitMain, ticks, ticks); | ||||
|         return bukkitPlotSquaredTask; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void taskAsync(final @NonNull Runnable runnable) { | ||||
|         if (this.bukkitMain.isEnabled()) { | ||||
|             new BukkitPlotSquaredTask(runnable).runTaskAsynchronously(this.bukkitMain); | ||||
|         } else { | ||||
|             runnable.run(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public <T> T sync(final @NonNull Callable<T> function, final int timeout) throws Exception { | ||||
|         if (PlotSquared.get().isMainThread(Thread.currentThread())) { | ||||
|             return function.call(); | ||||
|         } | ||||
|         return this.callMethodSync(function).get(timeout, TimeUnit.MILLISECONDS); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public <T> Future<T> callMethodSync(final @NonNull Callable<T> method) { | ||||
|         return Bukkit.getScheduler().callSyncMethod(this.bukkitMain, method); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void task(final @NonNull Runnable runnable) { | ||||
|         new BukkitPlotSquaredTask(runnable).runTask(this.bukkitMain); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void taskLater( | ||||
|             final @NonNull Runnable runnable, | ||||
|             final @NonNull TaskTime taskTime | ||||
|     ) { | ||||
|         final long delay = this.timeConverter.toTicks(taskTime); | ||||
|         new BukkitPlotSquaredTask(runnable).runTaskLater(this.bukkitMain, delay); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void taskLaterAsync( | ||||
|             final @NonNull Runnable runnable, | ||||
|             final @NonNull TaskTime taskTime | ||||
|     ) { | ||||
|         final long delay = this.timeConverter.toTicks(taskTime); | ||||
|         new BukkitPlotSquaredTask(runnable).runTaskLaterAsynchronously(this.bukkitMain, delay); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,50 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.util.task; | ||||
|  | ||||
| import com.plotsquared.core.util.task.TaskTime; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.checkerframework.checker.index.qual.NonNegative; | ||||
|  | ||||
| /** | ||||
|  * Time converter that uses the server MSPT count to convert between | ||||
|  * different time units | ||||
|  */ | ||||
| public final class PaperTimeConverter implements TaskTime.TimeConverter { | ||||
|  | ||||
|     private static final long MIN_MS_PER_TICKS = 50L; | ||||
|  | ||||
|     @Override | ||||
|     public long msToTicks(@NonNegative final long ms) { | ||||
|         return Math.max(1L, (long) (ms / Math.max(MIN_MS_PER_TICKS, Bukkit.getAverageTickTime()))); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public long ticksToMs(@NonNegative final long ticks) { | ||||
|         return Math.max(1L, (long) (ticks * Math.max(MIN_MS_PER_TICKS, Bukkit.getAverageTickTime()))); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,48 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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.util.task; | ||||
|  | ||||
| import com.plotsquared.core.util.task.TaskTime; | ||||
| import org.checkerframework.checker.index.qual.NonNegative; | ||||
|  | ||||
| /** | ||||
|  * Naive time converter that assumes that all ticks are 50 milliseconds | ||||
|  */ | ||||
| public final class SpigotTimeConverter implements TaskTime.TimeConverter { | ||||
|  | ||||
|     private static final long MS_PER_TICKS = 50L; | ||||
|  | ||||
|     @Override | ||||
|     public long msToTicks(@NonNegative final long ms) { | ||||
|         return Math.max(1L, ms / MS_PER_TICKS); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public long ticksToMs(@NonNegative final long ticks) { | ||||
|         return Math.max(1L, ticks * MS_PER_TICKS); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.uuid; | ||||
|  | ||||
| @@ -29,7 +29,7 @@ import com.earth2me.essentials.Essentials; | ||||
| import com.earth2me.essentials.User; | ||||
| import com.plotsquared.core.uuid.UUIDMapping; | ||||
| import com.plotsquared.core.uuid.UUIDService; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collections; | ||||
| @@ -47,11 +47,13 @@ public class EssentialsUUIDService implements UUIDService { | ||||
|         this.essentials = Essentials.getPlugin(Essentials.class); | ||||
|     } | ||||
|  | ||||
|     @Override @NotNull public List<UUIDMapping> getNames(@NotNull final List<UUID> uuids) { | ||||
|     @Override | ||||
|     public @NonNull List<UUIDMapping> getNames(final @NonNull List<UUID> uuids) { | ||||
|         return Collections.emptyList(); | ||||
|     } | ||||
|  | ||||
|     @Override @NotNull public List<UUIDMapping> getUUIDs(@NotNull final List<String> usernames) { | ||||
|     @Override | ||||
|     public @NonNull List<UUIDMapping> getUUIDs(final @NonNull List<String> usernames) { | ||||
|         final List<UUIDMapping> mappings = new ArrayList<>(usernames.size()); | ||||
|         for (final String username : usernames) { | ||||
|             try { | ||||
| @@ -62,7 +64,8 @@ public class EssentialsUUIDService implements UUIDService { | ||||
|                         mappings.add(new UUIDMapping(uuid, username)); | ||||
|                     } | ||||
|                 } | ||||
|             } catch (final Exception ignored){} | ||||
|             } catch (final Exception ignored) { | ||||
|             } | ||||
|         } | ||||
|         return mappings; | ||||
|     } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.uuid; | ||||
|  | ||||
| @@ -31,7 +31,7 @@ import net.luckperms.api.LuckPerms; | ||||
| import net.luckperms.api.model.user.UserManager; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.plugin.RegisteredServiceProvider; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| @@ -53,21 +53,24 @@ public class LuckPermsUUIDService implements UUIDService { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override @NotNull public List<UUIDMapping> getNames(@NotNull final List<UUID> uuids) { | ||||
|        final List<UUIDMapping> mappings = new ArrayList<>(uuids.size()); | ||||
|        final UserManager userManager = this.luckPerms.getUserManager(); | ||||
|        for (final UUID uuid : uuids) { | ||||
|            try { | ||||
|                final String username = userManager.lookupUsername(uuid).get(); | ||||
|                if (username != null) { | ||||
|                    mappings.add(new UUIDMapping(uuid, username)); | ||||
|                } | ||||
|            } catch (final Exception ignored) {} | ||||
|        } | ||||
|        return mappings; | ||||
|     @Override | ||||
|     public @NonNull List<UUIDMapping> getNames(final @NonNull List<UUID> uuids) { | ||||
|         final List<UUIDMapping> mappings = new ArrayList<>(uuids.size()); | ||||
|         final UserManager userManager = this.luckPerms.getUserManager(); | ||||
|         for (final UUID uuid : uuids) { | ||||
|             try { | ||||
|                 final String username = userManager.lookupUsername(uuid).get(); | ||||
|                 if (username != null) { | ||||
|                     mappings.add(new UUIDMapping(uuid, username)); | ||||
|                 } | ||||
|             } catch (final Exception ignored) { | ||||
|             } | ||||
|         } | ||||
|         return mappings; | ||||
|     } | ||||
|  | ||||
|     @Override @NotNull public List<UUIDMapping> getUUIDs(@NotNull final List<String> usernames) { | ||||
|     @Override | ||||
|     public @NonNull List<UUIDMapping> getUUIDs(final @NonNull List<String> usernames) { | ||||
|         final List<UUIDMapping> mappings = new ArrayList<>(usernames.size()); | ||||
|         final UserManager userManager = this.luckPerms.getUserManager(); | ||||
|         for (final String username : usernames) { | ||||
| @@ -76,7 +79,8 @@ public class LuckPermsUUIDService implements UUIDService { | ||||
|                 if (username != null) { | ||||
|                     mappings.add(new UUIDMapping(uuid, username)); | ||||
|                 } | ||||
|             } catch (final Exception ignored) {} | ||||
|             } catch (final Exception ignored) { | ||||
|             } | ||||
|         } | ||||
|         return mappings; | ||||
|     } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.uuid; | ||||
|  | ||||
| @@ -31,7 +31,7 @@ import com.plotsquared.core.uuid.UUIDMapping; | ||||
| import com.plotsquared.core.uuid.UUIDService; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.OfflinePlayer; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collections; | ||||
| @@ -43,7 +43,8 @@ import java.util.UUID; | ||||
|  */ | ||||
| public class OfflinePlayerUUIDService implements UUIDService { | ||||
|  | ||||
|     @Override @NotNull public List<UUIDMapping> getNames(@NotNull final List<UUID> uuids) { | ||||
|     @Override | ||||
|     public @NonNull List<UUIDMapping> getNames(final @NonNull List<UUID> uuids) { | ||||
|         if (Settings.UUID.FORCE_LOWERCASE || Bukkit.getWorlds().isEmpty()) { | ||||
|             return Collections.emptyList(); // This is useless now | ||||
|         } | ||||
| @@ -54,14 +55,16 @@ public class OfflinePlayerUUIDService implements UUIDService { | ||||
|                 if (offlinePlayer.hasPlayedBefore()) { | ||||
|                     wrappers.add(new UUIDMapping(uuid, offlinePlayer.getName())); | ||||
|                 } | ||||
|             } catch (final Exception ignored) {} /* This can be safely ignored. If this happens, it is | ||||
|             } catch (final Exception ignored) { | ||||
|             } /* This can be safely ignored. If this happens, it is | ||||
|                                                     probably because it's called before the worlds have | ||||
|                                                     been loaded. This is bad, but does not break anything */ | ||||
|         } | ||||
|         return wrappers; | ||||
|     } | ||||
|  | ||||
|     @Override @NotNull public List<UUIDMapping> getUUIDs(@NotNull final List<String> usernames) { | ||||
|     @Override | ||||
|     public @NonNull List<UUIDMapping> getUUIDs(final @NonNull List<String> usernames) { | ||||
|         final List<UUIDMapping> wrappers = new ArrayList<>(usernames.size()); | ||||
|         for (final String username : usernames) { | ||||
|             if (Settings.UUID.OFFLINE) { | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,7 +21,7 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.uuid; | ||||
|  | ||||
| @@ -29,7 +29,7 @@ import com.destroystokyo.paper.profile.PlayerProfile; | ||||
| import com.plotsquared.core.uuid.UUIDMapping; | ||||
| import com.plotsquared.core.uuid.UUIDService; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| @@ -40,7 +40,8 @@ import java.util.UUID; | ||||
|  */ | ||||
| public class PaperUUIDService implements UUIDService { | ||||
|  | ||||
|     @Override @NotNull public List<UUIDMapping> getNames(@NotNull final List<UUID> uuids) { | ||||
|     @Override | ||||
|     public @NonNull List<UUIDMapping> getNames(final @NonNull List<UUID> uuids) { | ||||
|         final List<UUIDMapping> mappings = new ArrayList<>(uuids.size()); | ||||
|         for (final UUID uuid : uuids) { | ||||
|             final PlayerProfile playerProfile = Bukkit.createProfile(uuid); | ||||
| @@ -51,7 +52,8 @@ public class PaperUUIDService implements UUIDService { | ||||
|         return mappings; | ||||
|     } | ||||
|  | ||||
|     @Override @NotNull public List<UUIDMapping> getUUIDs(@NotNull final List<String> usernames) { | ||||
|     @Override | ||||
|     public @NonNull List<UUIDMapping> getUUIDs(final @NonNull List<String> usernames) { | ||||
|         final List<UUIDMapping> mappings = new ArrayList<>(usernames.size()); | ||||
|         for (final String username : usernames) { | ||||
|             final PlayerProfile playerProfile = Bukkit.createProfile(username); | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *                  Copyright (C) 2021 IntellectualSites | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * | ||||
|  *     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 | ||||
| @@ -21,16 +21,16 @@ | ||||
|  *     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 <http://www.gnu.org/licenses/>. | ||||
|  *     along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.uuid; | ||||
|  | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.database.SQLite; | ||||
| import com.plotsquared.core.util.MainUtil; | ||||
| import com.plotsquared.core.util.FileUtils; | ||||
| import com.plotsquared.core.uuid.UUIDMapping; | ||||
| import com.plotsquared.core.uuid.UUIDService; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.sql.Connection; | ||||
| import java.sql.PreparedStatement; | ||||
| @@ -51,7 +51,7 @@ public class SQLiteUUIDService implements UUIDService, Consumer<List<UUIDMapping | ||||
|  | ||||
|     public SQLiteUUIDService(final String fileName) { | ||||
|         this.sqlite = | ||||
|             new SQLite(MainUtil.getFile(PlotSquared.get().IMP.getDirectory(), fileName)); | ||||
|                 new SQLite(FileUtils.getFile(PlotSquared.platform().getDirectory(), fileName)); | ||||
|         try { | ||||
|             this.sqlite.openConnection(); | ||||
|         } catch (ClassNotFoundException | SQLException e) { | ||||
| @@ -59,7 +59,7 @@ public class SQLiteUUIDService implements UUIDService, Consumer<List<UUIDMapping | ||||
|         } | ||||
|  | ||||
|         try (PreparedStatement stmt = getConnection().prepareStatement( | ||||
|             "CREATE TABLE IF NOT EXISTS `usercache` (uuid VARCHAR(32) NOT NULL, username VARCHAR(32) NOT NULL, PRIMARY KEY (uuid))")) { | ||||
|                 "CREATE TABLE IF NOT EXISTS `usercache` (uuid VARCHAR(32) NOT NULL, username VARCHAR(32) NOT NULL, PRIMARY KEY (uuid))")) { | ||||
|             stmt.execute(); | ||||
|         } catch (SQLException e) { | ||||
|             e.printStackTrace(); | ||||
| @@ -72,10 +72,11 @@ public class SQLiteUUIDService implements UUIDService, Consumer<List<UUIDMapping | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override @NotNull public List<UUIDMapping> getNames(@NotNull final List<UUID> uuids) { | ||||
|     @Override | ||||
|     public @NonNull List<UUIDMapping> getNames(final @NonNull List<UUID> uuids) { | ||||
|         final List<UUIDMapping> mappings = new ArrayList<>(uuids.size()); | ||||
|         try (final PreparedStatement statement = getConnection() | ||||
|             .prepareStatement("SELECT `username` FROM `usercache` WHERE `uuid` = ?")) { | ||||
|                 .prepareStatement("SELECT `username` FROM `usercache` WHERE `uuid` = ?")) { | ||||
|             for (final UUID uuid : uuids) { | ||||
|                 statement.setString(1, uuid.toString()); | ||||
|                 try (final ResultSet resultSet = statement.executeQuery()) { | ||||
| @@ -90,16 +91,19 @@ public class SQLiteUUIDService implements UUIDService, Consumer<List<UUIDMapping | ||||
|         return mappings; | ||||
|     } | ||||
|  | ||||
|     @Override @NotNull public List<UUIDMapping> getUUIDs(@NotNull List<String> usernames) { | ||||
|     @Override | ||||
|     public @NonNull List<UUIDMapping> getUUIDs(@NonNull List<String> usernames) { | ||||
|         final List<UUIDMapping> mappings = new ArrayList<>(usernames.size()); | ||||
|         try (final PreparedStatement statement = getConnection() | ||||
|             .prepareStatement("SELECT `uuid` FROM `usercache` WHERE `username` = ?")) { | ||||
|                 .prepareStatement("SELECT `uuid` FROM `usercache` WHERE `username` = ?")) { | ||||
|             for (final String username : usernames) { | ||||
|                 statement.setString(1, username); | ||||
|                 try (final ResultSet resultSet = statement.executeQuery()) { | ||||
|                     if (resultSet.next()) { | ||||
|                         mappings.add(new UUIDMapping(UUID.fromString(resultSet.getString("uuid")), | ||||
|                             username)); | ||||
|                         mappings.add(new UUIDMapping( | ||||
|                                 UUID.fromString(resultSet.getString("uuid")), | ||||
|                                 username | ||||
|                         )); | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
| @@ -109,9 +113,10 @@ public class SQLiteUUIDService implements UUIDService, Consumer<List<UUIDMapping | ||||
|         return mappings; | ||||
|     } | ||||
|  | ||||
|     @Override public void accept(final List<UUIDMapping> uuidWrappers) { | ||||
|     @Override | ||||
|     public void accept(final List<UUIDMapping> uuidWrappers) { | ||||
|         try (final PreparedStatement statement = getConnection() | ||||
|             .prepareStatement("INSERT OR REPLACE INTO `usercache` (`uuid`, `username`) VALUES(?, ?)")) { | ||||
|                 .prepareStatement("INSERT OR REPLACE INTO `usercache` (`uuid`, `username`) VALUES(?, ?)")) { | ||||
|             for (final UUIDMapping mapping : uuidWrappers) { | ||||
|                 statement.setString(1, mapping.getUuid().toString()); | ||||
|                 statement.setString(2, mapping.getUsername()); | ||||
| @@ -127,7 +132,7 @@ public class SQLiteUUIDService implements UUIDService, Consumer<List<UUIDMapping | ||||
|      * | ||||
|      * @return All read mappings | ||||
|      */ | ||||
|     @NotNull public List<UUIDMapping> getAll() { | ||||
|     public @NonNull List<UUIDMapping> getAll() { | ||||
|         final List<UUIDMapping> mappings = new LinkedList<>(); | ||||
|         try (final PreparedStatement statement = getConnection().prepareStatement("SELECT * FROM `usercache`")) { | ||||
|             try (final ResultSet resultSet = statement.executeQuery()) { | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user