mirror of
				https://github.com/IntellectualSites/PlotSquared.git
				synced 2025-10-25 07:33:44 +02:00 
			
		
		
		
	Compare commits
	
		
			962 Commits
		
	
	
		
			legacy/v5
			...
			feature/v6
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | d629e9eecb | ||
|   | 2802939112 | ||
|   | bf646be482 | ||
|   | cc7e17960b | ||
|   | 3c75b170f0 | ||
|   | a79c474957 | ||
|   | 4bb480a238 | ||
|   | 9ffa935c0c | ||
|   | 0a32268784 | ||
|   | ae3b8c06f6 | ||
|   | 713c4ad0d2 | ||
|   | 48386c0828 | ||
|   | 625b3921e1 | ||
|   | 48aa37d173 | ||
|   | 2c2314e95c | ||
|   | 5eb2fc3ad0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 82cd9a092c | ||
|   | 62754362c2 | ||
|   | 2e9dfd6f6f | ||
|   | fc9fe1462f | ||
|   | 41f546ca6b | ||
|   | d037da33cb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | dc2d08c67e | ||
|   | 96dfc27411 | ||
|   | 171d2e5e99 | ||
|   | 4433892431 | ||
|   | 98a07dad1b | ||
|   | 0ffa22b7a6 | ||
|   | 60a0129fe9 | ||
|   | d5f8a0842b | ||
|   | f7d55ce105 | ||
|   | 85911646f3 | ||
|   | 8b75dece69 | ||
|   | 7d6e515ba8 | ||
|   | 13d7357c85 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 16e26b910c | ||
|   | faadebd30e | ||
|   | 2aeacb3dcf | ||
|   | 9db7791835 | ||
|   | f49ddb819d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d71c62771e | ||
|   | a6aaa9538f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0974fb2834 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8982b33b6c | ||
|   | a7b3b3b7df | ||
|   | 888bb20e78 | ||
|   | b11bb6fa22 | ||
|   | e5764b958d | ||
|   | bfe3141ff1 | ||
|   | 73c82deeb0 | ||
|   | 38682ecff6 | ||
|   | 6a54dc7eff | ||
|   | 8454c29c91 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b2c9311a47 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c4aa497a2b | ||
|   | c13f544390 | ||
|   | c28177d6af | ||
|   | 8a80f252cf | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 93571c72d1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6fd7379221 | ||
|   | dc5c80d812 | ||
|   | 96e9a61e7c | ||
|   | b9bd9b81e6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ec77812879 | ||
|   | efc2083798 | ||
|   | 3a6f845c01 | ||
|   | 7efd42ae45 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f43f4cbf5d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 104cc4b7a4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 218be43143 | ||
|   | 308dba5601 | ||
|   | 5f233bb5d1 | ||
|   | 5e188d114f | ||
|   | 660e05d27f | ||
|   | 4c0ad148bf | ||
|   | 5c7bfb988c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b46f486680 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 05797d0c78 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 36ef2cf9ba | ||
|   | 26ec31a012 | ||
|   | fd4a542062 | ||
|   | dc13f2565f | ||
|   | 9f632af0ae | ||
|   | d698c6a1e5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 40e1bd9897 | ||
|   | 75f31c5bf6 | ||
|   | 954c813cef | ||
|   | 05e055e9cf | ||
|   | 2ea21c150f | ||
|   | c2fd4edad5 | ||
|   | 78b8696778 | ||
|   | e653961385 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b21d12fd52 | ||
|   | 7557df96c7 | ||
|   | 61797c3aff | ||
|   | 78125ff1e2 | ||
|   | d7c8715b25 | ||
|   | 34f005c244 | ||
|   | 6fbd1376ca | ||
|   | 951767dc64 | ||
|   | 16928b05f1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fff14b05cb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e619c867e9 | ||
|   | 543284e016 | ||
|   | 0d78ba5f35 | ||
|   | 9ba2b62fc2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f10ee27fdd | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3278ce1fe9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fefb0334bd | ||
|   | d06a827e31 | ||
|   | 10bb520f3a | ||
|   | 126aa53b61 | ||
|   | ac71046feb | ||
|   | 7c290e6bd0 | ||
|   | 4d297cc829 | ||
|   | 5ab410a5c5 | ||
|   | 1f28bac955 | ||
|   | 92c54de5e9 | ||
|   | ff5d79699d | ||
|   | a0594c19ee | ||
|   | da41c136fe | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6a5859ee0f | ||
|   | 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'] |  | ||||||
							
								
								
									
										15
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										15
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							| @@ -7,7 +7,8 @@ body: | |||||||
|     attributes: |     attributes: | ||||||
|       value: | |       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. |         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://intellectualsites.github.io/plotsquared-documentation/). | ||||||
|  |         Do NOT use the public issue tracker to report security vulnerabilities! They are disclosed using [this](https://forms.gle/btgdRn9yhGtzEiGW8) form! | ||||||
|  |  | ||||||
|   - type: dropdown |   - type: dropdown | ||||||
|     attributes: |     attributes: | ||||||
| @@ -17,8 +18,6 @@ body: | |||||||
|       options: |       options: | ||||||
|         - Paper |         - Paper | ||||||
|         - Spigot |         - Spigot | ||||||
|         - Tuinity |  | ||||||
|         - Purpur |  | ||||||
|     validations: |     validations: | ||||||
|       required: true |       required: true | ||||||
|  |  | ||||||
| @@ -28,10 +27,12 @@ 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. |       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 |       multiple: false | ||||||
|       options: |       options: | ||||||
|  |         - '1.19' | ||||||
|  |         - '1.18.2' | ||||||
|  |         - '1.18.1' | ||||||
|  |         - '1.18' | ||||||
|  |         - '1.17.1' | ||||||
|         - '1.16.5' |         - '1.16.5' | ||||||
|         - '1.15.2' |  | ||||||
|         - '1.14.4' |  | ||||||
|         - '1.13.2' |  | ||||||
|     validations: |     validations: | ||||||
|       required: true |       required: true | ||||||
|  |  | ||||||
| @@ -99,4 +100,4 @@ body: | |||||||
|   - type: textarea |   - type: textarea | ||||||
|     attributes: |     attributes: | ||||||
|       label: Anything else? |       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 |     url: https://discord.gg/intellectualsites | ||||||
|     about: Our support Discord, please ask questions and seek support here. |     about: Our support Discord, please ask questions and seek support here. | ||||||
|   - name: PlotSquared Wiki |   - name: PlotSquared Wiki | ||||||
|     url: https://github.com/IntellectualSites/PlotSquared/wiki |     url: https://intellectualsites.github.io/plotsquared-documentation/ | ||||||
|     about: Take a look at the wiki page for instructions how to setup PlotSquared and use its commands. |     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: |     attributes: | ||||||
|       value: | |       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. |         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://intellectualsites.github.io/plotsquared-documentation/). | ||||||
|  |  | ||||||
|   - type: textarea |   - type: textarea | ||||||
|     attributes: |     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 | daysUntilClose: 7 | ||||||
| only: issues | only: issues | ||||||
| exemptLabels: | exemptLabels: | ||||||
|   | |||||||
							
								
								
									
										54
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										54
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,20 +1,42 @@ | |||||||
| name: "build" | name: build | ||||||
|  |  | ||||||
|  | on: [ pull_request, push ] | ||||||
|  |  | ||||||
| on: ["pull_request", "push"] |  | ||||||
|   |  | ||||||
| jobs: | jobs: | ||||||
|   build: |   build: | ||||||
|     strategy: |     if: ${{ github.event_name != 'pull_request' || github.repository != github.event.pull_request.head.repo.full_name }} | ||||||
|       matrix: |     runs-on: ubuntu-latest | ||||||
|         java: ["1.8", "11"] |  | ||||||
|         os: ["ubuntu-18.04"] |  | ||||||
|     runs-on: "${{ matrix.os }}" |  | ||||||
|     steps: |     steps: | ||||||
|     - name: "Checkout Repository" |       - name: Checkout Repository | ||||||
|       uses: "actions/checkout@v2.3.4" |         uses: actions/checkout@v3 | ||||||
|     - name: "Setup JDK ${{ matrix.java }}" |       - name: Validate Gradle Wrapper" | ||||||
|       uses: "actions/setup-java@v1.4.3" |         uses: gradle/wrapper-validation-action@v1 | ||||||
|       with: |       - name: Setup Java | ||||||
|           java-version: "${{ matrix.java }}" |         uses: actions/setup-java@v3 | ||||||
|     - name: "Clean Build" |         with: | ||||||
|       run: "./gradlew clean build" |           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 }} | ||||||
|   | |||||||
							
								
								
									
										37
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | name: "CodeQL" | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: [ v6 ] | ||||||
|  |   pull_request: | ||||||
|  |     # The branches below must be a subset of the branches above | ||||||
|  |     branches: [ v6 ] | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   analyze: | ||||||
|  |     name: Analyze | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     permissions: | ||||||
|  |       actions: read | ||||||
|  |       contents: read | ||||||
|  |       security-events: write | ||||||
|  |  | ||||||
|  |     strategy: | ||||||
|  |       fail-fast: false | ||||||
|  |       matrix: | ||||||
|  |         language: [ 'java' ] | ||||||
|  |  | ||||||
|  |     steps: | ||||||
|  |       - name: Checkout repository | ||||||
|  |         uses: actions/checkout@v3 | ||||||
|  |  | ||||||
|  |       - name: Initialize CodeQL | ||||||
|  |         uses: github/codeql-action/init@v2 | ||||||
|  |         with: | ||||||
|  |           languages: ${{ matrix.language }} | ||||||
|  |  | ||||||
|  |       - name: Autobuild | ||||||
|  |         uses: github/codeql-action/autobuild@v2 | ||||||
|  |  | ||||||
|  |       - name: Perform CodeQL Analysis | ||||||
|  |         uses: github/codeql-action/analyze@v2 | ||||||
							
								
								
									
										19
									
								
								.github/workflows/release-drafter.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								.github/workflows/release-drafter.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | |||||||
|  | name: draft release | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - v6 | ||||||
|  |   pull_request: | ||||||
|  |     types: [ opened, reopened, synchronize ] | ||||||
|  |   pull_request_target: | ||||||
|  |     types: [ opened, reopened, synchronize ] | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   update_release_draft: | ||||||
|  |     if: ${{ github.event_name != 'pull_request' || github.repository != github.event.pull_request.head.repo.full_name }} | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - uses: release-drafter/release-drafter@v5 | ||||||
|  |         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" |  | ||||||
							
								
								
									
										18
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										18
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -2,11 +2,7 @@ | |||||||
| *.cmd | *.cmd | ||||||
| *.sh | *.sh | ||||||
| *.prefs | *.prefs | ||||||
| Sponge/build |  | ||||||
| Core/build | Core/build | ||||||
| Bukkit/build |  | ||||||
| Nukkit/build |  | ||||||
| buildSrc/ |  | ||||||
|  |  | ||||||
| ### Maven ### | ### Maven ### | ||||||
| /mvn | /mvn | ||||||
| @@ -45,7 +41,8 @@ hs_err_pid* | |||||||
| *.iml | *.iml | ||||||
|  |  | ||||||
| ## Directory-based project format: | ## Directory-based project format: | ||||||
| .idea/ | /.idea/* | ||||||
|  | !/.idea/icon.svg | ||||||
| # if you remove the above rule, at least ignore the following: | # if you remove the above rule, at least ignore the following: | ||||||
|  |  | ||||||
| # User-specific stuff: | # User-specific stuff: | ||||||
| @@ -78,9 +75,6 @@ hs_err_pid* | |||||||
| # IntelliJ | # IntelliJ | ||||||
| /out/ | /out/ | ||||||
|  |  | ||||||
| # mpeltonen/sbt-idea plugin |  | ||||||
| .idea_modules/ |  | ||||||
|  |  | ||||||
| # JIRA plugin | # JIRA plugin | ||||||
| atlassian-ide-plugin.xml | atlassian-ide-plugin.xml | ||||||
|  |  | ||||||
| @@ -134,12 +128,12 @@ local.properties | |||||||
| # STS (Spring Tool Suite) | # STS (Spring Tool Suite) | ||||||
| .springBeans | .springBeans | ||||||
| /target/ | /target/ | ||||||
| Nukkit/build/classes/ |  | ||||||
| Nukkit/build/dependency-cache/ |  | ||||||
| checkstyle.xml | checkstyle.xml | ||||||
| classes/ | classes/ | ||||||
| p2error.txt |  | ||||||
| *.bat | *.bat | ||||||
| Nukkit/build/resources/main/plugin.yml |  | ||||||
|  | # Other | ||||||
| docs/ | docs/ | ||||||
| build/ | build/ | ||||||
|  |  | ||||||
|  | .DS_Store | ||||||
|   | |||||||
							
								
								
									
										50
									
								
								.idea/icon.svg
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								.idea/icon.svg
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||||
|  | <svg xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:cc="http://creativecommons.org/ns#" xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns:svg="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg" xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" version="1.1" id="svg2" xml:space="preserve" width="4000" height="3333.3333" viewBox="0 0 4000 3333.3333" sodipodi:docname="PlotSquared.ai"> | ||||||
|  | 		<metadata id="metadata8"> | ||||||
|  | 			<rdf:RDF> | ||||||
|  | 				<cc:Work rdf:about=""> | ||||||
|  | 					<dc:format>image/svg+xml</dc:format> | ||||||
|  |                     <dc:type rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||||
|  | 				</cc:Work> | ||||||
|  | 			</rdf:RDF> | ||||||
|  | 		</metadata> | ||||||
|  |     <defs id="defs6"> | ||||||
|  | 			<clipPath clipPathUnits="userSpaceOnUse" id="clipPath18"> | ||||||
|  | 				<path d="M 0,2500 H 3000 V 0 H 0 Z" id="path16" /> | ||||||
|  | 			</clipPath> | ||||||
|  | 		</defs> | ||||||
|  |     <sodipodi:namedview pagecolor="#ffffff" bordercolor="#666666" borderopacity="1" objecttolerance="10" gridtolerance="10" guidetolerance="10" inkscape:pageopacity="0" inkscape:pageshadow="2" inkscape:window-width="640" inkscape:window-height="480" id="namedview4" /> | ||||||
|  |     <g id="g10" inkscape:groupmode="layer" inkscape:label="PlotSquared" transform="matrix(1.3333333,0,0,-1.3333333,0,3333.3333)"> | ||||||
|  | 			<g id="g12"> | ||||||
|  | 				<g id="g14" clip-path="url(#clipPath18)"> | ||||||
|  | 					<g id="g20" transform="translate(1486.1511,2242.6453)"> | ||||||
|  | 						<path d="m 0,0 c 16.533,10.846 33.211,21.453 50.104,31.699 78.972,-48.281 153.985,-102.704 227.269,-159.144 148.61,-115.422 287.884,-243.01 414.393,-382.333 111.39,-122.861 212.751,-255.152 298.898,-396.971 52.744,-87.322 100.544,-177.884 139.514,-272.214 -11.638,-3.551 -23.108,-7.655 -34.362,-12.286 l -0.24,0.288 c -11.135,12.982 -24.141,24.212 -34.915,37.506 -22.557,23.013 -45.425,45.737 -68.03,68.678 -19.725,20.253 -40.601,39.45 -58.958,60.974 -36.355,36.451 -72.517,73.093 -108.944,109.471 -22.628,26.013 -48.064,49.385 -71.965,74.197 -19.029,19.485 -38.706,38.346 -57.519,57.999 -12.166,14.998 -26.684,27.716 -39.93,41.658 -27.668,27.524 -54.903,55.479 -82.571,82.979 -23.924,27.956 -51.664,52.264 -76.692,79.164 -4.68,4.487 -8.855,10.774 -15.886,11.326 -22.34,34.027 -58.311,57.327 -97.377,67.502 -104.312,99.153 -215.487,191.202 -332.661,274.782 -117.942,-83.94 -229.476,-176.781 -334.484,-276.39 -26.684,-0.024 -53.368,0.024 -80.076,-0.024 0.024,-26.564 0.048,-53.104 0,-79.668 -72.229,-73.021 -139.491,-150.937 -202.385,-232.092 -63.758,-82.619 -121.973,-169.51 -173.541,-260.264 131.932,-69.061 257.864,-149.521 375.926,-240.275 0.096,-26.444 -0.12,-52.888 0.096,-79.332 l 0.744,-0.984 c 20.109,-24.14 43.409,-45.233 65.126,-67.861 15.118,-15.382 30.571,-30.404 45.569,-45.881 17.565,-20.733 37.698,-39.042 56.607,-58.503 19.917,-20.781 41.25,-40.218 59.967,-62.151 29.156,-29.299 58.167,-58.815 87.515,-87.922 29.155,-33.043 61.502,-63.111 92.169,-94.738 13.726,-12.67 25.124,-27.571 38.634,-40.457 25.029,-25.365 50.129,-50.657 75.325,-75.853 -37.914,-51.208 -73.741,-103.952 -107.192,-158.183 -167.83,273.317 -397.235,507.305 -662.37,687.158 -81.875,55.335 -167.23,105.584 -255.681,149.641 -52.815,26.276 -106.831,50.248 -162.239,70.381 99.393,233.628 242.795,446.715 410.289,636.79 93.562,106.088 194.634,205.433 301.466,298.13 C -217.335,-155.808 -111.439,-73.789 0,0" style="fill:#062f4c;fill-opacity:1;fill-rule:nonzero;stroke:none" id="path22" /> | ||||||
|  | 					</g> | ||||||
|  |                     <g id="g24" transform="translate(1201.7948,1741.5303)"> | ||||||
|  | 						<path d="M 0,0 C 105.008,99.609 216.543,192.45 334.485,276.39 451.659,192.81 562.833,100.76 667.146,1.608 c -34.987,8.83 -71.51,9.718 -107.264,6.431 -41.202,-4.296 -82.907,-19.077 -112.543,-48.953 -33.019,-32.155 -49.456,-77.604 -55.311,-122.501 -28.124,27.908 -56.104,55.983 -84.035,84.083 -2.976,2.976 -6.839,4.823 -10.391,6.911 -19.029,26.348 -45.953,46.673 -76.62,57.495 C 187.555,-2.472 151.513,-0.12 116.166,0 Z" style="fill:#4c8fcc;fill-opacity:1;fill-rule:nonzero;stroke:none" id="path26" /> | ||||||
|  | 					</g> | ||||||
|  |                     <g id="g28" transform="translate(919.3342,1429.7462)"> | ||||||
|  | 						<path d="m 0,0 c 62.894,81.156 130.156,159.072 202.385,232.092 0.048,-244.21 0.024,-488.421 0,-732.631 C 84.323,-409.785 -41.61,-329.325 -173.541,-260.264 -121.973,-169.51 -63.758,-82.619 0,0" style="fill:#4c8fcc;fill-opacity:1;fill-rule:nonzero;stroke:none" id="path30" /> | ||||||
|  | 					</g> | ||||||
|  |                     <g id="g32" transform="translate(1649.134,1700.6166)"> | ||||||
|  | 						<path d="m 0,0 c 29.635,29.875 71.341,44.657 112.543,48.952 35.754,3.288 72.277,2.4 107.263,-6.431 39.066,-10.174 75.037,-33.474 97.377,-67.501 11.879,-17.661 20.181,-37.411 26.42,-57.687 10.871,-38.802 11.95,-79.356 11.446,-119.358 -44.345,-0.072 -88.69,0.048 -133.035,-0.072 -1.032,30.907 3.263,63.686 -10.175,92.626 -9.526,20.325 -32.107,31.243 -53.751,32.131 -21.453,1.44 -45.065,-4.32 -59.175,-21.597 -12.79,-15.861 -15.382,-37.002 -16.558,-56.655 -1.295,-29.132 3.696,-59.031 17.518,-84.923 16.821,-30.619 39.378,-57.783 64.526,-81.9 31.387,-32.634 67.501,-60.374 97.857,-94.041 27.332,-28.988 51.256,-61.479 68.005,-97.785 20.541,-41.13 26.972,-87.827 25.82,-133.372 -0.912,-32.107 -5.231,-64.406 -16.149,-94.737 -11.59,-31.699 -31.123,-61.047 -58.335,-81.371 -25.124,-19.125 -55.696,-29.852 -86.651,-34.771 -49.552,-6.743 -101.888,-4.847 -148.465,14.854 -35.227,14.829 -64.238,42.689 -81.708,76.548 -20.996,40.242 -27.115,86.339 -27.259,131.212 0.048,17.829 0,35.658 0.048,53.463 44.345,0.048 88.69,-0.023 133.059,0.048 1.728,-35.538 -4.055,-72.06 5.663,-106.807 5.783,-22.173 26.204,-37.794 48.185,-41.754 20.733,-3.431 43.577,-2.015 61.622,9.791 15.502,9.43 23.949,26.78 26.78,44.225 5.903,35.922 1.872,74.293 -15.381,106.688 -16.918,30.595 -39.474,57.711 -64.55,81.899 -33.187,34.099 -71.173,63.254 -102.585,99.081 -26.756,28.867 -49.408,61.646 -65.486,97.641 -24.572,52.48 -26.731,112.422 -20.18,169.102 C -49.456,-77.604 -33.019,-32.155 0,0" style="fill:#feeeee;fill-opacity:1;fill-rule:nonzero;stroke:none" id="path34" /> | ||||||
|  | 					</g> | ||||||
|  |                     <g id="g36" transform="translate(1262.7214,1613.126)"> | ||||||
|  | 						<path d="m 0,0 v -301.13 c 23.204,0.024 46.409,-0.048 69.613,0.024 18.525,0.288 38.202,6.575 50.153,21.429 12.43,17.277 13.917,39.522 14.613,60.111 0.024,43.985 -0.048,87.994 0.024,131.979 -0.48,23.637 -0.983,50.369 -17.277,69.23 C 104.864,-5.711 86.867,-0.24 69.589,0 46.385,0.048 23.204,0.024 0,0 m -141.002,128.38 c 26.708,0.048 53.392,0 80.075,0.024 H 55.24 c 35.346,-0.12 71.389,-2.471 104.815,-14.925 30.668,-10.823 57.592,-31.148 76.621,-57.496 26.852,-39.09 36.69,-87.202 38.058,-133.947 0.024,-48.833 0.096,-97.689 -0.024,-146.521 -1.728,-47.993 -11.974,-97.953 -41.514,-136.971 -22.748,-30.644 -57.495,-50.801 -94.281,-59.583 -45.377,-11.878 -92.578,-6.791 -138.891,-7.847 -0.072,-111.799 0,-223.574 -0.024,-335.373 -13.942,0 -27.86,0.024 -41.778,-0.024 -32.802,0.072 -65.605,0 -98.384,0.048 l -0.744,0.984 c -0.216,26.444 0,52.888 -0.096,79.332 0.024,244.211 0.048,488.421 0,732.632 0.048,26.563 0.024,53.103 0,79.667" style="fill:#feeeee;fill-opacity:1;fill-rule:nonzero;stroke:none" id="path38" /> | ||||||
|  | 					</g> | ||||||
|  |                     <g id="g40" transform="translate(1966.3174,1675.6364)"> | ||||||
|  | 						<path d="m 0,0 c 7.031,-0.552 11.206,-6.839 15.885,-11.326 25.029,-26.9 52.768,-51.208 76.693,-79.164 27.667,-27.5 54.903,-55.456 82.571,-82.979 13.246,-13.942 27.764,-26.66 39.93,-41.658 18.813,-19.653 38.49,-38.514 57.519,-57.999 23.9,-24.812 49.337,-48.185 71.965,-74.197 36.427,-36.378 72.589,-73.02 108.943,-109.471 18.358,-21.524 39.234,-40.722 58.959,-60.974 22.605,-22.941 45.473,-45.665 68.03,-68.678 10.774,-13.294 23.78,-24.524 34.914,-37.506 -103.904,-41.97 -203.488,-94.114 -298.922,-152.761 -246.994,-152.28 -466.224,-350.298 -639.333,-583.398 -25.197,25.196 -50.297,50.488 -75.325,75.852 -13.51,12.886 -24.908,27.788 -38.634,40.458 -30.667,31.627 -63.014,61.695 -92.17,94.738 -29.347,29.107 -58.359,58.623 -87.514,87.922 -18.717,21.933 -40.05,41.37 -59.967,62.151 -18.909,19.461 -39.042,37.77 -56.607,58.503 -14.998,15.477 -30.452,30.499 -45.569,45.88 -21.717,22.629 -45.017,43.722 -65.126,67.862 32.779,-0.048 65.582,0.024 98.384,-0.048 114.391,-98.097 220.407,-205.984 315.384,-322.99 92.914,114.318 196.242,220.022 307.753,316.271 30.955,4.919 61.526,15.646 86.65,34.771 27.212,20.325 46.745,49.672 58.335,81.371 107.312,77.988 219.327,149.929 337.509,210.376 -35.299,64.67 -75.829,126.437 -118.254,186.643 C 176.253,-228.037 104.24,-140.115 26.42,-57.687 20.181,-37.41 11.878,-17.661 0,0" style="fill:#042338;fill-opacity:1;fill-rule:nonzero;stroke:none" id="path42" /> | ||||||
|  | 					</g> | ||||||
|  |                     <g id="g44" transform="translate(1499.3971,1669.1094)"> | ||||||
|  | 						<path d="m 0,0 c 3.551,-2.088 7.415,-3.935 10.39,-6.911 27.932,-28.1 55.912,-56.175 84.036,-84.083 -6.551,-56.679 -4.392,-116.622 20.18,-169.102 16.078,-35.994 38.73,-68.774 65.486,-97.641 31.412,-35.826 69.398,-64.982 102.585,-99.081 25.076,-24.188 47.632,-51.304 64.55,-81.899 17.253,-32.395 21.284,-70.765 15.381,-106.688 -2.831,-17.445 -11.278,-34.794 -26.78,-44.225 -18.045,-11.806 -40.889,-13.222 -61.622,-9.79 -21.981,3.959 -42.402,19.58 -48.185,41.753 -9.718,34.747 -3.935,71.269 -5.663,106.808 -44.369,-0.072 -88.714,0 -133.059,-0.048 -0.048,-17.806 0,-35.635 -0.048,-53.464 0.144,-44.873 6.263,-90.97 27.259,-131.212 17.47,-33.859 46.481,-61.718 81.708,-76.548 46.577,-19.701 98.913,-21.597 148.465,-14.854 -111.511,-96.249 -214.839,-201.953 -307.753,-316.271 -94.977,117.006 -200.993,224.893 -315.383,322.99 13.918,0.048 27.836,0.024 41.777,0.024 0.024,111.799 -0.048,223.574 0.024,335.372 46.313,1.056 93.514,-4.031 138.891,7.847 36.786,8.783 71.533,28.94 94.282,59.583 29.539,39.018 39.785,88.978 41.513,136.971 0.12,48.833 0.048,97.689 0.024,146.522 C 36.69,-87.203 26.852,-39.09 0,0" style="fill:#1c72ba;fill-opacity:1;fill-rule:nonzero;stroke:none" id="path46" /> | ||||||
|  | 					</g> | ||||||
|  |                     <g id="g48" transform="translate(1748.0469,1601.6797)"> | ||||||
|  | 						<path d="M 0,0 C 14.11,17.277 37.722,23.036 59.175,21.597 80.82,20.709 103.4,9.791 112.927,-10.534 c 13.438,-28.94 9.142,-61.719 10.174,-92.626 44.345,0.12 88.691,0 133.036,0.072 0.504,40.002 -0.576,80.556 -11.447,119.358 77.82,-82.428 149.833,-170.35 215.583,-262.664 42.426,-60.207 82.956,-121.973 118.254,-186.643 -118.182,-60.447 -230.196,-132.388 -337.508,-210.376 10.918,30.331 15.238,62.63 16.149,94.737 1.152,45.545 -5.279,92.242 -25.82,133.372 -16.749,36.306 -40.673,68.797 -68.005,97.785 -30.355,33.667 -66.47,61.406 -97.857,94.041 -25.148,24.117 -47.705,51.28 -64.526,81.9 -13.822,25.892 -18.813,55.791 -17.517,84.923 C -15.382,-37.002 -12.79,-15.862 0,0" style="fill:#1c72ba;fill-opacity:1;fill-rule:nonzero;stroke:none" id="path50" /> | ||||||
|  | 					</g> | ||||||
|  |                     <g id="g52" transform="translate(1262.7214,1613.126)"> | ||||||
|  | 						<path d="m 0,0 c 23.204,0.024 46.385,0.048 69.589,0 17.278,-0.24 35.275,-5.711 47.537,-18.357 16.294,-18.861 16.797,-45.593 17.277,-69.23 -0.072,-43.985 0,-87.994 -0.024,-131.979 -0.696,-20.589 -2.183,-42.834 -14.613,-60.111 -11.951,-14.854 -31.628,-21.141 -50.153,-21.429 -23.204,-0.072 -46.409,0 -69.613,-0.024 z" style="fill:#1c72ba;fill-opacity:1;fill-rule:nonzero;stroke:none" id="path54" /> | ||||||
|  | 					</g> | ||||||
|  | 				</g> | ||||||
|  | 			</g> | ||||||
|  | 		</g> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 11 KiB | 
| @@ -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) |  | ||||||
							
								
								
									
										107
									
								
								Bukkit/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										107
									
								
								Bukkit/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,107 @@ | |||||||
|  | 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://repo.papermc.io/repository/maven-public/") | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     maven { | ||||||
|  |         name = "EssentialsX" | ||||||
|  |         url = uri("https://repo.essentialsx.net/releases/") | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | dependencies { | ||||||
|  |     api(projects.plotSquaredCore) | ||||||
|  |  | ||||||
|  |     // Metrics | ||||||
|  |     implementation("org.bstats:bstats-bukkit") | ||||||
|  |  | ||||||
|  |     // Paper | ||||||
|  |     compileOnly(libs.paper) | ||||||
|  |     implementation("io.papermc:paperlib") | ||||||
|  |  | ||||||
|  |     // Plugins | ||||||
|  |     compileOnly(libs.worldeditBukkit) { | ||||||
|  |         exclude(group = "org.bukkit") | ||||||
|  |         exclude(group = "org.spigotmc") | ||||||
|  |     } | ||||||
|  |     compileOnly("com.fastasyncworldedit:FastAsyncWorldEdit-Bukkit") { isTransitive = false } | ||||||
|  |     testImplementation("com.fastasyncworldedit:FastAsyncWorldEdit-Bukkit") { isTransitive = false } | ||||||
|  |     compileOnly("com.github.MilkBowl:VaultAPI") { | ||||||
|  |         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("dev.notmyfault.serverlib:ServerLib") | ||||||
|  |  | ||||||
|  |     // Our libraries | ||||||
|  |     implementation(libs.arkitektonika) | ||||||
|  |     implementation(libs.http4j) | ||||||
|  |     implementation("com.intellectualsites.paster:Paster") | ||||||
|  |  | ||||||
|  |     // Adventure | ||||||
|  |     implementation("net.kyori:adventure-platform-bukkit") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | 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.github.spotbugs", "com.plotsquared.core.spotbugs") | ||||||
|  |     relocate("javax.inject", "com.plotsquared.core.annotation.inject") | ||||||
|  |     relocate("net.jcip", "com.plotsquared.core.annotations.jcip") | ||||||
|  |     relocate("edu.umd.cs.findbugs", "com.plotsquared.core.annotations.findbugs") | ||||||
|  |  | ||||||
|  |     // Get rid of all the libs which are 100% unused. | ||||||
|  |     minimize() | ||||||
|  |  | ||||||
|  |     mergeServiceFiles() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | tasks { | ||||||
|  |     withType<Javadoc> { | ||||||
|  |         val opt = options as StandardJavadocDocletOptions | ||||||
|  |         opt.links("https://jd.papermc.io/paper/1.18/") | ||||||
|  |         opt.links("https://docs.enginehub.org/javadoc/com.sk89q.worldedit/worldedit-bukkit/" + libs.worldeditBukkit.get().versionConstraint.toString()) | ||||||
|  |         opt.links("https://javadoc.io/doc/com.plotsquared/PlotSquared-Core/latest/") | ||||||
|  |         opt.links("https://jd.adventure.kyori.net/api/" + libs.adventure.get().versionConstraint.toString()) | ||||||
|  |         opt.links("https://google.github.io/guice/api-docs/" + libs.guice.get().versionConstraint.toString() + "/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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,12 +21,13 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit; | ||||||
|  |  | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
| import com.plotsquared.core.command.MainCommand; | import com.plotsquared.core.command.MainCommand; | ||||||
|  | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.player.ConsolePlayer; | import com.plotsquared.core.player.ConsolePlayer; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import org.bukkit.command.Command; | import org.bukkit.command.Command; | ||||||
| @@ -42,35 +43,43 @@ import java.util.ArrayList; | |||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
| import java.util.Collections; | import java.util.Collections; | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  | import java.util.Locale; | ||||||
|  |  | ||||||
| public class BukkitCommand implements CommandExecutor, TabCompleter { | public class BukkitCommand implements CommandExecutor, TabCompleter { | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(CommandSender commandSender, Command command, String commandLabel, |     public boolean onCommand( | ||||||
|         String[] args) { |             CommandSender commandSender, Command command, String commandLabel, | ||||||
|  |             String[] args | ||||||
|  |     ) { | ||||||
|         if (commandSender instanceof Player) { |         if (commandSender instanceof Player) { | ||||||
|             return MainCommand.onCommand(BukkitUtil.getPlayer((Player) commandSender), args); |             return MainCommand.onCommand(BukkitUtil.adapt((Player) commandSender), args); | ||||||
|         } |         } | ||||||
|         if (commandSender instanceof ConsoleCommandSender |         if (commandSender instanceof ConsoleCommandSender | ||||||
|             || commandSender instanceof ProxiedCommandSender |                 || commandSender instanceof ProxiedCommandSender | ||||||
|             || commandSender instanceof RemoteConsoleCommandSender) { |                 || commandSender instanceof RemoteConsoleCommandSender) { | ||||||
|             return MainCommand.onCommand(ConsolePlayer.getConsole(), args); |             return MainCommand.onCommand(ConsolePlayer.getConsole(), args); | ||||||
|         } |         } | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public List<String> onTabComplete(CommandSender commandSender, Command command, String s, |     public List<String> onTabComplete( | ||||||
|         String[] args) { |             CommandSender commandSender, Command command, String label, | ||||||
|  |             String[] args | ||||||
|  |     ) { | ||||||
|         if (!(commandSender instanceof Player)) { |         if (!(commandSender instanceof Player)) { | ||||||
|             return null; |             return null; | ||||||
|         } |         } | ||||||
|         PlotPlayer player = BukkitUtil.getPlayer((Player) commandSender); |         PlotPlayer<?> player = BukkitUtil.adapt((Player) commandSender); | ||||||
|         if (args.length == 0) { |         if (args.length == 0) { | ||||||
|             return Collections.singletonList("plots"); |             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 = |         Collection<com.plotsquared.core.command.Command> objects = | ||||||
|             MainCommand.getInstance().tab(player, args, s.endsWith(" ")); |                 MainCommand.getInstance().tab(player, args, label.endsWith(" ")); | ||||||
|         if (objects == null) { |         if (objects == null) { | ||||||
|             return null; |             return null; | ||||||
|         } |         } | ||||||
| @@ -80,4 +89,5 @@ public class BukkitCommand implements CommandExecutor, TabCompleter { | |||||||
|         } |         } | ||||||
|         return result; |         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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.entity; | ||||||
|  |  | ||||||
| @@ -30,4 +30,5 @@ class AgeableStats { | |||||||
|     int age; |     int age; | ||||||
|     boolean locked; |     boolean locked; | ||||||
|     boolean adult; |     boolean adult; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.entity; | ||||||
|  |  | ||||||
| @@ -37,4 +37,5 @@ class ArmorStandStats { | |||||||
|     boolean noPlate; |     boolean noPlate; | ||||||
|     boolean invisible; |     boolean invisible; | ||||||
|     boolean small; |     boolean small; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.entity; | ||||||
|  |  | ||||||
| @@ -34,4 +34,5 @@ class EntityBaseStats { | |||||||
|     double vZ; |     double vZ; | ||||||
|     double vY; |     double vY; | ||||||
|     double vX; |     double vX; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,18 +21,16 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.entity; | ||||||
|  |  | ||||||
| import lombok.Getter; |  | ||||||
| import lombok.NonNull; |  | ||||||
| import org.bukkit.Location; | import org.bukkit.Location; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
| import org.bukkit.entity.EntityType; | import org.bukkit.entity.EntityType; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| @Getter |  | ||||||
| public abstract class EntityWrapper { | public abstract class EntityWrapper { | ||||||
|  |  | ||||||
|     protected final float yaw; |     protected final float yaw; | ||||||
| @@ -43,7 +41,7 @@ public abstract class EntityWrapper { | |||||||
|     public double y; |     public double y; | ||||||
|     public double z; |     public double z; | ||||||
|  |  | ||||||
|     EntityWrapper(@NonNull final Entity entity) { |     EntityWrapper(final @NonNull Entity entity) { | ||||||
|         this.entity = entity; |         this.entity = entity; | ||||||
|         this.type = entity.getType(); |         this.type = entity.getType(); | ||||||
|  |  | ||||||
| @@ -55,7 +53,9 @@ public abstract class EntityWrapper { | |||||||
|         this.pitch = location.getPitch(); |         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); |         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 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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.entity; | ||||||
|  |  | ||||||
| @@ -34,4 +34,5 @@ class HorseStats { | |||||||
|     Horse.Variant variant; |     Horse.Variant variant; | ||||||
|     Horse.Color color; |     Horse.Color color; | ||||||
|     Horse.Style style; |     Horse.Style style; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.entity; | ||||||
|  |  | ||||||
| @@ -50,4 +50,5 @@ class LivingEntityStats { | |||||||
|     ItemStack chestplate; |     ItemStack chestplate; | ||||||
|     Collection<PotionEffect> potions; |     Collection<PotionEffect> potions; | ||||||
|     ItemStack offHand; |     ItemStack offHand; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,11 +21,13 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | 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.Art; | ||||||
| import org.bukkit.DyeColor; | import org.bukkit.DyeColor; | ||||||
| import org.bukkit.Location; | import org.bukkit.Location; | ||||||
| @@ -60,6 +62,8 @@ import java.util.List; | |||||||
|  |  | ||||||
| public final class ReplicatingEntityWrapper extends EntityWrapper { | public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||||
|  |  | ||||||
|  |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + ReplicatingEntityWrapper.class.getSimpleName()); | ||||||
|  |  | ||||||
|     private final short depth; |     private final short depth; | ||||||
|     private final int hash; |     private final int hash; | ||||||
|     private final EntityBaseStats base = new EntityBaseStats(); |     private final EntityBaseStats base = new EntityBaseStats(); | ||||||
| @@ -148,17 +152,17 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                 this.stack = item.getItemStack(); |                 this.stack = item.getItemStack(); | ||||||
|                 return; |                 return; | ||||||
|             case "ITEM_FRAME": |             case "ITEM_FRAME": | ||||||
|                 this.x = Math.floor(this.x); |                 this.x = Math.floor(this.getX()); | ||||||
|                 this.y = Math.floor(this.y); |                 this.y = Math.floor(this.getY()); | ||||||
|                 this.z = Math.floor(this.z); |                 this.z = Math.floor(this.getZ()); | ||||||
|                 ItemFrame itemFrame = (ItemFrame) entity; |                 ItemFrame itemFrame = (ItemFrame) entity; | ||||||
|                 this.dataByte = getOrdinal(Rotation.values(), itemFrame.getRotation()); |                 this.dataByte = getOrdinal(Rotation.values(), itemFrame.getRotation()); | ||||||
|                 this.stack = itemFrame.getItem().clone(); |                 this.stack = itemFrame.getItem().clone(); | ||||||
|                 return; |                 return; | ||||||
|             case "PAINTING": |             case "PAINTING": | ||||||
|                 this.x = Math.floor(this.x); |                 this.x = Math.floor(this.getX()); | ||||||
|                 this.y = Math.floor(this.y); |                 this.y = Math.floor(this.getY()); | ||||||
|                 this.z = Math.floor(this.z); |                 this.z = Math.floor(this.getZ()); | ||||||
|                 Painting painting = (Painting) entity; |                 Painting painting = (Painting) entity; | ||||||
|                 Art art = painting.getArt(); |                 Art art = painting.getArt(); | ||||||
|                 this.dataByte = getOrdinal(BlockFace.values(), painting.getFacing()); |                 this.dataByte = getOrdinal(BlockFace.values(), painting.getFacing()); | ||||||
| @@ -185,8 +189,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                 AbstractHorse horse = (AbstractHorse) entity; |                 AbstractHorse horse = (AbstractHorse) entity; | ||||||
|                 this.horse = new HorseStats(); |                 this.horse = new HorseStats(); | ||||||
|                 this.horse.jump = horse.getJumpStrength(); |                 this.horse.jump = horse.getJumpStrength(); | ||||||
|                 if (horse instanceof ChestedHorse) { |                 if (horse instanceof ChestedHorse horse1) { | ||||||
|                     ChestedHorse horse1 = (ChestedHorse) horse; |  | ||||||
|                     this.horse.chest = horse1.isCarryingChest(); |                     this.horse.chest = horse1.isCarryingChest(); | ||||||
|                 } |                 } | ||||||
|                 //todo these horse features need fixing |                 //todo these horse features need fixing | ||||||
| @@ -237,9 +240,9 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|             case "ARMOR_STAND": |             case "ARMOR_STAND": | ||||||
|                 ArmorStand stand = (ArmorStand) entity; |                 ArmorStand stand = (ArmorStand) entity; | ||||||
|                 this.inventory = |                 this.inventory = | ||||||
|                     new ItemStack[] {stand.getItemInHand().clone(), stand.getHelmet().clone(), |                         new ItemStack[]{stand.getItemInHand().clone(), stand.getHelmet().clone(), | ||||||
|                         stand.getChestplate().clone(), stand.getLeggings().clone(), |                                 stand.getChestplate().clone(), stand.getLeggings().clone(), | ||||||
|                         stand.getBoots().clone()}; |                                 stand.getBoots().clone()}; | ||||||
|                 storeLiving(stand); |                 storeLiving(stand); | ||||||
|                 this.stand = new ArmorStandStats(); |                 this.stand = new ArmorStandStats(); | ||||||
|  |  | ||||||
| @@ -332,18 +335,17 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                     this.dataByte = (byte) 0; |                     this.dataByte = (byte) 0; | ||||||
|                 } |                 } | ||||||
|                 storeLiving((LivingEntity) entity); |                 storeLiving((LivingEntity) entity); | ||||||
|                 return; |                 // END LIVING // | ||||||
|             // END LIVING // |  | ||||||
|             default: |  | ||||||
|                 PlotSquared.debug("&cCOULD NOT IDENTIFY ENTITY: " + entity.getType()); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public boolean equals(Object obj) { |     @Override | ||||||
|  |     public boolean equals(Object obj) { | ||||||
|         return this.hash == obj.hashCode(); |         return this.hash == obj.hashCode(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public int hashCode() { |     @Override | ||||||
|  |     public int hashCode() { | ||||||
|         return this.hash; |         return this.hash; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -391,7 +393,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|         try { |         try { | ||||||
|             entity.getInventory().setContents(this.inventory); |             entity.getInventory().setContents(this.inventory); | ||||||
|         } catch (IllegalArgumentException e) { |         } 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(); |         this.lived.leashed = lived.isLeashed(); | ||||||
|         if (this.lived.leashed) { |         if (this.lived.leashed) { | ||||||
|             Location location = lived.getLeashHolder().getLocation(); |             Location location = lived.getLeashHolder().getLocation(); | ||||||
|             this.lived.leashX = (short) (this.x - location.getBlockX()); |             this.lived.leashX = (short) (this.getX() - location.getBlockX()); | ||||||
|             this.lived.leashY = (short) (this.y - location.getBlockY()); |             this.lived.leashY = (short) (this.getY() - location.getBlockY()); | ||||||
|             this.lived.leashZ = (short) (this.z - location.getBlockZ()); |             this.lived.leashZ = (short) (this.getZ() - location.getBlockZ()); | ||||||
|         } |         } | ||||||
|         EntityEquipment equipment = lived.getEquipment(); |         EntityEquipment equipment = lived.getEquipment(); | ||||||
|         this.lived.equipped = equipment != null; |         this.lived.equipped = equipment != null; | ||||||
| @@ -459,8 +461,10 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|         this.tamed.tamed = tamed.isTamed(); |         this.tamed.tamed = tamed.isTamed(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public Entity spawn(World world, int xOffset, int zOffset) { |     @SuppressWarnings("deprecation") // Paper deprecation | ||||||
|         Location location = new Location(world, this.x + xOffset, this.y, this.z + zOffset); |     @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.setYaw(this.yaw); | ||||||
|         location.setPitch(this.pitch); |         location.setPitch(this.pitch); | ||||||
|         if (!this.getType().isSpawnable()) { |         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) { |                 if (this.stand.head[0] != 0 || this.stand.head[1] != 0 || this.stand.head[2] != 0) { | ||||||
|                     EulerAngle pose = |                     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); |                     stand.setHeadPose(pose); | ||||||
|                 } |                 } | ||||||
|                 if (this.stand.body[0] != 0 || this.stand.body[1] != 0 || this.stand.body[2] != 0) { |                 if (this.stand.body[0] != 0 || this.stand.body[1] != 0 || this.stand.body[2] != 0) { | ||||||
|                     EulerAngle pose = |                     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); |                     stand.setBodyPose(pose); | ||||||
|                 } |                 } | ||||||
|                 if (this.stand.leftLeg[0] != 0 || this.stand.leftLeg[1] != 0 |                 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], |                     EulerAngle pose = new EulerAngle(this.stand.leftLeg[0], this.stand.leftLeg[1], | ||||||
|                         this.stand.leftLeg[2]); |                             this.stand.leftLeg[2] | ||||||
|  |                     ); | ||||||
|                     stand.setLeftLegPose(pose); |                     stand.setLeftLegPose(pose); | ||||||
|                 } |                 } | ||||||
|                 if (this.stand.rightLeg[0] != 0 || this.stand.rightLeg[1] != 0 |                 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], |                     EulerAngle pose = new EulerAngle(this.stand.rightLeg[0], this.stand.rightLeg[1], | ||||||
|                         this.stand.rightLeg[2]); |                             this.stand.rightLeg[2] | ||||||
|  |                     ); | ||||||
|                     stand.setRightLegPose(pose); |                     stand.setRightLegPose(pose); | ||||||
|                 } |                 } | ||||||
|                 if (this.stand.leftArm[0] != 0 || this.stand.leftArm[1] != 0 |                 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], |                     EulerAngle pose = new EulerAngle(this.stand.leftArm[0], this.stand.leftArm[1], | ||||||
|                         this.stand.leftArm[2]); |                             this.stand.leftArm[2] | ||||||
|  |                     ); | ||||||
|                     stand.setLeftArmPose(pose); |                     stand.setLeftArmPose(pose); | ||||||
|                 } |                 } | ||||||
|                 if (this.stand.rightArm[0] != 0 || this.stand.rightArm[1] != 0 |                 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], |                     EulerAngle pose = new EulerAngle(this.stand.rightArm[0], this.stand.rightArm[1], | ||||||
|                         this.stand.rightArm[2]); |                             this.stand.rightArm[2] | ||||||
|  |                     ); | ||||||
|                     stand.setRightArmPose(pose); |                     stand.setRightArmPose(pose); | ||||||
|                 } |                 } | ||||||
|                 if (this.stand.invisible) { |                 if (this.stand.invisible) { | ||||||
| @@ -738,7 +746,9 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                 restoreLiving((LivingEntity) entity); |                 restoreLiving((LivingEntity) entity); | ||||||
|                 return entity; |                 return entity; | ||||||
|             default: |             default: | ||||||
|                 PlotSquared.debug("&cCOULD NOT IDENTIFY ENTITY: " + entity.getType()); |                 if (Settings.DEBUG) { | ||||||
|  |                     LOGGER.info("Could not identify entity: {}", entity.getType()); | ||||||
|  |                 } | ||||||
|                 return entity; |                 return entity; | ||||||
|             // END LIVING |             // END LIVING | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.entity; | ||||||
|  |  | ||||||
| @@ -31,4 +31,5 @@ class TameableStats { | |||||||
|  |  | ||||||
|     AnimalTamer owner; |     AnimalTamer owner; | ||||||
|     boolean tamed; |     boolean tamed; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,11 +21,11 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.entity; | ||||||
|  |  | ||||||
| import com.plotsquared.bukkit.BukkitMain; | import com.plotsquared.bukkit.BukkitPlatform; | ||||||
| import org.bukkit.Chunk; | import org.bukkit.Chunk; | ||||||
| import org.bukkit.Location; | import org.bukkit.Location; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| @@ -44,7 +44,8 @@ public class TeleportEntityWrapper extends EntityWrapper { | |||||||
|         super(entity); |         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())) { |         if (!getEntity().getLocation().getChunk().equals(oldLocation.getChunk())) { | ||||||
|             final Location oldLocation = this.oldLocation.clone(); |             final Location oldLocation = this.oldLocation.clone(); | ||||||
|             oldLocation.add(xOffset, 0, xOffset); |             oldLocation.add(xOffset, 0, xOffset); | ||||||
| @@ -53,12 +54,13 @@ public class TeleportEntityWrapper extends EntityWrapper { | |||||||
|             getEntity().setInvulnerable(invulnerableOld); |             getEntity().setInvulnerable(invulnerableOld); | ||||||
|             getEntity().setFireTicks(fireTicksOld); |             getEntity().setFireTicks(fireTicksOld); | ||||||
|             getEntity().setTicksLived(livingTicksOld); |             getEntity().setTicksLived(livingTicksOld); | ||||||
|             getEntity().removeMetadata("ps-tmp-teleport", BukkitMain.getPlugin(BukkitMain.class)); |             getEntity().removeMetadata("ps-tmp-teleport", BukkitPlatform.getPlugin(BukkitPlatform.class)); | ||||||
|         } |         } | ||||||
|         return getEntity(); |         return getEntity(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public void saveEntity() { |     @Override | ||||||
|  |     public void saveEntity() { | ||||||
|         if (getEntity().hasMetadata("ps-tmp-teleport")) { |         if (getEntity().hasMetadata("ps-tmp-teleport")) { | ||||||
|             this.oldLocation = (Location) this.getEntity().getMetadata("ps-tmp-teleport").get(0); |             this.oldLocation = (Location) this.getEntity().getMetadata("ps-tmp-teleport").get(0); | ||||||
|         } else { |         } else { | ||||||
| @@ -67,9 +69,9 @@ public class TeleportEntityWrapper extends EntityWrapper { | |||||||
|  |  | ||||||
|         // To account for offsets in the chunk manager |         // To account for offsets in the chunk manager | ||||||
|         this.oldLocation = oldLocation.clone(); |         this.oldLocation = oldLocation.clone(); | ||||||
|         this.oldLocation.setX(this.x); |         this.oldLocation.setX(this.getX()); | ||||||
|         this.oldLocation.setY(this.y); |         this.oldLocation.setY(this.getY()); | ||||||
|         this.oldLocation.setZ(this.z); |         this.oldLocation.setZ(this.getZ()); | ||||||
|  |  | ||||||
|         this.gravityOld = this.getEntity().hasGravity(); |         this.gravityOld = this.getEntity().hasGravity(); | ||||||
|         this.getEntity().setGravity(false); |         this.getEntity().setGravity(false); | ||||||
| @@ -77,11 +79,13 @@ public class TeleportEntityWrapper extends EntityWrapper { | |||||||
|         this.getEntity().setInvulnerable(true); |         this.getEntity().setInvulnerable(true); | ||||||
|         this.fireTicksOld = this.getEntity().getFireTicks(); |         this.fireTicksOld = this.getEntity().getFireTicks(); | ||||||
|         this.livingTicksOld = this.getEntity().getTicksLived(); |         this.livingTicksOld = this.getEntity().getTicksLived(); | ||||||
|         this.getEntity().setMetadata("ps-tmp-teleport", |         this.getEntity().setMetadata( | ||||||
|             new FixedMetadataValue(BukkitMain.getPlugin(BukkitMain.class), oldLocation)); |                 "ps-tmp-teleport", | ||||||
|  |                 new FixedMetadataValue(BukkitPlatform.getPlugin(BukkitPlatform.class), oldLocation) | ||||||
|  |         ); | ||||||
|         final Chunk newChunk = getNewChunk(); |         final Chunk newChunk = getNewChunk(); | ||||||
|         this.getEntity().teleport( |         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() { |     private Chunk getNewChunk() { | ||||||
| @@ -115,4 +119,5 @@ public class TeleportEntityWrapper extends EntityWrapper { | |||||||
|     private Chunk getChunkRelative(final Chunk chunk, final int dx, final int dz) { |     private Chunk getChunkRelative(final Chunk chunk, final int dx, final int dz) { | ||||||
|         return chunk.getWorld().getChunkAt(chunk.getX() + dx, chunk.getZ() + dz); |         return chunk.getWorld().getChunkAt(chunk.getX() + dx, chunk.getZ() + dz); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.generator; | ||||||
|  |  | ||||||
| @@ -29,37 +29,48 @@ import com.plotsquared.core.PlotSquared; | |||||||
| import com.plotsquared.core.generator.IndependentPlotGenerator; | import com.plotsquared.core.generator.IndependentPlotGenerator; | ||||||
| import com.plotsquared.core.location.ChunkWrapper; | import com.plotsquared.core.location.ChunkWrapper; | ||||||
| import com.plotsquared.core.plot.PlotArea; | import com.plotsquared.core.plot.PlotArea; | ||||||
| import com.plotsquared.core.queue.GlobalBlockQueue; | import com.plotsquared.core.plot.world.PlotAreaManager; | ||||||
| import com.plotsquared.core.queue.LocalBlockQueue; | import com.plotsquared.core.queue.QueueCoordinator; | ||||||
| import com.plotsquared.core.queue.ScopedLocalBlockQueue; | import com.plotsquared.core.queue.ScopedQueueCoordinator; | ||||||
|  | import com.sk89q.worldedit.bukkit.BukkitWorld; | ||||||
| import org.bukkit.Chunk; | import org.bukkit.Chunk; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.generator.BlockPopulator; | import org.bukkit.generator.BlockPopulator; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.util.Random; | import java.util.Random; | ||||||
|  |  | ||||||
| final class BlockStatePopulator extends BlockPopulator { | final class BlockStatePopulator extends BlockPopulator { | ||||||
|  |  | ||||||
|     private final IndependentPlotGenerator plotGenerator; |     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.plotGenerator = plotGenerator; | ||||||
|  |         this.plotAreaManager = plotAreaManager; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void populate(@NotNull final World world, @NotNull final Random random, |     public void populate(final @NonNull World world, final @NonNull Random random, final @NonNull Chunk source) { | ||||||
|         @NotNull final Chunk source) { |  | ||||||
|         if (this.queue == null) { |         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 PlotArea area = this.plotAreaManager.getPlotArea(world.getName(), null); | ||||||
|         final ChunkWrapper wrap = |         if (area == null) { | ||||||
|             new ChunkWrapper(area.getWorldName(), source.getX(), source.getZ()); |             return; | ||||||
|         final ScopedLocalBlockQueue chunk = this.queue.getForChunk(wrap.x, wrap.z); |         } | ||||||
|  |         final ChunkWrapper wrap = new ChunkWrapper(area.getWorldName(), source.getX(), source.getZ()); | ||||||
|  |         final ScopedQueueCoordinator chunk = this.queue.getForChunk(wrap.x, wrap.z, | ||||||
|  |                 com.plotsquared.bukkit.util.BukkitWorld.getMinWorldHeight(world), | ||||||
|  |                 com.plotsquared.bukkit.util.BukkitWorld.getMaxWorldHeight(world) - 1 | ||||||
|  |         ); | ||||||
|         if (this.plotGenerator.populateChunk(chunk, area)) { |         if (this.plotGenerator.populateChunk(chunk, area)) { | ||||||
|             this.queue.flush(); |             this.queue.enqueue(); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,15 +21,19 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.generator; | ||||||
|  |  | ||||||
|  | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.generator.AugmentedUtils; | import com.plotsquared.core.generator.AugmentedUtils; | ||||||
|  | import com.plotsquared.core.queue.QueueCoordinator; | ||||||
|  | import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||||
|  | import com.sk89q.worldedit.util.SideEffectSet; | ||||||
| import org.bukkit.Chunk; | import org.bukkit.Chunk; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.generator.BlockPopulator; | import org.bukkit.generator.BlockPopulator; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.util.Random; | import java.util.Random; | ||||||
|  |  | ||||||
| @@ -51,7 +55,15 @@ public class BukkitAugmentedGenerator extends BlockPopulator { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @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); |         QueueCoordinator queue = PlotSquared.platform().globalBlockQueue().getNewQueue(BukkitAdapter.adapt(world)); | ||||||
|  |         // The chunk is already loaded and we do not want to load the chunk in "fully" by using any PaperLib methods. | ||||||
|  |         queue.setForceSync(true); | ||||||
|  |         queue.setSideEffectSet(SideEffectSet.none()); | ||||||
|  |         queue.setBiomesEnabled(false); | ||||||
|  |         queue.setChunkObject(source); | ||||||
|  |         AugmentedUtils.generateChunk(world.getName(), source.getX(), source.getZ(), queue); | ||||||
|  |         queue.enqueue(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,28 +21,28 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.generator; | ||||||
|  |  | ||||||
| import com.plotsquared.bukkit.queue.GenChunk; | import com.plotsquared.bukkit.queue.GenChunk; | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
|  | import com.plotsquared.bukkit.util.BukkitWorld; | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.generator.GeneratorWrapper; | import com.plotsquared.core.generator.GeneratorWrapper; | ||||||
| import com.plotsquared.core.generator.IndependentPlotGenerator; | import com.plotsquared.core.generator.IndependentPlotGenerator; | ||||||
| import com.plotsquared.core.generator.SingleWorldGenerator; | import com.plotsquared.core.generator.SingleWorldGenerator; | ||||||
| import com.plotsquared.core.location.ChunkWrapper; | import com.plotsquared.core.location.ChunkWrapper; | ||||||
| import com.plotsquared.core.plot.PlotArea; | 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.ChunkManager; | ||||||
| import com.plotsquared.core.util.MainUtil; |  | ||||||
| import com.sk89q.worldedit.math.BlockVector2; | import com.sk89q.worldedit.math.BlockVector2; | ||||||
| import lombok.Getter; |  | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.block.Biome; | import org.bukkit.block.Biome; | ||||||
| import org.bukkit.generator.BlockPopulator; | import org.bukkit.generator.BlockPopulator; | ||||||
| import org.bukkit.generator.ChunkGenerator; | import org.bukkit.generator.ChunkGenerator; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| @@ -50,85 +50,69 @@ import java.util.Random; | |||||||
| import java.util.Set; | import java.util.Set; | ||||||
|  |  | ||||||
| public class BukkitPlotGenerator extends ChunkGenerator | 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 IndependentPlotGenerator plotGenerator; | ||||||
|     private final ChunkGenerator platformGenerator; |     private final ChunkGenerator platformGenerator; | ||||||
|     private final boolean full; |     private final boolean full; | ||||||
|  |     private final String levelName; | ||||||
|     private List<BlockPopulator> populators; |     private List<BlockPopulator> populators; | ||||||
|     private boolean loaded = false; |     private boolean loaded = false; | ||||||
|  |  | ||||||
|     @Getter private final String levelName; |     public BukkitPlotGenerator( | ||||||
|  |             final @NonNull String name, | ||||||
|     public BukkitPlotGenerator(String name, IndependentPlotGenerator generator) { |             final @NonNull IndependentPlotGenerator generator, | ||||||
|         if (generator == null) { |             final @NonNull PlotAreaManager plotAreaManager | ||||||
|             throw new IllegalArgumentException("Generator may not be null!"); |     ) { | ||||||
|         } |         this.plotAreaManager = plotAreaManager; | ||||||
|         this.levelName = name; |         this.levelName = name; | ||||||
|         this.plotGenerator = generator; |         this.plotGenerator = generator; | ||||||
|         this.platformGenerator = this; |         this.platformGenerator = this; | ||||||
|         this.populators = new ArrayList<>(); |         this.populators = new ArrayList<>(); | ||||||
|         this.populators.add(new BlockStatePopulator(this.plotGenerator)); |         this.populators.add(new BlockStatePopulator(this.plotGenerator, this.plotAreaManager)); | ||||||
|         this.full = true; |         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) { |         if (cg instanceof BukkitPlotGenerator) { | ||||||
|             throw new IllegalArgumentException("ChunkGenerator: " + cg.getClass().getName() |             throw new IllegalArgumentException("ChunkGenerator: " + cg.getClass().getName() | ||||||
|                 + " is already a BukkitPlotGenerator!"); |                     + " is already a BukkitPlotGenerator!"); | ||||||
|         } |         } | ||||||
|  |         this.plotAreaManager = plotAreaManager; | ||||||
|         this.levelName = world; |         this.levelName = world; | ||||||
|         this.full = false; |         this.full = false; | ||||||
|         this.platformGenerator = cg; |         this.platformGenerator = cg; | ||||||
|         this.plotGenerator = new DelegatePlotGenerator(cg, world); |         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())); |         BukkitAugmentedGenerator.get(BukkitUtil.getWorld(area.getWorldName())); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public boolean isFull() { |     @Override | ||||||
|  |     public boolean isFull() { | ||||||
|         return this.full; |         return this.full; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public IndependentPlotGenerator getPlotGenerator() { |     @Override | ||||||
|  |     public IndependentPlotGenerator getPlotGenerator() { | ||||||
|         return this.plotGenerator; |         return this.plotGenerator; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public ChunkGenerator getPlatformGenerator() { |     @Override | ||||||
|  |     public ChunkGenerator getPlatformGenerator() { | ||||||
|         return this.platformGenerator; |         return this.platformGenerator; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override @NotNull public List<BlockPopulator> getDefaultPopulators(@NotNull World world) { |     @Override | ||||||
|  |     public @NonNull List<BlockPopulator> getDefaultPopulators(@NonNull World world) { | ||||||
|         try { |         try { | ||||||
|             if (!this.loaded) { |             checkLoaded(world); | ||||||
|                 String name = world.getName(); |  | ||||||
|                 PlotSquared.get().loadWorld(name, this); |  | ||||||
|                 Set<PlotArea> areas = PlotSquared.get().getPlotAreas(name); |  | ||||||
|                 if (!areas.isEmpty()) { |  | ||||||
|                     PlotArea area = areas.iterator().next(); |  | ||||||
|                     if (!area.isMobSpawning()) { |  | ||||||
|                         if (!area.isSpawnEggs()) { |  | ||||||
|                             world.setSpawnFlags(false, false); |  | ||||||
|                         } |  | ||||||
|                         world.setAmbientSpawnLimit(0); |  | ||||||
|                         world.setAnimalSpawnLimit(0); |  | ||||||
|                         world.setMonsterSpawnLimit(0); |  | ||||||
|                         world.setWaterAnimalSpawnLimit(0); |  | ||||||
|                     } else { |  | ||||||
|                         world.setSpawnFlags(true, true); |  | ||||||
|                         world.setAmbientSpawnLimit(-1); |  | ||||||
|                         world.setAnimalSpawnLimit(-1); |  | ||||||
|                         world.setMonsterSpawnLimit(-1); |  | ||||||
|                         world.setWaterAnimalSpawnLimit(-1); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 this.loaded = true; |  | ||||||
|             } |  | ||||||
|         } catch (Exception e) { |         } catch (Exception e) { | ||||||
|             e.printStackTrace(); |             e.printStackTrace(); | ||||||
|         } |         } | ||||||
| @@ -147,16 +131,53 @@ public class BukkitPlotGenerator extends ChunkGenerator | |||||||
|         return toAdd; |         return toAdd; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override @NotNull |     private synchronized void checkLoaded(@NonNull World world) { | ||||||
|     public ChunkData generateChunkData(@NotNull World world, @NotNull Random random, int x, int z, |         // Do not attempt to load configurations until WorldEdit has a platform ready. | ||||||
|         @NotNull BiomeGrid biome) { |         if (!PlotSquared.get().isWeInitialised()) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (!this.loaded) { | ||||||
|  |             String name = world.getName(); | ||||||
|  |             PlotSquared.get().loadWorld(name, this); | ||||||
|  |             final Set<PlotArea> areas = this.plotAreaManager.getPlotAreasSet(name); | ||||||
|  |             if (!areas.isEmpty()) { | ||||||
|  |                 PlotArea area = areas.iterator().next(); | ||||||
|  |                 if (!area.isMobSpawning()) { | ||||||
|  |                     if (!area.isSpawnEggs()) { | ||||||
|  |                         world.setSpawnFlags(false, false); | ||||||
|  |                     } | ||||||
|  |                     setSpawnLimits(world, 0); | ||||||
|  |                 } else { | ||||||
|  |                     world.setSpawnFlags(true, true); | ||||||
|  |                     setSpawnLimits(world, -1); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             this.loaded = true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|         GenChunk result = new GenChunk(); |     @SuppressWarnings("deprecation") | ||||||
|  |     private void setSpawnLimits(@NonNull World world, int limit) { | ||||||
|  |         world.setAmbientSpawnLimit(limit); | ||||||
|  |         world.setAnimalSpawnLimit(limit); | ||||||
|  |         world.setMonsterSpawnLimit(limit); | ||||||
|  |         world.setWaterAnimalSpawnLimit(limit); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NonNull ChunkData generateChunkData( | ||||||
|  |             @NonNull World world, @NonNull Random random, int x, int z, | ||||||
|  |             @NonNull BiomeGrid biome | ||||||
|  |     ) { | ||||||
|  |  | ||||||
|  |         int minY = BukkitWorld.getMinWorldHeight(world); | ||||||
|  |         int maxY = BukkitWorld.getMaxWorldHeight(world); | ||||||
|  |         GenChunk result = new GenChunk(minY, maxY); | ||||||
|         if (this.getPlotGenerator() instanceof SingleWorldGenerator) { |         if (this.getPlotGenerator() instanceof SingleWorldGenerator) { | ||||||
|             if (result.getChunkData() != null) { |             if (result.getChunkData() != null) { | ||||||
|                 for (int chunkX = 0; chunkX < 16; chunkX++) { |                 for (int chunkX = 0; chunkX < 16; chunkX++) { | ||||||
|                     for (int chunkZ = 0; chunkZ < 16; chunkZ++) { |                     for (int chunkZ = 0; chunkZ < 16; chunkZ++) { | ||||||
|                         for (int y = 0; y < world.getMaxHeight(); y++) { |                         for (int y = minY; y < maxY; y++) { | ||||||
|                             biome.setBiome(chunkX, y, chunkZ, Biome.PLAINS); |                             biome.setBiome(chunkX, y, chunkZ, Biome.PLAINS); | ||||||
|  |  | ||||||
|                         } |                         } | ||||||
| @@ -187,22 +208,20 @@ public class BukkitPlotGenerator extends ChunkGenerator | |||||||
|         return result.getChunkData(); |         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 |         // Load if improperly loaded | ||||||
|         if (!this.loaded) { |         if (!this.loaded) { | ||||||
|             String name = world.getName(); |             checkLoaded(world); | ||||||
|             PlotSquared.get().loadWorld(name, this); |  | ||||||
|             this.loaded = true; |  | ||||||
|         } |         } | ||||||
|         // Process the chunk |         // Process the chunk | ||||||
|         if (ChunkManager.preProcessChunk(loc, result)) { |         if (ChunkManager.preProcessChunk(loc, result)) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         PlotArea area = PlotSquared.get().getPlotArea(world.getName(), null); |         PlotArea area = this.plotAreaManager.getPlotArea(world.getName(), null); | ||||||
|         if (area == null && (area = PlotSquared.get().getPlotArea(this.levelName, null)) == null) { |         if (area == null && (area = this.plotAreaManager.getPlotArea(this.levelName, null)) == null) { | ||||||
|             throw new IllegalStateException( |             throw new IllegalStateException( | ||||||
|                 "Cannot regenerate chunk that does not belong to a plot area." + " Location: " + loc |                     "Cannot regenerate chunk that does not belong to a plot area." + " Location: " + loc | ||||||
|                     + ", world: " + world); |                             + ", world: " + world); | ||||||
|         } |         } | ||||||
|         try { |         try { | ||||||
|             this.plotGenerator.generateChunk(result, area); |             this.plotGenerator.generateChunk(result, area); | ||||||
| @@ -213,7 +232,8 @@ public class BukkitPlotGenerator extends ChunkGenerator | |||||||
|         ChunkManager.postProcessChunk(loc, result); |         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; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -237,7 +257,8 @@ public class BukkitPlotGenerator extends ChunkGenerator | |||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public String toString() { |     @Override | ||||||
|  |     public String toString() { | ||||||
|         if (this.platformGenerator == this) { |         if (this.platformGenerator == this) { | ||||||
|             return this.plotGenerator.getName(); |             return this.plotGenerator.getName(); | ||||||
|         } |         } | ||||||
| @@ -248,11 +269,16 @@ public class BukkitPlotGenerator extends ChunkGenerator | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public boolean equals(final Object obj) { |     @Override | ||||||
|  |     public boolean equals(final Object obj) { | ||||||
|         if (obj == null) { |         if (obj == null) { | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|         return toString().equals(obj.toString()) || toString().equals(obj.getClass().getName()); |         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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | 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.location.Location; | ||||||
| import com.plotsquared.core.plot.PlotArea; | import com.plotsquared.core.plot.PlotArea; | ||||||
| import com.plotsquared.core.plot.PlotId; | 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.plotsquared.core.util.MathMan; | ||||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.block.Biome; | import org.bukkit.block.Biome; | ||||||
| import org.bukkit.generator.BlockPopulator; | import org.bukkit.generator.BlockPopulator; | ||||||
| import org.bukkit.generator.ChunkGenerator; | import org.bukkit.generator.ChunkGenerator; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
| import org.jetbrains.annotations.Range; |  | ||||||
|  |  | ||||||
| import java.util.Random; | import java.util.Random; | ||||||
|  |  | ||||||
| @@ -53,18 +52,22 @@ final class DelegatePlotGenerator extends IndependentPlotGenerator { | |||||||
|         this.world = world; |         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(); |         return this.chunkGenerator.getClass().getName(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public PlotArea getNewPlotArea(String world, String id, PlotId min, PlotId max) { |     @Override | ||||||
|         return PlotSquared.get().IMP.getDefaultGenerator().getNewPlotArea(world, id, min, max); |     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); |         World world = BukkitUtil.getWorld(this.world); | ||||||
|         Location min = result.getMin(); |         Location min = result.getMin(); | ||||||
|         int chunkX = min.getX() >> 4; |         int chunkX = min.getX() >> 4; | ||||||
| @@ -73,22 +76,24 @@ final class DelegatePlotGenerator extends IndependentPlotGenerator { | |||||||
|         try { |         try { | ||||||
|             ChunkGenerator.BiomeGrid grid = new ChunkGenerator.BiomeGrid() { |             ChunkGenerator.BiomeGrid grid = new ChunkGenerator.BiomeGrid() { | ||||||
|                 @Override |                 @Override | ||||||
|                 public void setBiome(@Range(from = 0, to = 15) int x, |                 public void setBiome(int x, int z, @NonNull Biome biome) { | ||||||
|                     @Range(from = 0, to = 15) int z, @NotNull Biome biome) { |  | ||||||
|                     result.setBiome(x, z, BukkitAdapter.adapt(biome)); |                     result.setBiome(x, z, BukkitAdapter.adapt(biome)); | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 //do not annotate with Override until we discontinue support for 1.4.4 |                 //do not annotate with Override until we discontinue support for 1.4.4 (we no longer support 1.4.4) | ||||||
|                 public void setBiome(int x, int y, int z, @NotNull Biome biome) { |                 @Override | ||||||
|  |                 public void setBiome(int x, int y, int z, @NonNull Biome biome) { | ||||||
|                     result.setBiome(x, z, BukkitAdapter.adapt(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; |                     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; |                     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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,18 +21,24 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.listener; | ||||||
|  |  | ||||||
|  | import com.google.inject.Inject; | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.location.Location; | import com.plotsquared.core.location.Location; | ||||||
| import com.plotsquared.core.plot.Plot; | 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.RefClass; | ||||||
| import com.plotsquared.core.util.ReflectionUtils.RefField; | import com.plotsquared.core.util.ReflectionUtils.RefField; | ||||||
| import com.plotsquared.core.util.ReflectionUtils.RefMethod; | 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.TaskManager; | ||||||
|  | import com.plotsquared.core.util.task.TaskTime; | ||||||
| import io.papermc.lib.PaperLib; | import io.papermc.lib.PaperLib; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
| import org.bukkit.Chunk; | import org.bukkit.Chunk; | ||||||
| @@ -51,65 +57,101 @@ import org.bukkit.event.entity.CreatureSpawnEvent; | |||||||
| import org.bukkit.event.entity.ItemSpawnEvent; | import org.bukkit.event.entity.ItemSpawnEvent; | ||||||
| import org.bukkit.event.world.ChunkLoadEvent; | import org.bukkit.event.world.ChunkLoadEvent; | ||||||
| import org.bukkit.event.world.ChunkUnloadEvent; | import org.bukkit.event.world.ChunkUnloadEvent; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.lang.reflect.Method; | import java.lang.reflect.Method; | ||||||
| import java.util.HashSet; | import java.util.HashSet; | ||||||
|  | import java.util.Objects; | ||||||
|  |  | ||||||
| import static com.plotsquared.core.util.ReflectionUtils.getRefClass; | import static com.plotsquared.core.util.ReflectionUtils.getRefClass; | ||||||
|  |  | ||||||
| @SuppressWarnings("unused") | @SuppressWarnings("unused") | ||||||
| public class ChunkListener implements Listener { | public class ChunkListener implements Listener { | ||||||
|  |  | ||||||
|  |     private final PlotAreaManager plotAreaManager; | ||||||
|  |     private final int version; | ||||||
|  |  | ||||||
|     private RefMethod methodGetHandleChunk; |     private RefMethod methodGetHandleChunk; | ||||||
|  |     private RefMethod methodGetHandleWorld; | ||||||
|     private RefField mustSave; |     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 Chunk lastChunk; | ||||||
|     private boolean ignoreUnload = false; |     private boolean ignoreUnload = false; | ||||||
|  |     private boolean isTrueForNotSave = true; | ||||||
|  |  | ||||||
|     public ChunkListener() { |     @Inject | ||||||
|         if (Settings.Chunk_Processor.AUTO_TRIM) { |     public ChunkListener(final @NonNull PlotAreaManager plotAreaManager) { | ||||||
|             try { |         this.plotAreaManager = plotAreaManager; | ||||||
|                 RefClass classChunk = getRefClass("{nms}.Chunk"); |         version = PlotSquared.platform().serverVersion()[1]; | ||||||
|                 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; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         if (!Settings.Chunk_Processor.AUTO_TRIM) { |         if (!Settings.Chunk_Processor.AUTO_TRIM) { | ||||||
|             return; |             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()) { |         for (World world : Bukkit.getWorlds()) { | ||||||
|             world.setAutoSave(false); |             world.setAutoSave(false); | ||||||
|         } |         } | ||||||
|  |         if (version > 13) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|         TaskManager.runTaskRepeat(() -> { |         TaskManager.runTaskRepeat(() -> { | ||||||
|             try { |             try { | ||||||
|                 HashSet<Chunk> toUnload = new HashSet<>(); |                 HashSet<Chunk> toUnload = new HashSet<>(); | ||||||
|                 for (World world : Bukkit.getWorlds()) { |                 for (World world : Bukkit.getWorlds()) { | ||||||
|                     String worldName = world.getName(); |                     String worldName = world.getName(); | ||||||
|                     if (!PlotSquared.get().hasPlotArea(worldName)) { |                     if (!this.plotAreaManager.hasPlotArea(worldName)) { | ||||||
|                         continue; |                         continue; | ||||||
|                     } |                     } | ||||||
|                     Object w = world.getClass().getDeclaredMethod("getHandle").invoke(world); |                     Object craftWorld = methodGetHandleWorld.of(world).call(); | ||||||
|                     Object chunkMap = w.getClass().getDeclaredMethod("getPlayerChunkMap").invoke(w); |                     if (version == 13) { | ||||||
|                     Method methodIsChunkInUse = |                         Object chunkMap = craftWorld.getClass().getDeclaredMethod("getPlayerChunkMap").invoke(craftWorld); | ||||||
|                         chunkMap.getClass().getDeclaredMethod("isChunkInUse", int.class, int.class); |                         Method methodIsChunkInUse = | ||||||
|                     Chunk[] chunks = world.getLoadedChunks(); |                                 chunkMap.getClass().getDeclaredMethod("isChunkInUse", int.class, int.class); | ||||||
|                     for (Chunk chunk : chunks) { |                         Chunk[] chunks = world.getLoadedChunks(); | ||||||
|                         if ((boolean) methodIsChunkInUse |                         for (Chunk chunk : chunks) { | ||||||
|                             .invoke(chunkMap, chunk.getX(), chunk.getZ())) { |                             if ((boolean) methodIsChunkInUse.invoke(chunkMap, chunk.getX(), chunk.getZ())) { | ||||||
|                             continue; |                                 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()) { |                 if (toUnload.isEmpty()) { | ||||||
| @@ -125,7 +167,7 @@ public class ChunkListener implements Listener { | |||||||
|             } catch (Throwable e) { |             } catch (Throwable e) { | ||||||
|                 e.printStackTrace(); |                 e.printStackTrace(); | ||||||
|             } |             } | ||||||
|         }, 1); |         }, TaskTime.ticks(1L)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public boolean unloadChunk(String world, Chunk chunk, boolean safe) { |     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(); |         Object c = this.methodGetHandleChunk.of(chunk).call(); | ||||||
|         RefField.RefExecutor field = this.mustSave.of(c); |         RefField.RefExecutor field = this.mustSave.of(c); | ||||||
|         if ((Boolean) field.get()) { |         if ((Boolean) field.get() != isTrueForNotSave) { | ||||||
|             field.set(false); |             field.set(isTrueForNotSave); | ||||||
|             if (chunk.isLoaded()) { |             if (chunk.isLoaded()) { | ||||||
|                 ignoreUnload = true; |                 ignoreUnload = true; | ||||||
|                 chunk.unload(false); |                 chunk.unload(false); | ||||||
| @@ -150,34 +192,56 @@ public class ChunkListener implements Listener { | |||||||
|         int z = chunkZ << 4; |         int z = chunkZ << 4; | ||||||
|         int x2 = x + 15; |         int x2 = x + 15; | ||||||
|         int z2 = z + 15; |         int z2 = z + 15; | ||||||
|         Plot plot = new Location(world, x, 1, z).getOwnedPlotAbs(); |         Location loc = Location.at(world, x, 1, z); | ||||||
|         if (plot != null && plot.hasOwner()) { |         PlotArea plotArea = plotAreaManager.getPlotArea(loc); | ||||||
|             return true; |         if (plotArea != null) { | ||||||
|  |             Plot plot = plotArea.getPlot(loc); | ||||||
|  |             if (plot != null && plot.hasOwner()) { | ||||||
|  |                 return true; | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|         plot = new Location(world, x2, 1, z2).getOwnedPlotAbs(); |         loc = Location.at(world, x2, 1, z2); | ||||||
|         if (plot != null && plot.hasOwner()) { |         plotArea = plotAreaManager.getPlotArea(loc); | ||||||
|             return true; |         if (plotArea != null) { | ||||||
|  |             Plot plot = plotArea.getPlot(loc); | ||||||
|  |             if (plot != null && plot.hasOwner()) { | ||||||
|  |                 return true; | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|         plot = new Location(world, x2, 1, z).getOwnedPlotAbs(); |         loc = Location.at(world, x2, 1, z); | ||||||
|         if (plot != null && plot.hasOwner()) { |         plotArea = plotAreaManager.getPlotArea(loc); | ||||||
|             return true; |         if (plotArea != null) { | ||||||
|  |             Plot plot = plotArea.getPlot(loc); | ||||||
|  |             if (plot != null && plot.hasOwner()) { | ||||||
|  |                 return true; | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|         plot = new Location(world, x, 1, z2).getOwnedPlotAbs(); |         loc = Location.at(world, x, 1, z2); | ||||||
|         if (plot != null && plot.hasOwner()) { |         plotArea = plotAreaManager.getPlotArea(loc); | ||||||
|             return true; |         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(); |         return plot != null && plot.hasOwner(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler public void onChunkUnload(ChunkUnloadEvent event) { |     @EventHandler | ||||||
|  |     public void onChunkUnload(ChunkUnloadEvent event) { | ||||||
|         if (ignoreUnload) { |         if (ignoreUnload) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Chunk chunk = event.getChunk(); |         Chunk chunk = event.getChunk(); | ||||||
|         if (Settings.Chunk_Processor.AUTO_TRIM) { |         if (Settings.Chunk_Processor.AUTO_TRIM) { | ||||||
|             String world = chunk.getWorld().getName(); |             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)) { |                 if (unloadChunk(world, chunk, true)) { | ||||||
|                     return; |                     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); |         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(); |         Item entity = event.getEntity(); | ||||||
|         PaperLib.getChunkAtAsync(event.getLocation()).thenAccept(chunk -> { |         PaperLib.getChunkAtAsync(event.getLocation()).thenAccept(chunk -> { | ||||||
|             if (chunk == this.lastChunk) { |             if (chunk == this.lastChunk) { | ||||||
| @@ -200,7 +266,7 @@ public class ChunkListener implements Listener { | |||||||
|                 event.setCancelled(true); |                 event.setCancelled(true); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             if (!PlotSquared.get().hasPlotArea(chunk.getWorld().getName())) { |             if (!this.plotAreaManager.hasPlotArea(chunk.getWorld().getName())) { | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             Entity[] entities = chunk.getEntities(); |             Entity[] entities = chunk.getEntities(); | ||||||
| @@ -230,7 +296,7 @@ public class ChunkListener implements Listener { | |||||||
|                 event.setCancelled(true); |                 event.setCancelled(true); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             if (!PlotSquared.get().hasPlotArea(chunk.getWorld().getName())) { |             if (!this.plotAreaManager.hasPlotArea(chunk.getWorld().getName())) { | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             Entity[] entities = chunk.getEntities(); |             Entity[] entities = chunk.getEntities(); | ||||||
| @@ -245,21 +311,16 @@ public class ChunkListener implements Listener { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void cleanChunk(final Chunk chunk) { |     private void cleanChunk(final Chunk chunk) { | ||||||
|         TaskManager.index.incrementAndGet(); |         final int currentIndex = TaskManager.index.incrementAndGet(); | ||||||
|         final Integer currentIndex = TaskManager.index.get(); |         PlotSquaredTask task = TaskManager.runTaskRepeat(() -> { | ||||||
|         Integer task = TaskManager.runTaskRepeat(() -> { |  | ||||||
|             if (!chunk.isLoaded()) { |             if (!chunk.isLoaded()) { | ||||||
|                 Bukkit.getScheduler().cancelTask(TaskManager.tasks.get(currentIndex)); |                 Objects.requireNonNull(TaskManager.removeTask(currentIndex)).cancel(); | ||||||
|                 TaskManager.tasks.remove(currentIndex); |  | ||||||
|                 PlotSquared.debug("Successfully processed and unloaded chunk!"); |  | ||||||
|                 chunk.unload(true); |                 chunk.unload(true); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             BlockState[] tiles = chunk.getTileEntities(); |             BlockState[] tiles = chunk.getTileEntities(); | ||||||
|             if (tiles.length == 0) { |             if (tiles.length == 0) { | ||||||
|                 Bukkit.getScheduler().cancelTask(TaskManager.tasks.get(currentIndex)); |                 Objects.requireNonNull(TaskManager.removeTask(currentIndex)).cancel(); | ||||||
|                 TaskManager.tasks.remove(currentIndex); |  | ||||||
|                 PlotSquared.debug("Successfully processed and unloaded chunk!"); |  | ||||||
|                 chunk.unload(true); |                 chunk.unload(true); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
| @@ -267,21 +328,19 @@ public class ChunkListener implements Listener { | |||||||
|             int i = 0; |             int i = 0; | ||||||
|             while (System.currentTimeMillis() - start < 250) { |             while (System.currentTimeMillis() - start < 250) { | ||||||
|                 if (i >= tiles.length - Settings.Chunk_Processor.MAX_TILES) { |                 if (i >= tiles.length - Settings.Chunk_Processor.MAX_TILES) { | ||||||
|                     Bukkit.getScheduler().cancelTask(TaskManager.tasks.get(currentIndex)); |                     Objects.requireNonNull(TaskManager.removeTask(currentIndex)).cancel(); | ||||||
|                     TaskManager.tasks.remove(currentIndex); |  | ||||||
|                     PlotSquared.debug("Successfully processed and unloaded chunk!"); |  | ||||||
|                     chunk.unload(true); |                     chunk.unload(true); | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|                 tiles[i].getBlock().setType(Material.AIR, false); |                 tiles[i].getBlock().setType(Material.AIR, false); | ||||||
|                 i++; |                 i++; | ||||||
|             } |             } | ||||||
|         }, 5); |         }, TaskTime.ticks(5L)); | ||||||
|         TaskManager.tasks.put(currentIndex, task); |         TaskManager.addTask(task, currentIndex); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public boolean processChunk(Chunk chunk, boolean unload) { |     public boolean processChunk(Chunk chunk, boolean unload) { | ||||||
|         if (!PlotSquared.get().hasPlotArea(chunk.getWorld().getName())) { |         if (!this.plotAreaManager.hasPlotArea(chunk.getWorld().getName())) { | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|         Entity[] entities = chunk.getEntities(); |         Entity[] entities = chunk.getEntities(); | ||||||
| @@ -296,24 +355,18 @@ public class ChunkListener implements Listener { | |||||||
|                     toRemove--; |                     toRemove--; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             PlotSquared.debug( |  | ||||||
|                 "PlotSquared detected unsafe chunk and processed: " + (chunk.getX() << 4) + "," + ( |  | ||||||
|                     chunk.getX() << 4)); |  | ||||||
|         } |         } | ||||||
|         if (tiles.length > Settings.Chunk_Processor.MAX_TILES) { |         if (tiles.length > Settings.Chunk_Processor.MAX_TILES) { | ||||||
|             if (unload) { |             if (unload) { | ||||||
|                 PlotSquared.debug( |  | ||||||
|                     "PlotSquared detected unsafe chunk: " + (chunk.getX() << 4) + "," + ( |  | ||||||
|                         chunk.getX() << 4)); |  | ||||||
|                 cleanChunk(chunk); |                 cleanChunk(chunk); | ||||||
|                 return true; |                 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); |                 tiles[i].getBlock().setType(Material.AIR, false); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,28 +21,43 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | 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.BukkitEntityUtil; | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
|  | import com.plotsquared.core.listener.PlayerBlockEventType; | ||||||
| import com.plotsquared.core.location.Location; | 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.Plot; | ||||||
| import com.plotsquared.core.plot.PlotArea; | 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.DisablePhysicsFlag; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.EntityChangeBlockFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.ExplosionFlag; | import com.plotsquared.core.plot.flag.implementations.ExplosionFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.InvincibleFlag; | import com.plotsquared.core.plot.flag.implementations.InvincibleFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.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.Material; | ||||||
|  | import org.bukkit.Particle; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.block.Block; | import org.bukkit.block.Block; | ||||||
| import org.bukkit.entity.Ageable; | import org.bukkit.entity.Ageable; | ||||||
|  | import org.bukkit.entity.Boat; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
| import org.bukkit.entity.EntityType; | import org.bukkit.entity.EntityType; | ||||||
| import org.bukkit.entity.FallingBlock; | import org.bukkit.entity.FallingBlock; | ||||||
|  | import org.bukkit.entity.Player; | ||||||
|  | import org.bukkit.entity.Projectile; | ||||||
| import org.bukkit.entity.TNTPrimed; | import org.bukkit.entity.TNTPrimed; | ||||||
| import org.bukkit.entity.Vehicle; | import org.bukkit.entity.Vehicle; | ||||||
| import org.bukkit.event.EventHandler; | import org.bukkit.event.EventHandler; | ||||||
| @@ -59,6 +74,9 @@ import org.bukkit.event.vehicle.VehicleCreateEvent; | |||||||
| import org.bukkit.metadata.FixedMetadataValue; | import org.bukkit.metadata.FixedMetadataValue; | ||||||
| import org.bukkit.metadata.MetadataValue; | import org.bukkit.metadata.MetadataValue; | ||||||
| import org.bukkit.plugin.Plugin; | import org.bukkit.plugin.Plugin; | ||||||
|  | import org.bukkit.projectiles.BlockProjectileSource; | ||||||
|  | import org.bukkit.projectiles.ProjectileSource; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.util.Iterator; | import java.util.Iterator; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| @@ -66,13 +84,28 @@ import java.util.List; | |||||||
| @SuppressWarnings("unused") | @SuppressWarnings("unused") | ||||||
| public class EntityEventListener implements Listener { | public class EntityEventListener implements Listener { | ||||||
|  |  | ||||||
|  |     private final PlotAreaManager plotAreaManager; | ||||||
|  |     private final EventDispatcher eventDispatcher; | ||||||
|     private float lastRadius; |     private float lastRadius; | ||||||
|  |  | ||||||
|  |     @Inject | ||||||
|  |     public EntityEventListener( | ||||||
|  |             final @NonNull PlotAreaManager plotAreaManager, | ||||||
|  |             final @NonNull EventDispatcher eventDispatcher | ||||||
|  |     ) { | ||||||
|  |         this.plotAreaManager = plotAreaManager; | ||||||
|  |         this.eventDispatcher = eventDispatcher; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST) |     @EventHandler(priority = EventPriority.HIGHEST) | ||||||
|     public void onEntityCombustByEntity(EntityCombustByEntityEvent event) { |     public void onEntityCombustByEntity(EntityCombustByEntityEvent event) { | ||||||
|         EntityDamageByEntityEvent eventChange = |         EntityDamageByEntityEvent eventChange = | ||||||
|             new EntityDamageByEntityEvent(event.getCombuster(), event.getEntity(), |                 new EntityDamageByEntityEvent( | ||||||
|                 EntityDamageEvent.DamageCause.FIRE_TICK, event.getDuration()); |                         event.getCombuster(), | ||||||
|  |                         event.getEntity(), | ||||||
|  |                         EntityDamageEvent.DamageCause.FIRE_TICK, | ||||||
|  |                         event.getDuration() | ||||||
|  |                 ); | ||||||
|         onEntityDamageByEntityEvent(eventChange); |         onEntityDamageByEntityEvent(eventChange); | ||||||
|         if (eventChange.isCancelled()) { |         if (eventChange.isCancelled()) { | ||||||
|             event.setCancelled(true); |             event.setCancelled(true); | ||||||
| @@ -82,8 +115,8 @@ public class EntityEventListener implements Listener { | |||||||
|     @EventHandler(priority = EventPriority.HIGHEST) |     @EventHandler(priority = EventPriority.HIGHEST) | ||||||
|     public void onEntityDamageByEntityEvent(EntityDamageByEntityEvent event) { |     public void onEntityDamageByEntityEvent(EntityDamageByEntityEvent event) { | ||||||
|         Entity damager = event.getDamager(); |         Entity damager = event.getDamager(); | ||||||
|         Location location = BukkitUtil.getLocation(damager); |         Location location = BukkitUtil.adapt(damager.getLocation()); | ||||||
|         if (!PlotSquared.get().hasPlotArea(location.getWorld())) { |         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Entity victim = event.getEntity(); |         Entity victim = event.getEntity(); | ||||||
| @@ -98,8 +131,7 @@ public class EntityEventListener implements Listener { | |||||||
| */ | */ | ||||||
|         if (!BukkitEntityUtil.entityDamage(damager, victim, event.getCause())) { |         if (!BukkitEntityUtil.entityDamage(damager, victim, event.getCause())) { | ||||||
|             if (event.isCancelled()) { |             if (event.isCancelled()) { | ||||||
|                 if (victim instanceof Ageable) { |                 if (victim instanceof Ageable ageable) { | ||||||
|                     Ageable ageable = (Ageable) victim; |  | ||||||
|                     if (ageable.getAge() == -24000) { |                     if (ageable.getAge() == -24000) { | ||||||
|                         ageable.setAge(0); |                         ageable.setAge(0); | ||||||
|                         ageable.setAdult(); |                         ageable.setAdult(); | ||||||
| @@ -113,7 +145,7 @@ public class EntityEventListener implements Listener { | |||||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|     public void creatureSpawnEvent(CreatureSpawnEvent event) { |     public void creatureSpawnEvent(CreatureSpawnEvent event) { | ||||||
|         Entity entity = event.getEntity(); |         Entity entity = event.getEntity(); | ||||||
|         Location location = BukkitUtil.getLocation(entity.getLocation()); |         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
|             return; |             return; | ||||||
| @@ -135,8 +167,8 @@ public class EntityEventListener implements Listener { | |||||||
|             case "PATROL": |             case "PATROL": | ||||||
|             case "RAID": |             case "RAID": | ||||||
|             case "SHEARED": |             case "SHEARED": | ||||||
|             case "SHOULDER_ENTITY": |  | ||||||
|             case "SILVERFISH_BLOCK": |             case "SILVERFISH_BLOCK": | ||||||
|  |             case "ENDER_PEARL": | ||||||
|             case "TRAP": |             case "TRAP": | ||||||
|             case "VILLAGE_DEFENSE": |             case "VILLAGE_DEFENSE": | ||||||
|             case "VILLAGE_INVASION": |             case "VILLAGE_INVASION": | ||||||
| @@ -176,7 +208,7 @@ public class EntityEventListener implements Listener { | |||||||
|             } |             } | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         if (BukkitEntityUtil.checkEntity(entity, plot)) { |         if (BukkitEntityUtil.checkEntity(entity, plot.getBasePlot(false))) { | ||||||
|             event.setCancelled(true); |             event.setCancelled(true); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -189,10 +221,10 @@ public class EntityEventListener implements Listener { | |||||||
|         Block block = event.getBlock(); |         Block block = event.getBlock(); | ||||||
|         World world = block.getWorld(); |         World world = block.getWorld(); | ||||||
|         String worldName = world.getName(); |         String worldName = world.getName(); | ||||||
|         if (!PlotSquared.get().hasPlotArea(worldName)) { |         if (!this.plotAreaManager.hasPlotArea(worldName)) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Location location = BukkitUtil.getLocation(block.getLocation()); |         Location location = BukkitUtil.adapt(block.getLocation()); | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
|             return; |             return; | ||||||
| @@ -220,16 +252,16 @@ public class EntityEventListener implements Listener { | |||||||
|                 entity.remove(); |                 entity.remove(); | ||||||
|             } |             } | ||||||
|         } else if (event.getTo() == Material.AIR) { |         } else if (event.getTo() == Material.AIR) { | ||||||
|             event.getEntity() |             event.getEntity().setMetadata("plot", new FixedMetadataValue((Plugin) PlotSquared.platform(), plot)); | ||||||
|                 .setMetadata("plot", new FixedMetadataValue((Plugin) PlotSquared.get().IMP, plot)); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGH) public void onDamage(EntityDamageEvent event) { |     @EventHandler(priority = EventPriority.HIGH) | ||||||
|  |     public void onDamage(EntityDamageEvent event) { | ||||||
|         if (event.getEntityType() != EntityType.PLAYER) { |         if (event.getEntityType() != EntityType.PLAYER) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Location location = BukkitUtil.getLocation(event.getEntity()); |         Location location = BukkitUtil.adapt(event.getEntity().getLocation()); | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
|             return; |             return; | ||||||
| @@ -242,19 +274,18 @@ public class EntityEventListener implements Listener { | |||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         if (plot.getFlag(InvincibleFlag.class)) { |         if (plot.getFlag(InvincibleFlag.class)) { | ||||||
|             plot.debug( |             plot.debug(event.getEntity().getName() + " could not take damage because invincible = true"); | ||||||
|                 event.getEntity().getName() + " could not take damage because invincible = true"); |  | ||||||
|             event.setCancelled(true); |             event.setCancelled(true); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|     public void onBigBoom(EntityExplodeEvent event) { |     public void onBigBoom(EntityExplodeEvent event) { | ||||||
|         Location location = BukkitUtil.getLocation(event.getLocation()); |         Location location = BukkitUtil.adapt(event.getLocation()); | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         boolean plotArea = location.isPlotArea(); |         boolean plotArea = location.isPlotArea(); | ||||||
|         if (!plotArea) { |         if (!plotArea) { | ||||||
|             if (!PlotSquared.get().hasPlotArea(location.getWorld())) { |             if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             return; |             return; | ||||||
| @@ -270,14 +301,11 @@ public class EntityEventListener implements Listener { | |||||||
|                     origin = (Plot) meta.get(0).value(); |                     origin = (Plot) meta.get(0).value(); | ||||||
|                 } |                 } | ||||||
|                 if (this.lastRadius != 0) { |                 if (this.lastRadius != 0) { | ||||||
|                     List<Entity> nearby = event.getEntity() |                     List<Entity> nearby = event.getEntity().getNearbyEntities(this.lastRadius, this.lastRadius, this.lastRadius); | ||||||
|                         .getNearbyEntities(this.lastRadius, this.lastRadius, this.lastRadius); |  | ||||||
|                     for (Entity near : nearby) { |                     for (Entity near : nearby) { | ||||||
|                         if (near instanceof TNTPrimed || near.getType() |                         if (near instanceof TNTPrimed || near.getType().equals(EntityType.MINECART_TNT)) { | ||||||
|                             .equals(EntityType.MINECART_TNT)) { |  | ||||||
|                             if (!near.hasMetadata("plot")) { |                             if (!near.hasMetadata("plot")) { | ||||||
|                                 near.setMetadata("plot", |                                 near.setMetadata("plot", new FixedMetadataValue((Plugin) PlotSquared.platform(), plot)); | ||||||
|                                     new FixedMetadataValue((Plugin) PlotSquared.get().IMP, plot)); |  | ||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
| @@ -286,9 +314,8 @@ public class EntityEventListener implements Listener { | |||||||
|                 Iterator<Block> iterator = event.blockList().iterator(); |                 Iterator<Block> iterator = event.blockList().iterator(); | ||||||
|                 while (iterator.hasNext()) { |                 while (iterator.hasNext()) { | ||||||
|                     Block block = iterator.next(); |                     Block block = iterator.next(); | ||||||
|                     location = BukkitUtil.getLocation(block.getLocation()); |                     location = BukkitUtil.adapt(block.getLocation()); | ||||||
|                     if (!area.contains(location.getX(), location.getZ()) || (origin != null |                     if (!area.contains(location.getX(), location.getZ()) || !origin.equals(area.getOwnedPlot(location))) { | ||||||
|                         && !origin.equals(area.getOwnedPlot(location)))) { |  | ||||||
|                         iterator.remove(); |                         iterator.remove(); | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
| @@ -298,35 +325,111 @@ public class EntityEventListener implements Listener { | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         event.setCancelled(true); |         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) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|     public void onPeskyMobsChangeTheWorldLikeWTFEvent(EntityChangeBlockEvent event) { |     public void onPeskyMobsChangeTheWorldLikeWTFEvent(EntityChangeBlockEvent event) { | ||||||
|         Entity e = event.getEntity(); |         Entity e = event.getEntity(); | ||||||
|         if (!(e instanceof FallingBlock)) { |         Material type = event.getBlock().getType(); | ||||||
|             Location location = BukkitUtil.getLocation(event.getBlock().getLocation()); |         Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||||
|             PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|             if (area != null) { |         if (area == null) { | ||||||
|                 Plot plot = area.getOwnedPlot(location); |             return; | ||||||
|                 if (plot != null && plot.getFlag(MobPlaceFlag.class)) { |         } | ||||||
|  |         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; |                     return; | ||||||
|                 } |                 } | ||||||
|                 if (plot != null) { |                 if (plot.isAdded(pp.getUUID()) || Permissions | ||||||
|                     plot.debug(e.getType() + " could not change block because mob-place = false"); |                         .hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||||
|  |                     return; | ||||||
|                 } |                 } | ||||||
|  |                 entity.remove(); | ||||||
|                 event.setCancelled(true); |                 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; |         this.lastRadius = event.getRadius() + 1; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|     public void onVehicleCreate(VehicleCreateEvent event) { |     public void onVehicleCreate(VehicleCreateEvent event) { | ||||||
|         Vehicle entity = event.getVehicle(); |         Vehicle entity = event.getVehicle(); | ||||||
|         Location location = BukkitUtil.getLocation(entity); |         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
|             return; |             return; | ||||||
| @@ -337,8 +440,8 @@ public class EntityEventListener implements Listener { | |||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         if (Settings.Enabled_Components.KILL_ROAD_VEHICLES) { |         if (Settings.Enabled_Components.KILL_ROAD_VEHICLES) { | ||||||
|             entity |             entity.setMetadata("plot", new FixedMetadataValue((Plugin) PlotSquared.platform(), plot)); | ||||||
|                 .setMetadata("plot", new FixedMetadataValue((Plugin) PlotSquared.get().IMP, plot)); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.listener; | ||||||
|  |  | ||||||
| @@ -36,6 +36,7 @@ import com.plotsquared.core.plot.flag.implementations.DoneFlag; | |||||||
| import io.papermc.lib.PaperLib; | import io.papermc.lib.PaperLib; | ||||||
| import org.bukkit.Chunk; | import org.bukkit.Chunk; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
|  | import org.bukkit.block.Block; | ||||||
| import org.bukkit.entity.ArmorStand; | import org.bukkit.entity.ArmorStand; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
| import org.bukkit.entity.EntityType; | import org.bukkit.entity.EntityType; | ||||||
| @@ -54,40 +55,43 @@ import org.bukkit.event.world.ChunkLoadEvent; | |||||||
| import org.bukkit.metadata.FixedMetadataValue; | import org.bukkit.metadata.FixedMetadataValue; | ||||||
| import org.bukkit.metadata.MetadataValue; | import org.bukkit.metadata.MetadataValue; | ||||||
| import org.bukkit.plugin.Plugin; | import org.bukkit.plugin.Plugin; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  |  | ||||||
| @SuppressWarnings("unused") |  | ||||||
| public class EntitySpawnListener implements Listener { | 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 ignoreTP = false; | ||||||
|  |     private static boolean hasPlotArea = false; | ||||||
|  |     private static String areaName = null; | ||||||
|  |  | ||||||
|     public static void testNether(final Entity entity) { |     public static void testNether(final Entity entity) { | ||||||
|         @NotNull World world = entity.getWorld(); |         @NonNull World world = entity.getWorld(); | ||||||
|         if (world.getEnvironment() != World.Environment.NETHER |         if (world.getEnvironment() != World.Environment.NETHER && world.getEnvironment() != World.Environment.THE_END) { | ||||||
|             && world.getEnvironment() != World.Environment.THE_END) { |  | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         test(entity); |         test(entity); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static void testCreate(final Entity entity) { |     public static void testCreate(final Entity entity) { | ||||||
|         @NotNull World world = entity.getWorld(); |         @NonNull World world = entity.getWorld(); | ||||||
|         if (!PlotSquared.get().hasPlotArea(entity.getWorld().getName())) { |         if (!world.getName().equals(areaName)) { | ||||||
|  |             areaName = world.getName(); | ||||||
|  |             hasPlotArea = PlotSquared.get().getPlotAreaManager().hasPlotArea(areaName); | ||||||
|  |         } | ||||||
|  |         if (!hasPlotArea) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         test(entity); |         test(entity); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static void test(Entity entity) { |     public static void test(Entity entity) { | ||||||
|         @NotNull World world = entity.getWorld(); |         @NonNull World world = entity.getWorld(); | ||||||
|         List<MetadataValue> meta = entity.getMetadata(KEY); |         List<MetadataValue> meta = entity.getMetadata(KEY); | ||||||
|         if (meta.isEmpty()) { |         if (meta.isEmpty()) { | ||||||
|             if (PlotSquared.get().hasPlotArea(world.getName())) { |             if (PlotSquared.get().getPlotAreaManager().hasPlotArea(world.getName())) { | ||||||
|                 entity.setMetadata(KEY, |                 entity.setMetadata(KEY, new FixedMetadataValue((Plugin) PlotSquared.platform(), entity.getLocation())); | ||||||
|                     new FixedMetadataValue((Plugin) PlotSquared.get().IMP, entity.getLocation())); |  | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             org.bukkit.Location origin = (org.bukkit.Location) meta.get(0).value(); |             org.bukkit.Location origin = (org.bukkit.Location) meta.get(0).value(); | ||||||
| @@ -121,7 +125,7 @@ public class EntitySpawnListener implements Listener { | |||||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|     public void creatureSpawnEvent(EntitySpawnEvent event) { |     public void creatureSpawnEvent(EntitySpawnEvent event) { | ||||||
|         Entity entity = event.getEntity(); |         Entity entity = event.getEntity(); | ||||||
|         Location location = BukkitUtil.getLocation(entity.getLocation()); |         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (!location.isPlotArea()) { |         if (!location.isPlotArea()) { | ||||||
|             return; |             return; | ||||||
| @@ -158,34 +162,62 @@ public class EntitySpawnListener implements Listener { | |||||||
|                 } |                 } | ||||||
|             case SHULKER: |             case SHULKER: | ||||||
|                 if (!entity.hasMetadata("shulkerPlot")) { |                 if (!entity.hasMetadata("shulkerPlot")) { | ||||||
|                     entity.setMetadata("shulkerPlot", |                     entity.setMetadata("shulkerPlot", new FixedMetadataValue((Plugin) PlotSquared.platform(), plot.getId())); | ||||||
|                         new FixedMetadataValue((Plugin) PlotSquared.get().IMP, plot.getId())); |  | ||||||
|                 } |                 } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler public void onChunkLoad(ChunkLoadEvent event) { |     @EventHandler | ||||||
|         @NotNull Chunk chunk = event.getChunk(); |     public void onChunkLoad(ChunkLoadEvent event) { | ||||||
|  |         @NonNull Chunk chunk = event.getChunk(); | ||||||
|         for (final Entity entity : chunk.getEntities()) { |         for (final Entity entity : chunk.getEntities()) { | ||||||
|             testCreate(entity); |             testCreate(entity); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler public void onVehicle(VehicleUpdateEvent event) { |     @EventHandler | ||||||
|  |     public void onVehicle(VehicleUpdateEvent event) { | ||||||
|         testNether(event.getVehicle()); |         testNether(event.getVehicle()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler public void onVehicle(VehicleCreateEvent event) { |     @EventHandler | ||||||
|  |     public void onVehicle(VehicleCreateEvent event) { | ||||||
|         testCreate(event.getVehicle()); |         testCreate(event.getVehicle()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler public void onVehicle(VehicleBlockCollisionEvent event) { |     @EventHandler | ||||||
|  |     public void onVehicle(VehicleBlockCollisionEvent event) { | ||||||
|         testNether(event.getVehicle()); |         testNether(event.getVehicle()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler public void onTeleport(EntityTeleportEvent event) { |     @EventHandler | ||||||
|         Entity ent = event.getEntity(); |     public void onTeleport(EntityTeleportEvent event) { | ||||||
|         if (ent instanceof Vehicle || ent instanceof ArmorStand) { |         Entity entity = event.getEntity(); | ||||||
|  |         Entity fromLocation = event.getEntity(); | ||||||
|  |         Block toLocation = event.getTo().getBlock(); | ||||||
|  |         final Location fromLocLocation = BukkitUtil.adapt(fromLocation.getLocation()); | ||||||
|  |         final PlotArea fromArea = fromLocLocation.getPlotArea(); | ||||||
|  |         Location toLocLocation = BukkitUtil.adapt(toLocation.getLocation()); | ||||||
|  |         PlotArea toArea = toLocLocation.getPlotArea(); | ||||||
|  |  | ||||||
|  |         if (toArea == null) { | ||||||
|  |             if (fromLocation.getType() == EntityType.SHULKER && fromArea != null) { | ||||||
|  |                 event.setCancelled(true); | ||||||
|  |             } | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         Plot toPlot = toArea.getOwnedPlot(toLocLocation); | ||||||
|  |         if (fromLocation.getType() == EntityType.SHULKER && fromArea != null) { | ||||||
|  |             final Plot fromPlot = fromArea.getOwnedPlot(fromLocLocation); | ||||||
|  |  | ||||||
|  |             if (fromPlot != null || toPlot != null) { | ||||||
|  |                 if ((fromPlot == null || !fromPlot.equals(toPlot)) && (toPlot == null || !toPlot.equals(fromPlot))) { | ||||||
|  |                     event.setCancelled(true); | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         if (entity instanceof Vehicle || entity instanceof ArmorStand) { | ||||||
|             testNether(event.getEntity()); |             testNether(event.getEntity()); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -195,9 +227,11 @@ public class EntitySpawnListener implements Listener { | |||||||
|         testNether(event.getVehicle()); |         testNether(event.getVehicle()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler public void spawn(CreatureSpawnEvent event) { |     @EventHandler | ||||||
|  |     public void spawn(CreatureSpawnEvent event) { | ||||||
|         if (event.getEntityType() == EntityType.ARMOR_STAND) { |         if (event.getEntityType() == EntityType.ARMOR_STAND) { | ||||||
|             testCreate(event.getEntity()); |             testCreate(event.getEntity()); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,15 +21,15 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.listener; | ||||||
|  |  | ||||||
| import com.google.common.collect.Iterables; | import com.google.common.collect.Iterables; | ||||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | import com.plotsquared.bukkit.player.BukkitPlayer; | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
| import com.plotsquared.core.configuration.Captions; |  | ||||||
| import com.plotsquared.core.location.Location; | import com.plotsquared.core.location.Location; | ||||||
|  | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| import com.plotsquared.core.plot.flag.implementations.ForcefieldFlag; | import com.plotsquared.core.plot.flag.implementations.ForcefieldFlag; | ||||||
| @@ -44,13 +44,13 @@ import java.util.UUID; | |||||||
| @SuppressWarnings("unused") | @SuppressWarnings("unused") | ||||||
| public class ForceFieldListener { | public class ForceFieldListener { | ||||||
|  |  | ||||||
|     private static Set<PlotPlayer> getNearbyPlayers(Player player, Plot plot) { |     private static Set<PlotPlayer<?>> getNearbyPlayers(Player player, Plot plot) { | ||||||
|         Set<PlotPlayer> players = new HashSet<>(); |         Set<PlotPlayer<?>> players = new HashSet<>(); | ||||||
|         for (Player nearPlayer : Iterables |         for (Player nearPlayer : Iterables | ||||||
|             .filter(player.getNearbyEntities(5d, 5d, 5d), Player.class)) { |                 .filter(player.getNearbyEntities(5d, 5d, 5d), Player.class)) { | ||||||
|             PlotPlayer plotPlayer; |             PlotPlayer<?> plotPlayer; | ||||||
|             if ((plotPlayer = BukkitUtil.getPlayer(nearPlayer)) == null || !plot |             if ((plotPlayer = BukkitUtil.adapt(nearPlayer)) == null || !plot | ||||||
|                 .equals(plotPlayer.getCurrentPlot())) { |                     .equals(plotPlayer.getCurrentPlot())) { | ||||||
|                 continue; |                 continue; | ||||||
|             } |             } | ||||||
|             if (!plot.isAdded(plotPlayer.getUUID())) { |             if (!plot.isAdded(plotPlayer.getUUID())) { | ||||||
| @@ -60,12 +60,12 @@ public class ForceFieldListener { | |||||||
|         return players; |         return players; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static PlotPlayer hasNearbyPermitted(Player player, Plot plot) { |     private static PlotPlayer<?> hasNearbyPermitted(Player player, Plot plot) { | ||||||
|         for (Player nearPlayer : Iterables |         for (Player nearPlayer : Iterables | ||||||
|             .filter(player.getNearbyEntities(5d, 5d, 5d), Player.class)) { |                 .filter(player.getNearbyEntities(5d, 5d, 5d), Player.class)) { | ||||||
|             PlotPlayer plotPlayer; |             PlotPlayer<?> plotPlayer; | ||||||
|             if ((plotPlayer = BukkitUtil.getPlayer(nearPlayer)) == null || !plot |             if ((plotPlayer = BukkitUtil.adapt(nearPlayer)) == null || !plot | ||||||
|                 .equals(plotPlayer.getCurrentPlot())) { |                     .equals(plotPlayer.getCurrentPlot())) { | ||||||
|                 continue; |                 continue; | ||||||
|             } |             } | ||||||
|             if (plot.isAdded(plotPlayer.getUUID())) { |             if (plot.isAdded(plotPlayer.getUUID())) { | ||||||
| @@ -75,7 +75,7 @@ public class ForceFieldListener { | |||||||
|         return null; |         return null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static Vector calculateVelocity(PlotPlayer player, PlotPlayer e) { |     private static Vector calculateVelocity(PlotPlayer<?> player, PlotPlayer<?> e) { | ||||||
|         Location playerLocation = player.getLocationFull(); |         Location playerLocation = player.getLocationFull(); | ||||||
|         Location oPlayerLocation = e.getLocation(); |         Location oPlayerLocation = e.getLocation(); | ||||||
|         double playerX = playerLocation.getX(); |         double playerX = playerLocation.getX(); | ||||||
| @@ -105,28 +105,29 @@ public class ForceFieldListener { | |||||||
|         return new Vector(x, y, z); |         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)) { |         if (plot.getFlag(ForcefieldFlag.class)) { | ||||||
|             UUID uuid = plotPlayer.getUUID(); |             UUID uuid = plotPlayer.getUUID(); | ||||||
|             if (plot.isAdded(uuid)) { |             if (plot.isAdded(uuid)) { | ||||||
|                 Set<PlotPlayer> players = getNearbyPlayers(player, plot); |                 Set<PlotPlayer<?>> players = getNearbyPlayers(player, plot); | ||||||
|                 for (PlotPlayer oPlayer : players) { |                 for (PlotPlayer<?> oPlayer : players) { | ||||||
|                     if (!Permissions |                     if (!Permissions | ||||||
|                         .hasPermission(oPlayer, Captions.PERMISSION_ADMIN_ENTRY_FORCEFIELD)) { |                             .hasPermission(oPlayer, Permission.PERMISSION_ADMIN_ENTRY_FORCEFIELD)) { | ||||||
|                         ((BukkitPlayer) oPlayer).player |                         ((BukkitPlayer) oPlayer).player | ||||||
|                             .setVelocity(calculateVelocity(plotPlayer, oPlayer)); |                                 .setVelocity(calculateVelocity(plotPlayer, oPlayer)); | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } else { |             } else { | ||||||
|                 PlotPlayer oPlayer = hasNearbyPermitted(player, plot); |                 PlotPlayer<?> oPlayer = hasNearbyPermitted(player, plot); | ||||||
|                 if (oPlayer == null) { |                 if (oPlayer == null) { | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|                 if (!Permissions |                 if (!Permissions | ||||||
|                     .hasPermission(plotPlayer, Captions.PERMISSION_ADMIN_ENTRY_FORCEFIELD)) { |                         .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_ENTRY_FORCEFIELD)) { | ||||||
|                     player.setVelocity(calculateVelocity(oPlayer, plotPlayer)); |                     player.setVelocity(calculateVelocity(oPlayer, plotPlayer)); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.listener; | ||||||
|  |  | ||||||
| @@ -32,17 +32,22 @@ import com.destroystokyo.paper.event.entity.PreSpawnerSpawnEvent; | |||||||
| import com.destroystokyo.paper.event.entity.SlimePathfindEvent; | import com.destroystokyo.paper.event.entity.SlimePathfindEvent; | ||||||
| import com.destroystokyo.paper.event.player.PlayerLaunchProjectileEvent; | import com.destroystokyo.paper.event.player.PlayerLaunchProjectileEvent; | ||||||
| import com.destroystokyo.paper.event.server.AsyncTabCompleteEvent; | import com.destroystokyo.paper.event.server.AsyncTabCompleteEvent; | ||||||
|  | import com.google.inject.Inject; | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
| import com.plotsquared.core.PlotSquared; |  | ||||||
| import com.plotsquared.core.command.Command; | import com.plotsquared.core.command.Command; | ||||||
| import com.plotsquared.core.command.MainCommand; | import com.plotsquared.core.command.MainCommand; | ||||||
| import com.plotsquared.core.configuration.Captions; |  | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
|  | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.location.Location; | import com.plotsquared.core.location.Location; | ||||||
|  | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| import com.plotsquared.core.plot.PlotArea; | import com.plotsquared.core.plot.PlotArea; | ||||||
| import com.plotsquared.core.plot.flag.implementations.DoneFlag; | import com.plotsquared.core.plot.flag.implementations.DoneFlag; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.ProjectilesFlag; | ||||||
|  | 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.Chunk; | ||||||
| import org.bukkit.block.Block; | import org.bukkit.block.Block; | ||||||
| import org.bukkit.block.TileState; | import org.bukkit.block.TileState; | ||||||
| @@ -51,13 +56,13 @@ import org.bukkit.entity.EntityType; | |||||||
| import org.bukkit.entity.Player; | import org.bukkit.entity.Player; | ||||||
| import org.bukkit.entity.Projectile; | import org.bukkit.entity.Projectile; | ||||||
| import org.bukkit.entity.Slime; | import org.bukkit.entity.Slime; | ||||||
| import org.bukkit.entity.ThrownPotion; |  | ||||||
| import org.bukkit.event.EventHandler; | import org.bukkit.event.EventHandler; | ||||||
| import org.bukkit.event.EventPriority; | import org.bukkit.event.EventPriority; | ||||||
| import org.bukkit.event.Listener; | import org.bukkit.event.Listener; | ||||||
| import org.bukkit.event.block.BlockPlaceEvent; | import org.bukkit.event.block.BlockPlaceEvent; | ||||||
| import org.bukkit.event.entity.CreatureSpawnEvent; | import org.bukkit.event.entity.CreatureSpawnEvent; | ||||||
| import org.bukkit.projectiles.ProjectileSource; | import org.bukkit.projectiles.ProjectileSource; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
| @@ -71,14 +76,21 @@ import java.util.regex.Pattern; | |||||||
| @SuppressWarnings("unused") | @SuppressWarnings("unused") | ||||||
| public class PaperListener implements Listener { | public class PaperListener implements Listener { | ||||||
|  |  | ||||||
|  |     private final PlotAreaManager plotAreaManager; | ||||||
|     private Chunk lastChunk; |     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) { |         if (!Settings.Paper_Components.ENTITY_PATHING) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Location toLoc = BukkitUtil.getLocation(event.getLoc()); |         Location toLoc = BukkitUtil.adapt(event.getLoc()); | ||||||
|         Location fromLoc = BukkitUtil.getLocation(event.getEntity().getLocation()); |         Location fromLoc = BukkitUtil.adapt(event.getEntity().getLocation()); | ||||||
|         PlotArea tarea = toLoc.getPlotArea(); |         PlotArea tarea = toLoc.getPlotArea(); | ||||||
|         if (tarea == null) { |         if (tarea == null) { | ||||||
|             return; |             return; | ||||||
| @@ -106,7 +118,8 @@ public class PaperListener implements Listener { | |||||||
|         event.setCancelled(true); |         event.setCancelled(true); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler public void onEntityPathfind(SlimePathfindEvent event) { |     @EventHandler | ||||||
|  |     public void onEntityPathfind(SlimePathfindEvent event) { | ||||||
|         if (!Settings.Paper_Components.ENTITY_PATHING) { |         if (!Settings.Paper_Components.ENTITY_PATHING) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| @@ -117,8 +130,8 @@ public class PaperListener implements Listener { | |||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Location toLoc = BukkitUtil.getLocation(b.getLocation()); |         Location toLoc = BukkitUtil.adapt(b.getLocation()); | ||||||
|         Location fromLoc = BukkitUtil.getLocation(event.getEntity().getLocation()); |         Location fromLoc = BukkitUtil.adapt(event.getEntity().getLocation()); | ||||||
|         PlotArea tarea = toLoc.getPlotArea(); |         PlotArea tarea = toLoc.getPlotArea(); | ||||||
|         if (tarea == null) { |         if (tarea == null) { | ||||||
|             return; |             return; | ||||||
| @@ -147,11 +160,12 @@ public class PaperListener implements Listener { | |||||||
|         event.setCancelled(true); |         event.setCancelled(true); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler public void onPreCreatureSpawnEvent(PreCreatureSpawnEvent event) { |     @EventHandler | ||||||
|  |     public void onPreCreatureSpawnEvent(PreCreatureSpawnEvent event) { | ||||||
|         if (!Settings.Paper_Components.CREATURE_SPAWN) { |         if (!Settings.Paper_Components.CREATURE_SPAWN) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Location location = BukkitUtil.getLocation(event.getSpawnLocation()); |         Location location = BukkitUtil.adapt(event.getSpawnLocation()); | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (!location.isPlotArea()) { |         if (!location.isPlotArea()) { | ||||||
|             return; |             return; | ||||||
| @@ -181,8 +195,8 @@ public class PaperListener implements Listener { | |||||||
|             case "PATROL": |             case "PATROL": | ||||||
|             case "RAID": |             case "RAID": | ||||||
|             case "SHEARED": |             case "SHEARED": | ||||||
|             case "SHOULDER_ENTITY": |  | ||||||
|             case "SILVERFISH_BLOCK": |             case "SILVERFISH_BLOCK": | ||||||
|  |             case "ENDER_PEARL": | ||||||
|             case "TRAP": |             case "TRAP": | ||||||
|             case "VILLAGE_DEFENSE": |             case "VILLAGE_DEFENSE": | ||||||
|             case "VILLAGE_INVASION": |             case "VILLAGE_INVASION": | ||||||
| @@ -253,7 +267,7 @@ public class PaperListener implements Listener { | |||||||
|     @EventHandler |     @EventHandler | ||||||
|     public void onPlayerNaturallySpawnCreaturesEvent(PlayerNaturallySpawnCreaturesEvent event) { |     public void onPlayerNaturallySpawnCreaturesEvent(PlayerNaturallySpawnCreaturesEvent event) { | ||||||
|         if (Settings.Paper_Components.CANCEL_CHUNK_SPAWN) { |         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(); |             PlotArea area = location.getPlotArea(); | ||||||
|             if (area != null && !area.isMobSpawning()) { |             if (area != null && !area.isMobSpawning()) { | ||||||
|                 event.setCancelled(true); |                 event.setCancelled(true); | ||||||
| @@ -261,9 +275,10 @@ public class PaperListener implements Listener { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler public void onPreSpawnerSpawnEvent(PreSpawnerSpawnEvent event) { |     @EventHandler | ||||||
|  |     public void onPreSpawnerSpawnEvent(PreSpawnerSpawnEvent event) { | ||||||
|         if (Settings.Paper_Components.SPAWNER_SPAWN) { |         if (Settings.Paper_Components.SPAWNER_SPAWN) { | ||||||
|             Location location = BukkitUtil.getLocation(event.getSpawnerLocation()); |             Location location = BukkitUtil.adapt(event.getSpawnerLocation()); | ||||||
|             PlotArea area = location.getPlotArea(); |             PlotArea area = location.getPlotArea(); | ||||||
|             if (area != null && !area.isMobSpawnerSpawning()) { |             if (area != null && !area.isMobSpawnerSpawning()) { | ||||||
|                 event.setCancelled(true); |                 event.setCancelled(true); | ||||||
| @@ -272,22 +287,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) { |         if (!Settings.Paper_Components.TILE_ENTITY_CHECK || !Settings.Enabled_Components.CHUNK_PROCESSOR) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         if (!(event.getBlock().getState(false) instanceof TileState)) { |         if (!(event.getBlock().getState(false) instanceof TileState)) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         final Location location = BukkitUtil.getLocation(event.getBlock().getLocation()); |         final Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||||
|         final PlotArea plotArea = location.getPlotArea(); |         final PlotArea plotArea = location.getPlotArea(); | ||||||
|         if (plotArea == null) { |         if (plotArea == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         final int tileEntityCount = event.getBlock().getChunk().getTileEntities(false).length; |         final int tileEntityCount = event.getBlock().getChunk().getTileEntities(false).length; | ||||||
|         if (tileEntityCount >= Settings.Chunk_Processor.MAX_TILES) { |         if (tileEntityCount >= Settings.Chunk_Processor.MAX_TILES) { | ||||||
|             final PlotPlayer<?> plotPlayer = BukkitUtil.getPlayer(event.getPlayer()); |             final PlotPlayer<?> plotPlayer = BukkitUtil.adapt(event.getPlayer()); | ||||||
|             Captions.TILE_ENTITY_CAP_REACHED.send(plotPlayer, Settings.Chunk_Processor.MAX_TILES); |             plotPlayer.sendMessage( | ||||||
|  |                     TranslatableCaption.of("errors.tile_entity_cap_reached"), | ||||||
|  |                     Template.of("amount", String.valueOf(Settings.Chunk_Processor.MAX_TILES)) | ||||||
|  |             ); | ||||||
|             event.setCancelled(true); |             event.setCancelled(true); | ||||||
|             event.setBuild(false); |             event.setBuild(false); | ||||||
|         } |         } | ||||||
| @@ -299,31 +318,57 @@ public class PaperListener implements Listener { | |||||||
|      * |      * | ||||||
|      * @param event Paper's PlayerLaunchProjectileEvent |      * @param event Paper's PlayerLaunchProjectileEvent | ||||||
|      */ |      */ | ||||||
|     @EventHandler public void onProjectileLaunch(PlayerLaunchProjectileEvent event) { |     @EventHandler | ||||||
|  |     public void onProjectileLaunch(PlayerLaunchProjectileEvent event) { | ||||||
|         if (!Settings.Paper_Components.PLAYER_PROJECTILE) { |         if (!Settings.Paper_Components.PLAYER_PROJECTILE) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Projectile entity = event.getProjectile(); |         Projectile entity = event.getProjectile(); | ||||||
|         if (!(entity instanceof ThrownPotion)) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         ProjectileSource shooter = entity.getShooter(); |         ProjectileSource shooter = entity.getShooter(); | ||||||
|         if (!(shooter instanceof Player)) { |         if (!(shooter instanceof Player)) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Location location = BukkitUtil.getLocation(entity); |         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||||
|         if (!PlotSquared.get().hasPlotArea(location.getWorld())) { |         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         PlotPlayer<?> pp = BukkitUtil.getPlayer((Player) shooter); |         PlotPlayer<Player> pp = BukkitUtil.adapt((Player) shooter); | ||||||
|         Plot plot = location.getOwnedPlot(); |         Plot plot = location.getOwnedPlot(); | ||||||
|         if (plot != null && !plot.isAdded(pp.getUUID())) { |  | ||||||
|             entity.remove(); |         if (plot == null) { | ||||||
|             event.setCancelled(true); |             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 (!plot.getFlag(ProjectilesFlag.class)) { | ||||||
|  |                 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) { |         if (!Settings.Paper_Components.ASYNC_TAB_COMPLETION) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| @@ -341,13 +386,13 @@ public class PaperListener implements Listener { | |||||||
|         if (unprocessedArgs.length == 1) { |         if (unprocessedArgs.length == 1) { | ||||||
|             return; // We don't do anything in this case |             return; // We don't do anything in this case | ||||||
|         } else if (!Settings.Enabled_Components.TAB_COMPLETED_ALIASES |         } else if (!Settings.Enabled_Components.TAB_COMPLETED_ALIASES | ||||||
|             .contains(unprocessedArgs[0].toLowerCase(Locale.ENGLISH))) { |                 .contains(unprocessedArgs[0].toLowerCase(Locale.ENGLISH))) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         final String[] args = new String[unprocessedArgs.length - 1]; |         final String[] args = new String[unprocessedArgs.length - 1]; | ||||||
|         System.arraycopy(unprocessedArgs, 1, args, 0, args.length); |         System.arraycopy(unprocessedArgs, 1, args, 0, args.length); | ||||||
|         try { |         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(" ")); |             final Collection<Command> objects = MainCommand.getInstance().tab(player, args, buffer.endsWith(" ")); | ||||||
|             if (objects == null) { |             if (objects == null) { | ||||||
|                 return; |                 return; | ||||||
| @@ -358,7 +403,8 @@ public class PaperListener implements Listener { | |||||||
|             } |             } | ||||||
|             event.setCompletions(result); |             event.setCompletions(result); | ||||||
|             event.setHandled(true); |             event.setHandled(true); | ||||||
|         } catch (final Exception ignored) {} |         } catch (final Exception ignored) { | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,19 +21,21 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.listener; | ||||||
|  |  | ||||||
|  | import com.google.inject.Inject; | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
| import com.plotsquared.core.configuration.Captions; |  | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
|  | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.location.Location; | import com.plotsquared.core.location.Location; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.PlotArea; | 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.Banner; | ||||||
| import org.bukkit.block.Beacon; | import org.bukkit.block.Beacon; | ||||||
| import org.bukkit.block.Bed; |  | ||||||
| import org.bukkit.block.BlockState; | import org.bukkit.block.BlockState; | ||||||
| import org.bukkit.block.CommandBlock; | import org.bukkit.block.CommandBlock; | ||||||
| import org.bukkit.block.Comparator; | import org.bukkit.block.Comparator; | ||||||
| @@ -48,36 +50,45 @@ import org.bukkit.block.Jukebox; | |||||||
| import org.bukkit.block.Sign; | import org.bukkit.block.Sign; | ||||||
| import org.bukkit.block.Skull; | import org.bukkit.block.Skull; | ||||||
| import org.bukkit.block.Structure; | import org.bukkit.block.Structure; | ||||||
|  | import org.bukkit.block.data.type.Bed; | ||||||
| import org.bukkit.event.EventHandler; | import org.bukkit.event.EventHandler; | ||||||
| import org.bukkit.event.block.BlockPlaceEvent; | import org.bukkit.event.block.BlockPlaceEvent; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| public class PaperListener113 extends PaperListener { | public class PaperListener113 extends PaperListener { | ||||||
|  |  | ||||||
|  |     @Inject | ||||||
|  |     public PaperListener113(@NonNull PlotAreaManager plotAreaManager) { | ||||||
|  |         super(plotAreaManager); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @EventHandler |     @EventHandler | ||||||
|     public void onBlockPlace(BlockPlaceEvent event) { |     public void onBlockPlace(BlockPlaceEvent event) { | ||||||
|         if (!Settings.Paper_Components.TILE_ENTITY_CHECK || !Settings.Enabled_Components.CHUNK_PROCESSOR) { |         if (!Settings.Paper_Components.TILE_ENTITY_CHECK || !Settings.Enabled_Components.CHUNK_PROCESSOR) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         BlockState state = event.getBlock().getState(false); |         BlockState state = event.getBlock().getState(false); | ||||||
|         if (!(state instanceof Banner || state instanceof Beacon || state instanceof Bed |         if (!(state instanceof Banner || state instanceof Beacon || state instanceof Bed || state instanceof CommandBlock | ||||||
|                 || state instanceof CommandBlock || state instanceof Comparator || state instanceof Conduit |                 || state instanceof Comparator || state instanceof Conduit || state instanceof Container || state instanceof CreatureSpawner | ||||||
|                 || state instanceof Container || state instanceof CreatureSpawner || state instanceof DaylightDetector |                 || state instanceof DaylightDetector || state instanceof EnchantingTable || state instanceof EnderChest || state instanceof EndGateway | ||||||
|                 || state instanceof EnchantingTable || state instanceof EnderChest || state instanceof EndGateway |                 || state instanceof Jukebox || state instanceof Sign || state instanceof Skull || state instanceof Structure)) { | ||||||
|                 || state instanceof Jukebox || state instanceof Sign || state instanceof Skull |  | ||||||
|                 || state instanceof Structure)) { |  | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         final Location location = BukkitUtil.getLocation(event.getBlock().getLocation()); |         final Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||||
|         final PlotArea plotArea = location.getPlotArea(); |         final PlotArea plotArea = location.getPlotArea(); | ||||||
|         if (plotArea == null) { |         if (plotArea == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         final int tileEntityCount = event.getBlock().getChunk().getTileEntities(false).length; |         final int tileEntityCount = event.getBlock().getChunk().getTileEntities(false).length; | ||||||
|         if (tileEntityCount >= Settings.Chunk_Processor.MAX_TILES) { |         if (tileEntityCount >= Settings.Chunk_Processor.MAX_TILES) { | ||||||
|             final PlotPlayer<?> plotPlayer = BukkitUtil.getPlayer(event.getPlayer()); |             final PlotPlayer<?> plotPlayer = BukkitUtil.adapt(event.getPlayer()); | ||||||
|             Captions.TILE_ENTITY_CAP_REACHED.send(plotPlayer, Settings.Chunk_Processor.MAX_TILES); |             plotPlayer.sendMessage( | ||||||
|  |                     TranslatableCaption.of("errors.tile_entity_cap_reached"), | ||||||
|  |                     Template.of("amount", String.valueOf(Settings.Chunk_Processor.MAX_TILES)) | ||||||
|  |             ); | ||||||
|             event.setCancelled(true); |             event.setCancelled(true); | ||||||
|             event.setBuild(false); |             event.setBuild(false); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,20 +21,24 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.listener; | ||||||
|  |  | ||||||
|  | import com.google.inject.Inject; | ||||||
| import com.plotsquared.bukkit.util.BukkitEntityUtil; | import com.plotsquared.bukkit.util.BukkitEntityUtil; | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.configuration.Captions; |  | ||||||
| import com.plotsquared.core.location.Location; | import com.plotsquared.core.location.Location; | ||||||
|  | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| import com.plotsquared.core.plot.PlotArea; | import com.plotsquared.core.plot.PlotArea; | ||||||
| import com.plotsquared.core.plot.PlotHandler; | import com.plotsquared.core.plot.PlotHandler; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.ProjectilesFlag; | ||||||
|  | import com.plotsquared.core.plot.world.PlotAreaManager; | ||||||
| import com.plotsquared.core.util.Permissions; | import com.plotsquared.core.util.Permissions; | ||||||
|  | import net.kyori.adventure.text.minimessage.Template; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
| import org.bukkit.entity.LivingEntity; | import org.bukkit.entity.LivingEntity; | ||||||
| import org.bukkit.entity.Player; | import org.bukkit.entity.Player; | ||||||
| @@ -49,27 +53,30 @@ import org.bukkit.event.entity.ProjectileHitEvent; | |||||||
| import org.bukkit.event.entity.ProjectileLaunchEvent; | import org.bukkit.event.entity.ProjectileLaunchEvent; | ||||||
| import org.bukkit.projectiles.BlockProjectileSource; | import org.bukkit.projectiles.BlockProjectileSource; | ||||||
| import org.bukkit.projectiles.ProjectileSource; | import org.bukkit.projectiles.ProjectileSource; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| @SuppressWarnings("unused") | @SuppressWarnings("unused") | ||||||
| public class ProjectileEventListener implements Listener { | 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) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|     public void onPotionSplash(LingeringPotionSplashEvent event) { |     public void onLingeringPotionSplash(LingeringPotionSplashEvent event) { | ||||||
|         Projectile entity = event.getEntity(); |         // Cancelling projectile hit events still results in area effect clouds. | ||||||
|         Location location = BukkitUtil.getLocation(entity); |         // We need to cancel the splash events to get rid of those. | ||||||
|         if (!PlotSquared.get().hasPlotArea(location.getWorld())) { |         onProjectileHit(event); | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         if (!this.onProjectileHit(event)) { |  | ||||||
|             event.setCancelled(true); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|     public void onPotionSplash(PotionSplashEvent event) { |     public void onPotionSplash(PotionSplashEvent event) { | ||||||
|         ThrownPotion damager = event.getPotion(); |         ThrownPotion damager = event.getPotion(); | ||||||
|         Location location = BukkitUtil.getLocation(damager); |         Location location = BukkitUtil.adapt(damager.getLocation()); | ||||||
|         if (!PlotSquared.get().hasPlotArea(location.getWorld())) { |         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         int count = 0; |         int count = 0; | ||||||
| @@ -79,78 +86,112 @@ public class ProjectileEventListener implements Listener { | |||||||
|                 count++; |                 count++; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         if ((count > 0 && count == event.getAffectedEntities().size()) || !onProjectileHit(event)) { |         if (count > 0 && count == event.getAffectedEntities().size()) { | ||||||
|             event.setCancelled(true); |             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(); |         Projectile entity = event.getEntity(); | ||||||
|         if (!(entity instanceof ThrownPotion)) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         ProjectileSource shooter = entity.getShooter(); |         ProjectileSource shooter = entity.getShooter(); | ||||||
|         if (!(shooter instanceof Player)) { |         if (!(shooter instanceof Player)) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Location location = BukkitUtil.getLocation(entity); |         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||||
|         if (!PlotSquared.get().hasPlotArea(location.getWorld())) { |         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         PlotPlayer<Player> pp = BukkitUtil.getPlayer((Player) shooter); |         PlotPlayer<Player> pp = BukkitUtil.adapt((Player) shooter); | ||||||
|         Plot plot = location.getOwnedPlot(); |         Plot plot = location.getOwnedPlot(); | ||||||
|         if (plot != null && !plot.isAdded(pp.getUUID())) { |  | ||||||
|             entity.remove(); |         if (plot == null) { | ||||||
|             event.setCancelled(true); |             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 (!plot.getFlag(ProjectilesFlag.class)) { | ||||||
|  |                 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 |     @EventHandler | ||||||
|     public boolean onProjectileHit(ProjectileHitEvent event) { |     public void onProjectileHit(ProjectileHitEvent event) { | ||||||
|         Projectile entity = event.getEntity(); |         Projectile entity = event.getEntity(); | ||||||
|         Location location = BukkitUtil.getLocation(entity); |         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||||
|         if (!PlotSquared.get().hasPlotArea(location.getWorld())) { |         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||||
|             return true; |             return; | ||||||
|         } |         } | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
|             return true; |             return; | ||||||
|         } |         } | ||||||
|         Plot plot = area.getPlot(location); |         Plot plot = area.getPlot(location); | ||||||
|         ProjectileSource shooter = entity.getShooter(); |         ProjectileSource shooter = entity.getShooter(); | ||||||
|         if (shooter instanceof Player) { |         if (shooter instanceof Player) { | ||||||
|             PlotPlayer<?> pp = BukkitUtil.getPlayer((Player) shooter); |             PlotPlayer<?> pp = BukkitUtil.adapt((Player) shooter); | ||||||
|             if (plot == null) { |             if (plot == null) { | ||||||
|                 if (!Permissions.hasPermission(pp, Captions.PERMISSION_PROJECTILE_UNOWNED)) { |                 if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { | ||||||
|                     entity.remove(); |                     entity.remove(); | ||||||
|                     return false; |                     event.setCancelled(true); | ||||||
|                 } |                 } | ||||||
|                 return true; |                 return; | ||||||
|             } |             } | ||||||
|             if (plot.isAdded(pp.getUUID()) || Permissions |             if (plot.isAdded(pp.getUUID()) || Permissions | ||||||
|                 .hasPermission(pp, Captions.PERMISSION_PROJECTILE_OTHER)) { |                     .hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_OTHER) || plot.getFlag(ProjectilesFlag.class)) { | ||||||
|                 return true; |                 return; | ||||||
|             } |             } | ||||||
|             entity.remove(); |             entity.remove(); | ||||||
|             return false; |             event.setCancelled(true); | ||||||
|  |             return; | ||||||
|         } |         } | ||||||
|         if (!(shooter instanceof Entity) && shooter != null) { |         if (!(shooter instanceof Entity) && shooter != null) { | ||||||
|             if (plot == null) { |             if (plot == null) { | ||||||
|                 entity.remove(); |                 entity.remove(); | ||||||
|                 return false; |                 event.setCancelled(true); | ||||||
|  |                 return; | ||||||
|             } |             } | ||||||
|             Location sLoc = |             Location sLoc = | ||||||
|                 BukkitUtil.getLocation(((BlockProjectileSource) shooter).getBlock().getLocation()); |                     BukkitUtil.adapt(((BlockProjectileSource) shooter).getBlock().getLocation()); | ||||||
|             if (!area.contains(sLoc.getX(), sLoc.getZ())) { |             if (!area.contains(sLoc.getX(), sLoc.getZ())) { | ||||||
|                 entity.remove(); |                 entity.remove(); | ||||||
|                 return false; |                 event.setCancelled(true); | ||||||
|  |                 return; | ||||||
|             } |             } | ||||||
|             Plot sPlot = area.getOwnedPlotAbs(sLoc); |             Plot sPlot = area.getOwnedPlotAbs(sLoc); | ||||||
|             if (sPlot == null || !PlotHandler.sameOwners(plot, sPlot)) { |             if (sPlot == null || !PlotHandler.sameOwners(plot, sPlot)) { | ||||||
|                 entity.remove(); |                 entity.remove(); | ||||||
|                 return false; |                 event.setCancelled(true); | ||||||
|  |                 return; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         return true; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,32 +21,37 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | 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.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.Settings; | ||||||
|  | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
|  | import com.plotsquared.core.player.ConsolePlayer; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
| import org.bukkit.event.EventHandler; | import org.bukkit.event.EventHandler; | ||||||
| import org.bukkit.event.Listener; | import org.bukkit.event.Listener; | ||||||
| import org.bukkit.event.server.ServerLoadEvent; | import org.bukkit.event.server.ServerLoadEvent; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| public class ServerListener implements Listener { | 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; |         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) { |         if (Bukkit.getPluginManager().getPlugin("MVdWPlaceholderAPI") != null && Settings.Enabled_Components.USE_MVDWAPI) { | ||||||
|             new MVdWPlaceholders(this.plugin, PlotSquared.get().getPlaceholderRegistry()); |             new MVdWPlaceholders(this.plugin, this.plugin.placeholderRegistry()); | ||||||
|             PlotSquared.log(Captions.PREFIX + "&6PlotSquared hooked into MVdWPlaceholderAPI"); |             ConsolePlayer.getConsole().sendMessage(TranslatableCaption.of("placeholder.hooked")); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,15 +21,15 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.listener; | ||||||
|  |  | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | 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.plot.world.SinglePlotAreaManager; | ||||||
| import com.plotsquared.core.util.ReflectionUtils; | import com.plotsquared.core.util.ReflectionUtils; | ||||||
| import org.bukkit.Bukkit; |  | ||||||
| import org.bukkit.Chunk; | import org.bukkit.Chunk; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.event.EventHandler; | import org.bukkit.event.EventHandler; | ||||||
| @@ -37,42 +37,45 @@ import org.bukkit.event.EventPriority; | |||||||
| import org.bukkit.event.Listener; | import org.bukkit.event.Listener; | ||||||
| import org.bukkit.event.world.ChunkEvent; | import org.bukkit.event.world.ChunkEvent; | ||||||
| import org.bukkit.event.world.ChunkLoadEvent; | import org.bukkit.event.world.ChunkLoadEvent; | ||||||
| import org.bukkit.plugin.Plugin; |  | ||||||
|  |  | ||||||
| import java.lang.reflect.Field; | import java.lang.reflect.Field; | ||||||
| import java.lang.reflect.Method; | import java.lang.reflect.Method; | ||||||
|  |  | ||||||
| import static com.plotsquared.core.util.ReflectionUtils.getRefClass; | import static com.plotsquared.core.util.ReflectionUtils.getRefClass; | ||||||
|  |  | ||||||
| @SuppressWarnings("unused") |  | ||||||
| public class SingleWorldListener implements Listener { | public class SingleWorldListener implements Listener { | ||||||
|  |  | ||||||
|     private Method methodGetHandleChunk; |     private final Method methodGetHandleChunk; | ||||||
|     private Field mustSave; |     private Field shouldSave = null; | ||||||
|     private boolean isTrueForNotSave = true; |  | ||||||
|  |  | ||||||
|     public SingleWorldListener(Plugin plugin) throws Exception { |     public SingleWorldListener() throws Exception { | ||||||
|         ReflectionUtils.RefClass classChunk = getRefClass("{nms}.Chunk"); |  | ||||||
|         ReflectionUtils.RefClass classCraftChunk = getRefClass("{cb}.CraftChunk"); |         ReflectionUtils.RefClass classCraftChunk = getRefClass("{cb}.CraftChunk"); | ||||||
|         this.methodGetHandleChunk = classCraftChunk.getMethod("getHandle").getRealMethod(); |         this.methodGetHandleChunk = classCraftChunk.getMethod("getHandle").getRealMethod(); | ||||||
|         try { |         try { | ||||||
|             if (PlotSquared.get().IMP.getServerVersion()[1] == 13) { |             if (PlotSquared.platform().serverVersion()[1] < 17) { | ||||||
|                 this.mustSave = classChunk.getField("mustSave").getRealField(); |                 ReflectionUtils.RefClass classChunk = getRefClass("{nms}.Chunk"); | ||||||
|                 this.isTrueForNotSave = false; |                 if (PlotSquared.platform().serverVersion()[1] == 13) { | ||||||
|             } else { |                     this.shouldSave = classChunk.getField("mustSave").getRealField(); | ||||||
|                 this.mustSave = classChunk.getField("mustNotSave").getRealField(); |                 } else { | ||||||
|  |                     this.shouldSave = classChunk.getField("s").getRealField(); | ||||||
|  |                 } | ||||||
|  |             } else if (PlotSquared.platform().serverVersion()[1] == 17) { | ||||||
|  |                 ReflectionUtils.RefClass classChunk = getRefClass("net.minecraft.world.level.chunk.Chunk"); | ||||||
|  |                 this.shouldSave = classChunk.getField("r").getRealField(); | ||||||
|  |             } else if (PlotSquared.platform().serverVersion()[1] == 18) { | ||||||
|  |                 ReflectionUtils.RefClass classChunk = getRefClass("net.minecraft.world.level.chunk.IChunkAccess"); | ||||||
|  |                 this.shouldSave = classChunk.getField("b").getRealField(); | ||||||
|             } |             } | ||||||
|         } catch (NoSuchFieldException e) { |         } catch (NoSuchFieldException e) { | ||||||
|             e.printStackTrace(); |             e.printStackTrace(); | ||||||
|         } |         } | ||||||
|         Bukkit.getPluginManager().registerEvents(this, plugin); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void markChunkAsClean(Chunk chunk) { |     public void markChunkAsClean(Chunk chunk) { | ||||||
|         try { |         try { | ||||||
|             Object nmsChunk = methodGetHandleChunk.invoke(chunk); |             Object nmsChunk = methodGetHandleChunk.invoke(chunk); | ||||||
|             if (mustSave != null) { |             if (shouldSave != null) { | ||||||
|                 this.mustSave.set(nmsChunk, false); |                 this.shouldSave.set(nmsChunk, false); | ||||||
|             } |             } | ||||||
|         } catch (Throwable e) { |         } catch (Throwable e) { | ||||||
|             e.printStackTrace(); |             e.printStackTrace(); | ||||||
| @@ -86,7 +89,7 @@ public class SingleWorldListener implements Listener { | |||||||
|         if (!(man instanceof SinglePlotAreaManager)) { |         if (!(man instanceof SinglePlotAreaManager)) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         if (!isPlotId(name)) { |         if (!SinglePlotArea.isSinglePlotWorld(name)) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -98,35 +101,9 @@ public class SingleWorldListener implements Listener { | |||||||
|     //        handle(event); |     //        handle(event); | ||||||
|     //    } |     //    } | ||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.LOWEST) public void onChunkLoad(ChunkLoadEvent event) { |     @EventHandler(priority = EventPriority.LOWEST) | ||||||
|  |     public void onChunkLoad(ChunkLoadEvent event) { | ||||||
|         handle(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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,10 +21,11 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.listener; | ||||||
|  |  | ||||||
|  | import com.google.inject.Inject; | ||||||
| import com.plotsquared.bukkit.generator.BukkitPlotGenerator; | import com.plotsquared.bukkit.generator.BukkitPlotGenerator; | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.generator.GeneratorWrapper; | import com.plotsquared.core.generator.GeneratorWrapper; | ||||||
| @@ -36,17 +37,23 @@ import org.bukkit.event.EventPriority; | |||||||
| import org.bukkit.event.Listener; | import org.bukkit.event.Listener; | ||||||
| import org.bukkit.event.world.WorldInitEvent; | import org.bukkit.event.world.WorldInitEvent; | ||||||
| import org.bukkit.generator.ChunkGenerator; | import org.bukkit.generator.ChunkGenerator; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| @SuppressWarnings("unused") | @SuppressWarnings("unused") | ||||||
| public class WorldEvents implements Listener { | 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) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|     public void onWorldInit(WorldInitEvent event) { |     public void onWorldInit(WorldInitEvent event) { | ||||||
|         World world = event.getWorld(); |         World world = event.getWorld(); | ||||||
|         String name = world.getName(); |         String name = world.getName(); | ||||||
|         PlotAreaManager manager = PlotSquared.get().getPlotAreaManager(); |         if (this.plotAreaManager instanceof final SinglePlotAreaManager single) { | ||||||
|         if (manager instanceof SinglePlotAreaManager) { |  | ||||||
|             SinglePlotAreaManager single = (SinglePlotAreaManager) manager; |  | ||||||
|             if (single.isWorld(name)) { |             if (single.isWorld(name)) { | ||||||
|                 world.setKeepSpawnInMemory(false); |                 world.setKeepSpawnInMemory(false); | ||||||
|                 return; |                 return; | ||||||
| @@ -56,7 +63,8 @@ public class WorldEvents implements Listener { | |||||||
|         if (gen instanceof GeneratorWrapper) { |         if (gen instanceof GeneratorWrapper) { | ||||||
|             PlotSquared.get().loadWorld(name, (GeneratorWrapper<?>) gen); |             PlotSquared.get().loadWorld(name, (GeneratorWrapper<?>) gen); | ||||||
|         } else { |         } 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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,18 +21,19 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.managers; | ||||||
|  |  | ||||||
|  | import com.google.inject.Singleton; | ||||||
| import com.plotsquared.core.configuration.file.YamlConfiguration; | import com.plotsquared.core.configuration.file.YamlConfiguration; | ||||||
| import com.plotsquared.core.util.PlatformWorldManager; | import com.plotsquared.core.util.PlatformWorldManager; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.WorldCreator; | import org.bukkit.WorldCreator; | ||||||
| import org.bukkit.WorldType; | import org.bukkit.WorldType; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
| import org.jetbrains.annotations.Nullable; | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  |  | ||||||
| import java.io.File; | import java.io.File; | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
| @@ -44,13 +45,15 @@ import java.util.List; | |||||||
|  * Default Bukkit world manager. It will handle world creation by |  * Default Bukkit world manager. It will handle world creation by | ||||||
|  * registering the generator in bukkit.yml |  * registering the generator in bukkit.yml | ||||||
|  */ |  */ | ||||||
|  | @Singleton | ||||||
| public class BukkitWorldManager implements PlatformWorldManager<World> { | public class BukkitWorldManager implements PlatformWorldManager<World> { | ||||||
|  |  | ||||||
|     @Override public void initialize() { |     @Override | ||||||
|  |     public void initialize() { | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override @Nullable |     @Override | ||||||
|     public World handleWorldCreation(@NotNull String worldName, @Nullable String generator) { |     public @Nullable World handleWorldCreation(@NonNull String worldName, @Nullable String generator) { | ||||||
|         this.setGenerator(worldName, generator); |         this.setGenerator(worldName, generator); | ||||||
|         final WorldCreator wc = new WorldCreator(worldName); |         final WorldCreator wc = new WorldCreator(worldName); | ||||||
|         wc.environment(World.Environment.NORMAL); |         wc.environment(World.Environment.NORMAL); | ||||||
| @@ -61,8 +64,8 @@ public class BukkitWorldManager implements PlatformWorldManager<World> { | |||||||
|         return Bukkit.createWorld(wc); |         return Bukkit.createWorld(wc); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     protected void setGenerator(@Nullable final String worldName, @Nullable final String generator) { |     protected void setGenerator(final @Nullable String worldName, final @Nullable String generator) { | ||||||
|         if (generator == null || worldName != null && worldName.contains(".")) { |         if (generator == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         File file = new File("bukkit.yml").getAbsoluteFile(); |         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"; |         return "bukkit"; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public Collection<String> getWorlds() { |     @Override | ||||||
|  |     public Collection<String> getWorlds() { | ||||||
|         final List<World> worlds = Bukkit.getWorlds(); |         final List<World> worlds = Bukkit.getWorlds(); | ||||||
|         final List<String> worldNames = new ArrayList<>(); |         final List<String> worldNames = new ArrayList<>(); | ||||||
|         for (final World world : worlds) { |         for (final World world : worlds) { | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,46 +21,50 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.managers; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  | import com.google.inject.Singleton; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
| import org.jetbrains.annotations.Nullable; | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
| import se.hyperver.hyperverse.Hyperverse; | import se.hyperver.hyperverse.Hyperverse; | ||||||
| import se.hyperver.hyperverse.world.WorldConfiguration; | import se.hyperver.hyperverse.world.WorldConfiguration; | ||||||
| import se.hyperver.hyperverse.world.WorldConfigurationBuilder; | import se.hyperver.hyperverse.world.WorldConfigurationBuilder; | ||||||
| import se.hyperver.hyperverse.world.WorldFeatures; | import se.hyperver.hyperverse.world.WorldFeatures; | ||||||
| import se.hyperver.hyperverse.world.WorldType; | import se.hyperver.hyperverse.world.WorldType; | ||||||
|  |  | ||||||
| /** | Hyperverse implementation is currently put on ice until Hyperverse is released on a stable line and deployed to the central | ||||||
|  * Hyperverse specific manager that creates worlds | repository. | ||||||
|  * using Hyperverse's API |  | ||||||
|  */ | @Singleton | ||||||
| public class HyperverseWorldManager extends BukkitWorldManager { | public class HyperverseWorldManager extends BukkitWorldManager { | ||||||
|  |  | ||||||
|     @Override @Nullable |     @Override | ||||||
|     public World handleWorldCreation(@NotNull String worldName, @Nullable String generator) { |     public @Nullable World handleWorldCreation(@NonNull String worldName, @Nullable String generator) { | ||||||
|         // First let Bukkit register the world |         // First let Bukkit register the world | ||||||
|         this.setGenerator(worldName, generator); |         this.setGenerator(worldName, generator); | ||||||
|         // Create the world |         // Create the world | ||||||
|         final WorldConfigurationBuilder worldConfigurationBuilder = WorldConfiguration.builder() |         final WorldConfigurationBuilder worldConfigurationBuilder = WorldConfiguration.builder() | ||||||
|             .setName(worldName).setType(WorldType.OVER_WORLD); |                 .setName(worldName).setType(WorldType.OVER_WORLD); | ||||||
|         if (generator != null) { |         if (generator != null) { | ||||||
|             worldConfigurationBuilder.setGenerator(generator).setWorldFeatures(WorldFeatures.FLATLAND); |             worldConfigurationBuilder.setGenerator(generator).setWorldFeatures(WorldFeatures.FLATLAND); | ||||||
|         } |         } | ||||||
|         try { |         try { | ||||||
|             return Hyperverse.getApi().createWorld(worldConfigurationBuilder.createWorldConfiguration()) |             return Hyperverse.getApi().createWorld(worldConfigurationBuilder.createWorldConfiguration()) | ||||||
|                 .getBukkitWorld(); |                     .getBukkitWorld(); | ||||||
|         } catch (final Exception e) { |         } catch (final Exception e) { | ||||||
|             e.printStackTrace(); |             e.printStackTrace(); | ||||||
|         } |         } | ||||||
|         return null; |         return null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public String getName() { |     @Override | ||||||
|  |     public String getName() { | ||||||
|         return "bukkit-hyperverse"; |         return "bukkit-hyperverse"; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  */ | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,28 +21,34 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.managers; | ||||||
|  |  | ||||||
|  | import com.google.inject.Singleton; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
| import org.jetbrains.annotations.Nullable; | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Multiverse specific manager that informs Multiverse of |  * Multiverse specific manager that informs Multiverse of | ||||||
|  * world creation by executing a console command |  * 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, since = "6.0.0") | ||||||
|  | @Singleton | ||||||
| public class MultiverseWorldManager extends BukkitWorldManager { | public class MultiverseWorldManager extends BukkitWorldManager { | ||||||
|  |  | ||||||
|     @Override @Nullable |     @Override | ||||||
|     public World handleWorldCreation(@NotNull final String worldName, @Nullable final String generator) { |     public @Nullable World handleWorldCreation(final @NonNull String worldName, final @Nullable String generator) { | ||||||
|         // First let Bukkit register the world |         // First let Bukkit register the world | ||||||
|         this.setGenerator(worldName, generator); |         this.setGenerator(worldName, generator); | ||||||
|         // Then we send the console command |         // Then we send the console command | ||||||
|         final StringBuilder commandBuilder = new StringBuilder("mv create ") |         final StringBuilder commandBuilder = new StringBuilder("mv create ") | ||||||
|             .append(worldName).append(" normal"); |                 .append(worldName).append(" normal"); | ||||||
|         if (generator != null) { |         if (generator != null) { | ||||||
|             commandBuilder.append(" -g ").append(generator); |             commandBuilder.append(" -g ").append(generator); | ||||||
|         } |         } | ||||||
| @@ -50,7 +56,8 @@ public class MultiverseWorldManager extends BukkitWorldManager { | |||||||
|         return Bukkit.getWorld(worldName); |         return Bukkit.getWorld(worldName); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public String getName() { |     @Override | ||||||
|  |     public String getName() { | ||||||
|         return "bukkit-multiverse"; |         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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.placeholder; | ||||||
|  |  | ||||||
| @@ -34,7 +34,7 @@ import com.plotsquared.core.util.placeholders.Placeholder; | |||||||
| import com.plotsquared.core.util.placeholders.PlaceholderRegistry; | import com.plotsquared.core.util.placeholders.PlaceholderRegistry; | ||||||
| import org.bukkit.entity.Player; | import org.bukkit.entity.Player; | ||||||
| import org.bukkit.plugin.Plugin; | import org.bukkit.plugin.Plugin; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Placeholder support for MVdWPlaceholderAPI |  * Placeholder support for MVdWPlaceholderAPI | ||||||
| @@ -45,8 +45,10 @@ public class MVdWPlaceholders { | |||||||
|     private final Plugin plugin; |     private final Plugin plugin; | ||||||
|     private final PlaceholderRegistry registry; |     private final PlaceholderRegistry registry; | ||||||
|  |  | ||||||
|     public MVdWPlaceholders(@NotNull final Plugin plugin, |     public MVdWPlaceholders( | ||||||
|                             @NotNull final PlaceholderRegistry registry) { |             final @NonNull Plugin plugin, | ||||||
|  |             final @NonNull PlaceholderRegistry registry | ||||||
|  |     ) { | ||||||
|         this.plugin = plugin; |         this.plugin = plugin; | ||||||
|         this.registry = registry; |         this.registry = registry; | ||||||
|         for (final Placeholder placeholder : registry.getPlaceholders()) { |         for (final Placeholder placeholder : registry.getPlaceholders()) { | ||||||
| @@ -55,24 +57,24 @@ public class MVdWPlaceholders { | |||||||
|         PlotSquared.get().getEventDispatcher().registerListener(this); |         PlotSquared.get().getEventDispatcher().registerListener(this); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Subscribe public void onNewPlaceholder(@NotNull final |     @Subscribe | ||||||
|         PlaceholderRegistry.PlaceholderAddedEvent event) { |     public void onNewPlaceholder(final PlaceholderRegistry.@NonNull PlaceholderAddedEvent event) { | ||||||
|         this.addPlaceholder(event.getPlaceholder()); |         this.addPlaceholder(event.getPlaceholder()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void addPlaceholder(@NotNull final Placeholder placeholder) { |     private void addPlaceholder(final @NonNull Placeholder placeholder) { | ||||||
|         PlaceholderAPI.registerPlaceholder(plugin, PREFIX + String.format("%s", placeholder.getKey()), |         PlaceholderAPI.registerPlaceholder( | ||||||
|             placeholderReplaceEvent -> { |                 plugin, | ||||||
|                 if (!placeholderReplaceEvent.isOnline() || placeholderReplaceEvent.getPlayer() == null) { |                 PREFIX + String.format("%s", placeholder.getKey()), | ||||||
|                     return ""; |                 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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.placeholder; | ||||||
|  |  | ||||||
| @@ -36,28 +36,34 @@ public class PAPIPlaceholders extends PlaceholderExpansion { | |||||||
|     public PAPIPlaceholders() { |     public PAPIPlaceholders() { | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public boolean persist() { |     @Override | ||||||
|  |     public boolean persist() { | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public boolean canRegister() { |     @Override | ||||||
|  |     public boolean canRegister() { | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public String getAuthor() { |     @Override | ||||||
|  |     public String getAuthor() { | ||||||
|         return "IntellectualSites"; |         return "IntellectualSites"; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public String getIdentifier() { |     @Override | ||||||
|  |     public String getIdentifier() { | ||||||
|         return "plotsquared"; |         return "plotsquared"; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public String getVersion() { |     @Override | ||||||
|  |     public String getVersion() { | ||||||
|         return "3"; |         return "3"; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public String onPlaceholderRequest(Player p, String identifier) { |     @Override | ||||||
|         final PlotPlayer<?> pl = PlotSquared.imp().getPlayerManager().getPlayerIfExists(p.getUniqueId()); |     public String onPlaceholderRequest(Player p, String identifier) { | ||||||
|  |         final PlotPlayer<?> pl = PlotSquared.platform().playerManager().getPlayerIfExists(p.getUniqueId()); | ||||||
|  |  | ||||||
|         if (pl == null) { |         if (pl == null) { | ||||||
|             return ""; |             return ""; | ||||||
| @@ -66,24 +72,26 @@ public class PAPIPlaceholders extends PlaceholderExpansion { | |||||||
|         // PAPI specific ones that don't translate well over into other placeholder APIs |         // PAPI specific ones that don't translate well over into other placeholder APIs | ||||||
|         if (identifier.startsWith("has_plot_")) { |         if (identifier.startsWith("has_plot_")) { | ||||||
|             identifier = identifier.substring("has_plot_".length()); |             identifier = identifier.substring("has_plot_".length()); | ||||||
|             if (identifier.isEmpty()) |             if (identifier.isEmpty()) { | ||||||
|                 return ""; |                 return ""; | ||||||
|  |             } | ||||||
|  |  | ||||||
|             return pl.getPlotCount(identifier) > 0 ? |             return pl.getPlotCount(identifier) > 0 ? | ||||||
|                 PlaceholderAPIPlugin.booleanTrue() : |                     PlaceholderAPIPlugin.booleanTrue() : | ||||||
|                 PlaceholderAPIPlugin.booleanFalse(); |                     PlaceholderAPIPlugin.booleanFalse(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (identifier.startsWith("plot_count_")) { |         if (identifier.startsWith("plot_count_")) { | ||||||
|             identifier = identifier.substring("plot_count_".length()); |             identifier = identifier.substring("plot_count_".length()); | ||||||
|             if (identifier.isEmpty()) |             if (identifier.isEmpty()) { | ||||||
|                 return ""; |                 return ""; | ||||||
|  |             } | ||||||
|  |  | ||||||
|             return String.valueOf(pl.getPlotCount(identifier)); |             return String.valueOf(pl.getPlotCount(identifier)); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         // PlotSquared placeholders |         // 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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,20 +21,22 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.placeholder; | ||||||
|  |  | ||||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | 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 com.plotsquared.core.player.PlotPlayer; | ||||||
| import me.clip.placeholderapi.PlaceholderAPI; | import me.clip.placeholderapi.PlaceholderAPI; | ||||||
| import org.bukkit.entity.Player; | import org.bukkit.entity.Player; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| public class PlaceholderFormatter implements ChatFormatter { | public class PlaceholderFormatter implements ChatFormatter { | ||||||
|  |  | ||||||
|     @Override public void format(final ChatContext context) { |     @Override | ||||||
|         final PlotPlayer recipient = context.getRecipient(); |     public void format(final @NonNull ChatContext context) { | ||||||
|  |         final PlotPlayer<?> recipient = context.getRecipient(); | ||||||
|         if (recipient instanceof BukkitPlayer) { |         if (recipient instanceof BukkitPlayer) { | ||||||
|             if (context.isRawOutput()) { |             if (context.isRawOutput()) { | ||||||
|                 context.setMessage(context.getMessage().replace('%', '\u2010')); |                 context.setMessage(context.getMessage().replace('%', '\u2010')); | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,43 +21,77 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | 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 com.plotsquared.core.player.OfflinePlotPlayer; | ||||||
| import org.bukkit.OfflinePlayer; | 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; | import java.util.UUID; | ||||||
|  |  | ||||||
| public class BukkitOfflinePlayer implements OfflinePlotPlayer { | public class BukkitOfflinePlayer implements OfflinePlotPlayer { | ||||||
|  |  | ||||||
|     public final OfflinePlayer player; |     public final OfflinePlayer player; | ||||||
|  |     private final PermissionProfile permissionProfile; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Please do not use this method. Instead use BukkitUtil.getPlayer(Player), |      * Please do not use this method. Instead use BukkitUtil.getPlayer(Player), | ||||||
|      * as it caches player objects. |      * 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.player = player; | ||||||
|  |         this.permissionProfile = permissionHandler.getPermissionProfile(this) | ||||||
|  |                 .orElse(NullPermissionProfile.INSTANCE); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @NotNull @Override public UUID getUUID() { |     @NonNull | ||||||
|  |     @Override | ||||||
|  |     public UUID getUUID() { | ||||||
|         return this.player.getUniqueId(); |         return this.player.getUniqueId(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public long getLastPlayed() { |     @Override | ||||||
|         return this.player.getLastPlayed(); |     @NonNegative | ||||||
|  |     public long getLastPlayed() { | ||||||
|  |         return this.player.getLastSeen(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public boolean isOnline() { |     @Override | ||||||
|         return this.player.isOnline(); |     public String getName() { | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override public String getName() { |  | ||||||
|         return this.player.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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,28 +21,30 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.player; | ||||||
|  |  | ||||||
| import com.google.common.base.Charsets; | import com.google.common.base.Charsets; | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.configuration.Captions; |  | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.events.TeleportCause; | import com.plotsquared.core.events.TeleportCause; | ||||||
| import com.plotsquared.core.location.Location; | import com.plotsquared.core.location.Location; | ||||||
|  | import com.plotsquared.core.permissions.Permission; | ||||||
|  | import com.plotsquared.core.permissions.PermissionHandler; | ||||||
| import com.plotsquared.core.player.ConsolePlayer; | import com.plotsquared.core.player.ConsolePlayer; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.PlotWeather; | import com.plotsquared.core.plot.PlotWeather; | ||||||
| import com.plotsquared.core.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.MathMan; | ||||||
| import com.plotsquared.core.util.StringMan; |  | ||||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||||
| import com.sk89q.worldedit.extension.platform.Actor; | import com.sk89q.worldedit.extension.platform.Actor; | ||||||
| import com.sk89q.worldedit.world.item.ItemType; | import com.sk89q.worldedit.world.item.ItemType; | ||||||
| import com.sk89q.worldedit.world.item.ItemTypes; | import com.sk89q.worldedit.world.item.ItemTypes; | ||||||
| import io.papermc.lib.PaperLib; | import io.papermc.lib.PaperLib; | ||||||
|  | import net.kyori.adventure.audience.Audience; | ||||||
| import org.bukkit.GameMode; | import org.bukkit.GameMode; | ||||||
| import org.bukkit.Sound; | import org.bukkit.Sound; | ||||||
| import org.bukkit.WeatherType; | import org.bukkit.WeatherType; | ||||||
| @@ -52,7 +54,8 @@ import org.bukkit.event.EventException; | |||||||
| import org.bukkit.event.player.PlayerTeleportEvent; | import org.bukkit.event.player.PlayerTeleportEvent; | ||||||
| import org.bukkit.permissions.PermissionAttachmentInfo; | import org.bukkit.permissions.PermissionAttachmentInfo; | ||||||
| import org.bukkit.plugin.RegisteredListener; | 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.Arrays; | ||||||
| import java.util.Set; | import java.util.Set; | ||||||
| @@ -68,63 +71,72 @@ public class BukkitPlayer extends PlotPlayer<Player> { | |||||||
|  |  | ||||||
|     private static boolean CHECK_EFFECTIVE = true; |     private static boolean CHECK_EFFECTIVE = true; | ||||||
|     public final Player player; |     public final Player player; | ||||||
|     private boolean offline; |  | ||||||
|     private String name; |     private String name; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * <p>Please do not use this method. Instead use |      * <p>Please do not use this method. Instead use | ||||||
|      * BukkitUtil.getPlayer(Player), as it caches player objects.</p> |      * 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) { |     public BukkitPlayer( | ||||||
|         this(player, false); |             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) { |     public BukkitPlayer( | ||||||
|         this(player, offline, true); |             final @NonNull PlotAreaManager plotAreaManager, final @NonNull | ||||||
|     } |             EventDispatcher eventDispatcher, final @NonNull Player player, | ||||||
|  |             final boolean realPlayer, | ||||||
|     public BukkitPlayer(@NotNull final Player player, final boolean offline, final boolean realPlayer) { |             final @NonNull PermissionHandler permissionHandler | ||||||
|  |     ) { | ||||||
|  |         super(plotAreaManager, eventDispatcher, permissionHandler); | ||||||
|         this.player = player; |         this.player = player; | ||||||
|         this.offline = offline; |         this.setupPermissionProfile(); | ||||||
|         if (realPlayer) { |         if (realPlayer) { | ||||||
|             super.populatePersistentMetaMap(); |             super.populatePersistentMetaMap(); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public Actor toActor() { |     @Override | ||||||
|  |     public Actor toActor() { | ||||||
|         return BukkitAdapter.adapt(player); |         return BukkitAdapter.adapt(player); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public Player getPlatformPlayer() { |     @Override | ||||||
|  |     public Player getPlatformPlayer() { | ||||||
|         return this.player; |         return this.player; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @NotNull @Override public Location getLocation() { |     @NonNull | ||||||
|         final Location location = super.getLocation(); |     @Override | ||||||
|         return location == null ? BukkitUtil.getLocation(this.player) : location; |     public UUID getUUID() { | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @NotNull @Override public UUID getUUID() { |  | ||||||
|         if (Settings.UUID.OFFLINE) { |         if (Settings.UUID.OFFLINE) { | ||||||
|             if (Settings.UUID.FORCE_LOWERCASE) { |             if (Settings.UUID.FORCE_LOWERCASE) { | ||||||
|                 return UUID.nameUUIDFromBytes(("OfflinePlayer:" + |                 return UUID.nameUUIDFromBytes(("OfflinePlayer:" + | ||||||
|                     getName().toLowerCase()).getBytes(Charsets.UTF_8)); |                         getName().toLowerCase()).getBytes(Charsets.UTF_8)); | ||||||
|             } else { |             } else { | ||||||
|                 return UUID.nameUUIDFromBytes(("OfflinePlayer:" + |                 return UUID.nameUUIDFromBytes(("OfflinePlayer:" + | ||||||
|                     getName()).getBytes(Charsets.UTF_8)); |                         getName()).getBytes(Charsets.UTF_8)); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         return player.getUniqueId(); |         return player.getUniqueId(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public long getLastPlayed() { |     @Override | ||||||
|         return this.player.getLastPlayed(); |     @NonNegative | ||||||
|  |     public long getLastPlayed() { | ||||||
|  |         return this.player.getLastSeen(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public boolean canTeleport(@NotNull final Location location) { |     @Override | ||||||
|         final org.bukkit.Location to = BukkitUtil.getLocation(location); |     public boolean canTeleport(final @NonNull Location location) { | ||||||
|  |         final org.bukkit.Location to = BukkitUtil.adapt(location); | ||||||
|         final org.bukkit.Location from = player.getLocation(); |         final org.bukkit.Location from = player.getLocation(); | ||||||
|         PlayerTeleportEvent event = new PlayerTeleportEvent(player, from, to); |         PlayerTeleportEvent event = new PlayerTeleportEvent(player, from, to); | ||||||
|         callEvent(event); |         callEvent(event); | ||||||
| @@ -136,15 +148,10 @@ public class BukkitPlayer extends PlotPlayer<Player> { | |||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     private void callEvent(final @NonNull Event event) { | ||||||
|     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) { |  | ||||||
|         final RegisteredListener[] listeners = event.getHandlers().getRegisteredListeners(); |         final RegisteredListener[] listeners = event.getHandlers().getRegisteredListeners(); | ||||||
|         for (final RegisteredListener listener : listeners) { |         for (final RegisteredListener listener : listeners) { | ||||||
|             if (listener.getPlugin().getName().equals(PlotSquared.imp().getPluginName())) { |             if (listener.getPlugin().getName().equals(PlotSquared.platform().pluginName())) { | ||||||
|                 continue; |                 continue; | ||||||
|             } |             } | ||||||
|             try { |             try { | ||||||
| @@ -155,23 +162,22 @@ public class BukkitPlayer extends PlotPlayer<Player> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public boolean hasPermission(final String permission) { |     @SuppressWarnings("StringSplitter") | ||||||
|         if (this.offline && EconHandler.getEconHandler() != null) { |     @Override | ||||||
|             return EconHandler.getEconHandler().hasPermission(getName(), permission); |     @NonNegative | ||||||
|         } |     public int hasPermissionRange( | ||||||
|         return this.player.hasPermission(permission); |             final @NonNull String stub, | ||||||
|     } |             @NonNegative final int range | ||||||
|  |     ) { | ||||||
|     @Override public int hasPermissionRange(final String stub, final int range) { |         if (hasPermission(Permission.PERMISSION_ADMIN.toString())) { | ||||||
|         if (hasPermission(Captions.PERMISSION_ADMIN.getTranslated())) { |  | ||||||
|             return Integer.MAX_VALUE; |             return Integer.MAX_VALUE; | ||||||
|         } |         } | ||||||
|         final String[] nodes = stub.split("\\."); |         final String[] nodes = stub.split("\\."); | ||||||
|         final StringBuilder n = new StringBuilder(); |         final StringBuilder n = new StringBuilder(); | ||||||
|         for (int i = 0; i < (nodes.length - 1); i++) { |         for (int i = 0; i < (nodes.length - 1); i++) { | ||||||
|             n.append(nodes[i]).append("."); |             n.append(nodes[i]).append("."); | ||||||
|             if (!stub.equals(n + Captions.PERMISSION_STAR.getTranslated())) { |             if (!stub.equals(n + Permission.PERMISSION_STAR.toString())) { | ||||||
|                 if (hasPermission(n + Captions.PERMISSION_STAR.getTranslated())) { |                 if (hasPermission(n + Permission.PERMISSION_STAR.toString())) { | ||||||
|                     return Integer.MAX_VALUE; |                     return Integer.MAX_VALUE; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -222,85 +228,63 @@ public class BukkitPlayer extends PlotPlayer<Player> { | |||||||
|         return max; |         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 |     @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) { |         if (Math.abs(location.getX()) >= 30000000 || Math.abs(location.getZ()) >= 30000000) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         final org.bukkit.Location bukkitLocation = |         final org.bukkit.Location bukkitLocation = | ||||||
|             new org.bukkit.Location(BukkitUtil.getWorld(location.getWorld()), location.getX() + 0.5, |                 new org.bukkit.Location(BukkitUtil.getWorld(location.getWorldName()), location.getX() + 0.5, | ||||||
|                 location.getY(), location.getZ() + 0.5, location.getYaw(), location.getPitch()); |                         location.getY(), location.getZ() + 0.5, location.getYaw(), location.getPitch() | ||||||
|  |                 ); | ||||||
|         PaperLib.teleportAsync(player, bukkitLocation, getTeleportCause(cause)); |         PaperLib.teleportAsync(player, bukkitLocation, getTeleportCause(cause)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public String getName() { |     @Override | ||||||
|  |     public String getName() { | ||||||
|         if (this.name == null) { |         if (this.name == null) { | ||||||
|             this.name = this.player.getName(); |             this.name = this.player.getName(); | ||||||
|         } |         } | ||||||
|         return this.name; |         return this.name; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public boolean isOnline() { |     @Override | ||||||
|         return !this.offline && this.player.isOnline(); |     public void setCompassTarget(Location location) { | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override public void setCompassTarget(Location location) { |  | ||||||
|         this.player.setCompassTarget( |         this.player.setCompassTarget( | ||||||
|             new org.bukkit.Location(BukkitUtil.getWorld(location.getWorld()), location.getX(), |                 new org.bukkit.Location(BukkitUtil.getWorld(location.getWorldName()), location.getX(), | ||||||
|                 location.getY(), location.getZ())); |                         location.getY(), location.getZ() | ||||||
|  |                 )); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public Location getLocationFull() { |     @Override | ||||||
|         return BukkitUtil.getLocationFull(this.player); |     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) { |         switch (weather) { | ||||||
|             case CLEAR: |             case CLEAR -> this.player.setPlayerWeather(WeatherType.CLEAR); | ||||||
|                 this.player.setPlayerWeather(WeatherType.CLEAR); |             case RAIN -> this.player.setPlayerWeather(WeatherType.DOWNFALL); | ||||||
|                 break; |             case WORLD -> this.player.resetPlayerWeather(); | ||||||
|             case RAIN: |             default -> { | ||||||
|                 this.player.setPlayerWeather(WeatherType.DOWNFALL); |  | ||||||
|                 break; |  | ||||||
|             case WORLD: |  | ||||||
|                 this.player.resetPlayerWeather(); |  | ||||||
|                 break; |  | ||||||
|             default: |  | ||||||
|                 //do nothing as this is PlotWeather.OFF |                 //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 |     @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)) { |         if (ADVENTURE.equals(gameMode)) { | ||||||
|             this.player.setGameMode(GameMode.ADVENTURE); |             this.player.setGameMode(GameMode.ADVENTURE); | ||||||
|         } else if (CREATIVE.equals(gameMode)) { |         } 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) { |         if (time != Long.MAX_VALUE) { | ||||||
|             this.player.setPlayerTime(time, false); |             this.player.setPlayerTime(time, false); | ||||||
|         } else { |         } else { | ||||||
| @@ -320,44 +305,58 @@ public class BukkitPlayer extends PlotPlayer<Player> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public boolean getFlight() { |     @Override | ||||||
|  |     public boolean getFlight() { | ||||||
|         return player.getAllowFlight(); |         return player.getAllowFlight(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public void setFlight(boolean fly) { |     @Override | ||||||
|  |     public void setFlight(boolean fly) { | ||||||
|         this.player.setAllowFlight(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) { |         if (id == ItemTypes.AIR) { | ||||||
|             // Let's just stop all the discs because why not? |             // Let's just stop all the discs because why not? | ||||||
|             for (final Sound sound : Arrays.stream(Sound.values()) |             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); |                 player.stopSound(sound); | ||||||
|             } |             } | ||||||
|             // this.player.playEffect(BukkitUtil.getLocation(location), Effect.RECORD_PLAY, Material.AIR); |             // this.player.playEffect(BukkitUtil.getLocation(location), Effect.RECORD_PLAY, Material.AIR); | ||||||
|         } else { |         } else { | ||||||
|             // this.player.playEffect(BukkitUtil.getLocation(location), Effect.RECORD_PLAY, id.to(Material.class)); |             // this.player.playEffect(BukkitUtil.getLocation(location), Effect.RECORD_PLAY, id.to(Material.class)); | ||||||
|             this.player.playSound(BukkitUtil.getLocation(location), |             this.player.playSound(BukkitUtil.adapt(location), | ||||||
|                 Sound.valueOf(BukkitAdapter.adapt(id).name()), Float.MAX_VALUE, 1f); |                     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); |         this.player.kickPlayer(message); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public void stopSpectating() { |     @Override | ||||||
|  |     public void stopSpectating() { | ||||||
|         if (getGameMode() == SPECTATOR) { |         if (getGameMode() == SPECTATOR) { | ||||||
|             this.player.setSpectatorTarget(null); |             this.player.setSpectatorTarget(null); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public boolean isBanned() { |     @Override | ||||||
|  |     public boolean isBanned() { | ||||||
|         return this.player.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) { |         if (other instanceof ConsolePlayer) { | ||||||
|             return true; |             return true; | ||||||
|         } else { |         } else { | ||||||
| @@ -365,14 +364,19 @@ public class BukkitPlayer extends PlotPlayer<Player> { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public PlayerTeleportEvent.TeleportCause getTeleportCause(@NotNull final TeleportCause cause) { |     /** | ||||||
|         switch (cause) { |      * Convert from PlotSquared's {@link TeleportCause} to Bukkit's {@link PlayerTeleportEvent.TeleportCause} | ||||||
|             case COMMAND: |      * | ||||||
|                 return PlayerTeleportEvent.TeleportCause.COMMAND; |      * @param cause PlotSquared teleport cause to convert | ||||||
|             case PLUGIN: |      * @return Bukkit's equivalent teleport cause | ||||||
|                 return PlayerTeleportEvent.TeleportCause.PLUGIN; |      */ | ||||||
|             default: |     public PlayerTeleportEvent.TeleportCause getTeleportCause(final @NonNull TeleportCause cause) { | ||||||
|                 return PlayerTeleportEvent.TeleportCause.UNKNOWN; |         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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,48 +21,78 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | 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 com.plotsquared.core.util.PlayerManager; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
| import org.bukkit.entity.Player; | import org.bukkit.entity.Player; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
| import org.jetbrains.annotations.Nullable; | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  |  | ||||||
| import java.util.UUID; | import java.util.UUID; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Player manager providing {@link BukkitPlayer Bukkit players} |  * Player manager providing {@link BukkitPlayer Bukkit players} | ||||||
|  */ |  */ | ||||||
|  | @Singleton | ||||||
| public class BukkitPlayerManager extends PlayerManager<BukkitPlayer, Player> { | public class BukkitPlayerManager extends PlayerManager<BukkitPlayer, Player> { | ||||||
|  |  | ||||||
|     @NotNull @Override public BukkitPlayer getPlayer(@NotNull final Player object) { |     private final PlotAreaManager plotAreaManager; | ||||||
|         try { |     private final EventDispatcher eventDispatcher; | ||||||
|             return getPlayer(object.getUniqueId()); |     private final PermissionHandler permissionHandler; | ||||||
|         } catch (final NoSuchPlayerException exception) { |  | ||||||
|             return new BukkitPlayer(object, object.isOnline(), false); |     @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); |         final Player player = Bukkit.getPlayer(uuid); | ||||||
|         if (player == null || !player.isOnline()) { |         if (player == null || !player.isOnline()) { | ||||||
|             throw new NoSuchPlayerException(uuid); |             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) { |         if (uuid == null) { | ||||||
|             return 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) { |     @NonNull | ||||||
|         return new BukkitOfflinePlayer(Bukkit.getOfflinePlayer(username)); |     @Override | ||||||
|  |     public BukkitOfflinePlayer getOfflinePlayer(final @NonNull String username) { | ||||||
|  |         return new BukkitOfflinePlayer(Bukkit.getOfflinePlayer(username), this.permissionHandler); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,288 @@ | |||||||
|  | /* | ||||||
|  |  *       _____  _       _    _____                                _ | ||||||
|  |  *      |  __ \| |     | |  / ____|                              | | | ||||||
|  |  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||||
|  |  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||||
|  |  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||||
|  |  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||||
|  |  *                                    | | | ||||||
|  |  *                                    |_| | ||||||
|  |  *            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 final AtomicInteger loadingChunks = new AtomicInteger(); | ||||||
|  |     private final boolean forceSync; | ||||||
|  |  | ||||||
|  |     private int batchSize; | ||||||
|  |     private PlotSquaredTask task; | ||||||
|  |     private volatile 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, | ||||||
|  |             @Assisted final boolean forceSync | ||||||
|  |     ) { | ||||||
|  |         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); | ||||||
|  |         this.forceSync = forceSync; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void start() { | ||||||
|  |         if (!forceSync) { | ||||||
|  |             // 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)); | ||||||
|  |         } else { | ||||||
|  |             try { | ||||||
|  |                 while (!shouldCancel && !requestedChunks.isEmpty()) { | ||||||
|  |                     chunkConsumer.accept(requestedChunks.poll()); | ||||||
|  |                 } | ||||||
|  |             } catch (Throwable t) { | ||||||
|  |                 throwableConsumer.accept(t); | ||||||
|  |             } finally { | ||||||
|  |                 finish(); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @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(); | ||||||
|  |             } | ||||||
|  |             if (task != null) { | ||||||
|  |                 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) { | ||||||
|  |             if (this.availableChunks.isEmpty()) { | ||||||
|  |                 if (this.requestedChunks.isEmpty() && loadingChunks.get() == 0) { | ||||||
|  |                     finish(); | ||||||
|  |                 } else { | ||||||
|  |                     requestBatch(); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             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 | ||||||
|  |             loadingChunks.incrementAndGet(); | ||||||
|  |             PaperLib | ||||||
|  |                     .getChunkAtAsync(this.bukkitWorld, chunk.getX(), chunk.getZ(), true, true) | ||||||
|  |                     .whenComplete((chunkObject, throwable) -> { | ||||||
|  |                         loadingChunks.decrementAndGet(); | ||||||
|  |                         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,408 @@ | |||||||
|  | /* | ||||||
|  |  *       _____  _       _    _____                                _ | ||||||
|  |  *      |  __ \| |     | |  / ____|                              | | | ||||||
|  |  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||||
|  |  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||||
|  |  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||||
|  |  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||||
|  |  *                                    | | | ||||||
|  |  *                                    |_| | ||||||
|  |  *            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.Chunk; | ||||||
|  | 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 static final SideEffectSet NO_SIDE_EFFECT_SET; | ||||||
|  |     private static final SideEffectSet EDGE_SIDE_EFFECT_SET; | ||||||
|  |     private static final SideEffectSet LIGHTING_SIDE_EFFECT_SET; | ||||||
|  |     private static final SideEffectSet EDGE_LIGHTING_SIDE_EFFECT_SET; | ||||||
|  |  | ||||||
|  |     static { | ||||||
|  |         NO_SIDE_EFFECT_SET = SideEffectSet.none().with(SideEffect.LIGHTING, SideEffect.State.OFF).with( | ||||||
|  |                 SideEffect.NEIGHBORS, | ||||||
|  |                 SideEffect.State.OFF | ||||||
|  |         ); | ||||||
|  |         EDGE_SIDE_EFFECT_SET = SideEffectSet.none().with(SideEffect.UPDATE, SideEffect.State.ON).with( | ||||||
|  |                 SideEffect.NEIGHBORS, | ||||||
|  |                 SideEffect.State.ON | ||||||
|  |         ); | ||||||
|  |         LIGHTING_SIDE_EFFECT_SET = SideEffectSet.none().with(SideEffect.NEIGHBORS, SideEffect.State.OFF); | ||||||
|  |         EDGE_LIGHTING_SIDE_EFFECT_SET = SideEffectSet.none().with(SideEffect.UPDATE, SideEffect.State.ON).with( | ||||||
|  |                 SideEffect.NEIGHBORS, | ||||||
|  |                 SideEffect.State.ON | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private org.bukkit.World bukkitWorld; | ||||||
|  |     @Inject | ||||||
|  |     private ChunkCoordinatorBuilderFactory chunkCoordinatorBuilderFactory; | ||||||
|  |     @Inject | ||||||
|  |     private ChunkCoordinatorFactory chunkCoordinatorFactory; | ||||||
|  |     private ChunkCoordinator chunkCoordinator; | ||||||
|  |  | ||||||
|  |     @Inject | ||||||
|  |     public BukkitQueueCoordinator(@NonNull World world) { | ||||||
|  |         super(world); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @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, getMinY(), getRegenStart()[1] << 4); | ||||||
|  |             BlockVector3 end = BlockVector3.at((getRegenEnd()[0] << 4) + 15, getMaxY(), (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 = getMinLayer(); layer <= getMaxLayer(); 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 + localChunk.getMinSection(), 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.getBiomes().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, getSideEffectSet(SideEffectState.NONE)); | ||||||
|  |                         } 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 = getSideEffectSet(edge ? SideEffectState.EDGE_LIGHTING : SideEffectState.LIGHTING); | ||||||
|  |             } else { | ||||||
|  |                 sideEffectSet = getSideEffectSet(edge ? SideEffectState.EDGE : SideEffectState.NONE); | ||||||
|  |             } | ||||||
|  |             getWorld().setBlock(loc, block, sideEffectSet); | ||||||
|  |         } catch (WorldEditException ignored) { | ||||||
|  |             // Fallback to not so nice method | ||||||
|  |             BlockData blockData = BukkitAdapter.adapt(block); | ||||||
|  |             Block existing; | ||||||
|  |             // Assume a chunk object has been given only when it should have been. | ||||||
|  |             if (getChunkObject() instanceof Chunk chunkObject) { | ||||||
|  |                 existing = chunkObject.getBlock(x & 15, y, z & 15); | ||||||
|  |             } else { | ||||||
|  |                  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(existing); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     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) { | ||||||
|  |         int layerIndex = (layer - localChunk.getMinSection()); | ||||||
|  |         if (layer == localChunk.getMinSection() || layerIndex == localChunk.getBaseblocks().length - 1) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         if (x == 0) { | ||||||
|  |             LocalChunk localChunkX = getBlockChunks().get(blockVector2.withX(blockVector2.getX() - 1)); | ||||||
|  |             if (localChunkX == null || localChunkX.getBaseblocks()[layerIndex] == null || | ||||||
|  |                     localChunkX.getBaseblocks()[layerIndex][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()[layerIndex] == null || | ||||||
|  |                     localChunkX.getBaseblocks()[layerIndex][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()[layerIndex] == null || | ||||||
|  |                     localChunkZ.getBaseblocks()[layerIndex][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()[layerIndex] == null || | ||||||
|  |                     localChunkZ.getBaseblocks()[layerIndex][ChunkUtil.getJ(x, y, 0)] != null) { | ||||||
|  |                 return true; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         if (y == 0) { | ||||||
|  |             if (localChunk.getBaseblocks()[layerIndex - 1] == null || | ||||||
|  |                     localChunk.getBaseblocks()[layerIndex][ChunkUtil.getJ(x, 15, z)] != null) { | ||||||
|  |                 return true; | ||||||
|  |             } | ||||||
|  |         } else if (y == 15) { | ||||||
|  |             if (localChunk.getBaseblocks()[layerIndex + 1] == null || | ||||||
|  |                     localChunk.getBaseblocks()[layerIndex][ChunkUtil.getJ(x, 0, z)] != null) { | ||||||
|  |                 return true; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         BaseBlock[] baseBlocks = localChunk.getBaseblocks()[layerIndex]; | ||||||
|  |         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; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private SideEffectSet getSideEffectSet(SideEffectState state) { | ||||||
|  |         if (getSideEffectSet() != null) { | ||||||
|  |             return getSideEffectSet(); | ||||||
|  |         } | ||||||
|  |         return switch (state) { | ||||||
|  |             case NONE -> NO_SIDE_EFFECT_SET; | ||||||
|  |             case EDGE -> EDGE_SIDE_EFFECT_SET; | ||||||
|  |             case LIGHTING -> LIGHTING_SIDE_EFFECT_SET; | ||||||
|  |             case EDGE_LIGHTING -> EDGE_LIGHTING_SIDE_EFFECT_SET; | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private enum SideEffectState { | ||||||
|  |         NONE, | ||||||
|  |         EDGE, | ||||||
|  |         LIGHTING, | ||||||
|  |         EDGE_LIGHTING | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.queue; | ||||||
|  |  | ||||||
| @@ -30,8 +30,9 @@ import com.plotsquared.bukkit.util.BukkitBlockUtil; | |||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
| import com.plotsquared.core.location.ChunkWrapper; | import com.plotsquared.core.location.ChunkWrapper; | ||||||
| import com.plotsquared.core.location.Location; | import com.plotsquared.core.location.Location; | ||||||
| import com.plotsquared.core.queue.ScopedLocalBlockQueue; | import com.plotsquared.core.queue.ScopedQueueCoordinator; | ||||||
| import com.plotsquared.core.util.MainUtil; | import com.plotsquared.core.util.AnnotationHelper; | ||||||
|  | import com.plotsquared.core.util.ChunkUtil; | ||||||
| import com.plotsquared.core.util.PatternUtil; | import com.plotsquared.core.util.PatternUtil; | ||||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||||
| import com.sk89q.worldedit.function.pattern.Pattern; | import com.sk89q.worldedit.function.pattern.Pattern; | ||||||
| @@ -39,18 +40,19 @@ import com.sk89q.worldedit.world.biome.BiomeType; | |||||||
| import com.sk89q.worldedit.world.block.BaseBlock; | import com.sk89q.worldedit.world.block.BaseBlock; | ||||||
| import com.sk89q.worldedit.world.block.BlockState; | import com.sk89q.worldedit.world.block.BlockState; | ||||||
| import com.sk89q.worldedit.world.block.BlockTypes; | import com.sk89q.worldedit.world.block.BlockTypes; | ||||||
| import lombok.Getter; | import org.bukkit.Bukkit; | ||||||
| import lombok.Setter; |  | ||||||
| import org.bukkit.Chunk; | import org.bukkit.Chunk; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.block.Biome; | import org.bukkit.block.Biome; | ||||||
| import org.bukkit.generator.ChunkGenerator.BiomeGrid; | import org.bukkit.generator.ChunkGenerator.BiomeGrid; | ||||||
| import org.bukkit.generator.ChunkGenerator.ChunkData; | 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; | import java.util.Arrays; | ||||||
|  |  | ||||||
| public class GenChunk extends ScopedLocalBlockQueue { | @AnnotationHelper.ApiDescription(info = "Internal use only. Subject to changes at any time.") | ||||||
|  | public class GenChunk extends ScopedQueueCoordinator { | ||||||
|  |  | ||||||
|     public final Biome[] biomes; |     public final Biome[] biomes; | ||||||
|     public BlockState[][] result; |     public BlockState[][] result; | ||||||
| @@ -59,14 +61,33 @@ public class GenChunk extends ScopedLocalBlockQueue { | |||||||
|     public String world; |     public String world; | ||||||
|     public int chunkX; |     public int chunkX; | ||||||
|     public int chunkZ; |     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)); |      * @param minY minimum world Y, inclusive | ||||||
|  |      * @param maxY maximum world Y, inclusive | ||||||
|  |      * | ||||||
|  |      * @since 6.6.0 | ||||||
|  |      */ | ||||||
|  |     public GenChunk(int minY, int maxY) { | ||||||
|  |         super(null, Location.at("", 0, minY, 0), Location.at("", 15, maxY, 15)); | ||||||
|         this.biomes = Biome.values(); |         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) { |         if (chunk == null) { | ||||||
|             World worldObj = BukkitUtil.getWorld(world); |             World worldObj = BukkitUtil.getWorld(world); | ||||||
|             if (worldObj != null) { |             if (worldObj != null) { | ||||||
| @@ -76,34 +97,48 @@ public class GenChunk extends ScopedLocalBlockQueue { | |||||||
|         return chunk; |         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; |         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; |         chunk = null; | ||||||
|         world = wrap.world; |         world = wrap.world; | ||||||
|         chunkX = wrap.x; |         chunkX = wrap.x; | ||||||
|         chunkZ = wrap.z; |         chunkZ = wrap.z; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public void fillBiome(BiomeType biomeType) { |     @Override | ||||||
|  |     public void fillBiome(@NonNull BiomeType biomeType) { | ||||||
|         if (biomeGrid == null) { |         if (biomeGrid == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Biome biome = BukkitAdapter.adapt(biomeType); |         Biome biome = BukkitAdapter.adapt(biomeType); | ||||||
|         for (int x = 0; x < 16; x++) { |         for (int y = getMin().getY(); y <= getMax().getY(); y++) { | ||||||
|             for (int z = 0; z < 16; z++) { |             for (int x = 0; x < 16; x++) { | ||||||
|                 this.biomeGrid.setBiome(x, z, biome); |                 for (int z = 0; z < 16; z++) { | ||||||
|  |                     this.biomeGrid.setBiome(x, y, z, biome); | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public void setCuboid(Location pos1, Location pos2, BlockState block) { |     @Override | ||||||
|         if (result != null && pos1.getX() == 0 && pos1.getZ() == 0 && pos2.getX() == 15 |     public void setCuboid(@NonNull Location pos1, @NonNull Location pos2, @NonNull BlockState block) { | ||||||
|             && pos2.getZ() == 15) { |         if (result != null && pos1.getX() == 0 && pos1.getZ() == 0 && pos2.getX() == 15 && pos2.getZ() == 15) { | ||||||
|             for (int y = pos1.getY(); y <= pos2.getY(); y++) { |             for (int y = pos1.getY(); y <= pos2.getY(); y++) { | ||||||
|                 int layer = y >> 4; |                 int layer = getLayerIndex(y); | ||||||
|                 BlockState[] data = result[layer]; |                 BlockState[] data = result[layer]; | ||||||
|                 if (data == null) { |                 if (data == null) { | ||||||
|                     result[layer] = data = new BlockState[4096]; |                     result[layer] = data = new BlockState[4096]; | ||||||
| @@ -119,28 +154,47 @@ public class GenChunk extends ScopedLocalBlockQueue { | |||||||
|         int maxX = Math.max(pos1.getX(), pos2.getX()); |         int maxX = Math.max(pos1.getX(), pos2.getX()); | ||||||
|         int maxY = Math.max(pos1.getY(), pos2.getY()); |         int maxY = Math.max(pos1.getY(), pos2.getY()); | ||||||
|         int maxZ = Math.max(pos1.getZ(), pos2.getZ()); |         int maxZ = Math.max(pos1.getZ(), pos2.getZ()); | ||||||
|         chunkData |         chunkData.setRegion(minX, minY, minZ, maxX + 1, maxY + 1, maxZ + 1, BukkitAdapter.adapt(block)); | ||||||
|             .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)); |         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) { |         if (this.biomeGrid != null) { | ||||||
|             this.biomeGrid.setBiome(x, z, biome); |             for (int y = getMin().getY(); y <= getMax().getY(); y++) { | ||||||
|  |                 this.setBiome(x, y, z, biome); | ||||||
|  |             } | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public boolean setBlock(int x, int y, int z, @NotNull Pattern pattern) { |     public boolean setBiome(int x, int y, int z, @NonNull Biome biome) { | ||||||
|         return setBlock(x, y, z, PatternUtil |         if (this.biomeGrid != null) { | ||||||
|             .apply(Preconditions.checkNotNull(pattern, "Pattern may not be null"), x, y, z)); |             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) { |         if (this.result == null) { | ||||||
|             this.chunkData.setBlock(x, y, z, BukkitAdapter.adapt(id)); |             this.chunkData.setBlock(x, y, z, BukkitAdapter.adapt(id)); | ||||||
|             return true; |             return true; | ||||||
| @@ -150,17 +204,18 @@ public class GenChunk extends ScopedLocalBlockQueue { | |||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void storeCache(final int x, final int y, final int z, final BlockState id) { |     private void storeCache(final int x, final int y, final int z, final @NonNull BlockState id) { | ||||||
|         int i = MainUtil.CACHE_I[y][x][z]; |         int i = getLayerIndex(y); | ||||||
|         BlockState[] v = this.result[i]; |         BlockState[] v = this.result[i]; | ||||||
|         if (v == null) { |         if (v == null) { | ||||||
|             this.result[i] = v = new BlockState[4096]; |             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; |         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) { |         if (this.result == null) { | ||||||
|             this.chunkData.setBlock(x, y, z, BukkitAdapter.adapt(id)); |             this.chunkData.setBlock(x, y, z, BukkitAdapter.adapt(id)); | ||||||
|             return true; |             return true; | ||||||
| @@ -170,8 +225,9 @@ public class GenChunk extends ScopedLocalBlockQueue { | |||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public BlockState getBlock(int x, int y, int z) { |     @Override | ||||||
|         int i = MainUtil.CACHE_I[y][x][z]; |     public @Nullable BlockState getBlock(int x, int y, int z) { | ||||||
|  |         int i = getLayerIndex(y); | ||||||
|         if (result == null) { |         if (result == null) { | ||||||
|             return BukkitBlockUtil.get(chunkData.getType(x, y, z)); |             return BukkitBlockUtil.get(chunkData.getType(x, y, z)); | ||||||
|         } |         } | ||||||
| @@ -179,7 +235,7 @@ public class GenChunk extends ScopedLocalBlockQueue { | |||||||
|         if (array == null) { |         if (array == null) { | ||||||
|             return BlockTypes.AIR.getDefaultState(); |             return BlockTypes.AIR.getDefaultState(); | ||||||
|         } |         } | ||||||
|         int j = MainUtil.CACHE_J[y][x][z]; |         int j = ChunkUtil.getJ(x, y, z); | ||||||
|         return array[j]; |         return array[j]; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -191,20 +247,23 @@ public class GenChunk extends ScopedLocalBlockQueue { | |||||||
|         return chunk == null ? chunkZ : chunk.getZ(); |         return chunk == null ? chunkZ : chunk.getZ(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public String getWorld() { |     @Override | ||||||
|         return chunk == null ? world : chunk.getWorld().getName(); |     public com.sk89q.worldedit.world.@NonNull World getWorld() { | ||||||
|  |         return chunk == null ? BukkitAdapter.adapt(Bukkit.getWorld(world)) : BukkitAdapter.adapt(chunk.getWorld()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public Location getMax() { |     @Override | ||||||
|         return new Location(getWorld(), 15 + (getX() << 4), 255, 15 + (getZ() << 4)); |     public @NonNull Location getMax() { | ||||||
|  |         return Location.at(getWorld().getName(), 15 + (getX() << 4), super.getMax().getY(), 15 + (getZ() << 4)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public Location getMin() { |     @Override | ||||||
|         return new Location(getWorld(), getX() << 4, 0, getZ() << 4); |     public @NonNull Location getMin() { | ||||||
|  |         return Location.at(getWorld().getName(), getX() << 4, super.getMin().getY(), getZ() << 4); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public GenChunk clone() { |     public @NonNull GenChunk clone() { | ||||||
|         GenChunk toReturn = new GenChunk(); |         GenChunk toReturn = new GenChunk(getMin().getY(), getMax().getY()); | ||||||
|         if (this.result != null) { |         if (this.result != null) { | ||||||
|             for (int i = 0; i < this.result.length; i++) { |             for (int i = 0; i < this.result.length; i++) { | ||||||
|                 BlockState[] matrix = this.result[i]; |                 BlockState[] matrix = this.result[i]; | ||||||
| @@ -217,4 +276,9 @@ public class GenChunk extends ScopedLocalBlockQueue { | |||||||
|         toReturn.chunkData = this.chunkData; |         toReturn.chunkData = this.chunkData; | ||||||
|         return toReturn; |         return toReturn; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private int getLayerIndex(int y) { | ||||||
|  |         return (y - getMin().getY()) >> 4; | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,21 +21,33 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | 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.SchematicHandler; | ||||||
|  | import com.plotsquared.core.util.WorldUtil; | ||||||
| import com.sk89q.jnbt.CompoundTag; | import com.sk89q.jnbt.CompoundTag; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Schematic Handler. |  * Schematic Handler. | ||||||
|  */ |  */ | ||||||
|  | @Singleton | ||||||
| public class BukkitSchematicHandler extends SchematicHandler { | public class BukkitSchematicHandler extends SchematicHandler { | ||||||
|  |  | ||||||
|     @Override |     @Inject | ||||||
|     public boolean restoreTile(LocalBlockQueue queue, CompoundTag ct, int x, int y, int z) { |     public BukkitSchematicHandler(final @NonNull WorldUtil worldUtil, @NonNull ProgressSubscriberFactory subscriberFactory) { | ||||||
|         return new StateWrapper(ct).restoreTag(queue.getWorld(), x, y, z); |         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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,12 +21,11 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.schematic; | ||||||
|  |  | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
| import com.plotsquared.core.configuration.Captions; |  | ||||||
| import com.sk89q.jnbt.ByteTag; | import com.sk89q.jnbt.ByteTag; | ||||||
| import com.sk89q.jnbt.CompoundTag; | import com.sk89q.jnbt.CompoundTag; | ||||||
| import com.sk89q.jnbt.ListTag; | import com.sk89q.jnbt.ListTag; | ||||||
| @@ -36,6 +35,7 @@ import com.sk89q.jnbt.Tag; | |||||||
| import com.sk89q.worldedit.blocks.BaseItemStack; | import com.sk89q.worldedit.blocks.BaseItemStack; | ||||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||||
| import com.sk89q.worldedit.world.item.ItemType; | import com.sk89q.worldedit.world.item.ItemType; | ||||||
|  | import org.bukkit.ChatColor; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.block.Block; | import org.bukkit.block.Block; | ||||||
| import org.bukkit.block.Container; | import org.bukkit.block.Container; | ||||||
| @@ -44,6 +44,7 @@ import org.bukkit.enchantments.Enchantment; | |||||||
| import org.bukkit.inventory.Inventory; | import org.bukkit.inventory.Inventory; | ||||||
| import org.bukkit.inventory.InventoryHolder; | import org.bukkit.inventory.InventoryHolder; | ||||||
| import org.bukkit.inventory.ItemStack; | import org.bukkit.inventory.ItemStack; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| @@ -66,132 +67,141 @@ public class StateWrapper { | |||||||
|  |  | ||||||
|     public static String jsonToColourCode(String str) { |     public static String jsonToColourCode(String str) { | ||||||
|         str = str.replace("{\"extra\":", "").replace("],\"text\":\"\"}", "]") |         str = str.replace("{\"extra\":", "").replace("],\"text\":\"\"}", "]") | ||||||
|             .replace("[{\"color\":\"black\",\"text\":\"", "&0") |                 .replace("[{\"color\":\"black\",\"text\":\"", "&0") | ||||||
|             .replace("[{\"color\":\"dark_blue\",\"text\":\"", "&1") |                 .replace("[{\"color\":\"dark_blue\",\"text\":\"", "&1") | ||||||
|             .replace("[{\"color\":\"dark_green\",\"text\":\"", "&2") |                 .replace("[{\"color\":\"dark_green\",\"text\":\"", "&2") | ||||||
|             .replace("[{\"color\":\"dark_aqua\",\"text\":\"", "&3") |                 .replace("[{\"color\":\"dark_aqua\",\"text\":\"", "&3") | ||||||
|             .replace("[{\"color\":\"dark_red\",\"text\":\"", "&4") |                 .replace("[{\"color\":\"dark_red\",\"text\":\"", "&4") | ||||||
|             .replace("[{\"color\":\"dark_purple\",\"text\":\"", "&5") |                 .replace("[{\"color\":\"dark_purple\",\"text\":\"", "&5") | ||||||
|             .replace("[{\"color\":\"gold\",\"text\":\"", "&6") |                 .replace("[{\"color\":\"gold\",\"text\":\"", "&6") | ||||||
|             .replace("[{\"color\":\"gray\",\"text\":\"", "&7") |                 .replace("[{\"color\":\"gray\",\"text\":\"", "&7") | ||||||
|             .replace("[{\"color\":\"dark_gray\",\"text\":\"", "&8") |                 .replace("[{\"color\":\"dark_gray\",\"text\":\"", "&8") | ||||||
|             .replace("[{\"color\":\"blue\",\"text\":\"", "&9") |                 .replace("[{\"color\":\"blue\",\"text\":\"", "&9") | ||||||
|             .replace("[{\"color\":\"green\",\"text\":\"", "&a") |                 .replace("[{\"color\":\"green\",\"text\":\"", "&a") | ||||||
|             .replace("[{\"color\":\"aqua\",\"text\":\"", "&b") |                 .replace("[{\"color\":\"aqua\",\"text\":\"", "&b") | ||||||
|             .replace("[{\"color\":\"red\",\"text\":\"", "&c") |                 .replace("[{\"color\":\"red\",\"text\":\"", "&c") | ||||||
|             .replace("[{\"color\":\"light_purple\",\"text\":\"", "&d") |                 .replace("[{\"color\":\"light_purple\",\"text\":\"", "&d") | ||||||
|             .replace("[{\"color\":\"yellow\",\"text\":\"", "&e") |                 .replace("[{\"color\":\"yellow\",\"text\":\"", "&e") | ||||||
|             .replace("[{\"color\":\"white\",\"text\":\"", "&f") |                 .replace("[{\"color\":\"white\",\"text\":\"", "&f") | ||||||
|             .replace("[{\"obfuscated\":true,\"text\":\"", "&k") |                 .replace("[{\"obfuscated\":true,\"text\":\"", "&k") | ||||||
|             .replace("[{\"bold\":true,\"text\":\"", "&l") |                 .replace("[{\"bold\":true,\"text\":\"", "&l") | ||||||
|             .replace("[{\"strikethrough\":true,\"text\":\"", "&m") |                 .replace("[{\"strikethrough\":true,\"text\":\"", "&m") | ||||||
|             .replace("[{\"underlined\":true,\"text\":\"", "&n") |                 .replace("[{\"underlined\":true,\"text\":\"", "&n") | ||||||
|             .replace("[{\"italic\":true,\"text\":\"", "&o").replace("[{\"color\":\"black\",", "&0") |                 .replace("[{\"italic\":true,\"text\":\"", "&o").replace("[{\"color\":\"black\",", "&0") | ||||||
|             .replace("[{\"color\":\"dark_blue\",", "&1") |                 .replace("[{\"color\":\"dark_blue\",", "&1") | ||||||
|             .replace("[{\"color\":\"dark_green\",", "&2") |                 .replace("[{\"color\":\"dark_green\",", "&2") | ||||||
|             .replace("[{\"color\":\"dark_aqua\",", "&3").replace("[{\"color\":\"dark_red\",", "&4") |                 .replace("[{\"color\":\"dark_aqua\",", "&3").replace("[{\"color\":\"dark_red\",", "&4") | ||||||
|             .replace("[{\"color\":\"dark_purple\",", "&5").replace("[{\"color\":\"gold\",", "&6") |                 .replace("[{\"color\":\"dark_purple\",", "&5").replace("[{\"color\":\"gold\",", "&6") | ||||||
|             .replace("[{\"color\":\"gray\",", "&7").replace("[{\"color\":\"dark_gray\",", "&8") |                 .replace("[{\"color\":\"gray\",", "&7").replace("[{\"color\":\"dark_gray\",", "&8") | ||||||
|             .replace("[{\"color\":\"blue\",", "&9").replace("[{\"color\":\"green\",", "&a") |                 .replace("[{\"color\":\"blue\",", "&9").replace("[{\"color\":\"green\",", "&a") | ||||||
|             .replace("[{\"color\":\"aqua\",", "&b").replace("[{\"color\":\"red\",", "&c") |                 .replace("[{\"color\":\"aqua\",", "&b").replace("[{\"color\":\"red\",", "&c") | ||||||
|             .replace("[{\"color\":\"light_purple\",", "&d").replace("[{\"color\":\"yellow\",", "&e") |                 .replace("[{\"color\":\"light_purple\",", "&d").replace("[{\"color\":\"yellow\",", "&e") | ||||||
|             .replace("[{\"color\":\"white\",", "&f").replace("[{\"obfuscated\":true,", "&k") |                 .replace("[{\"color\":\"white\",", "&f").replace("[{\"obfuscated\":true,", "&k") | ||||||
|             .replace("[{\"bold\":true,", "&l").replace("[{\"strikethrough\":true,", "&m") |                 .replace("[{\"bold\":true,", "&l").replace("[{\"strikethrough\":true,", "&m") | ||||||
|             .replace("[{\"underlined\":true,", "&n").replace("[{\"italic\":true,", "&o") |                 .replace("[{\"underlined\":true,", "&n").replace("[{\"italic\":true,", "&o") | ||||||
|             .replace("{\"color\":\"black\",\"text\":\"", "&0") |                 .replace("{\"color\":\"black\",\"text\":\"", "&0") | ||||||
|             .replace("{\"color\":\"dark_blue\",\"text\":\"", "&1") |                 .replace("{\"color\":\"dark_blue\",\"text\":\"", "&1") | ||||||
|             .replace("{\"color\":\"dark_green\",\"text\":\"", "&2") |                 .replace("{\"color\":\"dark_green\",\"text\":\"", "&2") | ||||||
|             .replace("{\"color\":\"dark_aqua\",\"text\":\"", "&3") |                 .replace("{\"color\":\"dark_aqua\",\"text\":\"", "&3") | ||||||
|             .replace("{\"color\":\"dark_red\",\"text\":\"", "&4") |                 .replace("{\"color\":\"dark_red\",\"text\":\"", "&4") | ||||||
|             .replace("{\"color\":\"dark_purple\",\"text\":\"", "&5") |                 .replace("{\"color\":\"dark_purple\",\"text\":\"", "&5") | ||||||
|             .replace("{\"color\":\"gold\",\"text\":\"", "&6") |                 .replace("{\"color\":\"gold\",\"text\":\"", "&6") | ||||||
|             .replace("{\"color\":\"gray\",\"text\":\"", "&7") |                 .replace("{\"color\":\"gray\",\"text\":\"", "&7") | ||||||
|             .replace("{\"color\":\"dark_gray\",\"text\":\"", "&8") |                 .replace("{\"color\":\"dark_gray\",\"text\":\"", "&8") | ||||||
|             .replace("{\"color\":\"blue\",\"text\":\"", "&9") |                 .replace("{\"color\":\"blue\",\"text\":\"", "&9") | ||||||
|             .replace("{\"color\":\"green\",\"text\":\"", "&a") |                 .replace("{\"color\":\"green\",\"text\":\"", "&a") | ||||||
|             .replace("{\"color\":\"aqua\",\"text\":\"", "&b") |                 .replace("{\"color\":\"aqua\",\"text\":\"", "&b") | ||||||
|             .replace("{\"color\":\"red\",\"text\":\"", "&c") |                 .replace("{\"color\":\"red\",\"text\":\"", "&c") | ||||||
|             .replace("{\"color\":\"light_purple\",\"text\":\"", "&d") |                 .replace("{\"color\":\"light_purple\",\"text\":\"", "&d") | ||||||
|             .replace("{\"color\":\"yellow\",\"text\":\"", "&e") |                 .replace("{\"color\":\"yellow\",\"text\":\"", "&e") | ||||||
|             .replace("{\"color\":\"white\",\"text\":\"", "&f") |                 .replace("{\"color\":\"white\",\"text\":\"", "&f") | ||||||
|             .replace("{\"obfuscated\":true,\"text\":\"", "&k") |                 .replace("{\"obfuscated\":true,\"text\":\"", "&k") | ||||||
|             .replace("{\"bold\":true,\"text\":\"", "&l") |                 .replace("{\"bold\":true,\"text\":\"", "&l") | ||||||
|             .replace("{\"strikethrough\":true,\"text\":\"", "&m") |                 .replace("{\"strikethrough\":true,\"text\":\"", "&m") | ||||||
|             .replace("{\"underlined\":true,\"text\":\"", "&n") |                 .replace("{\"underlined\":true,\"text\":\"", "&n") | ||||||
|             .replace("{\"italic\":true,\"text\":\"", "&o").replace("{\"color\":\"black\",", "&0") |                 .replace("{\"italic\":true,\"text\":\"", "&o").replace("{\"color\":\"black\",", "&0") | ||||||
|             .replace("{\"color\":\"dark_blue\",", "&1").replace("{\"color\":\"dark_green\",", "&2") |                 .replace("{\"color\":\"dark_blue\",", "&1").replace("{\"color\":\"dark_green\",", "&2") | ||||||
|             .replace("{\"color\":\"dark_aqua\",", "&3").replace("{\"color\":\"dark_red\",", "&4") |                 .replace("{\"color\":\"dark_aqua\",", "&3").replace("{\"color\":\"dark_red\",", "&4") | ||||||
|             .replace("{\"color\":\"dark_purple\",", "&5").replace("{\"color\":\"gold\",", "&6") |                 .replace("{\"color\":\"dark_purple\",", "&5").replace("{\"color\":\"gold\",", "&6") | ||||||
|             .replace("{\"color\":\"gray\",", "&7").replace("{\"color\":\"dark_gray\",", "&8") |                 .replace("{\"color\":\"gray\",", "&7").replace("{\"color\":\"dark_gray\",", "&8") | ||||||
|             .replace("{\"color\":\"blue\",", "&9").replace("{\"color\":\"green\",", "&a") |                 .replace("{\"color\":\"blue\",", "&9").replace("{\"color\":\"green\",", "&a") | ||||||
|             .replace("{\"color\":\"aqua\",", "&b").replace("{\"color\":\"red\",", "&c") |                 .replace("{\"color\":\"aqua\",", "&b").replace("{\"color\":\"red\",", "&c") | ||||||
|             .replace("{\"color\":\"light_purple\",", "&d").replace("{\"color\":\"yellow\",", "&e") |                 .replace("{\"color\":\"light_purple\",", "&d").replace("{\"color\":\"yellow\",", "&e") | ||||||
|             .replace("{\"color\":\"white\",", "&f").replace("{\"obfuscated\":true,", "&k") |                 .replace("{\"color\":\"white\",", "&f").replace("{\"obfuscated\":true,", "&k") | ||||||
|             .replace("{\"bold\":true,", "&l").replace("{\"strikethrough\":true,", "&m") |                 .replace("{\"bold\":true,", "&l").replace("{\"strikethrough\":true,", "&m") | ||||||
|             .replace("{\"underlined\":true,", "&n").replace("{\"italic\":true,", "&o") |                 .replace("{\"underlined\":true,", "&n").replace("{\"italic\":true,", "&o") | ||||||
|             .replace("\"color\":\"black\",\"text\":\"", "&0") |                 .replace("\"color\":\"black\",\"text\":\"", "&0") | ||||||
|             .replace("\"color\":\"dark_blue\",\"text\":\"", "&1") |                 .replace("\"color\":\"dark_blue\",\"text\":\"", "&1") | ||||||
|             .replace("\"color\":\"dark_green\",\"text\":\"", "&2") |                 .replace("\"color\":\"dark_green\",\"text\":\"", "&2") | ||||||
|             .replace("\"color\":\"dark_aqua\",\"text\":\"", "&3") |                 .replace("\"color\":\"dark_aqua\",\"text\":\"", "&3") | ||||||
|             .replace("\"color\":\"dark_red\",\"text\":\"", "&4") |                 .replace("\"color\":\"dark_red\",\"text\":\"", "&4") | ||||||
|             .replace("\"color\":\"dark_purple\",\"text\":\"", "&5") |                 .replace("\"color\":\"dark_purple\",\"text\":\"", "&5") | ||||||
|             .replace("\"color\":\"gold\",\"text\":\"", "&6") |                 .replace("\"color\":\"gold\",\"text\":\"", "&6") | ||||||
|             .replace("\"color\":\"gray\",\"text\":\"", "&7") |                 .replace("\"color\":\"gray\",\"text\":\"", "&7") | ||||||
|             .replace("\"color\":\"dark_gray\",\"text\":\"", "&8") |                 .replace("\"color\":\"dark_gray\",\"text\":\"", "&8") | ||||||
|             .replace("\"color\":\"blue\",\"text\":\"", "&9") |                 .replace("\"color\":\"blue\",\"text\":\"", "&9") | ||||||
|             .replace("\"color\":\"green\",\"text\":\"", "&a") |                 .replace("\"color\":\"green\",\"text\":\"", "&a") | ||||||
|             .replace("\"color\":\"aqua\",\"text\":\"", "&b") |                 .replace("\"color\":\"aqua\",\"text\":\"", "&b") | ||||||
|             .replace("\"color\":\"red\",\"text\":\"", "&c") |                 .replace("\"color\":\"red\",\"text\":\"", "&c") | ||||||
|             .replace("\"color\":\"light_purple\",\"text\":\"", "&d") |                 .replace("\"color\":\"light_purple\",\"text\":\"", "&d") | ||||||
|             .replace("\"color\":\"yellow\",\"text\":\"", "&e") |                 .replace("\"color\":\"yellow\",\"text\":\"", "&e") | ||||||
|             .replace("\"color\":\"white\",\"text\":\"", "&f") |                 .replace("\"color\":\"white\",\"text\":\"", "&f") | ||||||
|             .replace("\"obfuscated\":true,\"text\":\"", "&k") |                 .replace("\"obfuscated\":true,\"text\":\"", "&k") | ||||||
|             .replace("\"bold\":true,\"text\":\"", "&l") |                 .replace("\"bold\":true,\"text\":\"", "&l") | ||||||
|             .replace("\"strikethrough\":true,\"text\":\"", "&m") |                 .replace("\"strikethrough\":true,\"text\":\"", "&m") | ||||||
|             .replace("\"underlined\":true,\"text\":\"", "&n") |                 .replace("\"underlined\":true,\"text\":\"", "&n") | ||||||
|             .replace("\"italic\":true,\"text\":\"", "&o").replace("\"color\":\"black\",", "&0") |                 .replace("\"italic\":true,\"text\":\"", "&o").replace("\"color\":\"black\",", "&0") | ||||||
|             .replace("\"color\":\"dark_blue\",", "&1").replace("\"color\":\"dark_green\",", "&2") |                 .replace("\"color\":\"dark_blue\",", "&1").replace("\"color\":\"dark_green\",", "&2") | ||||||
|             .replace("\"color\":\"dark_aqua\",", "&3").replace("\"color\":\"dark_red\",", "&4") |                 .replace("\"color\":\"dark_aqua\",", "&3").replace("\"color\":\"dark_red\",", "&4") | ||||||
|             .replace("\"color\":\"dark_purple\",", "&5").replace("\"color\":\"gold\",", "&6") |                 .replace("\"color\":\"dark_purple\",", "&5").replace("\"color\":\"gold\",", "&6") | ||||||
|             .replace("\"color\":\"gray\",", "&7").replace("\"color\":\"dark_gray\",", "&8") |                 .replace("\"color\":\"gray\",", "&7").replace("\"color\":\"dark_gray\",", "&8") | ||||||
|             .replace("\"color\":\"blue\",", "&9").replace("\"color\":\"green\",", "&a") |                 .replace("\"color\":\"blue\",", "&9").replace("\"color\":\"green\",", "&a") | ||||||
|             .replace("\"color\":\"aqua\",", "&b").replace("\"color\":\"red\",", "&c") |                 .replace("\"color\":\"aqua\",", "&b").replace("\"color\":\"red\",", "&c") | ||||||
|             .replace("\"color\":\"light_purple\",", "&d").replace("\"color\":\"yellow\",", "&e") |                 .replace("\"color\":\"light_purple\",", "&d").replace("\"color\":\"yellow\",", "&e") | ||||||
|             .replace("\"color\":\"white\",", "&f").replace("\"obfuscated\":true,", "&k") |                 .replace("\"color\":\"white\",", "&f").replace("\"obfuscated\":true,", "&k") | ||||||
|             .replace("\"bold\":true,", "&l").replace("\"strikethrough\":true,", "&m") |                 .replace("\"bold\":true,", "&l").replace("\"strikethrough\":true,", "&m") | ||||||
|             .replace("\"underlined\":true,", "&n").replace("\"italic\":true,", "&o") |                 .replace("\"underlined\":true,", "&n").replace("\"italic\":true,", "&o") | ||||||
|             .replace("[{\"text\":\"", "&0").replace("{\"text\":\"", "&0").replace("\"},", "") |                 .replace("[{\"text\":\"", "&0").replace("{\"text\":\"", "&0").replace("\"},", "") | ||||||
|             .replace("\"}]", "").replace("\"}", ""); |                 .replace("\"}]", "").replace("\"}", ""); | ||||||
|         for (Entry<String, String> entry : Captions.replacements.entrySet()) { |         str = ChatColor.translateAlternateColorCodes('&', str); | ||||||
|             str = str.replace(entry.getKey(), entry.getValue()); |  | ||||||
|         } |  | ||||||
|         return str; |         return str; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Restore the TileEntity data to the given world at the given coordinates. | ||||||
|  |      * | ||||||
|  |      * @param worldName World name | ||||||
|  |      * @param x         x position | ||||||
|  |      * @param y         y position | ||||||
|  |      * @param z         z position | ||||||
|  |      * @return true if successful | ||||||
|  |      */ | ||||||
|     public boolean restoreTag(String worldName, int x, int y, int z) { |     public boolean restoreTag(String worldName, int x, int y, int z) { | ||||||
|         if (this.tag == null) { |         World world = BukkitUtil.getWorld(worldName); | ||||||
|  |         if (world == null) { | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|         World world = BukkitUtil.getWorld(worldName); |         return restoreTag(world.getBlockAt(x, y, z)); | ||||||
|         Block block = world.getBlockAt(x, y, z); |     } | ||||||
|         if (block == null) { |  | ||||||
|  |     /** | ||||||
|  |      * Restore the TileEntity data to the given block | ||||||
|  |      * | ||||||
|  |      * @param block Block to restore to | ||||||
|  |      * @return true if successful | ||||||
|  |      */ | ||||||
|  |     @SuppressWarnings("deprecation") // #setLine is needed for Spigot compatibility | ||||||
|  |     public boolean restoreTag(@NonNull Block block) { | ||||||
|  |         if (this.tag == null) { | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|         org.bukkit.block.BlockState state = block.getState(); |         org.bukkit.block.BlockState state = block.getState(); | ||||||
|         switch (getId()) { |         switch (getId()) { | ||||||
|             case "chest": |             case "chest", "beacon", "brewingstand", "dispenser", "dropper", "furnace", "hopper", "shulkerbox" -> { | ||||||
|             case "beacon": |                 if (!(state instanceof Container container)) { | ||||||
|             case "brewingstand": |  | ||||||
|             case "dispenser": |  | ||||||
|             case "dropper": |  | ||||||
|             case "furnace": |  | ||||||
|             case "hopper": |  | ||||||
|             case "shulkerbox": |  | ||||||
|                 if (!(state instanceof Container)) { |  | ||||||
|                     return false; |                     return false; | ||||||
|                 } |                 } | ||||||
|                 List<Tag> itemsTag = this.tag.getListTag("Items").getValue(); |                 List<Tag> itemsTag = this.tag.getListTag("Items").getValue(); | ||||||
|                 Container container = (Container) state; |  | ||||||
|                 Inventory inv = container.getSnapshotInventory(); |                 Inventory inv = container.getSnapshotInventory(); | ||||||
|                 for (Tag itemTag : itemsTag) { |                 for (Tag itemTag : itemsTag) { | ||||||
|                     CompoundTag itemComp = (CompoundTag) itemTag; |                     CompoundTag itemComp = (CompoundTag) itemTag; | ||||||
| @@ -208,9 +218,9 @@ public class StateWrapper { | |||||||
|                 } |                 } | ||||||
|                 container.update(true, false); |                 container.update(true, false); | ||||||
|                 return true; |                 return true; | ||||||
|             case "sign": |             } | ||||||
|                 if (state instanceof Sign) { |             case "sign" -> { | ||||||
|                     Sign sign = (Sign) state; |                 if (state instanceof Sign sign) { | ||||||
|                     sign.setLine(0, jsonToColourCode(tag.getString("Text1"))); |                     sign.setLine(0, jsonToColourCode(tag.getString("Text1"))); | ||||||
|                     sign.setLine(1, jsonToColourCode(tag.getString("Text2"))); |                     sign.setLine(1, jsonToColourCode(tag.getString("Text2"))); | ||||||
|                     sign.setLine(2, jsonToColourCode(tag.getString("Text3"))); |                     sign.setLine(2, jsonToColourCode(tag.getString("Text3"))); | ||||||
| @@ -219,6 +229,7 @@ public class StateWrapper { | |||||||
|                     return true; |                     return true; | ||||||
|                 } |                 } | ||||||
|                 return false; |                 return false; | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
| @@ -227,8 +238,7 @@ public class StateWrapper { | |||||||
|         if (this.tag != null) { |         if (this.tag != null) { | ||||||
|             return this.tag; |             return this.tag; | ||||||
|         } |         } | ||||||
|         if (this.state instanceof InventoryHolder) { |         if (this.state instanceof InventoryHolder inv) { | ||||||
|             InventoryHolder inv = (InventoryHolder) this.state; |  | ||||||
|             ItemStack[] contents = inv.getInventory().getContents(); |             ItemStack[] contents = inv.getInventory().getContents(); | ||||||
|             Map<String, Tag> values = new HashMap<>(); |             Map<String, Tag> values = new HashMap<>(); | ||||||
|             values.put("Items", new ListTag(CompoundTag.class, serializeInventory(contents))); |             values.put("Items", new ListTag(CompoundTag.class, serializeInventory(contents))); | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,30 +21,16 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.util; | ||||||
|  |  | ||||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||||
| import com.sk89q.worldedit.world.block.BlockState; | import com.sk89q.worldedit.world.block.BlockState; | ||||||
| import com.sk89q.worldedit.world.item.ItemType; |  | ||||||
| import org.bukkit.Material; | import org.bukkit.Material; | ||||||
| import org.bukkit.block.Block; | import org.bukkit.block.Block; | ||||||
|  |  | ||||||
| import java.util.function.Supplier; |  | ||||||
|  |  | ||||||
| public class BukkitBlockUtil { | 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) { |     public static BlockState get(Block block) { | ||||||
|         return get(block.getType()); |         return get(block.getType()); | ||||||
| @@ -53,4 +39,5 @@ public class BukkitBlockUtil { | |||||||
|     public static BlockState get(Material material) { |     public static BlockState get(Material material) { | ||||||
|         return BukkitAdapter.asBlockType(material).getDefaultState(); |         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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,145 +21,31 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.util; | ||||||
|  |  | ||||||
| import com.plotsquared.core.PlotSquared; | import com.google.inject.Singleton; | ||||||
| import com.plotsquared.core.listener.WEExtent; |  | ||||||
| import com.plotsquared.core.queue.GlobalBlockQueue; |  | ||||||
| import com.plotsquared.core.queue.LocalBlockQueue; |  | ||||||
| import com.plotsquared.core.util.ChunkManager; | 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.BlockVector2; | ||||||
| import com.sk89q.worldedit.math.BlockVector3; |  | ||||||
| import com.sk89q.worldedit.regions.CuboidRegion; | import com.sk89q.worldedit.regions.CuboidRegion; | ||||||
| import com.sk89q.worldedit.world.block.BaseBlock; |  | ||||||
| import io.papermc.lib.PaperLib; | 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 java.util.concurrent.CompletableFuture; | ||||||
|  |  | ||||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_ANIMAL; | @Singleton | ||||||
| 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; |  | ||||||
|  |  | ||||||
| public class BukkitChunkManager extends ChunkManager { | public class BukkitChunkManager extends ChunkManager { | ||||||
|  |  | ||||||
|     public static boolean isIn(CuboidRegion region, int x, int z) { |     public static boolean isIn(CuboidRegion region, int x, int z) { | ||||||
|         return x >= region.getMinimumPoint().getX() && x <= region.getMaximumPoint().getX() |         return x >= region.getMinimumPoint().getX() && x <= region.getMaximumPoint().getX() && z >= region | ||||||
|             && z >= region.getMinimumPoint().getZ() && z <= region.getMaximumPoint().getZ(); |                 .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; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public CompletableFuture<?> loadChunk(String world, BlockVector2 chunkLoc, boolean force) { |     public CompletableFuture<?> loadChunk(String world, BlockVector2 chunkLoc, boolean force) { | ||||||
|         return PaperLib |         return PaperLib.getChunkAtAsync(BukkitUtil.getWorld(world), chunkLoc.getX(), chunkLoc.getZ(), force); | ||||||
|             .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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,25 +21,32 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.util; | ||||||
|  |  | ||||||
|  | import com.google.inject.Singleton; | ||||||
| import com.plotsquared.bukkit.player.BukkitOfflinePlayer; | import com.plotsquared.bukkit.player.BukkitOfflinePlayer; | ||||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | import com.plotsquared.bukkit.player.BukkitPlayer; | ||||||
| import com.plotsquared.core.PlotSquared; |  | ||||||
| import com.plotsquared.core.player.OfflinePlotPlayer; | import com.plotsquared.core.player.OfflinePlotPlayer; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
|  | import com.plotsquared.core.plot.PlotArea; | ||||||
| import com.plotsquared.core.util.EconHandler; | import com.plotsquared.core.util.EconHandler; | ||||||
| import com.plotsquared.core.util.PermHandler; |  | ||||||
| import net.milkbowl.vault.economy.Economy; | import net.milkbowl.vault.economy.Economy; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
|  | import org.bukkit.OfflinePlayer; | ||||||
| import org.bukkit.plugin.RegisteredServiceProvider; | import org.bukkit.plugin.RegisteredServiceProvider; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
|  | @Singleton | ||||||
| public class BukkitEconHandler extends EconHandler { | public class BukkitEconHandler extends EconHandler { | ||||||
|  |  | ||||||
|     private Economy econ; |     private Economy econ; | ||||||
|  |  | ||||||
|  |     private static OfflinePlayer getBukkitOfflinePlayer(PlotPlayer<?> plotPlayer) { | ||||||
|  |         return ((BukkitPlayer) plotPlayer).player; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean init() { |     public boolean init() { | ||||||
|         if (this.econ == null) { |         if (this.econ == null) { | ||||||
| @@ -53,45 +60,54 @@ public class BukkitEconHandler extends EconHandler { | |||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         RegisteredServiceProvider<Economy> economyProvider = |         RegisteredServiceProvider<Economy> economyProvider = | ||||||
|             Bukkit.getServer().getServicesManager().getRegistration(Economy.class); |                 Bukkit.getServer().getServicesManager().getRegistration(Economy.class); | ||||||
|         if (economyProvider != null) { |         if (economyProvider != null) { | ||||||
|             this.econ = economyProvider.getProvider(); |             this.econ = economyProvider.getProvider(); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public double getMoney(PlotPlayer<?> player) { |     @Override | ||||||
|  |     public double getMoney(PlotPlayer<?> player) { | ||||||
|         double bal = super.getMoney(player); |         double bal = super.getMoney(player); | ||||||
|         if (Double.isNaN(bal)) { |         if (Double.isNaN(bal)) { | ||||||
|             return this.econ.getBalance(((BukkitPlayer) player).player); |             return this.econ.getBalance(getBukkitOfflinePlayer(player)); | ||||||
|         } |         } | ||||||
|         return bal; |         return bal; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public void withdrawMoney(PlotPlayer<?> player, double amount) { |     @Override | ||||||
|         this.econ.withdrawPlayer(((BukkitPlayer) player).player, amount); |     public void withdrawMoney(PlotPlayer<?> player, double amount) { | ||||||
|  |         this.econ.withdrawPlayer(getBukkitOfflinePlayer(player), amount); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public void depositMoney(PlotPlayer<?> player, double amount) { |     @Override | ||||||
|         this.econ.depositPlayer(((BukkitPlayer) player).player, amount); |     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); |         this.econ.depositPlayer(((BukkitOfflinePlayer) player).player, amount); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     @Override | ||||||
|      * @deprecated Use {@link PermHandler#hasPermission(String, String, String)} instead |     public boolean isEnabled(PlotArea plotArea) { | ||||||
|      */ |         return plotArea.useEconomy(); | ||||||
|     @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 double getBalance(PlotPlayer<?> player) { |     @Override | ||||||
|         return this.econ.getBalance(player.getName()); |     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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,14 +21,15 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.util; | ||||||
|  |  | ||||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | import com.plotsquared.bukkit.player.BukkitPlayer; | ||||||
| import com.plotsquared.core.configuration.Captions; |  | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
|  | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.location.Location; | import com.plotsquared.core.location.Location; | ||||||
|  | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| import com.plotsquared.core.plot.PlotArea; | import com.plotsquared.core.plot.PlotArea; | ||||||
| import com.plotsquared.core.plot.flag.implementations.AnimalAttackFlag; | 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.TamedAttackFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.VehicleCapFlag; | import com.plotsquared.core.plot.flag.implementations.VehicleCapFlag; | ||||||
| import com.plotsquared.core.util.EntityUtil; | import com.plotsquared.core.util.EntityUtil; | ||||||
| import com.plotsquared.core.util.MainUtil; |  | ||||||
| import com.plotsquared.core.util.Permissions; | import com.plotsquared.core.util.Permissions; | ||||||
| import com.plotsquared.core.util.entity.EntityCategories; | import com.plotsquared.core.util.entity.EntityCategories; | ||||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||||
|  | import net.kyori.adventure.text.minimessage.Template; | ||||||
| import org.bukkit.entity.Arrow; | import org.bukkit.entity.Arrow; | ||||||
| import org.bukkit.entity.Creature; | import org.bukkit.entity.Creature; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
| @@ -65,16 +66,15 @@ import java.util.Objects; | |||||||
| public class BukkitEntityUtil { | public class BukkitEntityUtil { | ||||||
|  |  | ||||||
|     public static final com.sk89q.worldedit.world.entity.EntityType FAKE_ENTITY_TYPE = |     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) { |     public static boolean entityDamage(Entity damager, Entity victim) { | ||||||
|         return entityDamage(damager, victim, null); |         return entityDamage(damager, victim, null); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static boolean entityDamage(Entity damager, Entity victim, |     public static boolean entityDamage(Entity damager, Entity victim, EntityDamageEvent.DamageCause cause) { | ||||||
|         EntityDamageEvent.DamageCause cause) { |         Location dloc = BukkitUtil.adapt(damager.getLocation()); | ||||||
|         Location dloc = BukkitUtil.getLocation(damager); |         Location vloc = BukkitUtil.adapt(victim.getLocation()); | ||||||
|         Location vloc = BukkitUtil.getLocation(victim); |  | ||||||
|         PlotArea dArea = dloc.getPlotArea(); |         PlotArea dArea = dloc.getPlotArea(); | ||||||
|         PlotArea vArea; |         PlotArea vArea; | ||||||
|         if (dArea != null && dArea.contains(vloc.getX(), vloc.getZ())) { |         if (dArea != null && dArea.contains(vloc.getX(), vloc.getZ())) { | ||||||
| @@ -144,15 +144,14 @@ public class BukkitEntityUtil { | |||||||
|         Player player; |         Player player; | ||||||
|         if (damager instanceof Player) { // attacker is player |         if (damager instanceof Player) { // attacker is player | ||||||
|             player = (Player) damager; |             player = (Player) damager; | ||||||
|         } else if (damager instanceof Projectile) { |         } else if (damager instanceof Projectile projectile) { | ||||||
|             Projectile projectile = (Projectile) damager; |  | ||||||
|             ProjectileSource shooter = projectile.getShooter(); |             ProjectileSource shooter = projectile.getShooter(); | ||||||
|             if (shooter instanceof Player) { // shooter is player |             if (shooter instanceof Player) { // shooter is player | ||||||
|                 player = (Player) shooter; |                 player = (Player) shooter; | ||||||
|             } else { // shooter is not player |             } else { // shooter is not player | ||||||
|                 if (shooter instanceof BlockProjectileSource) { |                 if (shooter instanceof BlockProjectileSource) { | ||||||
|                     Location sLoc = BukkitUtil |                     Location sLoc = BukkitUtil | ||||||
|                         .getLocation(((BlockProjectileSource) shooter).getBlock().getLocation()); |                             .adapt(((BlockProjectileSource) shooter).getBlock().getLocation()); | ||||||
|                     dplot = dArea.getPlot(sLoc); |                     dplot = dArea.getPlot(sLoc); | ||||||
|                 } |                 } | ||||||
|                 player = null; |                 player = null; | ||||||
| @@ -161,7 +160,7 @@ public class BukkitEntityUtil { | |||||||
|             player = null; |             player = null; | ||||||
|         } |         } | ||||||
|         if (player != null) { |         if (player != null) { | ||||||
|             BukkitPlayer plotPlayer = BukkitUtil.getPlayer(player); |             BukkitPlayer plotPlayer = BukkitUtil.adapt(player); | ||||||
|  |  | ||||||
|             final com.sk89q.worldedit.world.entity.EntityType entityType; |             final com.sk89q.worldedit.world.entity.EntityType entityType; | ||||||
|  |  | ||||||
| @@ -174,82 +173,93 @@ public class BukkitEntityUtil { | |||||||
|  |  | ||||||
|             if (EntityCategories.HANGING.contains(entityType)) { // hanging |             if (EntityCategories.HANGING.contains(entityType)) { // hanging | ||||||
|                 if (plot != null && (plot.getFlag(HangingBreakFlag.class) || plot |                 if (plot != null && (plot.getFlag(HangingBreakFlag.class) || plot | ||||||
|                     .isAdded(plotPlayer.getUUID()))) { |                         .isAdded(plotPlayer.getUUID()))) { | ||||||
|                     if (Settings.Done.RESTRICT_BUILDING && DoneFlag.isDone(plot)) { |                     if (Settings.Done.RESTRICT_BUILDING && DoneFlag.isDone(plot)) { | ||||||
|                         if (!Permissions |                         if (!Permissions | ||||||
|                             .hasPermission(plotPlayer, Captions.PERMISSION_ADMIN_BUILD_OTHER)) { |                                 .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_BUILD_OTHER)) { | ||||||
|                             MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, |                             plotPlayer.sendMessage( | ||||||
|                                 Captions.PERMISSION_ADMIN_BUILD_OTHER); |                                     TranslatableCaption.of("done.building_restricted") | ||||||
|  |                             ); | ||||||
|                             return false; |                             return false; | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                     return true; |                     return true; | ||||||
|                 } |                 } | ||||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.destroy." + stub)) { |                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_DESTROY + "." + stub)) { | ||||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, |                     plotPlayer.sendMessage( | ||||||
|                         "plots.admin.destroy." + stub); |                             TranslatableCaption.of("permission.no_permission_event"), | ||||||
|  |                             Template.of("node", Permission.PERMISSION_ADMIN_DESTROY + "." + stub) | ||||||
|  |                     ); | ||||||
|                     return false; |                     return false; | ||||||
|                 } |                 } | ||||||
|             } else if (victim.getType() == EntityType.ARMOR_STAND) { |             } else if (victim.getType() == EntityType.ARMOR_STAND) { | ||||||
|                 if (plot != null && (plot.getFlag(MiscBreakFlag.class) || plot |                 if (plot != null && (plot.getFlag(MiscBreakFlag.class) || plot | ||||||
|                     .isAdded(plotPlayer.getUUID()))) { |                         .isAdded(plotPlayer.getUUID()))) { | ||||||
|                     return true; |                     return true; | ||||||
|                 } |                 } | ||||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.destroy." + stub)) { |                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_DESTROY + "." + stub)) { | ||||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, |                     plotPlayer.sendMessage( | ||||||
|                         "plots.admin.destroy." + stub); |                             TranslatableCaption.of("permission.no_permission_event"), | ||||||
|  |                             Template.of("node", Permission.PERMISSION_ADMIN_DESTROY + "." + stub) | ||||||
|  |                     ); | ||||||
|                     if (plot != null) { |                     if (plot != null) { | ||||||
|                         plot.debug(player.getName() |                         plot.debug(player.getName() | ||||||
|                             + " could not break armor stand because misc-break = false"); |                                 + " could not break armor stand because misc-break = false"); | ||||||
|                     } |                     } | ||||||
|                     return false; |                     return false; | ||||||
|                 } |                 } | ||||||
|             } else if (EntityCategories.HOSTILE.contains(entityType)) { |             } else if (EntityCategories.HOSTILE.contains(entityType)) { | ||||||
|                 if (isPlot) { |                 if (isPlot) { | ||||||
|                     if (plot.getFlag(HostileAttackFlag.class) || plot.getFlag(PveFlag.class) || plot |                     if (plot.getFlag(HostileAttackFlag.class) || plot.getFlag(PveFlag.class) || plot | ||||||
|                         .isAdded(plotPlayer.getUUID())) { |                             .isAdded(plotPlayer.getUUID())) { | ||||||
|                         return true; |                         return true; | ||||||
|                     } |                     } | ||||||
|                 } else if (roadFlags && (area.getRoadFlag(HostileAttackFlag.class) || area |                 } else if (roadFlags && (area.getRoadFlag(HostileAttackFlag.class) || area | ||||||
|                     .getFlag(PveFlag.class))) { |                         .getFlag(PveFlag.class))) { | ||||||
|                     return true; |                     return true; | ||||||
|                 } |                 } | ||||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.pve." + stub)) { |                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, |                     plotPlayer.sendMessage( | ||||||
|                         "plots.admin.pve." + stub); |                             TranslatableCaption.of("permission.no_permission_event"), | ||||||
|  |                             Template.of("node", Permission.PERMISSION_ADMIN_PVE + "." + stub) | ||||||
|  |                     ); | ||||||
|                     if (plot != null) { |                     if (plot != null) { | ||||||
|                         plot.debug(player.getName() + " could not attack " + entityType |                         plot.debug(player.getName() + " could not attack " + entityType | ||||||
|                             + " because pve = false OR hostile-attack = false"); |                                 + " because pve = false OR hostile-attack = false"); | ||||||
|                     } |                     } | ||||||
|                     return false; |                     return false; | ||||||
|                 } |                 } | ||||||
|             } else if (EntityCategories.TAMEABLE.contains(entityType)) { // victim is tameable |             } else if (EntityCategories.TAMEABLE.contains(entityType)) { // victim is tameable | ||||||
|                 if (isPlot) { |                 if (isPlot) { | ||||||
|                     if (plot.getFlag(TamedAttackFlag.class) || plot.getFlag(PveFlag.class) || plot |                     if (plot.getFlag(TamedAttackFlag.class) || plot.getFlag(PveFlag.class) || plot | ||||||
|                         .isAdded(plotPlayer.getUUID())) { |                             .isAdded(plotPlayer.getUUID())) { | ||||||
|                         return true; |                         return true; | ||||||
|                     } |                     } | ||||||
|                 } else if (roadFlags && (area.getRoadFlag(TamedAttackFlag.class) || area |                 } else if (roadFlags && (area.getRoadFlag(TamedAttackFlag.class) || area | ||||||
|                     .getFlag(PveFlag.class))) { |                         .getFlag(PveFlag.class))) { | ||||||
|                     return true; |                     return true; | ||||||
|                 } |                 } | ||||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.pve." + stub)) { |                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, |                     plotPlayer.sendMessage( | ||||||
|                         "plots.admin.pve." + stub); |                             TranslatableCaption.of("permission.no_permission_event"), | ||||||
|  |                             Template.of("node", Permission.PERMISSION_ADMIN_PVE + "." + stub) | ||||||
|  |                     ); | ||||||
|                     if (plot != null) { |                     if (plot != null) { | ||||||
|                         plot.debug(player.getName() + " could not attack " + entityType |                         plot.debug(player.getName() + " could not attack " + entityType | ||||||
|                             + " because pve = false OR tamned-attack = false"); |                                 + " because pve = false OR tamed-attack = false"); | ||||||
|                     } |                     } | ||||||
|                     return false; |                     return false; | ||||||
|                 } |                 } | ||||||
|             } else if (EntityCategories.PLAYER.contains(entityType)) { |             } else if (EntityCategories.PLAYER.contains(entityType)) { | ||||||
|                 if (isPlot) { |                 if (isPlot) { | ||||||
|                     if (!plot.getFlag(PvpFlag.class) && !Permissions |                     if (!plot.getFlag(PvpFlag.class) && !Permissions | ||||||
|                         .hasPermission(plotPlayer, "plots.admin.pvp." + stub)) { |                             .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVP + "." + stub)) { | ||||||
|                         MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, |                         plotPlayer.sendMessage( | ||||||
|                             "plots.admin.pvp." + stub); |                                 TranslatableCaption.of("permission.no_permission_event"), | ||||||
|  |                                 Template.of("node", Permission.PERMISSION_ADMIN_PVP + "." + stub) | ||||||
|  |                         ); | ||||||
|                         plot.debug(player.getName() + " could not attack " + entityType |                         plot.debug(player.getName() + " could not attack " + entityType | ||||||
|                             + " because pve = false"); |                                 + " because pve = false"); | ||||||
|                         return false; |                         return false; | ||||||
|                     } else { |                     } else { | ||||||
|                         return true; |                         return true; | ||||||
| @@ -257,30 +267,36 @@ public class BukkitEntityUtil { | |||||||
|                 } else if (roadFlags && area.getRoadFlag(PvpFlag.class)) { |                 } else if (roadFlags && area.getRoadFlag(PvpFlag.class)) { | ||||||
|                     return true; |                     return true; | ||||||
|                 } |                 } | ||||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.pvp." + stub)) { |                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVP + "." + stub)) { | ||||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, |                     plotPlayer.sendMessage( | ||||||
|                         "plots.admin.pvp." + stub); |                             TranslatableCaption.of("permission.no_permission_event"), | ||||||
|  |                             Template.of("node", Permission.PERMISSION_ADMIN_PVP + "." + stub) | ||||||
|  |                     ); | ||||||
|                     return false; |                     return false; | ||||||
|                 } |                 } | ||||||
|             } else if (EntityCategories.ANIMAL.contains(entityType)) { // victim is animal |             } else if (EntityCategories.ANIMAL.contains(entityType)) { // victim is animal | ||||||
|                 if (isPlot) { |                 if (isPlot) { | ||||||
|                     if (plot.getFlag(AnimalAttackFlag.class) || plot.getFlag(PveFlag.class) || plot |                     if (plot.getFlag(AnimalAttackFlag.class) || plot.getFlag(PveFlag.class) || plot | ||||||
|                         .isAdded(plotPlayer.getUUID())) { |                             .isAdded(plotPlayer.getUUID())) { | ||||||
|                         plot.debug(player.getName() + " could not attack " + entityType |  | ||||||
|                             + " because pve = false OR animal-attack = false"); |  | ||||||
|                         return true; |                         return true; | ||||||
|                     } |                     } | ||||||
|                 } else if (roadFlags && (area.getRoadFlag(AnimalAttackFlag.class) || area |                 } else if (roadFlags && (area.getRoadFlag(AnimalAttackFlag.class) || area | ||||||
|                     .getFlag(PveFlag.class))) { |                         .getFlag(PveFlag.class))) { | ||||||
|                     return true; |                     return true; | ||||||
|                 } |                 } | ||||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.pve." + stub)) { |                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, |                     plotPlayer.sendMessage( | ||||||
|                         "plots.admin.pve." + stub); |                             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; |                     return false; | ||||||
|                 } |                 } | ||||||
|             } else if (EntityCategories.VEHICLE |             } else if (EntityCategories.VEHICLE | ||||||
|                 .contains(entityType)) { // Vehicles are managed in vehicle destroy event |                     .contains(entityType)) { // Vehicles are managed in vehicle destroy event | ||||||
|                 return true; |                 return true; | ||||||
|             } else { // victim is something else |             } else { // victim is something else | ||||||
|                 if (isPlot) { |                 if (isPlot) { | ||||||
| @@ -290,25 +306,27 @@ public class BukkitEntityUtil { | |||||||
|                 } else if (roadFlags && area.getRoadFlag(PveFlag.class)) { |                 } else if (roadFlags && area.getRoadFlag(PveFlag.class)) { | ||||||
|                     return true; |                     return true; | ||||||
|                 } |                 } | ||||||
|                 if (!Permissions.hasPermission(plotPlayer, "plots.admin.pve." + stub)) { |                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||||
|                     MainUtil.sendMessage(plotPlayer, Captions.NO_PERMISSION_EVENT, |                     plotPlayer.sendMessage( | ||||||
|                         "plots.admin.pve." + stub); |                             TranslatableCaption.of("permission.no_permission_event"), | ||||||
|  |                             Template.of("node", Permission.PERMISSION_ADMIN_PVE + "." + stub) | ||||||
|  |                     ); | ||||||
|                     if (plot != null) { |                     if (plot != null) { | ||||||
|                         plot.debug(player.getName() + " could not attack " + entityType |                         plot.debug(player.getName() + " could not attack " + entityType | ||||||
|                             + " because pve = false"); |                                 + " because pve = false"); | ||||||
|                     } |                     } | ||||||
|                     return false; |                     return false; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             return true; |             return true; | ||||||
|         } else if (dplot != null && (!dplot.equals(vplot) || Objects |         } 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); |             return vplot != null && vplot.getFlag(PveFlag.class); | ||||||
|         } |         } | ||||||
|         //disable the firework damage. too much of a headache to support at the moment. |         //disable the firework damage. too much of a headache to support at the moment. | ||||||
|         if (vplot != null) { |         if (vplot != null) { | ||||||
|             if (EntityDamageEvent.DamageCause.ENTITY_EXPLOSION == cause |             if (EntityDamageEvent.DamageCause.ENTITY_EXPLOSION == cause | ||||||
|                 && damager.getType() == EntityType.FIREWORK) { |                     && damager.getType() == EntityType.FIREWORK) { | ||||||
|                 return false; |                 return false; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @@ -316,46 +334,50 @@ public class BukkitEntityUtil { | |||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|         return ((vplot != null && vplot.getFlag(PveFlag.class)) || !(damager instanceof Arrow |         return ((vplot != null && vplot.getFlag(PveFlag.class)) || !(damager instanceof Arrow | ||||||
|             && !(victim instanceof Creature))); |                 && !(victim instanceof Creature))); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static boolean checkEntity(Entity entity, Plot plot) { |     public static boolean checkEntity(Entity entity, Plot plot) { | ||||||
|         if (plot == null || !plot.hasOwner() || plot.getFlags().isEmpty() && plot.getArea() |         if (plot == null || !plot.hasOwner() || plot.getFlags().isEmpty() && plot.getArea() | ||||||
|             .getFlagContainer().getFlagMap().isEmpty()) { |                 .getFlagContainer().getFlagMap().isEmpty()) { | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         final com.sk89q.worldedit.world.entity.EntityType entityType = |         final com.sk89q.worldedit.world.entity.EntityType entityType = | ||||||
|             BukkitAdapter.adapt(entity.getType()); |                 BukkitAdapter.adapt(entity.getType()); | ||||||
|  |  | ||||||
|         if (EntityCategories.PLAYER.contains(entityType)) { |         if (EntityCategories.PLAYER.contains(entityType)) { | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (EntityCategories.PROJECTILE.contains(entityType) || EntityCategories.OTHER |         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, |             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 |         // Has to go go before vehicle as horses are both | ||||||
|         // animals and vehicles |         // animals and vehicles | ||||||
|         if (EntityCategories.ANIMAL.contains(entityType) || EntityCategories.VILLAGER |         if (EntityCategories.ANIMAL.contains(entityType) || EntityCategories.VILLAGER | ||||||
|             .contains(entityType) || EntityCategories.TAMEABLE.contains(entityType)) { |                 .contains(entityType) || EntityCategories.TAMEABLE.contains(entityType)) { | ||||||
|             return EntityUtil |             return EntityUtil | ||||||
|                 .checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED, MobCapFlag.MOB_CAP_UNLIMITED, |                     .checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED, MobCapFlag.MOB_CAP_UNLIMITED, | ||||||
|                     AnimalCapFlag.ANIMAL_CAP_UNLIMITED); |                             AnimalCapFlag.ANIMAL_CAP_UNLIMITED | ||||||
|  |                     ); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (EntityCategories.HOSTILE.contains(entityType)) { |         if (EntityCategories.HOSTILE.contains(entityType)) { | ||||||
|             return EntityUtil |             return EntityUtil | ||||||
|                 .checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED, MobCapFlag.MOB_CAP_UNLIMITED, |                     .checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED, MobCapFlag.MOB_CAP_UNLIMITED, | ||||||
|                     HostileCapFlag.HOSTILE_CAP_UNLIMITED); |                             HostileCapFlag.HOSTILE_CAP_UNLIMITED | ||||||
|  |                     ); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (EntityCategories.VEHICLE.contains(entityType)) { |         if (EntityCategories.VEHICLE.contains(entityType)) { | ||||||
|             return EntityUtil.checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED, |             return EntityUtil.checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED, | ||||||
|                 VehicleCapFlag.VEHICLE_CAP_UNLIMITED); |                     VehicleCapFlag.VEHICLE_CAP_UNLIMITED | ||||||
|  |             ); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         return EntityUtil.checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED); |         return EntityUtil.checkEntity(plot, EntityCapFlag.ENTITY_CAP_UNLIMITED); | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,16 +21,18 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.util; | ||||||
|  |  | ||||||
|  | import com.google.inject.Singleton; | ||||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | import com.plotsquared.bukkit.player.BukkitPlayer; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.PlotInventory; | import com.plotsquared.core.plot.PlotInventory; | ||||||
| import com.plotsquared.core.plot.PlotItemStack; | import com.plotsquared.core.plot.PlotItemStack; | ||||||
| import com.plotsquared.core.util.InventoryUtil; | import com.plotsquared.core.util.InventoryUtil; | ||||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||||
|  | import net.kyori.adventure.text.Component; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
| import org.bukkit.ChatColor; | import org.bukkit.ChatColor; | ||||||
| import org.bukkit.Material; | import org.bukkit.Material; | ||||||
| @@ -40,62 +42,37 @@ import org.bukkit.inventory.InventoryView; | |||||||
| import org.bukkit.inventory.ItemStack; | import org.bukkit.inventory.ItemStack; | ||||||
| import org.bukkit.inventory.PlayerInventory; | import org.bukkit.inventory.PlayerInventory; | ||||||
| import org.bukkit.inventory.meta.ItemMeta; | import org.bukkit.inventory.meta.ItemMeta; | ||||||
|  | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  |  | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.stream.IntStream; | import java.util.stream.IntStream; | ||||||
|  |  | ||||||
|  | @Singleton | ||||||
| public class BukkitInventoryUtil extends InventoryUtil { | public class BukkitInventoryUtil extends InventoryUtil { | ||||||
|  |  | ||||||
|     @Override public void open(PlotInventory inv) { |     private static @Nullable ItemStack getItem(PlotItemStack item) { | ||||||
|         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) { |         if (item == null) { | ||||||
|             return null; |             return null; | ||||||
|         } |         } | ||||||
|         ItemStack stack = new ItemStack(BukkitAdapter.adapt(item.getType()), item.amount); |         Material material = BukkitAdapter.adapt(item.getType()); | ||||||
|         ItemMeta meta = null; |         if (material == null) { | ||||||
|         if (item.name != null) { |             return null; | ||||||
|             meta = stack.getItemMeta(); |  | ||||||
|             meta.setDisplayName(ChatColor.translateAlternateColorCodes('&', item.name)); |  | ||||||
|         } |         } | ||||||
|         if (item.lore != null) { |         ItemStack stack = new ItemStack(material, item.getAmount()); | ||||||
|  |         ItemMeta meta = null; | ||||||
|  |         if (item.getName() != null) { | ||||||
|  |             meta = stack.getItemMeta(); | ||||||
|  |             Component nameComponent = BukkitUtil.MINI_MESSAGE.parse(item.getName()); | ||||||
|  |             meta.setDisplayName(BukkitUtil.LEGACY_COMPONENT_SERIALIZER.serialize(nameComponent)); | ||||||
|  |         } | ||||||
|  |         if (item.getLore() != null) { | ||||||
|             if (meta == null) { |             if (meta == null) { | ||||||
|                 meta = stack.getItemMeta(); |                 meta = stack.getItemMeta(); | ||||||
|             } |             } | ||||||
|             List<String> lore = new ArrayList<>(); |             List<String> lore = new ArrayList<>(); | ||||||
|             for (String entry : item.lore) { |             for (String entry : item.getLore()) { | ||||||
|                 lore.add(ChatColor.translateAlternateColorCodes('&', entry)); |                 lore.add(BukkitUtil.LEGACY_COMPONENT_SERIALIZER.serialize(BukkitUtil.MINI_MESSAGE.deserialize(entry))); | ||||||
|             } |             } | ||||||
|             meta.setLore(lore); |             meta.setLore(lore); | ||||||
|         } |         } | ||||||
| @@ -105,6 +82,49 @@ public class BukkitInventoryUtil extends InventoryUtil { | |||||||
|         return stack; |         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 boolean setItemChecked(PlotInventory inv, int index, PlotItemStack item) { | ||||||
|  |         BukkitPlayer bp = (BukkitPlayer) inv.getPlayer(); | ||||||
|  |         InventoryView opened = bp.player.getOpenInventory(); | ||||||
|  |         ItemStack stack = getItem(item); | ||||||
|  |         if (stack == null) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         if (!inv.isOpen()) { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |         opened.setItem(index, stack); | ||||||
|  |         bp.player.updateInventory(); | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @SuppressWarnings("deprecation") // Paper deprecation | ||||||
|     public PlotItemStack getItem(ItemStack item) { |     public PlotItemStack getItem(ItemStack item) { | ||||||
|         if (item == null) { |         if (item == null) { | ||||||
|             return null; |             return null; | ||||||
| @@ -129,18 +149,21 @@ public class BukkitInventoryUtil extends InventoryUtil { | |||||||
|         return new PlotItemStack(id.name(), amount, name, lore); |         return new PlotItemStack(id.name(), amount, name, lore); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public PlotItemStack[] getItems(PlotPlayer player) { |     @Override | ||||||
|  |     public PlotItemStack[] getItems(PlotPlayer<?> player) { | ||||||
|         BukkitPlayer bp = (BukkitPlayer) player; |         BukkitPlayer bp = (BukkitPlayer) player; | ||||||
|         PlayerInventory inv = bp.player.getInventory(); |         PlayerInventory inv = bp.player.getInventory(); | ||||||
|         return IntStream.range(0, 36).mapToObj(i -> getItem(inv.getItem(i))) |         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()) { |         if (!plotInventory.isOpen()) { | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|         BukkitPlayer bp = (BukkitPlayer) plotInventory.player; |         BukkitPlayer bp = (BukkitPlayer) plotInventory.getPlayer(); | ||||||
|         InventoryView opened = bp.player.getOpenInventory(); |         InventoryView opened = bp.player.getOpenInventory(); | ||||||
|         if (plotInventory.isOpen()) { |         if (plotInventory.isOpen()) { | ||||||
|             if (opened.getType() == InventoryType.CRAFTING) { |             if (opened.getType() == InventoryType.CRAFTING) { | ||||||
| @@ -149,4 +172,5 @@ public class BukkitInventoryUtil extends InventoryUtil { | |||||||
|         } |         } | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,33 +21,31 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.util; | ||||||
|  |  | ||||||
| import com.plotsquared.bukkit.BukkitMain; | import com.google.inject.Inject; | ||||||
| import com.plotsquared.core.PlotSquared; | import com.google.inject.Singleton; | ||||||
| import com.plotsquared.core.generator.AugmentedUtils; | import com.plotsquared.core.generator.AugmentedUtils; | ||||||
|  | import com.plotsquared.core.inject.factory.ProgressSubscriberFactory; | ||||||
| import com.plotsquared.core.location.Location; | import com.plotsquared.core.location.Location; | ||||||
| import com.plotsquared.core.location.PlotLoc; | import com.plotsquared.core.location.PlotLoc; | ||||||
|  | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| import com.plotsquared.core.plot.PlotArea; | import com.plotsquared.core.plot.PlotArea; | ||||||
| import com.plotsquared.core.plot.PlotManager; | import com.plotsquared.core.plot.PlotManager; | ||||||
| import com.plotsquared.core.queue.GlobalBlockQueue; | import com.plotsquared.core.queue.GlobalBlockQueue; | ||||||
| import com.plotsquared.core.queue.LocalBlockQueue; | import com.plotsquared.core.queue.QueueCoordinator; | ||||||
| import com.plotsquared.core.queue.ScopedLocalBlockQueue; | import com.plotsquared.core.queue.ScopedQueueCoordinator; | ||||||
| import com.plotsquared.core.util.ChunkManager; | import com.plotsquared.core.util.ChunkManager; | ||||||
| import com.plotsquared.core.util.MainUtil; |  | ||||||
| import com.plotsquared.core.util.RegionManager; | 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.entity.EntityCategories; | ||||||
| import com.plotsquared.core.util.task.RunnableVal; | 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.BukkitAdapter; | ||||||
| import com.sk89q.worldedit.bukkit.BukkitWorld; | import com.sk89q.worldedit.bukkit.BukkitWorld; | ||||||
| import com.sk89q.worldedit.math.BlockVector2; |  | ||||||
| import com.sk89q.worldedit.regions.CuboidRegion; | 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.BaseBlock; | ||||||
| import com.sk89q.worldedit.world.block.BlockTypes; | import com.sk89q.worldedit.world.block.BlockTypes; | ||||||
| import io.papermc.lib.PaperLib; | import io.papermc.lib.PaperLib; | ||||||
| @@ -56,17 +54,14 @@ import org.bukkit.Chunk; | |||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
| import org.bukkit.entity.Player; | 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.ArrayList; | ||||||
| import java.util.HashSet; | import java.util.HashSet; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Map.Entry; |  | ||||||
| import java.util.Objects; |  | ||||||
| import java.util.Set; | 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_ANIMAL; | ||||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_ENTITY; | 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_MISC; | ||||||
| @@ -74,55 +69,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_MONSTER; | ||||||
| import static com.plotsquared.core.util.entity.EntityCategories.CAP_VEHICLE; | import static com.plotsquared.core.util.entity.EntityCategories.CAP_VEHICLE; | ||||||
|  |  | ||||||
|  | @Singleton | ||||||
| public class BukkitRegionManager extends RegionManager { | public class BukkitRegionManager extends RegionManager { | ||||||
|  |  | ||||||
|     public static boolean isIn(CuboidRegion region, int x, int z) { |     private final GlobalBlockQueue blockQueue; | ||||||
|         return x >= region.getMinimumPoint().getX() && x <= region.getMaximumPoint().getX() |  | ||||||
|             && z >= region.getMinimumPoint().getZ() && z <= region.getMaximumPoint().getZ(); |     @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) { |     @Override | ||||||
|         Set<BlockVector2> chunks = super.getChunkChunks(world); |     public boolean handleClear( | ||||||
|         if (Bukkit.isPrimaryThread()) { |             @NonNull Plot plot, | ||||||
|             for (Chunk chunk : Objects.requireNonNull(Bukkit.getWorld(world)).getLoadedChunks()) { |             @Nullable Runnable whenDone, | ||||||
|                 BlockVector2 loc = BlockVector2.at(chunk.getX() >> 5, chunk.getZ() >> 5); |             @NonNull PlotManager manager, | ||||||
|                 chunks.add(loc); |             @Nullable PlotPlayer<?> player | ||||||
|             } |     ) { | ||||||
|         } 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) { |  | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public int[] countEntities(Plot plot) { |     @Override | ||||||
|  |     public int[] countEntities(@NonNull Plot plot) { | ||||||
|         int[] existing = (int[]) plot.getMeta("EntityCount"); |         int[] existing = (int[]) plot.getMeta("EntityCount"); | ||||||
|         if (existing != null && (System.currentTimeMillis() - (long) plot.getMeta("EntityCountTime") |         if (existing != null && (System.currentTimeMillis() - (long) plot.getMeta("EntityCountTime") < 1000)) { | ||||||
|             < 1000)) { |  | ||||||
|             return existing; |             return existing; | ||||||
|         } |         } | ||||||
|         PlotArea area = plot.getArea(); |         PlotArea area = plot.getArea(); | ||||||
|         World world = BukkitUtil.getWorld(area.getWorldName()); |         World world = BukkitUtil.getWorld(area.getWorldName()); | ||||||
|  |  | ||||||
|         Location bot = plot.getBottomAbs(); |         Location bot = plot.getBottomAbs(); | ||||||
|         Location top = plot.getTopAbs(); |         Location top = plot.getTopAbs(); | ||||||
|         int bx = bot.getX() >> 4; |         int bx = bot.getX() >> 4; | ||||||
| @@ -162,7 +140,7 @@ public class BukkitRegionManager extends RegionManager { | |||||||
|                         if (X > bx && X < tx && Z > bz && Z < tz) { |                         if (X > bx && X < tx && Z > bz && Z < tz) { | ||||||
|                             count(count, entity); |                             count(count, entity); | ||||||
|                         } else { |                         } else { | ||||||
|                             Plot other = area.getPlot(BukkitUtil.getLocation(location)); |                             Plot other = area.getPlot(BukkitUtil.adapt(location)); | ||||||
|                             if (plot.equals(other)) { |                             if (plot.equals(other)) { | ||||||
|                                 count(count, entity); |                                 count(count, entity); | ||||||
|                             } |                             } | ||||||
| @@ -177,7 +155,7 @@ public class BukkitRegionManager extends RegionManager { | |||||||
|                 Entity[] entities1 = chunk.getEntities(); |                 Entity[] entities1 = chunk.getEntities(); | ||||||
|                 for (Entity entity : entities1) { |                 for (Entity entity : entities1) { | ||||||
|                     if (X == bx || X == tx || Z == bz || Z == tz) { |                     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)) { |                         if (plot.equals(other)) { | ||||||
|                             count(count, entity); |                             count(count, entity); | ||||||
|                         } |                         } | ||||||
| @@ -191,64 +169,13 @@ public class BukkitRegionManager extends RegionManager { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean copyRegion(Location pos1, Location pos2, Location newPos, |     public boolean regenerateRegion( | ||||||
|         final Runnable whenDone) { |             final @NonNull Location pos1, | ||||||
|         final int relX = newPos.getX() - pos1.getX(); |             final @NonNull Location pos2, | ||||||
|         final int relZ = newPos.getZ() - pos1.getZ(); |             final boolean ignoreAugment, | ||||||
|  |             final @Nullable Runnable whenDone | ||||||
|         final CuboidRegion region = |     ) { | ||||||
|             RegionUtil.createRegion(pos1.getX(), pos2.getX(), pos1.getZ(), pos2.getZ()); |         final BukkitWorld world = (BukkitWorld) worldUtil.getWeWorld(pos1.getWorldName()); | ||||||
|         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(); |  | ||||||
|  |  | ||||||
|         final int p1x = pos1.getX(); |         final int p1x = pos1.getX(); | ||||||
|         final int p1z = pos1.getZ(); |         final int p1z = pos1.getZ(); | ||||||
| @@ -259,150 +186,132 @@ public class BukkitRegionManager extends RegionManager { | |||||||
|         final int tcx = p2x >> 4; |         final int tcx = p2x >> 4; | ||||||
|         final int tcz = p2z >> 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++) { |             int x = chunk.getX(); | ||||||
|             for (int z = bcz; z <= tcz; z++) { |             int z = chunk.getZ(); | ||||||
|                 chunks.add(BlockVector2.at(x, z)); |             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; | ||||||
|             } |             } | ||||||
|         } |             boolean checkX1 = false; | ||||||
|         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; |  | ||||||
|  |  | ||||||
|                         int xxb2; |             int xxb2; | ||||||
|  |  | ||||||
|                         if (x == bcx) { |             if (x == bcx) { | ||||||
|                             xxb2 = p1x - 1; |                 xxb2 = p1x - 1; | ||||||
|                             checkX1 = true; |                 checkX1 = true; | ||||||
|                         } else { |             } else { | ||||||
|                             xxb2 = xxb; |                 xxb2 = xxb; | ||||||
|                         } |             } | ||||||
|                         boolean checkX2 = false; |             boolean checkX2 = false; | ||||||
|                         int xxt2; |             int xxt2; | ||||||
|                         if (x == tcx) { |             if (x == tcx) { | ||||||
|                             xxt2 = p2x + 1; |                 xxt2 = p2x + 1; | ||||||
|                             checkX2 = true; |                 checkX2 = true; | ||||||
|                         } else { |             } else { | ||||||
|                             xxt2 = xxt; |                 xxt2 = xxt; | ||||||
|                         } |             } | ||||||
|                         boolean checkZ1 = false; |             boolean checkZ1 = false; | ||||||
|                         int zzb2; |             int zzb2; | ||||||
|                         if (z == bcz) { |             if (z == bcz) { | ||||||
|                             zzb2 = p1z - 1; |                 zzb2 = p1z - 1; | ||||||
|                             checkZ1 = true; |                 checkZ1 = true; | ||||||
|                         } else { |             } else { | ||||||
|                             zzb2 = zzb; |                 zzb2 = zzb; | ||||||
|                         } |             } | ||||||
|                         boolean checkZ2 = false; |             boolean checkZ2 = false; | ||||||
|                         int zzt2; |             int zzt2; | ||||||
|                         if (z == tcz) { |             if (z == tcz) { | ||||||
|                             zzt2 = p2z + 1; |                 zzt2 = p2z + 1; | ||||||
|                             checkZ2 = true; |                 checkZ2 = true; | ||||||
|                         } else { |             } else { | ||||||
|                             zzt2 = zzt; |                 zzt2 = zzt; | ||||||
|                         } |             } | ||||||
|                         final ContentMap map = new ContentMap(); |             final ContentMap map = new ContentMap(); | ||||||
|                         if (checkX1) { |             if (checkX1) { | ||||||
|                             map.saveRegion(bukkitWorldObj, xxb, xxb2, zzb2, zzt2); // |                 map.saveRegion(world, xxb, xxb2, zzb2, zzt2); // | ||||||
|                         } |             } | ||||||
|                         if (checkX2) { |             if (checkX2) { | ||||||
|                             map.saveRegion(bukkitWorldObj, xxt2, xxt, zzb2, zzt2); // |                 map.saveRegion(world, xxt2, xxt, zzb2, zzt2); // | ||||||
|                         } |             } | ||||||
|                         if (checkZ1) { |             if (checkZ1) { | ||||||
|                             map.saveRegion(bukkitWorldObj, xxb2, xxt2, zzb, zzb2); // |                 map.saveRegion(world, xxb2, xxt2, zzb, zzb2); // | ||||||
|                         } |             } | ||||||
|                         if (checkZ2) { |             if (checkZ2) { | ||||||
|                             map.saveRegion(bukkitWorldObj, xxb2, xxt2, zzt2, zzt); // |                 map.saveRegion(world, xxb2, xxt2, zzt2, zzt); // | ||||||
|                         } |             } | ||||||
|                         if (checkX1 && checkZ1) { |             if (checkX1 && checkZ1) { | ||||||
|                             map.saveRegion(bukkitWorldObj, xxb, xxb2, zzb, zzb2); // |                 map.saveRegion(world, xxb, xxb2, zzb, zzb2); // | ||||||
|                         } |             } | ||||||
|                         if (checkX2 && checkZ1) { |             if (checkX2 && checkZ1) { | ||||||
|                             map.saveRegion(bukkitWorldObj, xxt2, xxt, zzb, zzb2); // ? |                 map.saveRegion(world, xxt2, xxt, zzb, zzb2); // ? | ||||||
|                         } |             } | ||||||
|                         if (checkX1 && checkZ2) { |             if (checkX1 && checkZ2) { | ||||||
|                             map.saveRegion(bukkitWorldObj, xxb, xxb2, zzt2, zzt); // ? |                 map.saveRegion(world, xxb, xxb2, zzt2, zzt); // ? | ||||||
|                         } |             } | ||||||
|                         if (checkX2 && checkZ2) { |             if (checkX2 && checkZ2) { | ||||||
|                             map.saveRegion(bukkitWorldObj, xxt2, xxt, zzt2, zzt); // |                 map.saveRegion(world, xxt2, xxt, zzt2, zzt); // | ||||||
|                         } |             } | ||||||
|                         CuboidRegion currentPlotClear = RegionUtil |             CuboidRegion currentPlotClear = new CuboidRegion(pos1.getBlockVector3(), pos2.getBlockVector3()); | ||||||
|                             .createRegion(pos1.getX(), pos2.getX(), pos1.getZ(), pos2.getZ()); |             map.saveEntitiesOut(Bukkit.getWorld(world.getName()).getChunkAt(x, z), currentPlotClear); | ||||||
|                         map.saveEntitiesOut(chunkObj, currentPlotClear); |             AugmentedUtils.bypass( | ||||||
|                         AugmentedUtils.bypass(ignoreAugment, () -> ChunkManager |                     ignoreAugment, | ||||||
|                             .setChunkInPlotArea(null, new RunnableVal<ScopedLocalBlockQueue>() { |                     () -> ChunkManager.setChunkInPlotArea(null, new RunnableVal<ScopedQueueCoordinator>() { | ||||||
|                                 @Override public void run(ScopedLocalBlockQueue value) { |                         @Override | ||||||
|                                     Location min = value.getMin(); |                         public void run(ScopedQueueCoordinator value) { | ||||||
|                                     int bx = min.getX(); |                             Location min = value.getMin(); | ||||||
|                                     int bz = min.getZ(); |                             int bx = min.getX(); | ||||||
|                                     for (int x1 = 0; x1 < 16; x1++) { |                             int bz = min.getZ(); | ||||||
|                                         for (int z1 = 0; z1 < 16; z1++) { |                             for (int x1 = 0; x1 < 16; x1++) { | ||||||
|                                             PlotLoc plotLoc = new PlotLoc(bx + x1, bz + z1); |                                 for (int z1 = 0; z1 < 16; z1++) { | ||||||
|                                             BaseBlock[] ids = map.allBlocks.get(plotLoc); |                                     PlotLoc plotLoc = new PlotLoc(bx + x1, bz + z1); | ||||||
|                                             if (ids != null) { |                                     BaseBlock[] ids = map.allBlocks.get(plotLoc); | ||||||
|                                                 for (int y = 0; |                                     if (ids != null) { | ||||||
|                                                      y < Math.min(128, ids.length); y++) { |                                         int minY = value.getMin().getY(); | ||||||
|                                                     BaseBlock id = ids[y]; |                                         for (int yIndex = 0; yIndex < ids.length; yIndex++) { | ||||||
|                                                     if (id != null) { |                                             int y = yIndex + minY; | ||||||
|                                                         value.setBlock(x1, y, z1, id); |                                             BaseBlock id = ids[yIndex]; | ||||||
|                                                     } else { |                                             if (id != null) { | ||||||
|                                                         value.setBlock(x1, y, z1, |                                                 value.setBlock(x1, y, z1, id); | ||||||
|                                                             BlockTypes.AIR.getDefaultState()); |                                             } 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); |                     }, world.getName(), chunk) | ||||||
|                     }); |             ); | ||||||
|                 } |             //map.restoreBlocks(worldObj, 0, 0); | ||||||
|                 if (!chunks.isEmpty()) { |             map.restoreEntities(Bukkit.getWorld(world.getName())); | ||||||
|                     TaskManager.runTaskLater(this, 1); |  | ||||||
|                 } else { |  | ||||||
|                     TaskManager.runTaskLater(whenDone, 1); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         }); |         }); | ||||||
|  |         regenQueue.setCompleteTask(whenDone); | ||||||
|  |         queue.setCompleteTask(regenQueue::enqueue); | ||||||
|  |         queue.enqueue(); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public void clearAllEntities(Location pos1, Location pos2) { |     @Override | ||||||
|         String world = pos1.getWorld(); |     public void clearAllEntities(@NonNull Location pos1, @NonNull Location pos2) { | ||||||
|         List<Entity> entities = BukkitUtil.getEntities(world); |         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 bx = pos1.getX(); | ||||||
|         int bz = pos1.getZ(); |         int bz = pos1.getZ(); | ||||||
|         int tx = pos2.getX(); |         int tx = pos2.getX(); | ||||||
| @@ -410,8 +319,7 @@ public class BukkitRegionManager extends RegionManager { | |||||||
|         for (Entity entity : entities) { |         for (Entity entity : entities) { | ||||||
|             if (!(entity instanceof Player)) { |             if (!(entity instanceof Player)) { | ||||||
|                 org.bukkit.Location location = entity.getLocation(); |                 org.bukkit.Location location = entity.getLocation(); | ||||||
|                 if (location.getX() >= bx && location.getX() <= tx && location.getZ() >= bz |                 if (location.getX() >= bx && location.getX() <= tx && location.getZ() >= bz && location.getZ() <= tz) { | ||||||
|                     && location.getZ() <= tz) { |  | ||||||
|                     if (entity.hasMetadata("ps-tmp-teleport")) { |                     if (entity.hasMetadata("ps-tmp-teleport")) { | ||||||
|                         continue; |                         continue; | ||||||
|                     } |                     } | ||||||
| @@ -421,69 +329,16 @@ public class BukkitRegionManager extends RegionManager { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     private void count(int[] count, @NonNull Entity entity) { | ||||||
|     public void swap(Location bot1, Location top1, Location bot2, Location top2, |         final com.sk89q.worldedit.world.entity.EntityType entityType = BukkitAdapter.adapt(entity.getType()); | ||||||
|         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()); |  | ||||||
|  |  | ||||||
|         if (EntityCategories.PLAYER.contains(entityType)) { |         if (EntityCategories.PLAYER.contains(entityType)) { | ||||||
|             return; |             return; | ||||||
|         } else if (EntityCategories.PROJECTILE.contains(entityType) || EntityCategories.OTHER |         } else if (EntityCategories.PROJECTILE.contains(entityType) || EntityCategories.OTHER.contains(entityType) || EntityCategories.HANGING | ||||||
|             .contains(entityType) || EntityCategories.HANGING.contains(entityType)) { |                 .contains(entityType)) { | ||||||
|             count[CAP_MISC]++; |             count[CAP_MISC]++; | ||||||
|         } else if (EntityCategories.ANIMAL.contains(entityType) || EntityCategories.VILLAGER |         } else if (EntityCategories.ANIMAL.contains(entityType) || EntityCategories.VILLAGER.contains(entityType) || EntityCategories.TAMEABLE | ||||||
|             .contains(entityType) || EntityCategories.TAMEABLE.contains(entityType)) { |                 .contains(entityType)) { | ||||||
|             count[CAP_MOB]++; |             count[CAP_MOB]++; | ||||||
|             count[CAP_ANIMAL]++; |             count[CAP_ANIMAL]++; | ||||||
|         } else if (EntityCategories.VEHICLE.contains(entityType)) { |         } else if (EntityCategories.VEHICLE.contains(entityType)) { | ||||||
| @@ -494,4 +349,5 @@ public class BukkitRegionManager extends RegionManager { | |||||||
|         } |         } | ||||||
|         count[CAP_ENTITY]++; |         count[CAP_ENTITY]++; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,37 +21,62 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.util; | ||||||
|  |  | ||||||
|  | import com.google.inject.Inject; | ||||||
|  | import com.google.inject.Singleton; | ||||||
| import com.plotsquared.bukkit.generator.BukkitPlotGenerator; | import com.plotsquared.bukkit.generator.BukkitPlotGenerator; | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.configuration.ConfigurationNode; | import com.plotsquared.core.configuration.ConfigurationNode; | ||||||
| import com.plotsquared.core.configuration.ConfigurationSection; | import com.plotsquared.core.configuration.ConfigurationSection; | ||||||
|  | import com.plotsquared.core.configuration.file.YamlConfiguration; | ||||||
| import com.plotsquared.core.generator.GeneratorWrapper; | 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.PlotArea; | ||||||
| import com.plotsquared.core.plot.PlotAreaType; | 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.setup.PlotAreaBuilder; | ||||||
| import com.plotsquared.core.util.SetupUtils; | 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.Bukkit; | ||||||
| import org.bukkit.Chunk; | import org.bukkit.Chunk; | ||||||
|  | import org.bukkit.Location; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.entity.Player; | import org.bukkit.entity.Player; | ||||||
| import org.bukkit.generator.ChunkGenerator; | import org.bukkit.generator.ChunkGenerator; | ||||||
| import org.bukkit.plugin.Plugin; | import org.bukkit.plugin.Plugin; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
|  | import java.io.File; | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.Map.Entry; | import java.util.Map.Entry; | ||||||
| import java.util.Objects; | import java.util.Objects; | ||||||
|  |  | ||||||
|  | @Singleton | ||||||
| public class BukkitSetupUtils extends SetupUtils { | public class BukkitSetupUtils extends SetupUtils { | ||||||
|  |  | ||||||
|     @Override public void updateGenerators() { |     private final PlotAreaManager plotAreaManager; | ||||||
|         if (!SetupUtils.generators.isEmpty()) { |     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 (loaded && !SetupUtils.generators.isEmpty() && !force) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         String testWorld = "CheckingPlotSquaredGenerator"; |         String testWorld = "CheckingPlotSquaredGenerator"; | ||||||
| @@ -66,7 +91,7 @@ public class BukkitSetupUtils extends SetupUtils { | |||||||
|                         if (generator instanceof GeneratorWrapper<?>) { |                         if (generator instanceof GeneratorWrapper<?>) { | ||||||
|                             wrapped = (GeneratorWrapper<?>) generator; |                             wrapped = (GeneratorWrapper<?>) generator; | ||||||
|                         } else { |                         } else { | ||||||
|                             wrapped = new BukkitPlotGenerator(testWorld, generator); |                             wrapped = new BukkitPlotGenerator(testWorld, generator, this.plotAreaManager); | ||||||
|                         } |                         } | ||||||
|                         SetupUtils.generators.put(name, wrapped); |                         SetupUtils.generators.put(name, wrapped); | ||||||
|                     } |                     } | ||||||
| @@ -75,156 +100,49 @@ public class BukkitSetupUtils extends SetupUtils { | |||||||
|                 e.printStackTrace(); |                 e.printStackTrace(); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |         loaded = true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public void unload(String worldName, boolean save) { |     @Override | ||||||
|         World world = Bukkit.getWorld(worldName); |     public void unload(String worldName, boolean save) { | ||||||
|         if (world == null) { |         TaskManager.runTask(() -> { | ||||||
|             return; |             World world = Bukkit.getWorld(worldName); | ||||||
|         } |             if (world == null) { | ||||||
|         World dw = Bukkit.getWorlds().get(0); |                 return; | ||||||
|         for (Player player : world.getPlayers()) { |  | ||||||
|             PaperLib.teleportAsync(player, dw.getSpawnLocation()); |  | ||||||
|         } |  | ||||||
|         if (save) { |  | ||||||
|             for (Chunk chunk : world.getLoadedChunks()) { |  | ||||||
|                 chunk.unload(true); |  | ||||||
|             } |             } | ||||||
|         } else { |             Location location = Bukkit.getWorlds().get(0).getSpawnLocation(); | ||||||
|             for (Chunk chunk : world.getLoadedChunks()) { |             for (Player player : world.getPlayers()) { | ||||||
|                 chunk.unload(false); |                 player.teleport(location); | ||||||
|             } |             } | ||||||
|         } |             if (save) { | ||||||
|         Bukkit.unloadWorld(world, false); |                 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) { |     @Override | ||||||
|         SetupUtils.manager.updateGenerators(); |     public String setupWorld(PlotAreaBuilder builder) { | ||||||
|         ConfigurationNode[] steps = object.step == null ? new ConfigurationNode[0] : object.step; |         this.updateGenerators(false); | ||||||
|         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(); |  | ||||||
|         ConfigurationNode[] steps = builder.settingsNodesWrapper() == null ? |         ConfigurationNode[] steps = builder.settingsNodesWrapper() == null ? | ||||||
|                 new ConfigurationNode[0] : builder.settingsNodesWrapper().getSettingsNodes(); |                 new ConfigurationNode[0] : builder.settingsNodesWrapper().getSettingsNodes(); | ||||||
|         String world = builder.worldName(); |         String world = builder.worldName(); | ||||||
|         PlotAreaType type = builder.plotAreaType(); |         PlotAreaType type = builder.plotAreaType(); | ||||||
|         String worldPath = "worlds." + builder.worldName(); |         String worldPath = "worlds." + builder.worldName(); | ||||||
|         switch (type) { |         switch (type) { | ||||||
|             case PARTIAL: { |             case PARTIAL -> { | ||||||
|                 if (builder.areaName() != null) { |                 if (builder.areaName() != null) { | ||||||
|                     if (!PlotSquared.get().worlds.contains(worldPath)) { |                     if (!this.worldConfiguration.contains(worldPath)) { | ||||||
|                         PlotSquared.get().worlds.createSection(worldPath); |                         this.worldConfiguration.createSection(worldPath); | ||||||
|                     } |                     } | ||||||
|                     ConfigurationSection worldSection = |                     ConfigurationSection worldSection = | ||||||
|                             PlotSquared.get().worlds.getConfigurationSection(worldPath); |                             this.worldConfiguration.getConfigurationSection(worldPath); | ||||||
|                     String areaName = builder.areaName() + "-" + builder.minimumId() + "-" + builder.maximumId(); |                     String areaName = builder.areaName() + "-" + builder.minimumId() + "-" + builder.maximumId(); | ||||||
|                     String areaPath = "areas." + areaName; |                     String areaPath = "areas." + areaName; | ||||||
|                     if (!worldSection.contains(areaPath)) { |                     if (!worldSection.contains(areaPath)) { | ||||||
| @@ -260,29 +178,28 @@ public class BukkitSetupUtils extends SetupUtils { | |||||||
|                 if (gen != null && gen.isFull()) { |                 if (gen != null && gen.isFull()) { | ||||||
|                     builder.generatorName(null); |                     builder.generatorName(null); | ||||||
|                 } |                 } | ||||||
|                 break; |  | ||||||
|             } |             } | ||||||
|             case AUGMENTED: { |             case AUGMENTED -> { | ||||||
|                 if (!builder.plotManager().endsWith(":single")) { |                 if (!builder.plotManager().endsWith(":single")) { | ||||||
|                     if (!PlotSquared.get().worlds.contains(worldPath)) { |                     if (!this.worldConfiguration.contains(worldPath)) { | ||||||
|                         PlotSquared.get().worlds.createSection(worldPath); |                         this.worldConfiguration.createSection(worldPath); | ||||||
|                     } |                     } | ||||||
|                     if (steps.length != 0) { |                     if (steps.length != 0) { | ||||||
|                         ConfigurationSection worldSection = |                         ConfigurationSection worldSection = | ||||||
|                                 PlotSquared.get().worlds.getConfigurationSection(worldPath); |                                 this.worldConfiguration.getConfigurationSection(worldPath); | ||||||
|                         for (ConfigurationNode step : steps) { |                         for (ConfigurationNode step : steps) { | ||||||
|                             worldSection.set(step.getConstant(), step.getValue()); |                             worldSection.set(step.getConstant(), step.getValue()); | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                     PlotSquared.get().worlds |                     this.worldConfiguration | ||||||
|                             .set("worlds." + world + ".generator.type", builder.plotAreaType().toString()); |                             .set("worlds." + world + ".generator.type", builder.plotAreaType().toString()); | ||||||
|                     PlotSquared.get().worlds |                     this.worldConfiguration | ||||||
|                             .set("worlds." + world + ".generator.terrain", builder.terrainType().toString()); |                             .set("worlds." + world + ".generator.terrain", builder.terrainType().toString()); | ||||||
|                     PlotSquared.get().worlds |                     this.worldConfiguration | ||||||
|                             .set("worlds." + world + ".generator.plugin", builder.plotManager()); |                             .set("worlds." + world + ".generator.plugin", builder.plotManager()); | ||||||
|                     if (builder.generatorName() != null && !builder.generatorName() |                     if (builder.generatorName() != null && !builder.generatorName() | ||||||
|                             .equals(builder.plotManager())) { |                             .equals(builder.plotManager())) { | ||||||
|                         PlotSquared.get().worlds |                         this.worldConfiguration | ||||||
|                                 .set("worlds." + world + ".generator.init", builder.generatorName()); |                                 .set("worlds." + world + ".generator.init", builder.generatorName()); | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
| @@ -290,30 +207,28 @@ public class BukkitSetupUtils extends SetupUtils { | |||||||
|                 if (gen != null && gen.isFull()) { |                 if (gen != null && gen.isFull()) { | ||||||
|                     builder.generatorName(null); |                     builder.generatorName(null); | ||||||
|                 } |                 } | ||||||
|                 break; |  | ||||||
|             } |             } | ||||||
|             case NORMAL: { |             case NORMAL -> { | ||||||
|                 if (steps.length != 0) { |                 if (steps.length != 0) { | ||||||
|                     if (!PlotSquared.get().worlds.contains(worldPath)) { |                     if (!this.worldConfiguration.contains(worldPath)) { | ||||||
|                         PlotSquared.get().worlds.createSection(worldPath); |                         this.worldConfiguration.createSection(worldPath); | ||||||
|                     } |                     } | ||||||
|                     ConfigurationSection worldSection = |                     ConfigurationSection worldSection = | ||||||
|                             PlotSquared.get().worlds.getConfigurationSection(worldPath); |                             this.worldConfiguration.getConfigurationSection(worldPath); | ||||||
|                     for (ConfigurationNode step : steps) { |                     for (ConfigurationNode step : steps) { | ||||||
|                         worldSection.set(step.getConstant(), step.getValue()); |                         worldSection.set(step.getConstant(), step.getValue()); | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|                 break; |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         try { |         try { | ||||||
|             PlotSquared.get().worlds.save(PlotSquared.get().worldsFile); |             this.worldConfiguration.save(this.worldFile); | ||||||
|         } catch (IOException e) { |         } catch (IOException e) { | ||||||
|             e.printStackTrace(); |             e.printStackTrace(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Objects.requireNonNull(PlotSquared.imp()).getWorldManager() |         Objects.requireNonNull(PlotSquared.platform()).worldManager() | ||||||
|                 .handleWorldCreation(builder.worldName(), builder.generatorName()); |                 .handleWorldCreation(builder.worldName(), builder.generatorName()); | ||||||
|  |  | ||||||
|         if (Bukkit.getWorld(world) != null) { |         if (Bukkit.getWorld(world) != null) { | ||||||
| @@ -323,9 +238,10 @@ public class BukkitSetupUtils extends SetupUtils { | |||||||
|         return builder.worldName(); |         return builder.worldName(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public String getGenerator(PlotArea plotArea) { |     @Override | ||||||
|  |     public String getGenerator(PlotArea plotArea) { | ||||||
|         if (SetupUtils.generators.isEmpty()) { |         if (SetupUtils.generators.isEmpty()) { | ||||||
|             updateGenerators(); |             updateGenerators(false); | ||||||
|         } |         } | ||||||
|         World world = Bukkit.getWorld(plotArea.getWorldName()); |         World world = Bukkit.getWorld(plotArea.getWorldName()); | ||||||
|         if (world == null) { |         if (world == null) { | ||||||
| @@ -343,4 +259,5 @@ public class BukkitSetupUtils extends SetupUtils { | |||||||
|         } |         } | ||||||
|         return null; |         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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,43 +21,45 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | 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.BukkitPlayer; | ||||||
| import com.plotsquared.bukkit.player.BukkitPlayerManager; | import com.plotsquared.bukkit.player.BukkitPlayerManager; | ||||||
| import com.plotsquared.core.PlotSquared; | 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.location.Location; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.PlotArea; | ||||||
| import com.plotsquared.core.util.BlockUtil; | import com.plotsquared.core.util.BlockUtil; | ||||||
| import com.plotsquared.core.util.MainUtil; |  | ||||||
| import com.plotsquared.core.util.MathMan; | import com.plotsquared.core.util.MathMan; | ||||||
| import com.plotsquared.core.util.PlayerManager; | import com.plotsquared.core.util.PlayerManager; | ||||||
| import com.plotsquared.core.util.StringComparison; | import com.plotsquared.core.util.StringComparison; | ||||||
| import com.plotsquared.core.util.WorldUtil; | import com.plotsquared.core.util.WorldUtil; | ||||||
| import com.plotsquared.core.util.task.RunnableVal; |  | ||||||
| import com.plotsquared.core.util.task.TaskManager; | import com.plotsquared.core.util.task.TaskManager; | ||||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||||
| import com.sk89q.worldedit.bukkit.BukkitWorld; | import com.sk89q.worldedit.bukkit.BukkitWorld; | ||||||
| import com.sk89q.worldedit.math.BlockVector2; | import com.sk89q.worldedit.math.BlockVector2; | ||||||
| import com.sk89q.worldedit.regions.CuboidRegion; |  | ||||||
| import com.sk89q.worldedit.world.biome.BiomeType; | import com.sk89q.worldedit.world.biome.BiomeType; | ||||||
| import com.sk89q.worldedit.world.block.BlockCategories; | import com.sk89q.worldedit.world.block.BlockCategories; | ||||||
| import com.sk89q.worldedit.world.block.BlockState; | import com.sk89q.worldedit.world.block.BlockState; | ||||||
| import com.sk89q.worldedit.world.block.BlockType; | import com.sk89q.worldedit.world.block.BlockType; | ||||||
| import com.sk89q.worldedit.world.block.BlockTypes; | import com.sk89q.worldedit.world.block.BlockTypes; | ||||||
| import io.papermc.lib.PaperLib; | 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.Bukkit; | ||||||
| import org.bukkit.Chunk; | import org.bukkit.Chunk; | ||||||
| import org.bukkit.Material; | import org.bukkit.Material; | ||||||
| import org.bukkit.OfflinePlayer; |  | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.block.Biome; |  | ||||||
| import org.bukkit.block.Block; | import org.bukkit.block.Block; | ||||||
| import org.bukkit.block.BlockFace; | import org.bukkit.block.BlockFace; | ||||||
| import org.bukkit.block.Sign; | import org.bukkit.block.Sign; | ||||||
| @@ -93,268 +95,158 @@ import org.bukkit.entity.Snowman; | |||||||
| import org.bukkit.entity.Tameable; | import org.bukkit.entity.Tameable; | ||||||
| import org.bukkit.entity.Vehicle; | import org.bukkit.entity.Vehicle; | ||||||
| import org.bukkit.entity.WaterMob; | import org.bukkit.entity.WaterMob; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.index.qual.NonNegative; | ||||||
| import org.jetbrains.annotations.Nullable; | 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.Collection; | ||||||
| import java.util.HashSet; | import java.util.HashSet; | ||||||
| import java.util.List; |  | ||||||
| import java.util.Objects; | import java.util.Objects; | ||||||
| import java.util.Set; | import java.util.Set; | ||||||
| import java.util.UUID; | import java.util.concurrent.Semaphore; | ||||||
| import java.util.function.Consumer; | import java.util.function.Consumer; | ||||||
| import java.util.function.IntConsumer; | import java.util.function.IntConsumer; | ||||||
| import java.util.stream.Stream; | import java.util.stream.Stream; | ||||||
|  |  | ||||||
| @SuppressWarnings({"unused", "WeakerAccess"}) | @SuppressWarnings({"unused", "WeakerAccess"}) | ||||||
|  | @Singleton | ||||||
| public class BukkitUtil extends WorldUtil { | public class BukkitUtil extends WorldUtil { | ||||||
|  |  | ||||||
|     private static String lastString = null; |     public static final BukkitAudiences BUKKIT_AUDIENCES = BukkitAudiences.create(BukkitPlatform.getPlugin(BukkitPlatform.class)); | ||||||
|     private static World lastWorld = null; |     public static final LegacyComponentSerializer LEGACY_COMPONENT_SERIALIZER = LegacyComponentSerializer.legacySection(); | ||||||
|  |     public static final MiniMessage MINI_MESSAGE = MiniMessage.builder().build(); | ||||||
|     private static Player lastPlayer = null; |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + BukkitUtil.class.getSimpleName()); | ||||||
|     private static BukkitPlayer lastPlotPlayer = null; |     private final Collection<BlockType> tileEntityTypes = new HashSet<>(); | ||||||
|  |  | ||||||
|     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); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get a plot based on the location. |      * Turn a Bukkit {@link Player} into a PlotSquared {@link PlotPlayer} | ||||||
|      * |      * | ||||||
|      * @param location the location to check |      * @param player Bukkit player | ||||||
|      * @return plot if found, otherwise it creates a temporary plot |      * @return PlotSquared player | ||||||
|      * @see Plot |  | ||||||
|      */ |      */ | ||||||
|     public static Plot getPlot(org.bukkit.Location location) { |     public static @NonNull BukkitPlayer adapt(final @NonNull Player player) { | ||||||
|         if (location == null) { |         final PlayerManager<?, ?> playerManager = PlotSquared.platform().playerManager(); | ||||||
|             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(); |  | ||||||
|         return ((BukkitPlayerManager) playerManager).getPlayer(player); |         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()), |      * Turn a Bukkit {@link org.bukkit.Location} into a PlotSquared {@link Location}. | ||||||
|             MathMan.roundInt(location.getY()), MathMan.roundInt(location.getZ())); |      * 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()), |      * Turn a Bukkit {@link org.bukkit.Location} into a PlotSquared {@link Location}. | ||||||
|             MathMan.roundInt(location.getY()), MathMan.roundInt(location.getZ()), location.getYaw(), |      * This copies the entire 6-tuple (world,x,y,z,yaw,pitch). | ||||||
|             location.getPitch()); |      * | ||||||
|  |      * @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(), |      * Turn a PlotSquared {@link Location} into a Bukkit {@link org.bukkit.Location}. | ||||||
|             location.getY(), location.getZ()); |      * 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); |         return Bukkit.getWorld(string); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static String getWorld(@NonNull final Entity entity) { |     private static void ensureLoaded( | ||||||
|         return entity.getWorld().getName(); |             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) { |     private static void ensureLoaded(final @NonNull Location location, final @NonNull Consumer<Chunk> chunkConsumer) { | ||||||
|         World world = getWorld(worldName); |         PaperLib.getChunkAtAsync(adapt(location), true).thenAccept(chunk -> ensureMainThread(chunkConsumer, chunk)); | ||||||
|         if (world != null) { |     } | ||||||
|             return world.getEntities(); |  | ||||||
|  |     private static <T> void ensureMainThread(final @NonNull Consumer<T> consumer, final @NonNull T value) { | ||||||
|  |         if (Bukkit.isPrimaryThread()) { | ||||||
|  |             consumer.accept(value); | ||||||
|         } else { |         } 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 |     @Override | ||||||
|     public void getHighestBlock(@NonNull final String world, final int x, final int z, |     public boolean isBlockSame(final @NonNull BlockState block1, final @NonNull BlockState block2) { | ||||||
|         final IntConsumer result) { |         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 -> { |         ensureLoaded(world, x, z, chunk -> { | ||||||
|             final World bukkitWorld = getWorld(world); |             final World bukkitWorld = Objects.requireNonNull(getWorld(world)); | ||||||
|             // Skip top and bottom block |             // Skip top and bottom block | ||||||
|             int air = 1; |             int air = 1; | ||||||
|             for (int y = bukkitWorld.getMaxHeight() - 1; y >= 0; y--) { |             int maxY = com.plotsquared.bukkit.util.BukkitWorld.getMaxWorldHeight(bukkitWorld); | ||||||
|  |             int minY = com.plotsquared.bukkit.util.BukkitWorld.getMinWorldHeight(bukkitWorld); | ||||||
|  |             for (int y = maxY - 1; y >= minY; y--) { | ||||||
|                 Block block = bukkitWorld.getBlockAt(x, y, z); |                 Block block = bukkitWorld.getBlockAt(x, y, z); | ||||||
|                 Material type = block.getType(); |                 Material type = block.getType(); | ||||||
|                 if (type.isSolid()) { |                 if (type.isSolid()) { | ||||||
| @@ -375,11 +267,15 @@ public class BukkitUtil extends WorldUtil { | |||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public int getHighestBlockSynchronous(String world, int x, int z) { |     @Override | ||||||
|         final World bukkitWorld = getWorld(world); |     @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 |         // Skip top and bottom block | ||||||
|         int air = 1; |         int air = 1; | ||||||
|         for (int y = bukkitWorld.getMaxHeight() - 1; y >= 0; y--) { |         int maxY = com.plotsquared.bukkit.util.BukkitWorld.getMaxWorldHeight(bukkitWorld); | ||||||
|  |         int minY = com.plotsquared.bukkit.util.BukkitWorld.getMinWorldHeight(bukkitWorld); | ||||||
|  |         for (int y = maxY - 1; y >= minY; y--) { | ||||||
|             Block block = bukkitWorld.getBlockAt(x, y, z); |             Block block = bukkitWorld.getBlockAt(x, y, z); | ||||||
|             Material type = block.getType(); |             Material type = block.getType(); | ||||||
|             if (type.isSolid()) { |             if (type.isSolid()) { | ||||||
| @@ -398,73 +294,75 @@ public class BukkitUtil extends WorldUtil { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void getSign(@NonNull final Location location, final Consumer<String[]> result) { |     public @NonNull String[] getSignSynchronous(final @NonNull Location location) { | ||||||
|         ensureLoaded(location, chunk -> { |         Block block = Objects.requireNonNull(getWorld(location.getWorldName())).getBlockAt( | ||||||
|             final Block block = chunk.getWorld().getBlockAt(getLocation(location)); |                 location.getX(), | ||||||
|             if (block.getState() instanceof Sign) { |                 location.getY(), | ||||||
|                 Sign sign = (Sign) block.getState(); |                 location.getZ() | ||||||
|                 result.accept(sign.getLines()); |         ); | ||||||
|             } |         try { | ||||||
|         }); |             return TaskManager.getPlatformImplementation().sync(() -> { | ||||||
|     } |                 if (block.getState() instanceof Sign sign) { | ||||||
|  |                     return 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(); |  | ||||||
|                 } |                 } | ||||||
|             } |                 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(); |         final org.bukkit.Location temp = getWorld(world).getSpawnLocation(); | ||||||
|         return new Location(world, temp.getBlockX(), temp.getBlockY(), temp.getBlockZ(), |         return Location.at(world, temp.getBlockX(), temp.getBlockY(), temp.getBlockZ(), temp.getYaw(), temp.getPitch()); | ||||||
|             temp.getYaw(), temp.getPitch()); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public void setSpawn(@NonNull final Location location) { |     @Override | ||||||
|         final World world = getWorld(location.getWorld()); |     public void setSpawn(final @NonNull Location location) { | ||||||
|  |         final World world = getWorld(location.getWorldName()); | ||||||
|         if (world != null) { |         if (world != null) { | ||||||
|             world.setSpawnLocation(location.getX(), location.getY(), location.getZ()); |             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); |         final World world = getWorld(worldName); | ||||||
|         if (world != null) { |         if (world != null) { | ||||||
|             world.save(); |             world.save(); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override @SuppressWarnings("deprecation") |     @Override | ||||||
|     public void setSign(@NonNull final String worldName, final int x, final int y, final int z, |     @SuppressWarnings("deprecation") | ||||||
|         @NonNull final String[] lines) { |     public void setSign( | ||||||
|         ensureLoaded(worldName, x, z, chunk -> { |             final @NonNull Location location, final @NonNull Caption[] lines, | ||||||
|             final World world = getWorld(worldName); |             final @NonNull Template... replacements | ||||||
|             final Block block = world.getBlockAt(x, y, z); |     ) { | ||||||
|             //        block.setType(Material.AIR); |         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(); |             final Material type = block.getType(); | ||||||
|             if (type != Material.LEGACY_SIGN && type != Material.LEGACY_WALL_SIGN) { |             if (type != Material.LEGACY_SIGN && type != Material.LEGACY_WALL_SIGN) { | ||||||
|                 BlockFace facing = BlockFace.EAST; |                 BlockFace facing = BlockFace.NORTH; | ||||||
|                 if (world.getBlockAt(x, y, z + 1).getType().isSolid()) { |                 if (!world.getBlockAt(location.getX(), location.getY(), location.getZ() + 1).getType().isSolid()) { | ||||||
|                     facing = BlockFace.NORTH; |                     if (world.getBlockAt(location.getX() - 1, location.getY(), location.getZ()).getType().isSolid()) { | ||||||
|                 } else if (world.getBlockAt(x + 1, y, z).getType().isSolid()) { |                         facing = BlockFace.EAST; | ||||||
|                     facing = BlockFace.WEST; |                     } else if (world.getBlockAt(location.getX() + 1, location.getY(), location.getZ()).getType().isSolid()) { | ||||||
|                 } else if (world.getBlockAt(x, y, z - 1).getType().isSolid()) { |                         facing = BlockFace.WEST; | ||||||
|                     facing = BlockFace.SOUTH; |                     } else if (world.getBlockAt(location.getX(), location.getY(), location.getZ() - 1).getType().isSolid()) { | ||||||
|  |                         facing = BlockFace.SOUTH; | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|                 if (PlotSquared.get().IMP.getServerVersion()[1] == 13) { |                 if (PlotSquared.platform().serverVersion()[1] == 13) { | ||||||
|                     block.setType(Material.valueOf("WALL_SIGN"), false); |                     block.setType(Material.valueOf(area.legacySignMaterial()), false); | ||||||
|                 } else { |                 } else { | ||||||
|                     block.setType(Material.valueOf("OAK_WALL_SIGN"), false); |                     block.setType(Material.valueOf(area.signMaterial()), false); | ||||||
|                 } |                 } | ||||||
|                 if (!(block.getBlockData() instanceof WallSign)) { |                 if (!(block.getBlockData() instanceof WallSign)) { | ||||||
|                     PlotSquared.debug(block.getBlockData().getAsString()); |  | ||||||
|                     throw new RuntimeException("Something went wrong generating a sign"); |                     throw new RuntimeException("Something went wrong generating a sign"); | ||||||
|                 } |                 } | ||||||
|                 final Directional sign = (Directional) block.getBlockData(); |                 final Directional sign = (Directional) block.getBlockData(); | ||||||
| @@ -472,114 +370,84 @@ public class BukkitUtil extends WorldUtil { | |||||||
|                 block.setBlockData(sign, false); |                 block.setBlockData(sign, false); | ||||||
|             } |             } | ||||||
|             final org.bukkit.block.BlockState blockstate = block.getState(); |             final org.bukkit.block.BlockState blockstate = block.getState(); | ||||||
|             if (blockstate instanceof Sign) { |             if (blockstate instanceof final Sign sign) { | ||||||
|                 final Sign sign = (Sign) blockstate; |  | ||||||
|                 for (int i = 0; i < lines.length; i++) { |                 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); |                 sign.update(true, false); | ||||||
|             } |             } | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public boolean isBlockSolid(@NonNull final BlockState block) { |     @Override | ||||||
|         return block.getBlockType().getMaterial().isSolid(); |     public @NonNull StringComparison<BlockState>.ComparisonResult getClosestBlock(@NonNull String name) { | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @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) { |  | ||||||
|         BlockState state = BlockUtil.get(name); |         BlockState state = BlockUtil.get(name); | ||||||
|         return new StringComparison<BlockState>().new ComparisonResult(1, state); |         return new StringComparison<BlockState>().new ComparisonResult(1, state); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void setBiomes(@NonNull final String worldName, @NonNull final CuboidRegion region, |     public com.sk89q.worldedit.world.@NonNull World getWeWorld(final @NonNull String world) { | ||||||
|         @NonNull final BiomeType biomeType) { |  | ||||||
|         final World world = getWorld(worldName); |  | ||||||
|         if (world == null) { |  | ||||||
|             PlotSquared.log("An error occurred setting the biome because the world was null."); |  | ||||||
|             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++) { |  | ||||||
|                 if (world.getBiome(x, z) != biome) { |  | ||||||
|                     world.setBiome(x, z, biome); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public com.sk89q.worldedit.world.World getWeWorld(String world) { |  | ||||||
|         return new BukkitWorld(Bukkit.getWorld(world)); |         return new BukkitWorld(Bukkit.getWorld(world)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @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 -> { |         ensureLoaded(location, chunk -> { | ||||||
|             final World world = getWorld(location.getWorld()); |             final World world = getWorld(location.getWorldName()); | ||||||
|             final Block block = world.getBlockAt(location.getX(), location.getY(), location.getZ()); |             final Block block = Objects.requireNonNull(world).getBlockAt(location.getX(), location.getY(), location.getZ()); | ||||||
|             result.accept(BukkitAdapter.asBlockType(block.getType()).getDefaultState()); |             result.accept(Objects.requireNonNull(BukkitAdapter.asBlockType(block.getType())).getDefaultState()); | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public BlockState getBlockSynchronous(@NonNull final Location location) { |     @Override | ||||||
|         final World world = getWorld(location.getWorld()); |     public @NonNull BlockState getBlockSynchronous(final @NonNull Location location) { | ||||||
|         final Block block = world.getBlockAt(location.getX(), location.getY(), location.getZ()); |         final World world = getWorld(location.getWorldName()); | ||||||
|         return BukkitAdapter.asBlockType(block.getType()).getDefaultState(); |         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() { |     @Override | ||||||
|         return Bukkit.getWorlds().get(0).getName(); |     @NonNegative | ||||||
|  |     public double getHealth(final @NonNull PlotPlayer<?> player) { | ||||||
|  |         return Objects.requireNonNull(Bukkit.getPlayer(player.getUUID())).getHealth(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public double getHealth(PlotPlayer player) { |     @Override | ||||||
|         return Bukkit.getPlayer(player.getUUID()).getHealth(); |     @NonNegative | ||||||
|  |     public int getFoodLevel(final @NonNull PlotPlayer<?> player) { | ||||||
|  |         return Objects.requireNonNull(Bukkit.getPlayer(player.getUUID())).getFoodLevel(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public int getFoodLevel(PlotPlayer player) { |     @Override | ||||||
|         return Bukkit.getPlayer(player.getUUID()).getFoodLevel(); |     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) { |     @Override | ||||||
|         Bukkit.getPlayer(player.getUUID()).setHealth(health); |     public void setFoodLevel(final @NonNull PlotPlayer<?> player, @NonNegative final int foodLevel) { | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override public void setFoodLevel(PlotPlayer player, int foodLevel) { |  | ||||||
|         Bukkit.getPlayer(player.getUUID()).setFoodLevel(foodLevel); |         Bukkit.getPlayer(player.getUUID()).setFoodLevel(foodLevel); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public Set<com.sk89q.worldedit.world.entity.EntityType> getTypesInCategory( |     public @NonNull Set<com.sk89q.worldedit.world.entity.EntityType> getTypesInCategory(final @NonNull String category) { | ||||||
|         final String category) { |  | ||||||
|         final Collection<Class<?>> allowedInterfaces = new HashSet<>(); |         final Collection<Class<?>> allowedInterfaces = new HashSet<>(); | ||||||
|         switch (category) { |         switch (category) { | ||||||
|             case "animal": { |             case "animal" -> { | ||||||
|                 allowedInterfaces.add(IronGolem.class); |                 allowedInterfaces.add(IronGolem.class); | ||||||
|                 allowedInterfaces.add(Snowman.class); |                 allowedInterfaces.add(Snowman.class); | ||||||
|                 allowedInterfaces.add(Animals.class); |                 allowedInterfaces.add(Animals.class); | ||||||
|                 allowedInterfaces.add(WaterMob.class); |                 allowedInterfaces.add(WaterMob.class); | ||||||
|                 allowedInterfaces.add(Ambient.class); |                 allowedInterfaces.add(Ambient.class); | ||||||
|             } |             } | ||||||
|             break; |             case "tameable" -> allowedInterfaces.add(Tameable.class); | ||||||
|             case "tameable": { |             case "vehicle" -> allowedInterfaces.add(Vehicle.class); | ||||||
|                 allowedInterfaces.add(Tameable.class); |             case "hostile" -> { | ||||||
|             } |  | ||||||
|             break; |  | ||||||
|             case "vehicle": { |  | ||||||
|                 allowedInterfaces.add(Vehicle.class); |  | ||||||
|             } |  | ||||||
|             break; |  | ||||||
|             case "hostile": { |  | ||||||
|                 allowedInterfaces.add(Shulker.class); |                 allowedInterfaces.add(Shulker.class); | ||||||
|                 allowedInterfaces.add(Monster.class); |                 allowedInterfaces.add(Monster.class); | ||||||
|                 allowedInterfaces.add(Boss.class); |                 allowedInterfaces.add(Boss.class); | ||||||
| @@ -588,20 +456,10 @@ public class BukkitUtil extends WorldUtil { | |||||||
|                 allowedInterfaces.add(Phantom.class); |                 allowedInterfaces.add(Phantom.class); | ||||||
|                 allowedInterfaces.add(EnderCrystal.class); |                 allowedInterfaces.add(EnderCrystal.class); | ||||||
|             } |             } | ||||||
|             break; |             case "hanging" -> allowedInterfaces.add(Hanging.class); | ||||||
|             case "hanging": { |             case "villager" -> allowedInterfaces.add(NPC.class); | ||||||
|                 allowedInterfaces.add(Hanging.class); |             case "projectile" -> allowedInterfaces.add(Projectile.class); | ||||||
|             } |             case "other" -> { | ||||||
|             break; |  | ||||||
|             case "villager": { |  | ||||||
|                 allowedInterfaces.add(NPC.class); |  | ||||||
|             } |  | ||||||
|             break; |  | ||||||
|             case "projectile": { |  | ||||||
|                 allowedInterfaces.add(Projectile.class); |  | ||||||
|             } |  | ||||||
|             break; |  | ||||||
|             case "other": { |  | ||||||
|                 allowedInterfaces.add(ArmorStand.class); |                 allowedInterfaces.add(ArmorStand.class); | ||||||
|                 allowedInterfaces.add(FallingBlock.class); |                 allowedInterfaces.add(FallingBlock.class); | ||||||
|                 allowedInterfaces.add(Item.class); |                 allowedInterfaces.add(Item.class); | ||||||
| @@ -613,15 +471,8 @@ public class BukkitUtil extends WorldUtil { | |||||||
|                 allowedInterfaces.add(EnderSignal.class); |                 allowedInterfaces.add(EnderSignal.class); | ||||||
|                 allowedInterfaces.add(Firework.class); |                 allowedInterfaces.add(Firework.class); | ||||||
|             } |             } | ||||||
|             break; |             case "player" -> allowedInterfaces.add(Player.class); | ||||||
|             case "player": { |             default -> LOGGER.error("Unknown entity category requested: {}", category); | ||||||
|                 allowedInterfaces.add(Player.class); |  | ||||||
|             } |  | ||||||
|             break; |  | ||||||
|             default: { |  | ||||||
|                 PlotSquared.log(Captions.PREFIX + "Unknown entity category requested: " + category); |  | ||||||
|             } |  | ||||||
|             break; |  | ||||||
|         } |         } | ||||||
|         final Set<com.sk89q.worldedit.world.entity.EntityType> types = new HashSet<>(); |         final Set<com.sk89q.worldedit.world.entity.EntityType> types = new HashSet<>(); | ||||||
|         outer: |         outer: | ||||||
| @@ -640,8 +491,8 @@ public class BukkitUtil extends WorldUtil { | |||||||
|         return types; |         return types; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private final Collection<BlockType> tileEntityTypes = new HashSet<>(); |     @Override | ||||||
|     @Override public Collection<BlockType> getTileEntityTypes() { |     public @NonNull Collection<BlockType> getTileEntityTypes() { | ||||||
|         if (this.tileEntityTypes.isEmpty()) { |         if (this.tileEntityTypes.isEmpty()) { | ||||||
|             // Categories |             // Categories | ||||||
|             tileEntityTypes.addAll(BlockCategories.BANNERS.getAll()); |             tileEntityTypes.addAll(BlockCategories.BANNERS.getAll()); | ||||||
| @@ -650,46 +501,83 @@ public class BukkitUtil extends WorldUtil { | |||||||
|             tileEntityTypes.addAll(BlockCategories.FLOWER_POTS.getAll()); |             tileEntityTypes.addAll(BlockCategories.FLOWER_POTS.getAll()); | ||||||
|             // Individual Types |             // Individual Types | ||||||
|             // Add these from strings |             // Add these from strings | ||||||
|             Stream.of("barrel", "beacon", "beehive", "bee_nest", "bell", "blast_furnace", |             Stream.of( | ||||||
|                 "brewing_stand", "campfire", "chest", "ender_chest", "trapped_chest", |                             "barrel", | ||||||
|                 "command_block", "end_gateway", "hopper", "jigsaw", "jubekox", |                             "beacon", | ||||||
|                 "lectern", "note_block", "black_shulker_box", "blue_shulker_box", |                             "beehive", | ||||||
|                 "brown_shulker_box", "cyan_shulker_box", "gray_shulker_box", "green_shulker_box", |                             "bee_nest", | ||||||
|                 "light_blue_shulker_box", "light_gray_shulker_box", "lime_shulker_box", |                             "bell", | ||||||
|                 "magenta_shulker_box", "orange_shulker_box", "pink_shulker_box", |                             "blast_furnace", | ||||||
|                 "purple_shulker_box", "red_shulker_box", "shulker_box", "white_shulker_box", |                             "brewing_stand", | ||||||
|                 "yellow_shulker_box", "smoker", "structure_block", "structure_void") |                             "campfire", | ||||||
|                 .map(BlockTypes::get) |                             "chest", | ||||||
|                 .filter(Objects::nonNull) |                             "ender_chest", | ||||||
|                 .forEach(tileEntityTypes::add); |                             "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; |         return this.tileEntityTypes; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public int getTileEntityCount(String world, BlockVector2 chunk) { |     @NonNegative | ||||||
|         return Bukkit.getWorld(world).getChunkAt(chunk.getBlockX(), chunk.getBlockZ()) |     public int getTileEntityCount(final @NonNull String world, final @NonNull BlockVector2 chunk) { | ||||||
|             .getTileEntities().length; |         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, |     @Override | ||||||
|         final Consumer<Chunk> chunkConsumer) { |     public Set<BlockVector2> getChunkChunks(String world) { | ||||||
|         PaperLib.getChunkAtAsync(getWorld(world), x >> 4, z >> 4, true) |         Set<BlockVector2> chunks = super.getChunkChunks(world); | ||||||
|             .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) { |  | ||||||
|         if (Bukkit.isPrimaryThread()) { |         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 { |         } else { | ||||||
|             Bukkit.getScheduler() |             final Semaphore semaphore = new Semaphore(1); | ||||||
|                 .runTask(BukkitMain.getPlugin(BukkitMain.class), () -> consumer.accept(value)); |             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,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.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 static final boolean HAS_MIN_Y; | ||||||
|  |  | ||||||
|  |     static { | ||||||
|  |         boolean temp; | ||||||
|  |         try { | ||||||
|  |             org.bukkit.World.class.getMethod("getMinHeight"); | ||||||
|  |             temp = true; | ||||||
|  |         } catch (NoSuchMethodException e) { | ||||||
|  |             temp = false; | ||||||
|  |         } | ||||||
|  |         HAS_MIN_Y = temp; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     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; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get the min world height from a Bukkit {@link org.bukkit.World}. Inclusive | ||||||
|  |      * | ||||||
|  |      * @since 6.6.0 | ||||||
|  |      */ | ||||||
|  |     public static int getMinWorldHeight(org.bukkit.World world) { | ||||||
|  |         return HAS_MIN_Y ? world.getMinHeight() : 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get the max world height from a Bukkit {@link org.bukkit.World}. Exclusive | ||||||
|  |      * | ||||||
|  |      * @since 6.6.0 | ||||||
|  |      */ | ||||||
|  |     public static int getMaxWorldHeight(org.bukkit.World world) { | ||||||
|  |         return HAS_MIN_Y ? world.getMaxHeight() : 256; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public org.bukkit.World getPlatformWorld() { | ||||||
|  |         return this.world; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NonNull String getName() { | ||||||
|  |         return this.world.getName(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public int getMinHeight() { | ||||||
|  |         return getMinWorldHeight(world); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public int getMaxHeight() { | ||||||
|  |         return getMaxWorldHeight(world) - 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean equals(final Object o) { | ||||||
|  |         if (this == o) { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |         if (o == null || getClass() != o.getClass()) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         final BukkitWorld that = (BukkitWorld) o; | ||||||
|  |         return world.equals(that.world); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public int hashCode() { | ||||||
|  |         return world.hashCode(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * @deprecated This method is not meant to be invoked or overridden, with no replacement. | ||||||
|  |      */ | ||||||
|  |     @Deprecated(forRemoval = true, since = "6.6.0") | ||||||
|  |     protected boolean canEqual(final Object other) { | ||||||
|  |         return other instanceof BukkitWorld; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public String toString() { | ||||||
|  |         return "BukkitWorld(world=" + this.world + ")"; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,19 +21,20 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.util; | ||||||
|  |  | ||||||
| import com.plotsquared.bukkit.entity.EntityWrapper; | import com.plotsquared.bukkit.entity.EntityWrapper; | ||||||
| import com.plotsquared.bukkit.entity.ReplicatingEntityWrapper; | import com.plotsquared.bukkit.entity.ReplicatingEntityWrapper; | ||||||
| import com.plotsquared.core.PlotSquared; |  | ||||||
| import com.plotsquared.core.location.Location; | import com.plotsquared.core.location.Location; | ||||||
| import com.plotsquared.core.location.PlotLoc; | import com.plotsquared.core.location.PlotLoc; | ||||||
| import com.sk89q.worldedit.bukkit.BukkitWorld; | import com.sk89q.worldedit.bukkit.BukkitWorld; | ||||||
| import com.sk89q.worldedit.math.BlockVector3; | import com.sk89q.worldedit.math.BlockVector3; | ||||||
| import com.sk89q.worldedit.regions.CuboidRegion; | import com.sk89q.worldedit.regions.CuboidRegion; | ||||||
| import com.sk89q.worldedit.world.block.BaseBlock; | 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.Chunk; | ||||||
| import org.bukkit.World; | import org.bukkit.World; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
| @@ -46,6 +47,8 @@ import java.util.Set; | |||||||
|  |  | ||||||
| public class ContentMap { | public class ContentMap { | ||||||
|  |  | ||||||
|  |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + ContentMap.class.getSimpleName()); | ||||||
|  |  | ||||||
|     final Set<EntityWrapper> entities; |     final Set<EntityWrapper> entities; | ||||||
|     final Map<PlotLoc, BaseBlock[]> allBlocks; |     final Map<PlotLoc, BaseBlock[]> allBlocks; | ||||||
|  |  | ||||||
| @@ -67,14 +70,14 @@ public class ContentMap { | |||||||
|         } |         } | ||||||
|         for (int x = x1; x <= x2; x++) { |         for (int x = x1; x <= x2; x++) { | ||||||
|             for (int z = z1; z <= z2; z++) { |             for (int z = z1; z <= z2; z++) { | ||||||
|                 saveBlocks(world, 256, x, z, 0, 0); |                 saveBlocks(world, x, z); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     void saveEntitiesOut(Chunk chunk, CuboidRegion region) { |     void saveEntitiesOut(Chunk chunk, CuboidRegion region) { | ||||||
|         for (Entity entity : chunk.getEntities()) { |         for (Entity entity : chunk.getEntities()) { | ||||||
|             Location location = BukkitUtil.getLocation(entity); |             Location location = BukkitUtil.adapt(entity.getLocation()); | ||||||
|             int x = location.getX(); |             int x = location.getX(); | ||||||
|             int z = location.getZ(); |             int z = location.getZ(); | ||||||
|             if (BukkitChunkManager.isIn(region, x, z)) { |             if (BukkitChunkManager.isIn(region, x, z)) { | ||||||
| @@ -89,14 +92,9 @@ public class ContentMap { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     void saveEntitiesIn(Chunk chunk, CuboidRegion region) { |     void saveEntitiesIn(Chunk chunk, CuboidRegion region, boolean delete) { | ||||||
|         saveEntitiesIn(chunk, region, 0, 0, false); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     void saveEntitiesIn(Chunk chunk, CuboidRegion region, int offsetX, int offsetZ, |  | ||||||
|         boolean delete) { |  | ||||||
|         for (Entity entity : chunk.getEntities()) { |         for (Entity entity : chunk.getEntities()) { | ||||||
|             Location location = BukkitUtil.getLocation(entity); |             Location location = BukkitUtil.adapt(entity.getLocation()); | ||||||
|             int x = location.getX(); |             int x = location.getX(); | ||||||
|             int z = location.getZ(); |             int z = location.getZ(); | ||||||
|             if (!BukkitChunkManager.isIn(region, x, z)) { |             if (!BukkitChunkManager.isIn(region, x, z)) { | ||||||
| @@ -106,8 +104,6 @@ public class ContentMap { | |||||||
|                 continue; |                 continue; | ||||||
|             } |             } | ||||||
|             EntityWrapper wrap = new ReplicatingEntityWrapper(entity, (short) 2); |             EntityWrapper wrap = new ReplicatingEntityWrapper(entity, (short) 2); | ||||||
|             wrap.x += offsetX; |  | ||||||
|             wrap.z += offsetZ; |  | ||||||
|             wrap.saveEntity(); |             wrap.saveEntity(); | ||||||
|             this.entities.add(wrap); |             this.entities.add(wrap); | ||||||
|             if (delete) { |             if (delete) { | ||||||
| @@ -118,27 +114,25 @@ public class ContentMap { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     void restoreEntities(World world, int xOffset, int zOffset) { |     void restoreEntities(World world) { | ||||||
|         for (EntityWrapper entity : this.entities) { |         for (EntityWrapper entity : this.entities) { | ||||||
|             try { |             try { | ||||||
|                 entity.spawn(world, xOffset, zOffset); |                 entity.spawn(world, 0, 0); | ||||||
|             } catch (Exception e) { |             } catch (Exception e) { | ||||||
|                 PlotSquared.debug("Failed to restore entity (e): " + e.toString()); |                 LOGGER.error("Failed to restore entity", e); | ||||||
|                 e.printStackTrace(); |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         this.entities.clear(); |         this.entities.clear(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     //todo optimize maxY |     private void saveBlocks(BukkitWorld world, int x, int z) { | ||||||
|     void saveBlocks(BukkitWorld world, int maxY, int x, int z, int offsetX, int offsetZ) { |         BaseBlock[] ids = new BaseBlock[world.getMaxY() - world.getMinY() + 1]; | ||||||
|         maxY = Math.min(255, maxY); |         for (short yIndex = 0; yIndex <= world.getMaxY() - world.getMinY(); yIndex++) { | ||||||
|         BaseBlock[] ids = new BaseBlock[maxY + 1]; |             BaseBlock block = world.getFullBlock(BlockVector3.at(x, yIndex + world.getMinY(), z)); | ||||||
|         for (short y = 0; y <= maxY; y++) { |             ids[yIndex] = block; | ||||||
|             BaseBlock block = world.getFullBlock(BlockVector3.at(x, y, z)); |  | ||||||
|             ids[y] = block; |  | ||||||
|         } |         } | ||||||
|         PlotLoc loc = new PlotLoc(x + offsetX, z + offsetZ); |         PlotLoc loc = new PlotLoc(x, z); | ||||||
|         this.allBlocks.put(loc, ids); |         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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.util; | ||||||
|  |  | ||||||
| @@ -38,7 +38,7 @@ import java.util.ArrayList; | |||||||
| public class SetGenCB { | public class SetGenCB { | ||||||
|  |  | ||||||
|     public static void setGenerator(World world) throws Exception { |     public static void setGenerator(World world) throws Exception { | ||||||
|         SetupUtils.manager.updateGenerators(); |         PlotSquared.platform().setupUtils().updateGenerators(false); | ||||||
|         PlotSquared.get().removePlotAreas(world.getName()); |         PlotSquared.get().removePlotAreas(world.getName()); | ||||||
|         ChunkGenerator gen = world.getGenerator(); |         ChunkGenerator gen = world.getGenerator(); | ||||||
|         if (gen == null) { |         if (gen == null) { | ||||||
| @@ -69,9 +69,10 @@ public class SetGenCB { | |||||||
|         } |         } | ||||||
|         if (!set) { |         if (!set) { | ||||||
|             world.getPopulators() |             world.getPopulators() | ||||||
|                 .removeIf(blockPopulator -> blockPopulator instanceof BukkitAugmentedGenerator); |                     .removeIf(blockPopulator -> blockPopulator instanceof BukkitAugmentedGenerator); | ||||||
|         } |         } | ||||||
|         PlotSquared.get() |         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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,17 +21,19 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.util; | ||||||
|  |  | ||||||
| import com.google.gson.JsonObject; | import com.google.gson.JsonObject; | ||||||
| import com.google.gson.JsonParser; | import com.google.gson.JsonParser; | ||||||
| import com.google.gson.stream.JsonReader; | import com.google.gson.stream.JsonReader; | ||||||
|  | import com.google.inject.Inject; | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.PlotVersion; | import com.plotsquared.core.PlotVersion; | ||||||
| import com.plotsquared.core.configuration.Captions; |  | ||||||
| import com.plotsquared.core.configuration.Settings; | 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.Bukkit; | ||||||
| import org.bukkit.event.Listener; | import org.bukkit.event.Listener; | ||||||
| import org.bukkit.plugin.java.JavaPlugin; | import org.bukkit.plugin.java.JavaPlugin; | ||||||
| @@ -44,6 +46,8 @@ import java.net.URL; | |||||||
|  |  | ||||||
| public class UpdateUtility implements Listener { | public class UpdateUtility implements Listener { | ||||||
|  |  | ||||||
|  |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + UpdateUtility.class.getSimpleName()); | ||||||
|  |  | ||||||
|     public static PlotVersion internalVersion; |     public static PlotVersion internalVersion; | ||||||
|     public static String spigotVersion; |     public static String spigotVersion; | ||||||
|     public static boolean hasUpdate; |     public static boolean hasUpdate; | ||||||
| @@ -51,48 +55,48 @@ public class UpdateUtility implements Listener { | |||||||
|     public final JavaPlugin javaPlugin; |     public final JavaPlugin javaPlugin; | ||||||
|     private boolean notify = true; |     private boolean notify = true; | ||||||
|  |  | ||||||
|  |     @Inject | ||||||
|     public UpdateUtility(final JavaPlugin javaPlugin) { |     public UpdateUtility(final JavaPlugin javaPlugin) { | ||||||
|         this.javaPlugin = javaPlugin; |         this.javaPlugin = javaPlugin; | ||||||
|         internalVersion = PlotSquared.get().getVersion(); |         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() { |     public void updateChecker() { | ||||||
|         task = Bukkit.getScheduler().runTaskTimerAsynchronously(this.javaPlugin, () -> { |         task = Bukkit.getScheduler().runTaskTimerAsynchronously(this.javaPlugin, () -> { | ||||||
|             try { |             try { | ||||||
|                 HttpsURLConnection connection = (HttpsURLConnection) new URL( |                 HttpsURLConnection connection = (HttpsURLConnection) new URL( | ||||||
|                     "https://api.spigotmc.org/simple/0.1/index.php?action=getResource&id=77506") |                         "https://api.spigotmc.org/simple/0.1/index.php?action=getResource&id=77506") | ||||||
|                     .openConnection(); |                         .openConnection(); | ||||||
|                 connection.setRequestMethod("GET"); |                 connection.setRequestMethod("GET"); | ||||||
|                 JsonObject result = (new JsonParser()) |                 JsonObject result = new JsonParser() | ||||||
|                     .parse(new JsonReader(new InputStreamReader(connection.getInputStream()))) |                         .parse(new JsonReader(new InputStreamReader(connection.getInputStream()))) | ||||||
|                     .getAsJsonObject(); |                         .getAsJsonObject(); | ||||||
|                 spigotVersion = result.get("current_version").getAsString(); |                 spigotVersion = result.get("current_version").getAsString(); | ||||||
|             } catch (IOException e) { |             } 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; |                 return; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             if (internalVersion.isLaterVersion(spigotVersion)) { |             if (internalVersion.isLaterVersion(spigotVersion)) { | ||||||
|                 PlotSquared |                 LOGGER.info("There appears to be a PlotSquared update available!"); | ||||||
|                     .log(Captions.PREFIX + "&6There appears to be a PlotSquared update available!"); |                 LOGGER.info("You are running version {}, the latest version is {}", | ||||||
|                 PlotSquared.log( |                         internalVersion.versionString(), spigotVersion | ||||||
|                     Captions.PREFIX + "&6You are running version " + internalVersion.versionString() |                 ); | ||||||
|                         + ", &6latest version is " + spigotVersion); |                 LOGGER.info("https://www.spigotmc.org/resources/77506/updates"); | ||||||
|                 PlotSquared |  | ||||||
|                     .log(Captions.PREFIX + "&6https://www.spigotmc.org/resources/77506/updates"); |  | ||||||
|                 hasUpdate = true; |                 hasUpdate = true; | ||||||
|                 if (Settings.UpdateChecker.NOTIFY_ONCE) { |                 if (Settings.UpdateChecker.NOTIFY_ONCE) { | ||||||
|                     cancelTask(); |                     cancelTask(); | ||||||
|                 } |                 } | ||||||
|             } else if (notify) { |             } else if (notify) { | ||||||
|                 notify = false; |                 notify = false; | ||||||
|                 PlotSquared.log(Captions.PREFIX |                 LOGGER.info("Congratulations! You are running the latest PlotSquared version"); | ||||||
|                     + "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() { |     private void cancelTask() { | ||||||
|         Bukkit.getScheduler().runTaskLater(javaPlugin, () -> task.cancel(), 20L); |         Bukkit.getScheduler().runTaskLater(javaPlugin, () -> task.cancel(), 20L); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,134 @@ | |||||||
|  | /* | ||||||
|  |  *       _____  _       _    _____                                _ | ||||||
|  |  *      |  __ \| |     | |  / ____|                              | | | ||||||
|  |  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||||
|  |  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||||
|  |  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||||
|  |  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||||
|  |  *                                    | | | ||||||
|  |  *                                    |_| | ||||||
|  |  *            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 void setBiome(CuboidRegion region, int extendBiome, BiomeType biome, PlotArea area, Runnable whenDone) { | ||||||
|  |         delegate.setBiome(region, extendBiome, biome, area.getWorldName(), 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, actor, 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 |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.uuid; | ||||||
|  |  | ||||||
| @@ -29,7 +29,7 @@ import com.earth2me.essentials.Essentials; | |||||||
| import com.earth2me.essentials.User; | import com.earth2me.essentials.User; | ||||||
| import com.plotsquared.core.uuid.UUIDMapping; | import com.plotsquared.core.uuid.UUIDMapping; | ||||||
| import com.plotsquared.core.uuid.UUIDService; | 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.ArrayList; | ||||||
| import java.util.Collections; | import java.util.Collections; | ||||||
| @@ -47,11 +47,13 @@ public class EssentialsUUIDService implements UUIDService { | |||||||
|         this.essentials = Essentials.getPlugin(Essentials.class); |         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(); |         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()); |         final List<UUIDMapping> mappings = new ArrayList<>(usernames.size()); | ||||||
|         for (final String username : usernames) { |         for (final String username : usernames) { | ||||||
|             try { |             try { | ||||||
| @@ -62,7 +64,8 @@ public class EssentialsUUIDService implements UUIDService { | |||||||
|                         mappings.add(new UUIDMapping(uuid, username)); |                         mappings.add(new UUIDMapping(uuid, username)); | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } catch (final Exception ignored){} |             } catch (final Exception ignored) { | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|         return mappings; |         return mappings; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.uuid; | ||||||
|  |  | ||||||
| @@ -31,7 +31,7 @@ import net.luckperms.api.LuckPerms; | |||||||
| import net.luckperms.api.model.user.UserManager; | import net.luckperms.api.model.user.UserManager; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
| import org.bukkit.plugin.RegisteredServiceProvider; | import org.bukkit.plugin.RegisteredServiceProvider; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| @@ -53,21 +53,24 @@ public class LuckPermsUUIDService implements UUIDService { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override @NotNull public List<UUIDMapping> getNames(@NotNull final List<UUID> uuids) { |     @Override | ||||||
|        final List<UUIDMapping> mappings = new ArrayList<>(uuids.size()); |     public @NonNull List<UUIDMapping> getNames(final @NonNull List<UUID> uuids) { | ||||||
|        final UserManager userManager = this.luckPerms.getUserManager(); |         final List<UUIDMapping> mappings = new ArrayList<>(uuids.size()); | ||||||
|        for (final UUID uuid : uuids) { |         final UserManager userManager = this.luckPerms.getUserManager(); | ||||||
|            try { |         for (final UUID uuid : uuids) { | ||||||
|                final String username = userManager.lookupUsername(uuid).get(); |             try { | ||||||
|                if (username != null) { |                 final String username = userManager.lookupUsername(uuid).get(); | ||||||
|                    mappings.add(new UUIDMapping(uuid, username)); |                 if (username != null) { | ||||||
|                } |                     mappings.add(new UUIDMapping(uuid, username)); | ||||||
|            } catch (final Exception ignored) {} |                 } | ||||||
|        } |             } catch (final Exception ignored) { | ||||||
|        return mappings; |             } | ||||||
|  |         } | ||||||
|  |         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 List<UUIDMapping> mappings = new ArrayList<>(usernames.size()); | ||||||
|         final UserManager userManager = this.luckPerms.getUserManager(); |         final UserManager userManager = this.luckPerms.getUserManager(); | ||||||
|         for (final String username : usernames) { |         for (final String username : usernames) { | ||||||
| @@ -76,7 +79,8 @@ public class LuckPermsUUIDService implements UUIDService { | |||||||
|                 if (username != null) { |                 if (username != null) { | ||||||
|                     mappings.add(new UUIDMapping(uuid, username)); |                     mappings.add(new UUIDMapping(uuid, username)); | ||||||
|                 } |                 } | ||||||
|             } catch (final Exception ignored) {} |             } catch (final Exception ignored) { | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|         return mappings; |         return mappings; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.uuid; | ||||||
|  |  | ||||||
| @@ -31,7 +31,7 @@ import com.plotsquared.core.uuid.UUIDMapping; | |||||||
| import com.plotsquared.core.uuid.UUIDService; | import com.plotsquared.core.uuid.UUIDService; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
| import org.bukkit.OfflinePlayer; | import org.bukkit.OfflinePlayer; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.Collections; | import java.util.Collections; | ||||||
| @@ -43,7 +43,8 @@ import java.util.UUID; | |||||||
|  */ |  */ | ||||||
| public class OfflinePlayerUUIDService implements UUIDService { | 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()) { |         if (Settings.UUID.FORCE_LOWERCASE || Bukkit.getWorlds().isEmpty()) { | ||||||
|             return Collections.emptyList(); // This is useless now |             return Collections.emptyList(); // This is useless now | ||||||
|         } |         } | ||||||
| @@ -54,14 +55,16 @@ public class OfflinePlayerUUIDService implements UUIDService { | |||||||
|                 if (offlinePlayer.hasPlayedBefore()) { |                 if (offlinePlayer.hasPlayedBefore()) { | ||||||
|                     wrappers.add(new UUIDMapping(uuid, offlinePlayer.getName())); |                     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 |                                                     probably because it's called before the worlds have | ||||||
|                                                     been loaded. This is bad, but does not break anything */ |                                                     been loaded. This is bad, but does not break anything */ | ||||||
|         } |         } | ||||||
|         return wrappers; |         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()); |         final List<UUIDMapping> wrappers = new ArrayList<>(usernames.size()); | ||||||
|         for (final String username : usernames) { |         for (final String username : usernames) { | ||||||
|             if (Settings.UUID.OFFLINE) { |             if (Settings.UUID.OFFLINE) { | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,7 +21,7 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | 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.UUIDMapping; | ||||||
| import com.plotsquared.core.uuid.UUIDService; | import com.plotsquared.core.uuid.UUIDService; | ||||||
| import org.bukkit.Bukkit; | import org.bukkit.Bukkit; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| @@ -40,7 +40,8 @@ import java.util.UUID; | |||||||
|  */ |  */ | ||||||
| public class PaperUUIDService implements UUIDService { | 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()); |         final List<UUIDMapping> mappings = new ArrayList<>(uuids.size()); | ||||||
|         for (final UUID uuid : uuids) { |         for (final UUID uuid : uuids) { | ||||||
|             final PlayerProfile playerProfile = Bukkit.createProfile(uuid); |             final PlayerProfile playerProfile = Bukkit.createProfile(uuid); | ||||||
| @@ -51,7 +52,8 @@ public class PaperUUIDService implements UUIDService { | |||||||
|         return mappings; |         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 List<UUIDMapping> mappings = new ArrayList<>(usernames.size()); | ||||||
|         for (final String username : usernames) { |         for (final String username : usernames) { | ||||||
|             final PlayerProfile playerProfile = Bukkit.createProfile(username); |             final PlayerProfile playerProfile = Bukkit.createProfile(username); | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,16 +21,16 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.uuid; | ||||||
|  |  | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.database.SQLite; | 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.UUIDMapping; | ||||||
| import com.plotsquared.core.uuid.UUIDService; | 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.Connection; | ||||||
| import java.sql.PreparedStatement; | import java.sql.PreparedStatement; | ||||||
| @@ -51,7 +51,7 @@ public class SQLiteUUIDService implements UUIDService, Consumer<List<UUIDMapping | |||||||
|  |  | ||||||
|     public SQLiteUUIDService(final String fileName) { |     public SQLiteUUIDService(final String fileName) { | ||||||
|         this.sqlite = |         this.sqlite = | ||||||
|             new SQLite(MainUtil.getFile(PlotSquared.get().IMP.getDirectory(), fileName)); |                 new SQLite(FileUtils.getFile(PlotSquared.platform().getDirectory(), fileName)); | ||||||
|         try { |         try { | ||||||
|             this.sqlite.openConnection(); |             this.sqlite.openConnection(); | ||||||
|         } catch (ClassNotFoundException | SQLException e) { |         } catch (ClassNotFoundException | SQLException e) { | ||||||
| @@ -59,7 +59,7 @@ public class SQLiteUUIDService implements UUIDService, Consumer<List<UUIDMapping | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         try (PreparedStatement stmt = getConnection().prepareStatement( |         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(); |             stmt.execute(); | ||||||
|         } catch (SQLException e) { |         } catch (SQLException e) { | ||||||
|             e.printStackTrace(); |             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()); |         final List<UUIDMapping> mappings = new ArrayList<>(uuids.size()); | ||||||
|         try (final PreparedStatement statement = getConnection() |         try (final PreparedStatement statement = getConnection() | ||||||
|             .prepareStatement("SELECT `username` FROM `usercache` WHERE `uuid` = ?")) { |                 .prepareStatement("SELECT `username` FROM `usercache` WHERE `uuid` = ?")) { | ||||||
|             for (final UUID uuid : uuids) { |             for (final UUID uuid : uuids) { | ||||||
|                 statement.setString(1, uuid.toString()); |                 statement.setString(1, uuid.toString()); | ||||||
|                 try (final ResultSet resultSet = statement.executeQuery()) { |                 try (final ResultSet resultSet = statement.executeQuery()) { | ||||||
| @@ -90,16 +91,19 @@ public class SQLiteUUIDService implements UUIDService, Consumer<List<UUIDMapping | |||||||
|         return mappings; |         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()); |         final List<UUIDMapping> mappings = new ArrayList<>(usernames.size()); | ||||||
|         try (final PreparedStatement statement = getConnection() |         try (final PreparedStatement statement = getConnection() | ||||||
|             .prepareStatement("SELECT `uuid` FROM `usercache` WHERE `username` = ?")) { |                 .prepareStatement("SELECT `uuid` FROM `usercache` WHERE `username` = ?")) { | ||||||
|             for (final String username : usernames) { |             for (final String username : usernames) { | ||||||
|                 statement.setString(1, username); |                 statement.setString(1, username); | ||||||
|                 try (final ResultSet resultSet = statement.executeQuery()) { |                 try (final ResultSet resultSet = statement.executeQuery()) { | ||||||
|                     if (resultSet.next()) { |                     if (resultSet.next()) { | ||||||
|                         mappings.add(new UUIDMapping(UUID.fromString(resultSet.getString("uuid")), |                         mappings.add(new UUIDMapping( | ||||||
|                             username)); |                                 UUID.fromString(resultSet.getString("uuid")), | ||||||
|  |                                 username | ||||||
|  |                         )); | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -109,9 +113,10 @@ public class SQLiteUUIDService implements UUIDService, Consumer<List<UUIDMapping | |||||||
|         return mappings; |         return mappings; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override public void accept(final List<UUIDMapping> uuidWrappers) { |     @Override | ||||||
|  |     public void accept(final List<UUIDMapping> uuidWrappers) { | ||||||
|         try (final PreparedStatement statement = getConnection() |         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) { |             for (final UUIDMapping mapping : uuidWrappers) { | ||||||
|                 statement.setString(1, mapping.getUuid().toString()); |                 statement.setString(1, mapping.getUuid().toString()); | ||||||
|                 statement.setString(2, mapping.getUsername()); |                 statement.setString(2, mapping.getUsername()); | ||||||
| @@ -127,7 +132,7 @@ public class SQLiteUUIDService implements UUIDService, Consumer<List<UUIDMapping | |||||||
|      * |      * | ||||||
|      * @return All read mappings |      * @return All read mappings | ||||||
|      */ |      */ | ||||||
|     @NotNull public List<UUIDMapping> getAll() { |     public @NonNull List<UUIDMapping> getAll() { | ||||||
|         final List<UUIDMapping> mappings = new LinkedList<>(); |         final List<UUIDMapping> mappings = new LinkedList<>(); | ||||||
|         try (final PreparedStatement statement = getConnection().prepareStatement("SELECT * FROM `usercache`")) { |         try (final PreparedStatement statement = getConnection().prepareStatement("SELECT * FROM `usercache`")) { | ||||||
|             try (final ResultSet resultSet = statement.executeQuery()) { |             try (final ResultSet resultSet = statement.executeQuery()) { | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  *                                    | | |  *                                    | | | ||||||
|  *                                    |_| |  *                                    |_| | ||||||
|  *            PlotSquared plot management system for Minecraft |  *            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 |  *     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 |  *     it under the terms of the GNU General Public License as published by | ||||||
| @@ -21,19 +21,20 @@ | |||||||
|  *     GNU General Public License for more details. |  *     GNU General Public License for more details. | ||||||
|  * |  * | ||||||
|  *     You should have received a copy of the GNU General Public License |  *     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; | package com.plotsquared.bukkit.uuid; | ||||||
|  |  | ||||||
| import com.google.common.util.concurrent.RateLimiter; | import com.google.common.util.concurrent.RateLimiter; | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.configuration.Captions; |  | ||||||
| import com.plotsquared.core.uuid.UUIDMapping; | import com.plotsquared.core.uuid.UUIDMapping; | ||||||
| import com.plotsquared.core.uuid.UUIDService; | import com.plotsquared.core.uuid.UUIDService; | ||||||
| import com.sk89q.squirrelid.Profile; | import org.apache.logging.log4j.LogManager; | ||||||
| import com.sk89q.squirrelid.resolver.HttpRepositoryService; | import org.apache.logging.log4j.Logger; | ||||||
| import com.sk89q.squirrelid.resolver.ProfileService; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.enginehub.squirrelid.Profile; | ||||||
|  | import org.enginehub.squirrelid.resolver.HttpRepositoryService; | ||||||
|  | import org.enginehub.squirrelid.resolver.ProfileService; | ||||||
|  |  | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| @@ -47,6 +48,8 @@ import java.util.UUID; | |||||||
| @SuppressWarnings("UnstableApiUsage") | @SuppressWarnings("UnstableApiUsage") | ||||||
| public class SquirrelIdUUIDService implements UUIDService { | public class SquirrelIdUUIDService implements UUIDService { | ||||||
|  |  | ||||||
|  |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + SquirrelIdUUIDService.class.getSimpleName()); | ||||||
|  |  | ||||||
|     private final ProfileService profileService; |     private final ProfileService profileService; | ||||||
|     private final RateLimiter rateLimiter; |     private final RateLimiter rateLimiter; | ||||||
|  |  | ||||||
| @@ -64,12 +67,13 @@ public class SquirrelIdUUIDService implements UUIDService { | |||||||
|         this.rateLimiter = RateLimiter.create(rateLimit / 600.0D); |         this.rateLimiter = RateLimiter.create(rateLimit / 600.0D); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @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> results = new ArrayList<>(uuids.size()); |         final List<UUIDMapping> results = new ArrayList<>(uuids.size()); | ||||||
|         this.rateLimiter.acquire(uuids.size()); |         this.rateLimiter.acquire(uuids.size()); | ||||||
|         try { |         try { | ||||||
|             try { |             try { | ||||||
|                 for (final Profile profile : this.profileService.findAllById(uuids)) { |                 for (final Profile profile : this.profileService.findAllByUuid(uuids)) { | ||||||
|                     results.add(new UUIDMapping(profile.getUniqueId(), profile.getName())); |                     results.add(new UUIDMapping(profile.getUniqueId(), profile.getName())); | ||||||
|                 } |                 } | ||||||
|             } catch (final IllegalArgumentException illegalArgumentException) { |             } catch (final IllegalArgumentException illegalArgumentException) { | ||||||
| @@ -78,7 +82,9 @@ public class SquirrelIdUUIDService implements UUIDService { | |||||||
|                 // go through them one by one |                 // go through them one by one | ||||||
|                 // |                 // | ||||||
|                 if (uuids.size() >= 2) { |                 if (uuids.size() >= 2) { | ||||||
|                     PlotSquared.debug(Captions.PREFIX + "(UUID) Found invalid UUID in batch. Will try each UUID individually."); |                     if (Settings.DEBUG) { | ||||||
|  |                         LOGGER.info("(UUID) Found invalid UUID in batch. Will try each UUID individually."); | ||||||
|  |                     } | ||||||
|                     for (final UUID uuid : uuids) { |                     for (final UUID uuid : uuids) { | ||||||
|                         final List<UUIDMapping> result = this.getNames(Collections.singletonList(uuid)); |                         final List<UUIDMapping> result = this.getNames(Collections.singletonList(uuid)); | ||||||
|                         if (result.isEmpty()) { |                         if (result.isEmpty()) { | ||||||
| @@ -86,8 +92,8 @@ public class SquirrelIdUUIDService implements UUIDService { | |||||||
|                         } |                         } | ||||||
|                         results.add(result.get(0)); |                         results.add(result.get(0)); | ||||||
|                     } |                     } | ||||||
|                 } else if (uuids.size() == 1) { |                 } else if (uuids.size() == 1 && Settings.DEBUG) { | ||||||
|                     PlotSquared.debug(Captions.PREFIX + "(UUID) Found invalid UUID: " + uuids.get(0)); |                     LOGGER.info("(UUID) Found invalid UUID: {}", uuids.get(0)); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } catch (IOException | InterruptedException e) { |         } catch (IOException | InterruptedException e) { | ||||||
| @@ -96,7 +102,8 @@ public class SquirrelIdUUIDService implements UUIDService { | |||||||
|         return results; |         return results; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @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> results = new ArrayList<>(usernames.size()); |         final List<UUIDMapping> results = new ArrayList<>(usernames.size()); | ||||||
|         this.rateLimiter.acquire(usernames.size()); |         this.rateLimiter.acquire(usernames.size()); | ||||||
|         try { |         try { | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user