mirror of
				https://github.com/IntellectualSites/PlotSquared.git
				synced 2025-10-25 15:43:44 +02:00 
			
		
		
		
	Compare commits
	
		
			824 Commits
		
	
	
		
			refactor/v
			...
			fix/genera
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | d1a6c021fc | ||
|   | a388629d62 | ||
|   | 6991bc79d6 | ||
|   | a191000142 | ||
|   | 2cf57dda47 | ||
|   | e60a016164 | ||
|   | e2f6b0a6fe | ||
|   | 8002d170f5 | ||
|   | 52e8ba6ddd | ||
|   | 2f1fe633af | ||
|   | dc30408ed0 | ||
|   | 313ea8b266 | ||
|   | 1ad0e700a5 | ||
|   | 590f442a0f | ||
|   | 308527e822 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1f32909ffd | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1785693fe4 | ||
|   | df738b279d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b7b25252c6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7123f51bb5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 897afa894e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2238609551 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 255959232b | ||
|   | 622c9f1d13 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e7aff3982e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | db7ea780f9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | db188150d7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cfd8401515 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c887cbe28c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 59183c1412 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 439fb3a8ea | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 78d6ca1deb | ||
|   | 03aa1be5a3 | ||
|   | 2e3832f1bd | ||
|   | 05af41f832 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b613318a29 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c2f8356042 | ||
|   | 109e6059f8 | ||
|   | f01c287f89 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fd8cf3c475 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8c9957edeb | ||
|   | aaba2b5b1a | ||
|   | 02a65c8855 | ||
|   | 8a5fa26796 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 13cbb7e083 | ||
|   | 4d8d5b3a9f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cd6a32cf44 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 273c0ad989 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 774da7183b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e083015ab2 | ||
|   | 3f577d039b | ||
|   | 4d2e4a3d1a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e5d36579b1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f0cde251bd | ||
|   | 58016bb1c8 | ||
|   | e5943ba627 | ||
|   | 07dfdeef2c | ||
|   | 025b08e716 | ||
|   | 921435689e | ||
|   | aae154b23a | ||
|   | 0f33465d76 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 438f1d9656 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cb38aeef93 | ||
|   | 7be0655b86 | ||
|   | 774298bef5 | ||
|   | 6fc25bc034 | ||
|   | 1054018e1e | ||
|   | 0508a7f6b6 | ||
|   | da0a57a48c | ||
|   | 87859b002b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4f4ba07bd2 | ||
|   | 53771a3ece | ||
|   | f020a6c6da | ||
|   | d7e158747e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4d64ea83ec | ||
|   | 7ab334562c | ||
|   | fe1ef36f7e | ||
|   | 70bc02985f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3ec7e992a3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9acaa9c554 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e132c01331 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 24e4e51884 | ||
|   | 84ec090df1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6c6ea1c1b4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 97989face1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f471c02330 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 94322d5982 | ||
|   | 6cbb894249 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6f0fa19601 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0b692459e6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0d410ed869 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 711fba0b2a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 671a27fa6f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 03de685dc4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f616885206 | ||
|   | 7da4eb1ab5 | ||
|   | 629646ab06 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 74a1a1f954 | ||
|   | aa44078018 | ||
|   | bfbf406418 | ||
|   | 2accedf264 | ||
|   | 6ef0d58480 | ||
|   | fbf4a638b4 | ||
|   | 9abfa21078 | ||
|   | b84599b4b3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 058983cdd5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6ba3694121 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8e8e31b80e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | da2e66c1f8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d5d6fcb859 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 96f73331f9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 020947d90c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7dbd0bcff8 | ||
|   | 9626302f04 | ||
|   | 1b4a347e8b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 19e6ed4b9b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6b1b0f2d6a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e499bc02ec | ||
|   | 62084fffdd | ||
|   | d012f79349 | ||
|   | 139d6efc70 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6f5cb917f2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9bf66f1b7c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 92875ebe7f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 25a4545f14 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3fdeed019b | ||
|   | f3400df811 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | beb7cb40f4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2c0bb03e5c | ||
|   | e8c170686c | ||
|   | ff8676cde3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 817effb735 | ||
|   | a0a3d8828a | ||
|   | 8741bfcf88 | ||
|   | 6a6c113e5b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3f573b4d46 | ||
|   | 2f6db9c3db | ||
|   | 2f050b7b47 | ||
|   | eb0d854870 | ||
|   | d4f10422e3 | ||
|   | 4e2ea67992 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f533e194f4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 98bc2e7407 | ||
|   | 1b9d0d5317 | ||
|   | 8bb15d5c65 | ||
|   | e9baa802ec | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 63e2d325cf | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 94abd69e22 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 661e4ae8d3 | ||
|   | 974c639a51 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 11b806bd4d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0275372051 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c18cf3acfe | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fa52149394 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f5108ec253 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a7058eeff2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | eabae8db44 | ||
|   | a836e8e763 | ||
|   | 6930a9cecb | ||
|   | effbacb823 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 47d1f1e0cb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6bedd9b25f | ||
|   | 960b7b2a8b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 198052b7a8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a5af3a9d16 | ||
|   | bc4e2c51da | ||
|   | c46b4ddeaa | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c8e7367987 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1a0450eefb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0cb8075184 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5d979b0a4f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fe1554c03c | ||
|   | f0fd9986b4 | ||
|   | bab6f20a5d | ||
|   | 32d36b28fa | ||
|   | a11c560d4e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 66bb0a6214 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7218e9829f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2e17c941fc | ||
|   | 5e628cc758 | ||
|   | a42e08dc0e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 10bf45c128 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9fe61e5053 | ||
|   | 8f5bdf5dbb | ||
|   | 9df1387f81 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | dcf5a7d940 | ||
|   | 641e3840cb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5642061d6f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 003898f6a7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fecf8104e9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8a9dab4f8e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0832656a12 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5525085e6d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7a429fd05c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2c0ad36939 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5e0957c14a | ||
|   | c83306a2ea | ||
|   | 6f4c156585 | ||
|   | afb36d98c7 | ||
|   | 001ae78fb2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5fc8e06c22 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f9401dda94 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d1a48dba4d | ||
|   | db9b51a535 | ||
|   | 511db0af37 | ||
|   | e1ccda3e6d | ||
|   | a69cd609b9 | ||
|   | 5d4e6c5819 | ||
|   | db05f1481a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ee3dd00225 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 346a48225d | ||
|   | dfd80c4723 | ||
|   | fad038ef78 | ||
|   | 84b1af8856 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 13c5a67cb1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8ae894d51d | ||
|   | bbb3736846 | ||
|   | 1ebcbf60a6 | ||
|   | 494144dc4f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 895cf0da66 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 05811d80ce | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4de9967ef4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 034bd866eb | ||
|   | 98aab56616 | ||
|   | 8f980c726b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7f59c03f06 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 480a5925b6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cee4493723 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b8ac1a22c1 | ||
|   | 67e69e3fc1 | ||
|   | 670f5a802e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c4bd6b6500 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 161e3ffdc7 | ||
|   | be8f07c556 | ||
|   | 215053e364 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4839a83279 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0649ef33f0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1590a4b6eb | ||
|   | c793b4454a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7ab1a3eafb | ||
|   | c65c9e7827 | ||
|   | f88ea94bfe | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c8e8eb919f | ||
|   | 83fe761fe4 | ||
|   | a7447c9d75 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5867cc51a7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 86e21f3e1a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d7d884ad6d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1c45e01a14 | ||
|   | 6ef1163325 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c57d784df7 | ||
|   | c239908aa3 | ||
|   | a6412581a6 | ||
|   | f20c5f46e3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4db5954490 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9f68654614 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2e4c6199e5 | ||
|   | 7edca600fd | ||
|   | bc1cc074b8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d383187c6e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 125a3f6772 | ||
|   | faca8c2da0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0ad5ef4f94 | ||
|   | 5e8d8629c2 | ||
|   | 9f4f213a8c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ce14036949 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2dbb6ee025 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0da1d9f17a | ||
|   | f1f41b0523 | ||
|   | fe324d3ea9 | ||
|   | ff83868cbc | ||
|   | 111ea7029e | ||
|   | 9be2eedf7f | ||
|   | 82f868ae7d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e46dbd826c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 809ddce2b3 | ||
|   | 1b40cea51f | ||
|   | 022e0fc224 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b32137a650 | ||
|   | 17c41c0494 | ||
|   | 1ee76bf2d9 | ||
|   | 2321831044 | ||
|   | 25e98618b9 | ||
|   | 5344efd1b7 | ||
|   | 68701b6201 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 64c610ef37 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 931bb90600 | ||
|   | 1dfa3b4e66 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8f236a56a6 | ||
|   | e51121960d | ||
|   | cc011de032 | ||
|   | 28298ffdd6 | ||
|   | 499d3c39bc | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3d56937f14 | ||
|   | 0f1c2cb4e4 | ||
|   | 7b233c944a | ||
|   | d9537ee9df | ||
|   | 0de6887526 | ||
|   | a2e3274215 | ||
|   | b369683b9c | ||
|   | 7f1f1e025e | ||
|   | 59787fe7f3 | ||
|   | 6783d5ece6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d9aa2a496c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 809ed6778c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2fe44053a2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c36b87ca14 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5aec7653b6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cc5d01e225 | ||
|   | 448577774a | ||
|   | 966c878a72 | ||
|   | 5021f5b379 | ||
|   | 76ea9e0d3c | ||
|   | 951f08bc8b | ||
|   | ae941e67a4 | ||
|   | 9566af5fda | ||
|   | fccc146053 | ||
|   | a1d94af242 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6371cd4c5a | ||
|   | e4613cfc62 | ||
|   | 8c44b2d2d2 | ||
|   | 449af2f3a4 | ||
|   | ead7acdd76 | ||
|   | 1991142d48 | ||
|   | 63ae11b3d3 | ||
|   | 86fe3c6846 | ||
|   | a90e179338 | ||
|   | a6ae287908 | ||
|   | 1a33997099 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6edd4b8220 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9b0d1e484c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6971fa4c10 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3c818f3e33 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 31be2e5eb3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 945a8ad306 | ||
|   | c6b0b99cd6 | ||
|   | dbfc43e3cd | ||
|   | c8b4a2fa39 | ||
|   | d851e27aed | ||
|   | 4a45729c9e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7931c0864e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1456b29d93 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 761477b76d | ||
|   | e61bcf905f | ||
|   | 85bec710df | ||
|   | d130794453 | ||
|   | f5f875eb11 | ||
|   | 89511f07f9 | ||
|   | 1a18adcd95 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 65858c5f3e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5c7520b5f5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f3b9cd5ded | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8a3eb25805 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 48bbd3c018 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | bf85013f70 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d36a2d236b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 79f111ec0a | ||
|   | 31ae62b62c | ||
|   | cdb44d4884 | ||
|   | eb63e4351d | ||
|   | ba7880241b | ||
|   | be6838f29e | ||
|   | dc73116401 | ||
|   | b6a87df072 | ||
|   | 8195afaa2f | ||
|   | 561eac2fbd | ||
|   | fdc887850c | ||
|   | e3bfd9b8bf | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e689337188 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ee6ae6cba0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | dc8d7809bd | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | dcd63ed4d9 | ||
|   | 3cc770970f | ||
|   | 1c3776b605 | ||
|   | 95c7f621fb | ||
|   | 15b4cbdb0f | ||
|   | 812eac18d3 | ||
|   | 16a4ee835c | ||
|   | c013b92e62 | ||
|   | b00a46b286 | ||
|   | 44b1127181 | ||
|   | c7bfd48a21 | ||
|   | dc13783db8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0a390ab342 | ||
|   | d111740f64 | ||
|   | 28e97e8441 | ||
|   | a30cdb37d6 | ||
|   | f848162066 | ||
|   | 40c70aa98d | ||
|   | 0d2b36bac8 | ||
|   | d7e5bcdaa5 | ||
|   | fc783574a3 | ||
|   | 5f7bb784f0 | ||
|   | 26c55a318f | ||
|   | ee68bc3d9e | ||
|   | a3bc3968a5 | ||
|   | 79454da1a6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 12a4c92ad9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 167692d464 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ae26e8155c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 286ea62a21 | ||
|   | d95c74d8c9 | ||
|   | c1555ddbc7 | ||
|   | 4fe0c586d9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | aae6ea4fee | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 385d018504 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f4def082c1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 69c9f1df83 | ||
|   | e138dc0267 | ||
|   | ca50b53f94 | ||
|   | f705487055 | ||
|   | b7c9453a1a | ||
|   | 1aa370d562 | ||
|   | d3dab0d736 | ||
|   | 764156b267 | ||
|   | 665f5251bf | ||
|   | 7c328095d7 | ||
|   | 7884c91d52 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e9a19e0821 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 022847fc4b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1ee673be58 | ||
|   | 3c2aa99e86 | ||
|   | 11fac3f060 | ||
|   | 3e57e524b9 | ||
|   | f582ec03c5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 893be136f0 | ||
|   | b74ba30281 | ||
|   | ba9dab1f73 | ||
|   | 8e60fdb477 | ||
|   | 443fe8dd47 | ||
|   | e56e52ba4f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cd008bed9b | ||
|   | d4c90283d6 | ||
|   | dc04ec955a | ||
|   | 72f511ce99 | ||
|   | 0d63c2bdb6 | ||
|   | 49e13384cf | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1ddc19ff69 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a6d436e841 | ||
|   | 9b0b39ac2e | ||
|   | 638f0bd078 | ||
|   | c27b838dad | ||
|   | e0cb2949df | ||
|   | 59be582c28 | ||
|   | f6cbb3792f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a68918f830 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1a7ded864e | ||
|   | cbc8bc8879 | ||
|   | 21f79d1c13 | ||
|   | 293d7acf2d | ||
|   | d876d3722a | ||
|   | dffb7672ff | ||
|   | f867867a42 | ||
|   | 59eefd6865 | ||
|   | 587a286d05 | ||
|   | e10caf6aa0 | ||
|   | 08b325e37d | ||
|   | c394108ba6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 31e89019f1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3a7075e28d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8373b7874e | ||
|   | fe13882b97 | ||
|   | f45064c4c4 | ||
|   | af32399dd2 | ||
|   | f3c03348d9 | ||
|   | a53330e39b | ||
|   | e2ba93dab9 | ||
|   | 9d43434e40 | ||
|   | 4f421167d1 | ||
|   | 94f4619c2c | ||
|   | 9885d3e506 | ||
|   | a54276d3b2 | ||
|   | cbb284b0fd | ||
|   | ed22b22e9c | ||
|   | 444ccda807 | ||
|   | db361cc420 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 079dc02cfe | ||
|   | e98791c865 | ||
|   | 7c3112f30f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c01f5f5c7d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 95caa19505 | ||
|   | b8055201df | ||
|   | 81daefae4a | ||
|   | 02437a8c72 | ||
|   | 958c66b28f | ||
|   | c656190e14 | ||
|   | e914cb210e | ||
|   | 94c6af74d2 | ||
|   | ebb82bd66d | ||
|   | 66f907eb5d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9ffede2c5c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 97172df0dc | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c9746b182c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 10a2b80ffc | ||
|   | ca0f38255d | ||
|   | 0484ac73af | ||
|   | 1a712ad3c1 | ||
|   | 2d1f483469 | ||
|   | 91830e233b | ||
|   | 561edb83bf | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0e09cf223a | ||
|   | d78360d6eb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1464804c11 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8629eae5fc | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7b8ba7c3ac | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 77c7466c17 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 82fe76fd37 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3f81ea4ef8 | ||
|   | 745b06a008 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 77b2bd166a | ||
|   | 8e02336c44 | ||
|   | 3adfbde45a | ||
|   | e6db8e2750 | ||
|   | 9cd0ee9b49 | ||
|   | d455d1fcd7 | ||
|   | ea19ff783f | ||
|   | 447e4c7d58 | ||
|   | 89031447f2 | ||
|   | 4210a3a555 | ||
|   | 42e146b8c7 | ||
|   | a5fdcda673 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 52823f5024 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1326c257a0 | ||
|   | fc3137cd96 | ||
|   | a5c53a96d1 | ||
|   | c46cc73f52 | ||
|   | 276e619caa | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f11acacedd | ||
|   | f636a5ec63 | ||
|   | d1bac90745 | ||
|   | 785362c576 | ||
|   | e98f628d34 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b2ab61559c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | dd6eb8e74f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 97cdd03ea4 | ||
|   | f5118e6802 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 94ca5cf679 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5a55a1f602 | ||
|   | 1e7ba7d173 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ed33635a15 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 888682e5d0 | ||
|   | 72bb5f00b0 | ||
|   | 773fd6f59f | ||
|   | aa784e98f8 | ||
|   | 5cce86d924 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 84567bcb00 | ||
|   | d7c2ab1d16 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0d359ade0c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ffbec24290 | ||
|   | 586474c8e6 | ||
|   | 70b6636f50 | ||
|   | ab357deb48 | ||
|   | bd62d1a1c7 | ||
|   | 6130c3dfa5 | ||
|   | 26692d6633 | ||
|   | bb0f200429 | ||
|   | 5787588500 | ||
|   | bed62edc02 | ||
|   | ee0f389c78 | ||
|   | b40383b5a4 | ||
|   | be8903128d | ||
|   | 08800ec16d | ||
|   | 83e274ff9f | ||
|   | 0dd8b1053c | ||
|   | 0558fcf5d5 | ||
|   | 5af8be4293 | ||
|   | cbacdd67eb | ||
|   | c45bbe3ec5 | ||
|   | 746028afbc | ||
|   | b79537ebbc | ||
|   | 6efd581500 | ||
|   | 07e598e48f | ||
|   | f6f00dfcda | ||
|   | 63a6bdc1d6 | ||
|   | abbac057ed | ||
|   | c978322036 | ||
|   | 39d2f1a72c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 86919b8841 | ||
|   | 1448d8d4af | ||
|   | 8d9a387587 | ||
|   | 00722bc463 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6a34a1996f | ||
|   | 1d201b04ba | ||
|   | e887a59158 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d1b8f652a7 | ||
|   | 0707aa47c9 | ||
|   | 4aa7bc51bc | ||
|   | 0f0030916f | ||
|   | 8c57d616cf | ||
|   | 66660507e0 | ||
|   | a12490c3eb | ||
|   | bbf1e4fe61 | ||
|   | febac6fa40 | ||
|   | 99ee8a780d | ||
|   | 577a0d8ed9 | ||
|   | 02ae14894a | ||
|   | 36e5f36660 | ||
|   | 78dbe7fbbc | ||
|   | 669293566b | ||
|   | 350eae7813 | ||
|   | 12dc198a86 | ||
|   | fb2533d66a | ||
|   | 707c7be5bd | ||
|   | 37d6dcc7ea | ||
|   | becd8c4eaf | ||
|   | 530fcc0fea | ||
|   | 7135bdd6aa | ||
|   | 3198c3b081 | ||
|   | 339ca8e30f | ||
|   | 742d78a505 | ||
|   | 120bf37196 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2a40a6b35e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ea1f35b45a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 15e63378a7 | ||
|   | f3bc504a6f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3a8fae47a0 | ||
|   | 70cb1cd100 | ||
|   | 2067cc1670 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e6338976dd | ||
|   | 90ebd5d5ed | ||
|   | c973ee8649 | ||
|   | c1543f034c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6baf339ecb | ||
|   | 11906ef1c9 | ||
|   | ee8902154a | ||
|   | b5bc1988e5 | ||
|   | 2147012beb | ||
|   | 52bb561689 | ||
|   | 25ce7a83f1 | ||
|   | 55c8a590e7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 28bd993680 | ||
|   | 8330f37d8a | ||
|   | 985fae65b6 | ||
|   | db2d590e8e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c8d356783a | ||
|   | 4947450ff0 | ||
|   | de4e91ff62 | ||
|   | fe5e3d5f6d | ||
|   | b8b3098022 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0ae8fc46b8 | ||
|   | e0762f63a8 | ||
|   | 9f3850000c | ||
|   | 76c6be9ba7 | ||
|   | 308a5aa781 | ||
|   | e244527538 | ||
|   | 11dd013333 | ||
|   | b740d5854c | ||
|   | d5445cfbef | ||
|   | 3effaefda7 | ||
|   | 49b19e0eaf | ||
|   | caef3a923b | ||
|   | d4c3ceaf2b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 25e6aecf13 | ||
|   | 33c11fdee9 | ||
|   | 316dd92667 | ||
|   | e53d2ac449 | ||
|   | 5786e8cc7a | ||
|   | 1b717c9b10 | ||
|   | 35abae99ca | ||
|   | d1a85982fb | ||
|   | 3446b913cd | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8b8fd6aab7 | ||
|   | 520bb64eca | ||
|   | 60d266b2d7 | ||
|   | 8deeef4f7d | ||
|   | 9b0b071c0c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 81a3f1098d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | efc248dcdb | ||
|   | 476f3d328d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 69f5f88183 | ||
|   | 6df63f7fc7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 295b8a0135 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fcc5bc5473 | ||
|   | 408b834376 | ||
|   | 986812b9e4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8d4333ad9d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9ff9097ff9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1ef424a2f1 | ||
|   | 9fd96dbaa2 | ||
|   | b0a4e11c46 | ||
|   | 77bce43ace | ||
|   | cba1927cc7 | ||
|   | 3d19c5c2ad | ||
|   | e0eff15694 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0bdeeea83b | ||
|   | 7669e79da1 | ||
|   | 6f96daae56 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d1021d19da | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ee589ac7f0 | ||
|   | 3b747ffecf | ||
|   | 4e5a2b9f96 | ||
|   | aeb4350ccb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9609990832 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0e4319b757 | ||
|   | c8f4907f77 | ||
|   | dcf98c2298 | ||
|   | ae59c7442f | ||
|   | 98708118d8 | ||
|   | 276d8f8e1e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 87f89541b5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 93f6de7029 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 73d2686b17 | ||
|   | c446a95b07 | ||
|   | 3676e1df35 | ||
|   | 12e2705260 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 31e777a03a | ||
|   | 7f436c405b | ||
|   | 4d4d2ab087 | ||
|   | 91017acce4 | ||
|   | c0bfa297bb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e90fd231d9 | ||
|   | cd9f0789de | ||
|   | 5f4c8d92df | ||
|   | 263cb47a21 | ||
|   | 005600c99e | ||
|   | 26bec7fe2f | ||
|   | f4b886d977 | ||
|   | 75fd9b2631 | ||
|   | c09d0d882e | ||
|   | 312cb2996c | ||
|   | f218902581 | ||
|   | f27009216c | ||
|   | 6b680fb2c0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | dda52ebc2e | ||
|   | 9ac8d38bab | ||
|   | 6a54328f7d | ||
|   | 7279862def | ||
|   | 08ce4c872c | ||
|   | 27ffe4fcdc | ||
|   | 8afcaccb8a | ||
|   | c83b13e374 | ||
|   | 2b0c5b1e21 | ||
|   | 3d5c694daa | ||
|   | 23360057b9 | ||
|   | d153232969 | ||
|   | bb0aa8d5cc | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d69f3b0893 | ||
|   | 565838ad43 | ||
|   | 8b52461271 | ||
|   | d08381dfed | ||
|   | b6c45f2df3 | ||
|   | b9479405e1 | ||
|   | a238ff19bf | ||
|   | c93b08d0c7 | ||
|   | 1470b7117a | ||
|   | 7cbc67f4fc | ||
|   | 0a76bbb2b0 | ||
|   | 09cc59a1c1 | ||
|   | 60f7113105 | ||
|   | 26c0c1b7cd | ||
|   | bf646be482 | ||
|   | cc7e17960b | ||
|   | 3c75b170f0 | ||
|   | 764c94c9cc | ||
|   | a79c474957 | ||
|   | 4bb480a238 | ||
|   | 9ffa935c0c | ||
|   | 3d87ee41b3 | ||
|   | 0a32268784 | ||
|   | ae3b8c06f6 | ||
|   | 713c4ad0d2 | ||
|   | fd8832ac98 | ||
|   | 48386c0828 | ||
|   | 625b3921e1 | ||
|   | 48aa37d173 | ||
|   | 228acc196c | ||
|   | 62197f3deb | ||
|   | 2c2314e95c | ||
|   | 5eb2fc3ad0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 82cd9a092c | ||
|   | 62754362c2 | ||
|   | daa9348993 | ||
|   | 2e9dfd6f6f | ||
|   | 7e4499e092 | ||
|   | 396a1575d2 | ||
|   | fc9fe1462f | ||
|   | 41f546ca6b | ||
|   | d037da33cb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | dc2d08c67e | ||
|   | 953d57d1b5 | ||
|   | 96dfc27411 | ||
|   | 171d2e5e99 | ||
|   | 4433892431 | ||
|   | b53d2d03a4 | ||
|   | c1431c0971 | ||
|   | 98a07dad1b | ||
|   | 0ffa22b7a6 | ||
|   | 60a0129fe9 | ||
|   | 62ee60a76c | ||
|   | 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 | 
							
								
								
									
										630
									
								
								.editorconfig
									
									
									
									
									
								
							
							
						
						
									
										630
									
								
								.editorconfig
									
									
									
									
									
								
							| @@ -13,40 +13,6 @@ ij_formatter_tags_enabled = false | ||||
| ij_smart_tabs = false | ||||
| ij_wrap_on_typing = true | ||||
|  | ||||
| [*.conf] | ||||
| indent_size = 2 | ||||
| tab_width = 2 | ||||
| ij_continuation_indent_size = 2 | ||||
| ij_hocon_keep_blank_lines_before_right_brace = 2 | ||||
| ij_hocon_keep_indents_on_empty_lines = false | ||||
| ij_hocon_keep_line_breaks = true | ||||
| ij_hocon_space_after_colon = true | ||||
| ij_hocon_space_after_comma = true | ||||
| ij_hocon_space_before_colon = true | ||||
| ij_hocon_space_before_comma = false | ||||
| ij_hocon_spaces_within_braces = false | ||||
| ij_hocon_spaces_within_brackets = false | ||||
| ij_hocon_spaces_within_method_call_parentheses = false | ||||
|  | ||||
| [*.css] | ||||
| ij_css_align_closing_brace_with_properties = false | ||||
| ij_css_blank_lines_around_nested_selector = 1 | ||||
| ij_css_blank_lines_between_blocks = 1 | ||||
| ij_css_brace_placement = end_of_line | ||||
| ij_css_enforce_quotes_on_format = false | ||||
| ij_css_hex_color_long_format = false | ||||
| ij_css_hex_color_lower_case = false | ||||
| ij_css_hex_color_short_format = false | ||||
| ij_css_hex_color_upper_case = false | ||||
| ij_css_keep_blank_lines_in_code = 2 | ||||
| ij_css_keep_indents_on_empty_lines = false | ||||
| ij_css_keep_single_line_blocks = false | ||||
| ij_css_properties_order = font, font-family, font-size, font-weight, font-style, font-variant, font-size-adjust, font-stretch, line-height, position, z-index, top, right, bottom, left, display, visibility, float, clear, overflow, overflow-x, overflow-y, clip, zoom, align-content, align-items, align-self, flex, flex-flow, flex-basis, flex-direction, flex-grow, flex-shrink, flex-wrap, justify-content, order, box-sizing, width, min-width, max-width, height, min-height, max-height, margin, margin-top, margin-right, margin-bottom, margin-left, padding, padding-top, padding-right, padding-bottom, padding-left, table-layout, empty-cells, caption-side, border-spacing, border-collapse, list-style, list-style-position, list-style-type, list-style-image, content, quotes, counter-reset, counter-increment, resize, cursor, user-select, nav-index, nav-up, nav-right, nav-down, nav-left, transition, transition-delay, transition-timing-function, transition-duration, transition-property, transform, transform-origin, animation, animation-name, animation-duration, animation-play-state, animation-timing-function, animation-delay, animation-iteration-count, animation-direction, text-align, text-align-last, vertical-align, white-space, text-decoration, text-emphasis, text-emphasis-color, text-emphasis-style, text-emphasis-position, text-indent, text-justify, letter-spacing, word-spacing, text-outline, text-transform, text-wrap, text-overflow, text-overflow-ellipsis, text-overflow-mode, word-wrap, word-break, tab-size, hyphens, pointer-events, opacity, color, border, border-width, border-style, border-color, border-top, border-top-width, border-top-style, border-top-color, border-right, border-right-width, border-right-style, border-right-color, border-bottom, border-bottom-width, border-bottom-style, border-bottom-color, border-left, border-left-width, border-left-style, border-left-color, border-radius, border-top-left-radius, border-top-right-radius, border-bottom-right-radius, border-bottom-left-radius, border-image, border-image-source, border-image-slice, border-image-width, border-image-outset, border-image-repeat, outline, outline-width, outline-style, outline-color, outline-offset, background, background-color, background-image, background-repeat, background-attachment, background-position, background-position-x, background-position-y, background-clip, background-origin, background-size, box-decoration-break, box-shadow, text-shadow | ||||
| ij_css_space_after_colon = true | ||||
| ij_css_space_before_opening_brace = true | ||||
| ij_css_use_double_quotes = true | ||||
| ij_css_value_alignment = do_not_align | ||||
|  | ||||
| [*.java] | ||||
| ij_java_align_consecutive_assignments = false | ||||
| ij_java_align_consecutive_variable_declarations = false | ||||
| @@ -277,63 +243,12 @@ ij_java_wrap_comments = false | ||||
| ij_java_wrap_first_method_in_call_chain = true | ||||
| ij_java_wrap_long_lines = false | ||||
|  | ||||
| [*.nbtt] | ||||
| max_line_length = 150 | ||||
| ij_continuation_indent_size = 4 | ||||
| ij_nbtt_keep_indents_on_empty_lines = false | ||||
| ij_nbtt_space_after_colon = true | ||||
| ij_nbtt_space_after_comma = true | ||||
| ij_nbtt_space_before_colon = true | ||||
| ij_nbtt_space_before_comma = false | ||||
| ij_nbtt_spaces_within_brackets = false | ||||
| ij_nbtt_spaces_within_parentheses = false | ||||
|  | ||||
| [*.properties] | ||||
| ij_properties_align_group_field_declarations = false | ||||
| ij_properties_keep_blank_lines = false | ||||
| ij_properties_key_value_delimiter = equals | ||||
| ij_properties_spaces_around_key_value_delimiter = false | ||||
|  | ||||
| [*.sass] | ||||
| indent_size = 2 | ||||
| ij_sass_align_closing_brace_with_properties = false | ||||
| ij_sass_blank_lines_around_nested_selector = 1 | ||||
| ij_sass_blank_lines_between_blocks = 1 | ||||
| ij_sass_brace_placement = 0 | ||||
| ij_sass_enforce_quotes_on_format = false | ||||
| ij_sass_hex_color_long_format = false | ||||
| ij_sass_hex_color_lower_case = false | ||||
| ij_sass_hex_color_short_format = false | ||||
| ij_sass_hex_color_upper_case = false | ||||
| ij_sass_keep_blank_lines_in_code = 2 | ||||
| ij_sass_keep_indents_on_empty_lines = false | ||||
| ij_sass_keep_single_line_blocks = false | ||||
| ij_sass_properties_order = font, font-family, font-size, font-weight, font-style, font-variant, font-size-adjust, font-stretch, line-height, position, z-index, top, right, bottom, left, display, visibility, float, clear, overflow, overflow-x, overflow-y, clip, zoom, align-content, align-items, align-self, flex, flex-flow, flex-basis, flex-direction, flex-grow, flex-shrink, flex-wrap, justify-content, order, box-sizing, width, min-width, max-width, height, min-height, max-height, margin, margin-top, margin-right, margin-bottom, margin-left, padding, padding-top, padding-right, padding-bottom, padding-left, table-layout, empty-cells, caption-side, border-spacing, border-collapse, list-style, list-style-position, list-style-type, list-style-image, content, quotes, counter-reset, counter-increment, resize, cursor, user-select, nav-index, nav-up, nav-right, nav-down, nav-left, transition, transition-delay, transition-timing-function, transition-duration, transition-property, transform, transform-origin, animation, animation-name, animation-duration, animation-play-state, animation-timing-function, animation-delay, animation-iteration-count, animation-direction, text-align, text-align-last, vertical-align, white-space, text-decoration, text-emphasis, text-emphasis-color, text-emphasis-style, text-emphasis-position, text-indent, text-justify, letter-spacing, word-spacing, text-outline, text-transform, text-wrap, text-overflow, text-overflow-ellipsis, text-overflow-mode, word-wrap, word-break, tab-size, hyphens, pointer-events, opacity, color, border, border-width, border-style, border-color, border-top, border-top-width, border-top-style, border-top-color, border-right, border-right-width, border-right-style, border-right-color, border-bottom, border-bottom-width, border-bottom-style, border-bottom-color, border-left, border-left-width, border-left-style, border-left-color, border-radius, border-top-left-radius, border-top-right-radius, border-bottom-right-radius, border-bottom-left-radius, border-image, border-image-source, border-image-slice, border-image-width, border-image-outset, border-image-repeat, outline, outline-width, outline-style, outline-color, outline-offset, background, background-color, background-image, background-repeat, background-attachment, background-position, background-position-x, background-position-y, background-clip, background-origin, background-size, box-decoration-break, box-shadow, text-shadow | ||||
| ij_sass_space_after_colon = true | ||||
| ij_sass_space_before_opening_brace = true | ||||
| ij_sass_use_double_quotes = true | ||||
| ij_sass_value_alignment = 0 | ||||
|  | ||||
| [*.scss] | ||||
| indent_size = 2 | ||||
| ij_scss_align_closing_brace_with_properties = false | ||||
| ij_scss_blank_lines_around_nested_selector = 1 | ||||
| ij_scss_blank_lines_between_blocks = 1 | ||||
| ij_scss_brace_placement = 0 | ||||
| ij_scss_enforce_quotes_on_format = false | ||||
| ij_scss_hex_color_long_format = false | ||||
| ij_scss_hex_color_lower_case = false | ||||
| ij_scss_hex_color_short_format = false | ||||
| ij_scss_hex_color_upper_case = false | ||||
| ij_scss_keep_blank_lines_in_code = 2 | ||||
| ij_scss_keep_indents_on_empty_lines = false | ||||
| ij_scss_keep_single_line_blocks = false | ||||
| ij_scss_properties_order = font, font-family, font-size, font-weight, font-style, font-variant, font-size-adjust, font-stretch, line-height, position, z-index, top, right, bottom, left, display, visibility, float, clear, overflow, overflow-x, overflow-y, clip, zoom, align-content, align-items, align-self, flex, flex-flow, flex-basis, flex-direction, flex-grow, flex-shrink, flex-wrap, justify-content, order, box-sizing, width, min-width, max-width, height, min-height, max-height, margin, margin-top, margin-right, margin-bottom, margin-left, padding, padding-top, padding-right, padding-bottom, padding-left, table-layout, empty-cells, caption-side, border-spacing, border-collapse, list-style, list-style-position, list-style-type, list-style-image, content, quotes, counter-reset, counter-increment, resize, cursor, user-select, nav-index, nav-up, nav-right, nav-down, nav-left, transition, transition-delay, transition-timing-function, transition-duration, transition-property, transform, transform-origin, animation, animation-name, animation-duration, animation-play-state, animation-timing-function, animation-delay, animation-iteration-count, animation-direction, text-align, text-align-last, vertical-align, white-space, text-decoration, text-emphasis, text-emphasis-color, text-emphasis-style, text-emphasis-position, text-indent, text-justify, letter-spacing, word-spacing, text-outline, text-transform, text-wrap, text-overflow, text-overflow-ellipsis, text-overflow-mode, word-wrap, word-break, tab-size, hyphens, pointer-events, opacity, color, border, border-width, border-style, border-color, border-top, border-top-width, border-top-style, border-top-color, border-right, border-right-width, border-right-style, border-right-color, border-bottom, border-bottom-width, border-bottom-style, border-bottom-color, border-left, border-left-width, border-left-style, border-left-color, border-radius, border-top-left-radius, border-top-right-radius, border-bottom-right-radius, border-bottom-left-radius, border-image, border-image-source, border-image-slice, border-image-width, border-image-outset, border-image-repeat, outline, outline-width, outline-style, outline-color, outline-offset, background, background-color, background-image, background-repeat, background-attachment, background-position, background-position-x, background-position-y, background-clip, background-origin, background-size, box-decoration-break, box-shadow, text-shadow | ||||
| ij_scss_space_after_colon = true | ||||
| ij_scss_space_before_opening_brace = true | ||||
| ij_scss_use_double_quotes = true | ||||
| ij_scss_value_alignment = 0 | ||||
|  | ||||
| [.editorconfig] | ||||
| ij_editorconfig_align_group_field_declarations = false | ||||
| ij_editorconfig_space_after_colon = false | ||||
| @@ -342,548 +257,6 @@ ij_editorconfig_space_before_colon = false | ||||
| ij_editorconfig_space_before_comma = false | ||||
| ij_editorconfig_spaces_around_assignment_operators = true | ||||
|  | ||||
| [{*.ant, *.fxml, *.jhm, *.jnlp, *.jrxml, *.pom, *.rng, *.tld, *.wsdl, *.xml, *.xsd, *.xsl, *.xslt, *.xul}] | ||||
| ij_xml_align_attributes = true | ||||
| ij_xml_align_text = false | ||||
| ij_xml_attribute_wrap = normal | ||||
| ij_xml_block_comment_at_first_column = true | ||||
| ij_xml_keep_blank_lines = 2 | ||||
| ij_xml_keep_indents_on_empty_lines = false | ||||
| ij_xml_keep_line_breaks = true | ||||
| ij_xml_keep_line_breaks_in_text = true | ||||
| ij_xml_keep_whitespaces = false | ||||
| ij_xml_keep_whitespaces_around_cdata = preserve | ||||
| ij_xml_keep_whitespaces_inside_cdata = false | ||||
| ij_xml_line_comment_at_first_column = true | ||||
| ij_xml_space_after_tag_name = false | ||||
| ij_xml_space_around_equals_in_attribute = false | ||||
| ij_xml_space_inside_empty_tag = false | ||||
| ij_xml_text_wrap = normal | ||||
|  | ||||
| [{*.ats, *.ts}] | ||||
| ij_continuation_indent_size = 4 | ||||
| ij_typescript_align_imports = false | ||||
| ij_typescript_align_multiline_array_initializer_expression = false | ||||
| ij_typescript_align_multiline_binary_operation = false | ||||
| ij_typescript_align_multiline_chained_methods = false | ||||
| ij_typescript_align_multiline_extends_list = false | ||||
| ij_typescript_align_multiline_for = true | ||||
| ij_typescript_align_multiline_parameters = true | ||||
| ij_typescript_align_multiline_parameters_in_calls = false | ||||
| ij_typescript_align_multiline_ternary_operation = false | ||||
| ij_typescript_align_object_properties = 0 | ||||
| ij_typescript_align_union_types = false | ||||
| ij_typescript_align_var_statements = 0 | ||||
| ij_typescript_array_initializer_new_line_after_left_brace = false | ||||
| ij_typescript_array_initializer_right_brace_on_new_line = false | ||||
| ij_typescript_array_initializer_wrap = off | ||||
| ij_typescript_assignment_wrap = off | ||||
| ij_typescript_binary_operation_sign_on_next_line = false | ||||
| ij_typescript_binary_operation_wrap = off | ||||
| ij_typescript_blacklist_imports = rxjs/Rx, node_modules/**, **/node_modules/**, @angular/material, @angular/material/typings/** | ||||
| ij_typescript_blank_lines_after_imports = 1 | ||||
| ij_typescript_blank_lines_around_class = 1 | ||||
| ij_typescript_blank_lines_around_field = 0 | ||||
| ij_typescript_blank_lines_around_field_in_interface = 0 | ||||
| ij_typescript_blank_lines_around_function = 1 | ||||
| ij_typescript_blank_lines_around_method = 1 | ||||
| ij_typescript_blank_lines_around_method_in_interface = 1 | ||||
| ij_typescript_block_brace_style = end_of_line | ||||
| ij_typescript_call_parameters_new_line_after_left_paren = false | ||||
| ij_typescript_call_parameters_right_paren_on_new_line = false | ||||
| ij_typescript_call_parameters_wrap = off | ||||
| ij_typescript_catch_on_new_line = false | ||||
| ij_typescript_chained_call_dot_on_new_line = true | ||||
| ij_typescript_class_brace_style = end_of_line | ||||
| ij_typescript_comma_on_new_line = false | ||||
| ij_typescript_do_while_brace_force = never | ||||
| ij_typescript_else_on_new_line = false | ||||
| ij_typescript_enforce_trailing_comma = keep | ||||
| ij_typescript_extends_keyword_wrap = off | ||||
| ij_typescript_extends_list_wrap = off | ||||
| ij_typescript_field_prefix = _ | ||||
| ij_typescript_file_name_style = relaxed | ||||
| ij_typescript_finally_on_new_line = false | ||||
| ij_typescript_for_brace_force = never | ||||
| ij_typescript_for_statement_new_line_after_left_paren = false | ||||
| ij_typescript_for_statement_right_paren_on_new_line = false | ||||
| ij_typescript_for_statement_wrap = off | ||||
| ij_typescript_force_quote_style = false | ||||
| ij_typescript_force_semicolon_style = false | ||||
| ij_typescript_function_expression_brace_style = end_of_line | ||||
| ij_typescript_if_brace_force = never | ||||
| ij_typescript_import_merge_members = global | ||||
| ij_typescript_import_prefer_absolute_path = global | ||||
| ij_typescript_import_sort_members = true | ||||
| ij_typescript_import_sort_module_name = false | ||||
| ij_typescript_import_use_node_resolution = true | ||||
| ij_typescript_imports_wrap = on_every_item | ||||
| ij_typescript_indent_case_from_switch = true | ||||
| ij_typescript_indent_chained_calls = true | ||||
| ij_typescript_indent_package_children = 0 | ||||
| ij_typescript_jsdoc_include_types = false | ||||
| ij_typescript_jsx_attribute_value = braces | ||||
| ij_typescript_keep_blank_lines_in_code = 2 | ||||
| ij_typescript_keep_first_column_comment = true | ||||
| ij_typescript_keep_indents_on_empty_lines = false | ||||
| ij_typescript_keep_line_breaks = true | ||||
| ij_typescript_keep_simple_blocks_in_one_line = false | ||||
| ij_typescript_keep_simple_methods_in_one_line = false | ||||
| ij_typescript_line_comment_add_space = true | ||||
| ij_typescript_line_comment_at_first_column = false | ||||
| ij_typescript_method_brace_style = end_of_line | ||||
| ij_typescript_method_call_chain_wrap = off | ||||
| ij_typescript_method_parameters_new_line_after_left_paren = false | ||||
| ij_typescript_method_parameters_right_paren_on_new_line = false | ||||
| ij_typescript_method_parameters_wrap = off | ||||
| ij_typescript_object_literal_wrap = on_every_item | ||||
| ij_typescript_parentheses_expression_new_line_after_left_paren = false | ||||
| ij_typescript_parentheses_expression_right_paren_on_new_line = false | ||||
| ij_typescript_place_assignment_sign_on_next_line = false | ||||
| ij_typescript_prefer_as_type_cast = false | ||||
| ij_typescript_prefer_explicit_types_function_expression_returns = false | ||||
| ij_typescript_prefer_explicit_types_function_returns = false | ||||
| ij_typescript_prefer_explicit_types_vars_fields = false | ||||
| ij_typescript_prefer_parameters_wrap = false | ||||
| ij_typescript_reformat_c_style_comments = false | ||||
| ij_typescript_space_after_colon = true | ||||
| ij_typescript_space_after_comma = true | ||||
| ij_typescript_space_after_dots_in_rest_parameter = false | ||||
| ij_typescript_space_after_generator_mult = true | ||||
| ij_typescript_space_after_property_colon = true | ||||
| ij_typescript_space_after_quest = true | ||||
| ij_typescript_space_after_type_colon = true | ||||
| ij_typescript_space_after_unary_not = false | ||||
| ij_typescript_space_before_async_arrow_lparen = true | ||||
| ij_typescript_space_before_catch_keyword = true | ||||
| ij_typescript_space_before_catch_left_brace = true | ||||
| ij_typescript_space_before_catch_parentheses = true | ||||
| ij_typescript_space_before_class_lbrace = true | ||||
| ij_typescript_space_before_class_left_brace = true | ||||
| ij_typescript_space_before_colon = true | ||||
| ij_typescript_space_before_comma = false | ||||
| ij_typescript_space_before_do_left_brace = true | ||||
| ij_typescript_space_before_else_keyword = true | ||||
| ij_typescript_space_before_else_left_brace = true | ||||
| ij_typescript_space_before_finally_keyword = true | ||||
| ij_typescript_space_before_finally_left_brace = true | ||||
| ij_typescript_space_before_for_left_brace = true | ||||
| ij_typescript_space_before_for_parentheses = true | ||||
| ij_typescript_space_before_for_semicolon = false | ||||
| ij_typescript_space_before_function_left_parenth = true | ||||
| ij_typescript_space_before_generator_mult = false | ||||
| ij_typescript_space_before_if_left_brace = true | ||||
| ij_typescript_space_before_if_parentheses = true | ||||
| ij_typescript_space_before_method_call_parentheses = false | ||||
| ij_typescript_space_before_method_left_brace = true | ||||
| ij_typescript_space_before_method_parentheses = false | ||||
| ij_typescript_space_before_property_colon = false | ||||
| ij_typescript_space_before_quest = true | ||||
| ij_typescript_space_before_switch_left_brace = true | ||||
| ij_typescript_space_before_switch_parentheses = true | ||||
| ij_typescript_space_before_try_left_brace = true | ||||
| ij_typescript_space_before_type_colon = false | ||||
| ij_typescript_space_before_unary_not = false | ||||
| ij_typescript_space_before_while_keyword = true | ||||
| ij_typescript_space_before_while_left_brace = true | ||||
| ij_typescript_space_before_while_parentheses = true | ||||
| ij_typescript_spaces_around_additive_operators = true | ||||
| ij_typescript_spaces_around_arrow_function_operator = true | ||||
| ij_typescript_spaces_around_assignment_operators = true | ||||
| ij_typescript_spaces_around_bitwise_operators = true | ||||
| ij_typescript_spaces_around_equality_operators = true | ||||
| ij_typescript_spaces_around_logical_operators = true | ||||
| ij_typescript_spaces_around_multiplicative_operators = true | ||||
| ij_typescript_spaces_around_relational_operators = true | ||||
| ij_typescript_spaces_around_shift_operators = true | ||||
| ij_typescript_spaces_around_unary_operator = false | ||||
| ij_typescript_spaces_within_array_initializer_brackets = false | ||||
| ij_typescript_spaces_within_brackets = false | ||||
| ij_typescript_spaces_within_catch_parentheses = false | ||||
| ij_typescript_spaces_within_for_parentheses = false | ||||
| ij_typescript_spaces_within_if_parentheses = false | ||||
| ij_typescript_spaces_within_imports = false | ||||
| ij_typescript_spaces_within_interpolation_expressions = false | ||||
| ij_typescript_spaces_within_method_call_parentheses = false | ||||
| ij_typescript_spaces_within_method_parentheses = false | ||||
| ij_typescript_spaces_within_object_literal_braces = false | ||||
| ij_typescript_spaces_within_object_type_braces = true | ||||
| ij_typescript_spaces_within_parentheses = false | ||||
| ij_typescript_spaces_within_switch_parentheses = false | ||||
| ij_typescript_spaces_within_type_assertion = false | ||||
| ij_typescript_spaces_within_union_types = true | ||||
| ij_typescript_spaces_within_while_parentheses = false | ||||
| ij_typescript_special_else_if_treatment = true | ||||
| ij_typescript_ternary_operation_signs_on_next_line = false | ||||
| ij_typescript_ternary_operation_wrap = off | ||||
| ij_typescript_union_types_wrap = on_every_item | ||||
| ij_typescript_use_chained_calls_group_indents = false | ||||
| ij_typescript_use_double_quotes = true | ||||
| ij_typescript_use_explicit_js_extension = global | ||||
| ij_typescript_use_path_mapping = always | ||||
| ij_typescript_use_public_modifier = false | ||||
| ij_typescript_use_semicolon_after_statement = true | ||||
| ij_typescript_var_declaration_wrap = normal | ||||
| ij_typescript_while_brace_force = never | ||||
| ij_typescript_while_on_new_line = false | ||||
| ij_typescript_wrap_comments = false | ||||
|  | ||||
| [{*.bash, *.sh, *.zsh}] | ||||
| indent_size = 2 | ||||
| tab_width = 2 | ||||
| ij_shell_binary_ops_start_line = false | ||||
| ij_shell_keep_column_alignment_padding = false | ||||
| ij_shell_minify_program = false | ||||
| ij_shell_redirect_followed_by_space = false | ||||
| ij_shell_switch_cases_indented = false | ||||
|  | ||||
| [{*.cjs, *.js}] | ||||
| ij_continuation_indent_size = 4 | ||||
| ij_javascript_align_imports = false | ||||
| ij_javascript_align_multiline_array_initializer_expression = false | ||||
| ij_javascript_align_multiline_binary_operation = false | ||||
| ij_javascript_align_multiline_chained_methods = false | ||||
| ij_javascript_align_multiline_extends_list = false | ||||
| ij_javascript_align_multiline_for = true | ||||
| ij_javascript_align_multiline_parameters = true | ||||
| ij_javascript_align_multiline_parameters_in_calls = false | ||||
| ij_javascript_align_multiline_ternary_operation = false | ||||
| ij_javascript_align_object_properties = 0 | ||||
| ij_javascript_align_union_types = false | ||||
| ij_javascript_align_var_statements = 0 | ||||
| ij_javascript_array_initializer_new_line_after_left_brace = false | ||||
| ij_javascript_array_initializer_right_brace_on_new_line = false | ||||
| ij_javascript_array_initializer_wrap = off | ||||
| ij_javascript_assignment_wrap = off | ||||
| ij_javascript_binary_operation_sign_on_next_line = false | ||||
| ij_javascript_binary_operation_wrap = off | ||||
| ij_javascript_blacklist_imports = rxjs/Rx, node_modules/**, **/node_modules/**, @angular/material, @angular/material/typings/** | ||||
| ij_javascript_blank_lines_after_imports = 1 | ||||
| ij_javascript_blank_lines_around_class = 1 | ||||
| ij_javascript_blank_lines_around_field = 0 | ||||
| ij_javascript_blank_lines_around_function = 1 | ||||
| ij_javascript_blank_lines_around_method = 1 | ||||
| ij_javascript_block_brace_style = end_of_line | ||||
| ij_javascript_call_parameters_new_line_after_left_paren = false | ||||
| ij_javascript_call_parameters_right_paren_on_new_line = false | ||||
| ij_javascript_call_parameters_wrap = off | ||||
| ij_javascript_catch_on_new_line = false | ||||
| ij_javascript_chained_call_dot_on_new_line = true | ||||
| ij_javascript_class_brace_style = end_of_line | ||||
| ij_javascript_comma_on_new_line = false | ||||
| ij_javascript_do_while_brace_force = never | ||||
| ij_javascript_else_on_new_line = false | ||||
| ij_javascript_enforce_trailing_comma = keep | ||||
| ij_javascript_extends_keyword_wrap = off | ||||
| ij_javascript_extends_list_wrap = off | ||||
| ij_javascript_field_prefix = _ | ||||
| ij_javascript_file_name_style = relaxed | ||||
| ij_javascript_finally_on_new_line = false | ||||
| ij_javascript_for_brace_force = never | ||||
| ij_javascript_for_statement_new_line_after_left_paren = false | ||||
| ij_javascript_for_statement_right_paren_on_new_line = false | ||||
| ij_javascript_for_statement_wrap = off | ||||
| ij_javascript_force_quote_style = false | ||||
| ij_javascript_force_semicolon_style = false | ||||
| ij_javascript_function_expression_brace_style = end_of_line | ||||
| ij_javascript_if_brace_force = never | ||||
| ij_javascript_import_merge_members = global | ||||
| ij_javascript_import_prefer_absolute_path = global | ||||
| ij_javascript_import_sort_members = true | ||||
| ij_javascript_import_sort_module_name = false | ||||
| ij_javascript_import_use_node_resolution = true | ||||
| ij_javascript_imports_wrap = on_every_item | ||||
| ij_javascript_indent_case_from_switch = true | ||||
| ij_javascript_indent_chained_calls = true | ||||
| ij_javascript_indent_package_children = 0 | ||||
| ij_javascript_jsx_attribute_value = braces | ||||
| ij_javascript_keep_blank_lines_in_code = 2 | ||||
| ij_javascript_keep_first_column_comment = true | ||||
| ij_javascript_keep_indents_on_empty_lines = false | ||||
| ij_javascript_keep_line_breaks = true | ||||
| ij_javascript_keep_simple_blocks_in_one_line = false | ||||
| ij_javascript_keep_simple_methods_in_one_line = false | ||||
| ij_javascript_line_comment_add_space = true | ||||
| ij_javascript_line_comment_at_first_column = false | ||||
| ij_javascript_method_brace_style = end_of_line | ||||
| ij_javascript_method_call_chain_wrap = off | ||||
| ij_javascript_method_parameters_new_line_after_left_paren = false | ||||
| ij_javascript_method_parameters_right_paren_on_new_line = false | ||||
| ij_javascript_method_parameters_wrap = off | ||||
| ij_javascript_object_literal_wrap = on_every_item | ||||
| ij_javascript_parentheses_expression_new_line_after_left_paren = false | ||||
| ij_javascript_parentheses_expression_right_paren_on_new_line = false | ||||
| ij_javascript_place_assignment_sign_on_next_line = false | ||||
| ij_javascript_prefer_as_type_cast = false | ||||
| ij_javascript_prefer_explicit_types_function_expression_returns = false | ||||
| ij_javascript_prefer_explicit_types_function_returns = false | ||||
| ij_javascript_prefer_explicit_types_vars_fields = false | ||||
| ij_javascript_prefer_parameters_wrap = false | ||||
| ij_javascript_reformat_c_style_comments = false | ||||
| ij_javascript_space_after_colon = true | ||||
| ij_javascript_space_after_comma = true | ||||
| ij_javascript_space_after_dots_in_rest_parameter = false | ||||
| ij_javascript_space_after_generator_mult = true | ||||
| ij_javascript_space_after_property_colon = true | ||||
| ij_javascript_space_after_quest = true | ||||
| ij_javascript_space_after_type_colon = true | ||||
| ij_javascript_space_after_unary_not = false | ||||
| ij_javascript_space_before_async_arrow_lparen = true | ||||
| ij_javascript_space_before_catch_keyword = true | ||||
| ij_javascript_space_before_catch_left_brace = true | ||||
| ij_javascript_space_before_catch_parentheses = true | ||||
| ij_javascript_space_before_class_lbrace = true | ||||
| ij_javascript_space_before_class_left_brace = true | ||||
| ij_javascript_space_before_colon = true | ||||
| ij_javascript_space_before_comma = false | ||||
| ij_javascript_space_before_do_left_brace = true | ||||
| ij_javascript_space_before_else_keyword = true | ||||
| ij_javascript_space_before_else_left_brace = true | ||||
| ij_javascript_space_before_finally_keyword = true | ||||
| ij_javascript_space_before_finally_left_brace = true | ||||
| ij_javascript_space_before_for_left_brace = true | ||||
| ij_javascript_space_before_for_parentheses = true | ||||
| ij_javascript_space_before_for_semicolon = false | ||||
| ij_javascript_space_before_function_left_parenth = true | ||||
| ij_javascript_space_before_generator_mult = false | ||||
| ij_javascript_space_before_if_left_brace = true | ||||
| ij_javascript_space_before_if_parentheses = true | ||||
| ij_javascript_space_before_method_call_parentheses = false | ||||
| ij_javascript_space_before_method_left_brace = true | ||||
| ij_javascript_space_before_method_parentheses = false | ||||
| ij_javascript_space_before_property_colon = false | ||||
| ij_javascript_space_before_quest = true | ||||
| ij_javascript_space_before_switch_left_brace = true | ||||
| ij_javascript_space_before_switch_parentheses = true | ||||
| ij_javascript_space_before_try_left_brace = true | ||||
| ij_javascript_space_before_type_colon = false | ||||
| ij_javascript_space_before_unary_not = false | ||||
| ij_javascript_space_before_while_keyword = true | ||||
| ij_javascript_space_before_while_left_brace = true | ||||
| ij_javascript_space_before_while_parentheses = true | ||||
| ij_javascript_spaces_around_additive_operators = true | ||||
| ij_javascript_spaces_around_arrow_function_operator = true | ||||
| ij_javascript_spaces_around_assignment_operators = true | ||||
| ij_javascript_spaces_around_bitwise_operators = true | ||||
| ij_javascript_spaces_around_equality_operators = true | ||||
| ij_javascript_spaces_around_logical_operators = true | ||||
| ij_javascript_spaces_around_multiplicative_operators = true | ||||
| ij_javascript_spaces_around_relational_operators = true | ||||
| ij_javascript_spaces_around_shift_operators = true | ||||
| ij_javascript_spaces_around_unary_operator = false | ||||
| ij_javascript_spaces_within_array_initializer_brackets = false | ||||
| ij_javascript_spaces_within_brackets = false | ||||
| ij_javascript_spaces_within_catch_parentheses = false | ||||
| ij_javascript_spaces_within_for_parentheses = false | ||||
| ij_javascript_spaces_within_if_parentheses = false | ||||
| ij_javascript_spaces_within_imports = false | ||||
| ij_javascript_spaces_within_interpolation_expressions = false | ||||
| ij_javascript_spaces_within_method_call_parentheses = false | ||||
| ij_javascript_spaces_within_method_parentheses = false | ||||
| ij_javascript_spaces_within_object_literal_braces = false | ||||
| ij_javascript_spaces_within_object_type_braces = true | ||||
| ij_javascript_spaces_within_parentheses = false | ||||
| ij_javascript_spaces_within_switch_parentheses = false | ||||
| ij_javascript_spaces_within_type_assertion = false | ||||
| ij_javascript_spaces_within_union_types = true | ||||
| ij_javascript_spaces_within_while_parentheses = false | ||||
| ij_javascript_special_else_if_treatment = true | ||||
| ij_javascript_ternary_operation_signs_on_next_line = false | ||||
| ij_javascript_ternary_operation_wrap = off | ||||
| ij_javascript_union_types_wrap = on_every_item | ||||
| ij_javascript_use_chained_calls_group_indents = false | ||||
| ij_javascript_use_double_quotes = true | ||||
| ij_javascript_use_explicit_js_extension = global | ||||
| ij_javascript_use_path_mapping = always | ||||
| ij_javascript_use_public_modifier = false | ||||
| ij_javascript_use_semicolon_after_statement = true | ||||
| ij_javascript_var_declaration_wrap = normal | ||||
| ij_javascript_while_brace_force = never | ||||
| ij_javascript_while_on_new_line = false | ||||
| ij_javascript_wrap_comments = false | ||||
|  | ||||
| [{*.ft, *.vm, *.vsl}] | ||||
| ij_vtl_keep_indents_on_empty_lines = false | ||||
|  | ||||
| [{*.gant, *.gradle, *.groovy, *.gy}] | ||||
| ij_groovy_align_group_field_declarations = false | ||||
| ij_groovy_align_multiline_array_initializer_expression = false | ||||
| ij_groovy_align_multiline_assignment = false | ||||
| ij_groovy_align_multiline_binary_operation = false | ||||
| ij_groovy_align_multiline_chained_methods = false | ||||
| ij_groovy_align_multiline_extends_list = false | ||||
| ij_groovy_align_multiline_for = true | ||||
| ij_groovy_align_multiline_list_or_map = true | ||||
| ij_groovy_align_multiline_method_parentheses = false | ||||
| ij_groovy_align_multiline_parameters = true | ||||
| ij_groovy_align_multiline_parameters_in_calls = false | ||||
| ij_groovy_align_multiline_resources = true | ||||
| ij_groovy_align_multiline_ternary_operation = false | ||||
| ij_groovy_align_multiline_throws_list = false | ||||
| ij_groovy_align_named_args_in_map = true | ||||
| ij_groovy_align_throws_keyword = false | ||||
| ij_groovy_array_initializer_new_line_after_left_brace = false | ||||
| ij_groovy_array_initializer_right_brace_on_new_line = false | ||||
| ij_groovy_array_initializer_wrap = off | ||||
| ij_groovy_assert_statement_wrap = off | ||||
| ij_groovy_assignment_wrap = off | ||||
| ij_groovy_binary_operation_wrap = off | ||||
| ij_groovy_blank_lines_after_class_header = 0 | ||||
| ij_groovy_blank_lines_after_imports = 1 | ||||
| ij_groovy_blank_lines_after_package = 1 | ||||
| ij_groovy_blank_lines_around_class = 1 | ||||
| ij_groovy_blank_lines_around_field = 0 | ||||
| ij_groovy_blank_lines_around_field_in_interface = 0 | ||||
| ij_groovy_blank_lines_around_method = 1 | ||||
| ij_groovy_blank_lines_around_method_in_interface = 1 | ||||
| ij_groovy_blank_lines_before_imports = 1 | ||||
| ij_groovy_blank_lines_before_method_body = 0 | ||||
| ij_groovy_blank_lines_before_package = 0 | ||||
| ij_groovy_block_brace_style = end_of_line | ||||
| ij_groovy_block_comment_at_first_column = true | ||||
| ij_groovy_call_parameters_new_line_after_left_paren = false | ||||
| ij_groovy_call_parameters_right_paren_on_new_line = false | ||||
| ij_groovy_call_parameters_wrap = off | ||||
| ij_groovy_catch_on_new_line = false | ||||
| ij_groovy_class_annotation_wrap = split_into_lines | ||||
| ij_groovy_class_brace_style = end_of_line | ||||
| ij_groovy_class_count_to_use_import_on_demand = 5 | ||||
| ij_groovy_do_while_brace_force = never | ||||
| ij_groovy_else_on_new_line = false | ||||
| ij_groovy_enum_constants_wrap = off | ||||
| ij_groovy_extends_keyword_wrap = off | ||||
| ij_groovy_extends_list_wrap = off | ||||
| ij_groovy_field_annotation_wrap = split_into_lines | ||||
| ij_groovy_finally_on_new_line = false | ||||
| ij_groovy_for_brace_force = never | ||||
| ij_groovy_for_statement_new_line_after_left_paren = false | ||||
| ij_groovy_for_statement_right_paren_on_new_line = false | ||||
| ij_groovy_for_statement_wrap = off | ||||
| ij_groovy_if_brace_force = never | ||||
| ij_groovy_import_annotation_wrap = 2 | ||||
| ij_groovy_imports_layout = *, |, javax.**, java.**, |, $* | ||||
| ij_groovy_indent_case_from_switch = true | ||||
| ij_groovy_indent_label_blocks = true | ||||
| ij_groovy_insert_inner_class_imports = false | ||||
| ij_groovy_keep_blank_lines_before_right_brace = 2 | ||||
| ij_groovy_keep_blank_lines_in_code = 2 | ||||
| ij_groovy_keep_blank_lines_in_declarations = 2 | ||||
| ij_groovy_keep_control_statement_in_one_line = true | ||||
| ij_groovy_keep_first_column_comment = true | ||||
| ij_groovy_keep_indents_on_empty_lines = false | ||||
| ij_groovy_keep_line_breaks = true | ||||
| ij_groovy_keep_multiple_expressions_in_one_line = false | ||||
| ij_groovy_keep_simple_blocks_in_one_line = false | ||||
| ij_groovy_keep_simple_classes_in_one_line = true | ||||
| ij_groovy_keep_simple_lambdas_in_one_line = true | ||||
| ij_groovy_keep_simple_methods_in_one_line = true | ||||
| ij_groovy_label_indent_absolute = false | ||||
| ij_groovy_label_indent_size = 0 | ||||
| ij_groovy_lambda_brace_style = end_of_line | ||||
| ij_groovy_layout_static_imports_separately = true | ||||
| ij_groovy_line_comment_add_space = false | ||||
| ij_groovy_line_comment_at_first_column = true | ||||
| ij_groovy_method_annotation_wrap = split_into_lines | ||||
| ij_groovy_method_brace_style = end_of_line | ||||
| ij_groovy_method_call_chain_wrap = off | ||||
| ij_groovy_method_parameters_new_line_after_left_paren = false | ||||
| ij_groovy_method_parameters_right_paren_on_new_line = false | ||||
| ij_groovy_method_parameters_wrap = off | ||||
| ij_groovy_modifier_list_wrap = false | ||||
| ij_groovy_names_count_to_use_import_on_demand = 3 | ||||
| ij_groovy_parameter_annotation_wrap = off | ||||
| ij_groovy_parentheses_expression_new_line_after_left_paren = false | ||||
| ij_groovy_parentheses_expression_right_paren_on_new_line = false | ||||
| ij_groovy_prefer_parameters_wrap = false | ||||
| ij_groovy_resource_list_new_line_after_left_paren = false | ||||
| ij_groovy_resource_list_right_paren_on_new_line = false | ||||
| ij_groovy_resource_list_wrap = off | ||||
| ij_groovy_space_after_assert_separator = true | ||||
| ij_groovy_space_after_colon = true | ||||
| ij_groovy_space_after_comma = true | ||||
| ij_groovy_space_after_comma_in_type_arguments = true | ||||
| ij_groovy_space_after_for_semicolon = true | ||||
| ij_groovy_space_after_quest = true | ||||
| ij_groovy_space_after_type_cast = true | ||||
| ij_groovy_space_before_annotation_parameter_list = false | ||||
| ij_groovy_space_before_array_initializer_left_brace = false | ||||
| ij_groovy_space_before_assert_separator = false | ||||
| ij_groovy_space_before_catch_keyword = true | ||||
| ij_groovy_space_before_catch_left_brace = true | ||||
| ij_groovy_space_before_catch_parentheses = true | ||||
| ij_groovy_space_before_class_left_brace = true | ||||
| ij_groovy_space_before_closure_left_brace = true | ||||
| ij_groovy_space_before_colon = true | ||||
| ij_groovy_space_before_comma = false | ||||
| ij_groovy_space_before_do_left_brace = true | ||||
| ij_groovy_space_before_else_keyword = true | ||||
| ij_groovy_space_before_else_left_brace = true | ||||
| ij_groovy_space_before_finally_keyword = true | ||||
| ij_groovy_space_before_finally_left_brace = true | ||||
| ij_groovy_space_before_for_left_brace = true | ||||
| ij_groovy_space_before_for_parentheses = true | ||||
| ij_groovy_space_before_for_semicolon = false | ||||
| ij_groovy_space_before_if_left_brace = true | ||||
| ij_groovy_space_before_if_parentheses = true | ||||
| ij_groovy_space_before_method_call_parentheses = false | ||||
| ij_groovy_space_before_method_left_brace = true | ||||
| ij_groovy_space_before_method_parentheses = false | ||||
| ij_groovy_space_before_quest = true | ||||
| ij_groovy_space_before_switch_left_brace = true | ||||
| ij_groovy_space_before_switch_parentheses = true | ||||
| ij_groovy_space_before_synchronized_left_brace = true | ||||
| ij_groovy_space_before_synchronized_parentheses = true | ||||
| ij_groovy_space_before_try_left_brace = true | ||||
| ij_groovy_space_before_try_parentheses = true | ||||
| ij_groovy_space_before_while_keyword = true | ||||
| ij_groovy_space_before_while_left_brace = true | ||||
| ij_groovy_space_before_while_parentheses = true | ||||
| ij_groovy_space_in_named_argument = true | ||||
| ij_groovy_space_in_named_argument_before_colon = false | ||||
| ij_groovy_space_within_empty_array_initializer_braces = false | ||||
| ij_groovy_space_within_empty_method_call_parentheses = false | ||||
| ij_groovy_spaces_around_additive_operators = true | ||||
| ij_groovy_spaces_around_assignment_operators = true | ||||
| ij_groovy_spaces_around_bitwise_operators = true | ||||
| ij_groovy_spaces_around_equality_operators = true | ||||
| ij_groovy_spaces_around_lambda_arrow = true | ||||
| ij_groovy_spaces_around_logical_operators = true | ||||
| ij_groovy_spaces_around_multiplicative_operators = true | ||||
| ij_groovy_spaces_around_regex_operators = true | ||||
| ij_groovy_spaces_around_relational_operators = true | ||||
| ij_groovy_spaces_around_shift_operators = true | ||||
| ij_groovy_spaces_within_annotation_parentheses = false | ||||
| ij_groovy_spaces_within_array_initializer_braces = false | ||||
| ij_groovy_spaces_within_braces = true | ||||
| ij_groovy_spaces_within_brackets = false | ||||
| ij_groovy_spaces_within_cast_parentheses = false | ||||
| ij_groovy_spaces_within_catch_parentheses = false | ||||
| ij_groovy_spaces_within_for_parentheses = false | ||||
| ij_groovy_spaces_within_gstring_injection_braces = false | ||||
| ij_groovy_spaces_within_if_parentheses = false | ||||
| ij_groovy_spaces_within_list_or_map = false | ||||
| ij_groovy_spaces_within_method_call_parentheses = false | ||||
| ij_groovy_spaces_within_method_parentheses = false | ||||
| ij_groovy_spaces_within_parentheses = false | ||||
| ij_groovy_spaces_within_switch_parentheses = false | ||||
| ij_groovy_spaces_within_synchronized_parentheses = false | ||||
| ij_groovy_spaces_within_try_parentheses = false | ||||
| ij_groovy_spaces_within_tuple_expression = false | ||||
| ij_groovy_spaces_within_while_parentheses = false | ||||
| ij_groovy_special_else_if_treatment = true | ||||
| ij_groovy_ternary_operation_wrap = off | ||||
| ij_groovy_throws_keyword_wrap = off | ||||
| ij_groovy_throws_list_wrap = off | ||||
| ij_groovy_use_flying_geese_braces = false | ||||
| ij_groovy_use_fq_class_names = false | ||||
| ij_groovy_use_fq_class_names_in_javadoc = true | ||||
| ij_groovy_use_relative_indents = false | ||||
| ij_groovy_use_single_class_imports = true | ||||
| ij_groovy_variable_annotation_wrap = off | ||||
| ij_groovy_while_brace_force = never | ||||
| ij_groovy_while_on_new_line = false | ||||
| ij_groovy_wrap_long_lines = false | ||||
|  | ||||
| [{*.gradle.kts, *.kt, *.kts, *.main.kts}] | ||||
| ij_kotlin_align_in_columns_case_branch = false | ||||
| ij_kotlin_align_multiline_binary_operation = false | ||||
| @@ -963,7 +336,8 @@ ij_kotlin_wrap_elvis_expressions = 1 | ||||
| ij_kotlin_wrap_expression_body_functions = 0 | ||||
| ij_kotlin_wrap_first_method_in_call_chain = false | ||||
|  | ||||
| [{*.har, *.jsb2, *.jsb3, *.json, .babelrc, .eslintrc, .stylelintrc, bowerrc, jest.config, mcmod.info}] | ||||
|  | ||||
| [*.json] | ||||
| indent_size = 2 | ||||
| ij_json_keep_blank_lines_in_code = 0 | ||||
| ij_json_keep_indents_on_empty_lines = false | ||||
|   | ||||
							
								
								
									
										4
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| * text=auto | ||||
|  | ||||
| *.java text | ||||
| *.jar binary | ||||
							
								
								
									
										21
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										21
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							| @@ -7,8 +7,8 @@ body: | ||||
|     attributes: | ||||
|       value: | | ||||
|         Thanks for taking the time to fill out this bug report for PlotSquared! Fill out the following form to your best ability to help us fix the problem. | ||||
|         Only use this if you're absolutely sure that you found a bug and can reproduce it. For anything else, use: [our Discord server](https://discord.gg/intellectualsites) or [the wiki](https://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! | ||||
|         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.gitbook.io/plotsquared/). | ||||
|         Do NOT use the public issue tracker to report security vulnerabilities! They are disclosed using [this](https://github.com/IntellectualSites/PlotSquared/security/policy) GitHub form! | ||||
|  | ||||
|   - type: dropdown | ||||
|     attributes: | ||||
| @@ -24,17 +24,16 @@ body: | ||||
|   - type: dropdown | ||||
|     attributes: | ||||
|       label: Server Version | ||||
|       description: Which server version version you using? If your server version is not listed, it is not supported. Update to a supported version first. | ||||
|       description: Which server version are you using? If your server version is not listed, it is not supported. Update to a supported version first. | ||||
|       multiple: false | ||||
|       options: | ||||
|         - '1.18.2' | ||||
|         - '1.18.1' | ||||
|         - '1.18' | ||||
|         - '1.17.1' | ||||
|         - '1.16.5' | ||||
|         - '1.15.2' | ||||
|         - '1.14.4' | ||||
|         - '1.13.2' | ||||
|         - '1.21.4' | ||||
|         - '1.21.3' | ||||
|         - '1.21.1' | ||||
|         - '1.20.6' | ||||
|         - '1.20.4' | ||||
|         - '1.20' | ||||
|         - '1.19.4' | ||||
|     validations: | ||||
|       required: true | ||||
|  | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							| @@ -4,5 +4,5 @@ contact_links: | ||||
|     url: https://discord.gg/intellectualsites | ||||
|     about: Our support Discord, please ask questions and seek support here. | ||||
|   - name: PlotSquared Wiki | ||||
|     url: https://intellectualsites.github.io/plotsquared-documentation/ | ||||
|     url: https://intellectualsites.gitbook.io/plotsquared/ | ||||
|     about: Take a look at the wiki page for instructions how to setup PlotSquared and use its commands. | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/ISSUE_TEMPLATE/feature_request.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/ISSUE_TEMPLATE/feature_request.yml
									
									
									
									
										vendored
									
									
								
							| @@ -7,7 +7,7 @@ body: | ||||
|     attributes: | ||||
|       value: | | ||||
|         Thanks for taking the time to fill out this feature request for PlotSquared! Fill out the following form to your best ability to help us understand your feature request and greately improve the change of it getting added. | ||||
|         For anything else than a feature request, use: [our Discord server](https://discord.gg/intellectualsites) or [the wiki](https://intellectualsites.github.io/plotsquared-documentation/). | ||||
|         For anything else than a feature request, use: [our Discord server](https://discord.gg/intellectualsites) or [the wiki](https://intellectualsites.gitbook.io/plotsquared/). | ||||
|  | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|   | ||||
							
								
								
									
										19
									
								
								.github/renovate.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								.github/renovate.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| { | ||||
|   "$schema": "https://docs.renovatebot.com/renovate-schema.json", | ||||
|   "extends": [ | ||||
|     "config:recommended", | ||||
|     ":semanticCommitsDisabled", | ||||
|     "schedule:earlyMondays" | ||||
|   ], | ||||
|   "automerge": true, | ||||
|   "labels": [ | ||||
|     "dependencies" | ||||
|   ], | ||||
|   "rebaseWhen": "conflicted", | ||||
|   "ignoreDeps": [ | ||||
|     "com.google.code.gson:gson", | ||||
|     "com.google.guava:guava", | ||||
|     "org.yaml:snakeyaml", | ||||
|     "org.apache.logging.log4j:log4j-api" | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										24
									
								
								.github/workflows/announce-release-on-discord.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								.github/workflows/announce-release-on-discord.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| name: Announce release on discord | ||||
| on: | ||||
|   release: | ||||
|     types: [ published ] | ||||
| jobs: | ||||
|   send_announcement: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: send custom message with args | ||||
|         env: | ||||
|           DISCORD_WEBHOOK: ${{ secrets.DISCORD_WEBHOOK }} | ||||
|           DISCORD_USERNAME: PlotSquared Release | ||||
|           DISCORD_AVATAR: https://raw.githubusercontent.com/IntellectualSites/Assets/main/plugins/PlotSquared/PlotSquared.png | ||||
|         uses: Ilshidur/action-discord@0.4.0 | ||||
|         with: | ||||
|           args: | | ||||
|             "<@&525015541815967744> <@&679322738552471574> <@&699293353862496266>" | ||||
|             "" | ||||
|             "<:plotsquared:730750385886593039> **PlotSquared ${{ github.event.release.tag_name }} has been released!**" | ||||
|             "" | ||||
|             "Click here to view changelog: https://github.com/IntellectualSites/PlotSquared/releases/tag/${{ github.event.release.tag_name }}" | ||||
|             "" | ||||
|             "The download is available at:" | ||||
|             "- Spigot: <https://www.spigotmc.org/resources/77506/>" | ||||
							
								
								
									
										21
									
								
								.github/workflows/build-pr.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								.github/workflows/build-pr.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| name: Build PR | ||||
| on: [ pull_request ] | ||||
| jobs: | ||||
|   build_pr: | ||||
|     if: github.repository_owner == 'IntellectualSites' | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ ubuntu-latest, windows-latest, macos-latest ] | ||||
|     steps: | ||||
|       - name: Checkout Repository | ||||
|         uses: actions/checkout@v5 | ||||
|       - name: Validate Gradle Wrapper | ||||
|         uses: gradle/actions/wrapper-validation@v5 | ||||
|       - name: Setup Java | ||||
|         uses: actions/setup-java@v5 | ||||
|         with: | ||||
|           distribution: temurin | ||||
|           java-version: 21 | ||||
|       - name: Clean Build | ||||
|         run: ./gradlew clean build | ||||
							
								
								
									
										59
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										59
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,21 +1,22 @@ | ||||
| name: build | ||||
|  | ||||
| on: [ pull_request, push ] | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - main | ||||
| jobs: | ||||
|   build: | ||||
|     if: ${{ github.event_name != 'pull_request' || github.repository != github.event.pull_request.head.repo.full_name }} | ||||
|     if: github.repository_owner == 'IntellectualSites' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout Repository | ||||
|         uses: actions/checkout@v3.0.0 | ||||
|       - name: Validate Gradle Wrapper" | ||||
|         uses: gradle/wrapper-validation-action@v1.0.4 | ||||
|         uses: actions/checkout@v5 | ||||
|       - name: Validate Gradle Wrapper | ||||
|         uses: gradle/actions/wrapper-validation@v5 | ||||
|       - name: Setup Java | ||||
|         uses: actions/setup-java@v3.0.0 | ||||
|         uses: actions/setup-java@v5 | ||||
|         with: | ||||
|           distribution: temurin | ||||
|           java-version: 17 | ||||
|           java-version: 21 | ||||
|       - name: Clean Build | ||||
|         run: ./gradlew clean build | ||||
|       - name: Determine release status | ||||
| @@ -27,16 +28,40 @@ jobs: | ||||
|             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 | ||||
|         if: ${{ runner.os == 'Linux' && env.STATUS == 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main'}} | ||||
|         run: ./gradlew publishAndReleaseToMavenCentral --no-configuration-cache | ||||
|         env: | ||||
|           ORG_GRADLE_PROJECT_sonatypeUsername: ${{ secrets.SONATYPE_USERNAME }} | ||||
|           ORG_GRADLE_PROJECT_sonatypePassword: ${{ secrets.SONATYPE_PASSWORD }} | ||||
|           ORG_GRADLE_PROJECT_mavenCentralUsername: ${{ secrets.CENTRAL_USERNAME }} | ||||
|           ORG_GRADLE_PROJECT_mavenCentralPassword: ${{ secrets.CENTRAL_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 | ||||
|         if: ${{ runner.os == 'Linux' && env.STATUS != 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main' }} | ||||
|         run: ./gradlew publishAllPublicationsToMavenCentralRepository | ||||
|         env: | ||||
|           ORG_GRADLE_PROJECT_sonatypeUsername: ${{ secrets.SONATYPE_USERNAME }} | ||||
|           ORG_GRADLE_PROJECT_sonatypePassword: ${{ secrets.SONATYPE_PASSWORD }} | ||||
|           ORG_GRADLE_PROJECT_mavenCentralUsername: ${{ secrets.CENTRAL_USERNAME }} | ||||
|           ORG_GRADLE_PROJECT_mavenCentralPassword: ${{ secrets.CENTRAL_PASSWORD }} | ||||
|       - name: Publish core javadoc | ||||
|         if: ${{ runner.os == 'Linux' && env.STATUS == 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main'}} | ||||
|         uses: cpina/github-action-push-to-another-repository@main | ||||
|         env: | ||||
|           SSH_DEPLOY_KEY: ${{ secrets.SSH_DEPLOY_KEY }} | ||||
|         with: | ||||
|           source-directory: 'Core/build/docs/javadoc' | ||||
|           destination-github-username: 'IntellectualSites' | ||||
|           destination-repository-name: 'plotsquared-javadocs' | ||||
|           user-email: ${{ secrets.USER_EMAIL }} | ||||
|           target-branch: main | ||||
|           target-directory: v7/core | ||||
|       - name: Publish bukkit javadoc | ||||
|         if: ${{ runner.os == 'Linux' && env.STATUS == 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main'}} | ||||
|         uses: cpina/github-action-push-to-another-repository@main | ||||
|         env: | ||||
|           SSH_DEPLOY_KEY: ${{ secrets.SSH_DEPLOY_KEY }} | ||||
|         with: | ||||
|           source-directory: 'Bukkit/build/docs/javadoc' | ||||
|           destination-github-username: 'IntellectualSites' | ||||
|           destination-repository-name: 'plotsquared-javadocs' | ||||
|           user-email: ${{ secrets.USER_EMAIL }} | ||||
|           target-branch: main | ||||
|           target-directory: v7/bukkit | ||||
|   | ||||
							
								
								
									
										23
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										23
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,11 +1,10 @@ | ||||
| name: "CodeQL" | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: [ v6 ] | ||||
|     branches: [ main ] | ||||
|   pull_request: | ||||
|     # The branches below must be a subset of the branches above | ||||
|     branches: [ v6 ] | ||||
|     branches: [ main ] | ||||
|  | ||||
| jobs: | ||||
|   analyze: | ||||
| @@ -15,23 +14,23 @@ jobs: | ||||
|       actions: read | ||||
|       contents: read | ||||
|       security-events: write | ||||
|  | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         language: [ 'java' ] | ||||
|  | ||||
|     steps: | ||||
|       - name: Checkout repository | ||||
|         uses: actions/checkout@v3 | ||||
|  | ||||
|         uses: actions/checkout@v5 | ||||
|       - name: Setup Java | ||||
|         uses: actions/setup-java@v5 | ||||
|         with: | ||||
|           distribution: temurin | ||||
|           java-version: 21 | ||||
|       - name: Initialize CodeQL | ||||
|         uses: github/codeql-action/init@v1 | ||||
|         uses: github/codeql-action/init@v4 | ||||
|         with: | ||||
|           languages: ${{ matrix.language }} | ||||
|  | ||||
|       - name: Autobuild | ||||
|         uses: github/codeql-action/autobuild@v1 | ||||
|  | ||||
|         uses: github/codeql-action/autobuild@v4 | ||||
|       - name: Perform CodeQL Analysis | ||||
|         uses: github/codeql-action/analyze@v1 | ||||
|         uses: github/codeql-action/analyze@v4 | ||||
|   | ||||
							
								
								
									
										23
									
								
								.github/workflows/label-merge-conflicts.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								.github/workflows/label-merge-conflicts.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| name: "Label conflicting PRs" | ||||
| on: | ||||
|   push: | ||||
|   pull_request_target: | ||||
|     types: [ synchronize ] | ||||
|   pull_request: | ||||
|     types: [ synchronize ] | ||||
|  | ||||
| permissions: | ||||
|   pull-requests: write | ||||
|  | ||||
| jobs: | ||||
|   main: | ||||
|     if: github.event.pull_request.user.login != 'dependabot[bot]' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Label conflicting PRs | ||||
|         uses: eps1lon/actions-label-merge-conflict@v3.0.3 | ||||
|         with: | ||||
|           dirtyLabel: "unresolved-merge-conflict" | ||||
|           repoToken: "${{ secrets.GITHUB_TOKEN }}" | ||||
|           commentOnDirty: "Please take a moment and address the merge conflicts of your pull request. Thanks!" | ||||
|           continueOnMissingPermissions: true | ||||
							
								
								
									
										6
									
								
								.github/workflows/release-drafter.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/release-drafter.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,19 +1,17 @@ | ||||
| name: draft release | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - v6 | ||||
|       - main | ||||
|   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.19.0 | ||||
|       - uses: release-drafter/release-drafter@v6 | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|   | ||||
							
								
								
									
										13
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										13
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -41,7 +41,8 @@ hs_err_pid* | ||||
| *.iml | ||||
|  | ||||
| ## Directory-based project format: | ||||
| .idea/ | ||||
| /.idea/* | ||||
| !/.idea/icon.svg | ||||
| # if you remove the above rule, at least ignore the following: | ||||
|  | ||||
| # User-specific stuff: | ||||
| @@ -74,9 +75,6 @@ hs_err_pid* | ||||
| # IntelliJ | ||||
| /out/ | ||||
|  | ||||
| # mpeltonen/sbt-idea plugin | ||||
| .idea_modules/ | ||||
|  | ||||
| # JIRA plugin | ||||
| atlassian-ide-plugin.xml | ||||
|  | ||||
| @@ -133,5 +131,12 @@ local.properties | ||||
| checkstyle.xml | ||||
| classes/ | ||||
| *.bat | ||||
|  | ||||
| # Other | ||||
| docs/ | ||||
| build/ | ||||
|  | ||||
| .DS_Store | ||||
| # Ignore run folders | ||||
| run-[0-9].[0-9][0-9]/ | ||||
| run-[0-9].[0-9][0-9].[0-9]/ | ||||
|   | ||||
							
								
								
									
										146
									
								
								.idea/icon.svg
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										146
									
								
								.idea/icon.svg
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,146 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <svg | ||||
|    version="1.1" | ||||
|    id="svg2" | ||||
|    xml:space="preserve" | ||||
|    width="512" | ||||
|    height="512" | ||||
|    viewBox="0 0 512 512.00001" | ||||
|    sodipodi:docname="icon.svg" | ||||
|    inkscape:version="1.1.2 (b8e25be8, 2022-02-05)" | ||||
|    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" | ||||
|    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" | ||||
|    xmlns="http://www.w3.org/2000/svg" | ||||
|    xmlns:svg="http://www.w3.org/2000/svg" | ||||
|    xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
|    xmlns:cc="http://creativecommons.org/ns#" | ||||
|    xmlns:dc="http://purl.org/dc/elements/1.1/"> | ||||
| 		<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="1440" | ||||
|    inkscape:window-height="900" | ||||
|    id="namedview4" | ||||
|    inkscape:pagecheckerboard="0" | ||||
|    showgrid="false" | ||||
|    inkscape:zoom="0.1632" | ||||
|    inkscape:cx="1087.6225" | ||||
|    inkscape:cy="1666.6666" | ||||
|    inkscape:window-x="0" | ||||
|    inkscape:window-y="0" | ||||
|    inkscape:window-maximized="0" | ||||
|    inkscape:current-layer="g10" /> | ||||
|     <g | ||||
|    id="g10" | ||||
|    inkscape:groupmode="layer" | ||||
|    inkscape:label="PlotSquared" | ||||
|    transform="matrix(1.3333333,0,0,-1.3333333,0,3333.3333)"> | ||||
| 			<g | ||||
|    id="g12" | ||||
|    transform="matrix(0.16955078,0,0,0.16955078,-68.456969,2101.8529)"> | ||||
| 				<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: 12 KiB | 
| @@ -1,4 +0,0 @@ | ||||
| jdkVersion = "17" | ||||
| build = "gradle clean build -x test" | ||||
| tools = ["findsecbugs", "ErrorProne", "Semgrep", "Detekt", "Infer"] | ||||
| ignoreRules = ["CatchAndPrintStackTrace", "ReferenceEquality", "FallThrough", "FutureReturnValueIgnored", "MixedMutabilityReturnType", "EmptyCatch", "MissingCasesInEnumSwitch", "OperatorPrecedence", "StaticAssignmentInConstructor", "ReferenceEquality", "EqualsHashCode", "EqualsGetClass", "TypeParameterUnusedInFormals", "StringSplitter", "InlineMeSuggester", "NULL_DEREFERENCE"] | ||||
| @@ -3,12 +3,12 @@ import com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar | ||||
| repositories { | ||||
|     maven { | ||||
|         name = "PlaceholderAPI" | ||||
|         url = uri("https://repo.extendedclip.com/content/repositories/placeholderapi/") | ||||
|         url = uri("https://repo.extendedclip.com/releases/") | ||||
|     } | ||||
|  | ||||
|     maven { | ||||
|         name = "PaperMC" | ||||
|         url = uri("https://papermc.io/repo/repository/maven-public/") | ||||
|         url = uri("https://repo.papermc.io/repository/maven-public/") | ||||
|     } | ||||
|  | ||||
|     maven { | ||||
| @@ -17,26 +17,34 @@ repositories { | ||||
|     } | ||||
| } | ||||
|  | ||||
| // Make sure we control the exact version of paper being included, while dropping spigot + bukkit | ||||
| configurations.all { | ||||
|     exclude("org.bukkit") | ||||
|     exclude("org.spigotmc") | ||||
|  | ||||
|     resolutionStrategy.eachDependency { | ||||
|         if (requested.group == "io.papermc.paper" && requested.name == "paper-api") { | ||||
|             useVersion(checkNotNull(libs.paper.orNull?.version)) | ||||
|             because("specific paper version is required to prevent binary incompatibilities on older versions") | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     api(projects.plotSquaredCore) | ||||
|     api(projects.plotsquaredCore) | ||||
|  | ||||
|     // Metrics | ||||
|     implementation(libs.bstats) | ||||
|     implementation(libs.bstatsBukkit) | ||||
|  | ||||
|     // Paper | ||||
|     compileOnly(libs.paper) | ||||
|     implementation(libs.paperlib) | ||||
|  | ||||
|     // Plugins | ||||
|     compileOnly(libs.worldeditBukkit) { | ||||
|         exclude(group = "org.bukkit") | ||||
|         exclude(group = "org.spigotmc") | ||||
|     } | ||||
|     compileOnly(libs.fastasyncworldeditBukkit) { isTransitive = false } | ||||
|     testImplementation(libs.fastasyncworldeditBukkit) { isTransitive = false } | ||||
|     compileOnly(libs.vault) { | ||||
|         exclude(group = "org.bukkit") | ||||
|     } | ||||
|     compileOnly(libs.worldeditBukkit) | ||||
|     compileOnly(libs.faweBukkit) { isTransitive = false } | ||||
|     testImplementation(libs.faweBukkit) { isTransitive = false } | ||||
|     compileOnly(libs.vault) | ||||
|     compileOnly(libs.placeholderapi) | ||||
|     compileOnly(libs.luckperms) | ||||
|     compileOnly(libs.essentialsx) | ||||
| @@ -48,11 +56,11 @@ dependencies { | ||||
|  | ||||
|     // Our libraries | ||||
|     implementation(libs.arkitektonika) | ||||
|     implementation(libs.http4j) | ||||
|     implementation(libs.paster) | ||||
|     implementation(libs.informativeAnnotations) | ||||
|  | ||||
|     // Adventure | ||||
|     implementation(libs.adventurePlatformBukkit) | ||||
|     implementation(libs.adventureBukkit) | ||||
| } | ||||
|  | ||||
| tasks.processResources { | ||||
| @@ -62,15 +70,17 @@ tasks.processResources { | ||||
| } | ||||
|  | ||||
| tasks.named<ShadowJar>("shadowJar") { | ||||
|     dependsOn(":plotsquared-core:shadowJar") | ||||
|     dependencies { | ||||
|         exclude(dependency("org.checkerframework:")) | ||||
|     } | ||||
|  | ||||
|     relocate("net.kyori.option", "com.plotsquared.core.configuration.option") | ||||
|     relocate("net.kyori.adventure", "com.plotsquared.core.configuration.adventure") | ||||
|     relocate("net.kyori.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.enginehub.squirrelid", "com.plotsquared.squirrelid") | ||||
|     relocate("org.khelekore.prtree", "com.plotsquared.prtree") | ||||
|     relocate("com.google.inject", "com.plotsquared.google") | ||||
|     relocate("org.aopalliance", "com.plotsquared.core.aopalliance") | ||||
| @@ -83,8 +93,11 @@ tasks.named<ShadowJar>("shadowJar") { | ||||
|     relocate("org.jetbrains", "com.plotsquared.core.annotations") | ||||
|     relocate("org.intellij.lang", "com.plotsquared.core.intellij.annotations") | ||||
|     relocate("javax.annotation", "com.plotsquared.core.annotation") | ||||
|     relocate("com.google.code.findbugs", "com.plotsquared.core.findbugs") | ||||
|     relocate("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") | ||||
|     relocate("com.intellectualsites.annotations", "com.plotsquared.core.annotations.informative") | ||||
|  | ||||
|     // Get rid of all the libs which are 100% unused. | ||||
|     minimize() | ||||
| @@ -94,12 +107,20 @@ tasks.named<ShadowJar>("shadowJar") { | ||||
|  | ||||
| tasks { | ||||
|     withType<Javadoc> { | ||||
|         val isRelease = if (rootProject.version.toString().endsWith("-SNAPSHOT")) "TODO" else rootProject.version.toString() | ||||
|         val opt = options as StandardJavadocDocletOptions | ||||
|         opt.links("https://papermc.io/javadocs/paper/1.18/") | ||||
|         opt.links("https://jd.papermc.io/paper/1.20.4/") | ||||
|         opt.links("https://docs.enginehub.org/javadoc/com.sk89q.worldedit/worldedit-bukkit/" + libs.worldeditBukkit.get().versionConstraint.toString()) | ||||
|         opt.links("https://javadoc.io/doc/com.plotsquared/PlotSquared-Core/latest/") | ||||
|         opt.links("https://jd.adventure.kyori.net/api/" + libs.adventure.get().versionConstraint.toString()) | ||||
|         opt.links("https://intellectualsites.github.io/plotsquared-javadocs/core/") | ||||
|         opt.links("https://jd.advntr.dev/api/" + libs.adventureApi.get().versionConstraint.toString()) | ||||
|         opt.links("https://google.github.io/guice/api-docs/" + libs.guice.get().versionConstraint.toString() + "/javadoc/") | ||||
|         opt.links("https://checkerframework.org/api/") | ||||
|         opt.isLinkSource = true | ||||
|         opt.bottom(File("$rootDir/javadocfooter.html").readText()) | ||||
|         opt.isUse = true | ||||
|         opt.encoding("UTF-8") | ||||
|         opt.keyWords() | ||||
|         opt.addStringOption("-since", isRelease) | ||||
|         opt.noTimestamp() | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -31,7 +24,6 @@ import com.google.inject.Injector; | ||||
| import com.google.inject.Key; | ||||
| import com.google.inject.Singleton; | ||||
| import com.google.inject.Stage; | ||||
| import com.google.inject.TypeLiteral; | ||||
| import com.plotsquared.bukkit.generator.BukkitPlotGenerator; | ||||
| import com.plotsquared.bukkit.inject.BackupModule; | ||||
| import com.plotsquared.bukkit.inject.BukkitModule; | ||||
| @@ -42,20 +34,23 @@ import com.plotsquared.bukkit.listener.BlockEventListener117; | ||||
| import com.plotsquared.bukkit.listener.ChunkListener; | ||||
| import com.plotsquared.bukkit.listener.EntityEventListener; | ||||
| import com.plotsquared.bukkit.listener.EntitySpawnListener; | ||||
| import com.plotsquared.bukkit.listener.HighFreqBlockEventListener; | ||||
| import com.plotsquared.bukkit.listener.PaperListener; | ||||
| import com.plotsquared.bukkit.listener.PaperListener113; | ||||
| import com.plotsquared.bukkit.listener.PlayerEventListener; | ||||
| import com.plotsquared.bukkit.listener.PlayerEventListener1201; | ||||
| import com.plotsquared.bukkit.listener.ProjectileEventListener; | ||||
| import com.plotsquared.bukkit.listener.ServerListener; | ||||
| import com.plotsquared.bukkit.listener.SingleWorldListener; | ||||
| import com.plotsquared.bukkit.listener.SpigotListener; | ||||
| import com.plotsquared.bukkit.listener.WorldEvents; | ||||
| import com.plotsquared.bukkit.placeholder.PAPIPlaceholders; | ||||
| import com.plotsquared.bukkit.placeholder.PlaceholderFormatter; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayerManager; | ||||
| import com.plotsquared.bukkit.schematic.StateWrapper; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.bukkit.util.BukkitWorld; | ||||
| import com.plotsquared.bukkit.util.SetGenCB; | ||||
| import com.plotsquared.bukkit.util.TranslationUpdateManager; | ||||
| import com.plotsquared.bukkit.util.UpdateUtility; | ||||
| import com.plotsquared.bukkit.util.task.BukkitTaskManager; | ||||
| import com.plotsquared.bukkit.util.task.PaperTimeConverter; | ||||
| @@ -78,6 +73,8 @@ import com.plotsquared.core.configuration.Storage; | ||||
| import com.plotsquared.core.configuration.caption.ChatFormatter; | ||||
| import com.plotsquared.core.configuration.file.YamlConfiguration; | ||||
| import com.plotsquared.core.database.DBFunc; | ||||
| import com.plotsquared.core.events.RemoveRoadEntityEvent; | ||||
| import com.plotsquared.core.events.Result; | ||||
| import com.plotsquared.core.generator.GeneratorWrapper; | ||||
| import com.plotsquared.core.generator.IndependentPlotGenerator; | ||||
| import com.plotsquared.core.generator.SingleWorldGenerator; | ||||
| @@ -116,6 +113,7 @@ import com.plotsquared.core.uuid.CacheUUIDService; | ||||
| import com.plotsquared.core.uuid.UUIDPipeline; | ||||
| import com.plotsquared.core.uuid.offline.OfflineModeUUIDService; | ||||
| import com.sk89q.worldedit.WorldEdit; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import io.papermc.lib.PaperLib; | ||||
| import net.kyori.adventure.audience.Audience; | ||||
| import net.kyori.adventure.text.Component; | ||||
| @@ -138,12 +136,14 @@ import org.bukkit.generator.ChunkGenerator; | ||||
| import org.bukkit.metadata.FixedMetadataValue; | ||||
| import org.bukkit.metadata.MetadataValue; | ||||
| import org.bukkit.plugin.Plugin; | ||||
| import org.bukkit.plugin.RegisteredServiceProvider; | ||||
| import org.bukkit.plugin.java.JavaPlugin; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
| import org.incendo.serverlib.ServerLib; | ||||
|  | ||||
| import java.io.File; | ||||
| import java.io.IOException; | ||||
| import java.lang.reflect.Method; | ||||
| import java.util.ArrayList; | ||||
| import java.util.Arrays; | ||||
| @@ -254,6 +254,12 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public @NonNull String serverBrand() { | ||||
|         return Bukkit.getName(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     @SuppressWarnings("deprecation") // Paper deprecation | ||||
|     public void onEnable() { | ||||
|         this.pluginName = getDescription().getName(); | ||||
|  | ||||
| @@ -270,7 +276,7 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|  | ||||
|         final PlotSquared plotSquared = new PlotSquared(this, "Bukkit"); | ||||
|  | ||||
|         // FAWE | ||||
|         // FastAsyncWorldEdit | ||||
|         if (Settings.FAWE_Components.FAWE_HOOK) { | ||||
|             Plugin fawe = getServer().getPluginManager().getPlugin("FastAsyncWorldEdit"); | ||||
|             if (fawe != null) { | ||||
| @@ -278,12 +284,24 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                     Class.forName("com.fastasyncworldedit.bukkit.regions.plotsquared.FaweQueueCoordinator"); | ||||
|                     faweHook = true; | ||||
|                 } catch (Exception ignored) { | ||||
|                     LOGGER.error("Incompatible version of FAWE to enable hook, please upgrade: https://ci.athion" + | ||||
|                     LOGGER.error("Incompatible version of FastAsyncWorldEdit to enable hook, please upgrade: https://ci.athion" + | ||||
|                             ".net/job/FastAsyncWorldEdit/"); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Validate compatibility of StateWrapper with the current running server version | ||||
|         // Do this always, even if it's not required, to prevent running servers which fail to restore plot backups or | ||||
|         // inserting broken plot / road templates. | ||||
|         try { | ||||
|             var instance = StateWrapper.INSTANCE; | ||||
|         } catch (Exception e) { | ||||
|             LOGGER.error("Failed to initialize required classes for restoring tile entities. " + | ||||
|                     "PlotSquared will disable itself to prevent possible damages.", e); | ||||
|             getServer().getPluginManager().disablePlugin(this); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // We create the injector after PlotSquared has been initialized, so that we have access | ||||
|         // to generated instances and settings | ||||
|         this.injector = Guice | ||||
| @@ -297,6 +315,12 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                 ); | ||||
|         this.injector.injectMembers(this); | ||||
|  | ||||
|         try { | ||||
|             this.injector.getInstance(TranslationUpdateManager.class).upgradeTranslationFile(); | ||||
|         } catch (IOException e) { | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|  | ||||
|         this.serverLocale = Locale.forLanguageTag(Settings.Enabled_Components.DEFAULT_LOCALE); | ||||
|  | ||||
|         if (PremiumVerification.isPremium() && Settings.Enabled_Components.UPDATE_NOTIFICATIONS) { | ||||
| @@ -353,7 +377,13 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|  | ||||
|         if (Settings.Enabled_Components.EVENTS) { | ||||
|             getServer().getPluginManager().registerEvents(injector().getInstance(PlayerEventListener.class), this); | ||||
|             if ((serverVersion()[1] == 20 && serverVersion()[2] >= 1) || serverVersion()[1] > 20) { | ||||
|                 getServer().getPluginManager().registerEvents(injector().getInstance(PlayerEventListener1201.class), this); | ||||
|             } | ||||
|             getServer().getPluginManager().registerEvents(injector().getInstance(BlockEventListener.class), this); | ||||
|             if (Settings.HIGH_FREQUENCY_LISTENER) { | ||||
|                 getServer().getPluginManager().registerEvents(injector().getInstance(HighFreqBlockEventListener.class), this); | ||||
|             } | ||||
|             if (serverVersion()[1] >= 17) { | ||||
|                 getServer().getPluginManager().registerEvents(injector().getInstance(BlockEventListener117.class), this); | ||||
|             } | ||||
| @@ -362,11 +392,9 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|             getServer().getPluginManager().registerEvents(injector().getInstance(ServerListener.class), this); | ||||
|             getServer().getPluginManager().registerEvents(injector().getInstance(EntitySpawnListener.class), this); | ||||
|             if (PaperLib.isPaper() && Settings.Paper_Components.PAPER_LISTENERS) { | ||||
|                 if (serverVersion()[1] == 13) { | ||||
|                     getServer().getPluginManager().registerEvents(injector().getInstance(PaperListener113.class), this); | ||||
|                 } else { | ||||
|                 getServer().getPluginManager().registerEvents(injector().getInstance(PaperListener.class), this); | ||||
|                 } | ||||
|             } else { | ||||
|                 getServer().getPluginManager().registerEvents(injector().getInstance(SpigotListener.class), this); | ||||
|             } | ||||
|             this.plotListener.startRunnable(); | ||||
|         } | ||||
| @@ -432,7 +460,7 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|  | ||||
|         plotSquared.startExpiryTasks(); | ||||
|  | ||||
|         // Once the server has loaded force updating all generators known to P2 | ||||
|         // Once the server has loaded force updating all generators known to PlotSquared | ||||
|         TaskManager.runTaskLater(() -> PlotSquared.platform().setupUtils().updateGenerators(true), TaskTime.ticks(1L)); | ||||
|  | ||||
|         // Services are accessed in order | ||||
| @@ -513,8 +541,10 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                 this.backgroundPipeline.registerService(essentialsUUIDService); | ||||
|             } | ||||
|  | ||||
|             if (Settings.UUID.IMPROMPTU_SERVICE_MOJANG_API) { | ||||
|                 final SquirrelIdUUIDService impromptuMojangService = new SquirrelIdUUIDService(Settings.UUID.IMPROMPTU_LIMIT); | ||||
|                 this.impromptuPipeline.registerService(impromptuMojangService); | ||||
|             } | ||||
|             final SquirrelIdUUIDService backgroundMojangService = new SquirrelIdUUIDService(Settings.UUID.BACKGROUND_LIMIT); | ||||
|             this.backgroundPipeline.registerService(backgroundMojangService); | ||||
|         } else { | ||||
| @@ -546,7 +576,7 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|         this.startMetrics(); | ||||
|  | ||||
|         if (Settings.Enabled_Components.WORLDS) { | ||||
|             TaskManager.getPlatformImplementation().taskRepeat(this::unload, TaskTime.seconds(1L)); | ||||
|             TaskManager.getPlatformImplementation().taskRepeat(this::unload, TaskTime.seconds(10L)); | ||||
|             try { | ||||
|                 singleWorldListener = injector().getInstance(SingleWorldListener.class); | ||||
|                 Bukkit.getPluginManager().registerEvents(singleWorldListener, this); | ||||
| @@ -661,20 +691,15 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|             final @NonNull SQLiteUUIDService sqLiteUUIDService, | ||||
|             final @NonNull CacheUUIDService cacheUUIDService | ||||
|     ) { | ||||
|         // Load all uuids into a big chunky boi queue | ||||
|         final Queue<UUID> uuidQueue = new LinkedBlockingQueue<>(); | ||||
|         // Record all unique UUID's and put them into a queue | ||||
|         final Set<UUID> uuidSet = new HashSet<>(); | ||||
|         PlotSquared.get().forEachPlotRaw(plot -> { | ||||
|             final Set<UUID> uuids = new HashSet<>(); | ||||
|             uuids.add(plot.getOwnerAbs()); | ||||
|             uuids.addAll(plot.getMembers()); | ||||
|             uuids.addAll(plot.getTrusted()); | ||||
|             uuids.addAll(plot.getDenied()); | ||||
|             for (final UUID uuid : uuids) { | ||||
|                 if (!uuidQueue.contains(uuid)) { | ||||
|                     uuidQueue.add(uuid); | ||||
|                 } | ||||
|             } | ||||
|             uuidSet.add(plot.getOwnerAbs()); | ||||
|             uuidSet.addAll(plot.getMembers()); | ||||
|             uuidSet.addAll(plot.getTrusted()); | ||||
|             uuidSet.addAll(plot.getDenied()); | ||||
|         }); | ||||
|         final Queue<UUID> uuidQueue = new LinkedBlockingQueue<>(uuidSet); | ||||
|  | ||||
|         LOGGER.info("(UUID) {} UUIDs will be cached", uuidQueue.size()); | ||||
|  | ||||
| @@ -737,6 +762,11 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|         this.getServer().getPluginManager().disablePlugin(this); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void shutdownServer() { | ||||
|         getServer().shutdown(); | ||||
|     } | ||||
|  | ||||
|     private void registerCommands() { | ||||
|         final BukkitCommand bukkitCommand = new BukkitCommand(); | ||||
|         final PluginCommand plotCommand = getCommand("plots"); | ||||
| @@ -769,22 +799,31 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                 Iterator<Entity> iterator = entities.iterator(); | ||||
|                 while (iterator.hasNext()) { | ||||
|                     Entity entity = iterator.next(); | ||||
|                     //noinspection ConstantValue - getEntitySpawnReason annotated as NotNull, but is not NotNull. lol. | ||||
|                     if (PaperLib.isPaper() && entity.getEntitySpawnReason() != null && "CUSTOM".equals(entity.getEntitySpawnReason().name())) { | ||||
|                         continue; | ||||
|                     } | ||||
|                     // Fallback for Spigot not having Entity#getEntitySpawnReason | ||||
|                     if (entity.getMetadata("ps_custom_spawned").stream().anyMatch(MetadataValue::asBoolean)) { | ||||
|                         continue; | ||||
|                     } | ||||
|                     // TODO: use (type) pattern matching when targeting java 21 | ||||
|                     switch (entity.getType().toString()) { | ||||
|                         case "EGG": | ||||
|                         case "FISHING_HOOK": | ||||
|                         case "ENDER_SIGNAL": | ||||
|                         case "FISHING_HOOK", "FISHING_BOBBER": | ||||
|                         case "ENDER_SIGNAL", "EYE_OF_ENDER": | ||||
|                         case "AREA_EFFECT_CLOUD": | ||||
|                         case "EXPERIENCE_ORB": | ||||
|                         case "LEASH_HITCH": | ||||
|                         case "FIREWORK": | ||||
|                         case "LIGHTNING": | ||||
|                         case "LEASH_HITCH", "LEASH_KNOT": | ||||
|                         case "FIREWORK", "FIREWORK_ROCKET": | ||||
|                         case "LIGHTNING", "LIGHTNING_BOLT": | ||||
|                         case "WITHER_SKULL": | ||||
|                         case "UNKNOWN": | ||||
|                         case "PLAYER": | ||||
|                             // non moving / unmovable | ||||
|                             continue; | ||||
|                         case "THROWN_EXP_BOTTLE": | ||||
|                         case "SPLASH_POTION": | ||||
|                         case "THROWN_EXP_BOTTLE", "EXPERIENCE_BOTTLE": | ||||
|                         case "SPLASH_POTION", "POTION": | ||||
|                         case "SNOWBALL": | ||||
|                         case "SHULKER_BULLET": | ||||
|                         case "SPECTRAL_ARROW": | ||||
| @@ -802,13 +841,26 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                             // Temporarily classify as vehicle | ||||
|                         case "MINECART": | ||||
|                         case "MINECART_CHEST": | ||||
|                         case "CHEST_MINECART": | ||||
|                         case "MINECART_COMMAND": | ||||
|                         case "COMMAND_BLOCK_MINECART": | ||||
|                         case "MINECART_FURNACE": | ||||
|                         case "FURNACE_MINECART": | ||||
|                         case "MINECART_HOPPER": | ||||
|                         case "HOPPER_MINECART": | ||||
|                         case "MINECART_MOB_SPAWNER": | ||||
|                         case "ENDER_CRYSTAL": | ||||
|                         case "SPAWNER_MINECART": | ||||
|                         case "END_CRYSTAL": | ||||
|                         case "ENDER_CRYSTAL": // Backwards compatibility for 1.20.4 | ||||
|                         case "MINECART_TNT": | ||||
|                         case "TNT_MINECART": | ||||
|                         case "CHEST_BOAT": | ||||
|                         case "BOAT": | ||||
|                         case "ACACIA_BOAT", "BIRCH_BOAT", "CHERRY_BOAT", "DARK_OAK_BOAT", "JUNGLE_BOAT", "MANGROVE_BOAT", | ||||
|                              "OAK_BOAT", "PALE_OAK_BOAT", "SPRUCE_BOAT", "BAMBOO_RAFT": | ||||
|                         case "ACACIA_CHEST_BOAT", "BIRCH_CHEST_BOAT", "CHERRY_CHEST_BOAT", "DARK_OAK_CHEST_BOAT", | ||||
|                              "JUNGLE_CHEST_BOAT", "MANGROVE_CHEST_BOAT", "OAK_CHEST_BOAT", "PALE_OAK_CHEST_BOAT", | ||||
|                              "SPRUCE_CHEST_BOAT", "BAMBOO_CHEST_RAFT": | ||||
|                             if (Settings.Enabled_Components.KILL_ROAD_VEHICLES) { | ||||
|                                 com.plotsquared.core.location.Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|                                 Plot plot = location.getPlot(); | ||||
| @@ -817,8 +869,7 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                                         if (entity.hasMetadata("ps-tmp-teleport")) { | ||||
|                                             continue; | ||||
|                                         } | ||||
|                                         iterator.remove(); | ||||
|                                         entity.remove(); | ||||
|                                         this.removeRoadEntity(entity, iterator); | ||||
|                                     } | ||||
|                                     continue; | ||||
|                                 } | ||||
| @@ -831,35 +882,33 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                                     if (entity.hasMetadata("ps-tmp-teleport")) { | ||||
|                                         continue; | ||||
|                                     } | ||||
|                                     iterator.remove(); | ||||
|                                     entity.remove(); | ||||
|                                     this.removeRoadEntity(entity, iterator); | ||||
|                                 } | ||||
|                             } | ||||
|                             continue; | ||||
|                         case "SMALL_FIREBALL": | ||||
|                         case "FIREBALL": | ||||
|                         case "DRAGON_FIREBALL": | ||||
|                         case "DROPPED_ITEM": | ||||
|                         case "DROPPED_ITEM", "ITEM": | ||||
|                             if (Settings.Enabled_Components.KILL_ROAD_ITEMS | ||||
|                                     && plotArea.getOwnedPlotAbs(BukkitUtil.adapt(entity.getLocation())) == null) { | ||||
|                                 entity.remove(); | ||||
|                                 this.removeRoadEntity(entity, iterator); | ||||
|                             } | ||||
|                             // dropped item | ||||
|                             continue; | ||||
|                         case "PRIMED_TNT": | ||||
|                         case "PRIMED_TNT", "TNT": | ||||
|                         case "FALLING_BLOCK": | ||||
|                             // managed elsewhere | ||||
|                             continue; | ||||
|                         case "SHULKER": | ||||
|                             if (Settings.Enabled_Components.KILL_ROAD_MOBS) { | ||||
|                             if (Settings.Enabled_Components.KILL_ROAD_MOBS && (Settings.Enabled_Components.KILL_NAMED_ROAD_MOBS || entity.getCustomName() == null)) { | ||||
|                                 LivingEntity livingEntity = (LivingEntity) entity; | ||||
|                                 List<MetadataValue> meta = entity.getMetadata("shulkerPlot"); | ||||
|                                 if (!meta.isEmpty()) { | ||||
|                                     if (livingEntity.isLeashed()) { | ||||
|                                     if (livingEntity.isLeashed() && !Settings.Enabled_Components.KILL_OWNED_ROAD_MOBS) { | ||||
|                                         continue; | ||||
|                                     } | ||||
|                                     List<MetadataValue> keep = entity.getMetadata("keep"); | ||||
|                                     if (!keep.isEmpty()) { | ||||
|                                     if (entity.hasMetadata("keep")) { | ||||
|                                         continue; | ||||
|                                     } | ||||
|  | ||||
| @@ -868,15 +917,12 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                                         com.plotsquared.core.location.Location pLoc = BukkitUtil.adapt(entity.getLocation()); | ||||
|                                         PlotArea area = pLoc.getPlotArea(); | ||||
|                                         if (area != null) { | ||||
|                                             PlotId currentPlotId = area.getPlotAbs(pLoc).getId(); | ||||
|                                             if (!originalPlotId.equals(currentPlotId) && (currentPlotId == null || !area.getPlot( | ||||
|                                                             originalPlotId) | ||||
|                                                     .equals(area.getPlot(currentPlotId)))) { | ||||
|                                             Plot currentPlot = area.getPlotAbs(pLoc); | ||||
|                                             if (currentPlot == null || !originalPlotId.equals(currentPlot.getId())) { | ||||
|                                                 if (entity.hasMetadata("ps-tmp-teleport")) { | ||||
|                                                     continue; | ||||
|                                                 } | ||||
|                                                 iterator.remove(); | ||||
|                                                 entity.remove(); | ||||
|                                                 this.removeRoadEntity(entity, iterator); | ||||
|                                             } | ||||
|                                         } | ||||
|                                     } | ||||
| @@ -885,11 +931,11 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                                     com.plotsquared.core.location.Location pLoc = BukkitUtil.adapt(entity.getLocation()); | ||||
|                                     PlotArea area = pLoc.getPlotArea(); | ||||
|                                     if (area != null) { | ||||
|                                         PlotId currentPlotId = area.getPlotAbs(pLoc).getId(); | ||||
|                                         if (currentPlotId != null) { | ||||
|                                         Plot currentPlot = area.getPlotAbs(pLoc); | ||||
|                                         if (currentPlot != null) { | ||||
|                                             entity.setMetadata( | ||||
|                                                     "shulkerPlot", | ||||
|                                                     new FixedMetadataValue((Plugin) PlotSquared.platform(), currentPlotId) | ||||
|                                                     new FixedMetadataValue((Plugin) PlotSquared.platform(), currentPlot.getId()) | ||||
|                                             ); | ||||
|                                         } | ||||
|                                     } | ||||
| @@ -923,12 +969,14 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                         case "ENDERMITE": | ||||
|                         case "ENDER_DRAGON": | ||||
|                         case "GHAST": | ||||
|                         case "HAPPY_GHAST": // 1.21.6+ | ||||
|                         case "GHASTLING": // 1.21.6+ | ||||
|                         case "GIANT": | ||||
|                         case "GUARDIAN": | ||||
|                         case "HORSE": | ||||
|                         case "IRON_GOLEM": | ||||
|                         case "MAGMA_CUBE": | ||||
|                         case "MUSHROOM_COW": | ||||
|                         case "MUSHROOM_COW", "MOOSHROOM": | ||||
|                         case "OCELOT": | ||||
|                         case "PIG": | ||||
|                         case "PIG_ZOMBIE": | ||||
| @@ -937,7 +985,7 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                         case "SILVERFISH": | ||||
|                         case "SKELETON": | ||||
|                         case "SLIME": | ||||
|                         case "SNOWMAN": | ||||
|                         case "SNOWMAN", "SNOW_GOLEM": | ||||
|                         case "SPIDER": | ||||
|                         case "SQUID": | ||||
|                         case "VILLAGER": | ||||
| @@ -975,23 +1023,24 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                                                 || !entity.hasMetadata("keep")) { | ||||
|                                             Entity passenger = entity.getPassenger(); | ||||
|                                             if ((Settings.Enabled_Components.KILL_OWNED_ROAD_MOBS | ||||
|                                                     || !(passenger instanceof Player)) && entity.getMetadata("keep").isEmpty()) { | ||||
|                                                     || !((passenger instanceof Player) || livingEntity.isLeashed())) | ||||
|                                                     && (Settings.Enabled_Components.KILL_NAMED_ROAD_MOBS || entity.getCustomName() == null) | ||||
|                                                     && entity.getMetadata("keep").isEmpty()) { | ||||
|                                                 if (entity.hasMetadata("ps-tmp-teleport")) { | ||||
|                                                     continue; | ||||
|                                                 } | ||||
|                                                 iterator.remove(); | ||||
|                                                 entity.remove(); | ||||
|                                                 this.removeRoadEntity(entity, iterator); | ||||
|                                             } | ||||
|                                         } | ||||
|                                     } else { | ||||
|                                         Entity passenger = entity.getPassenger(); | ||||
|                                         if ((Settings.Enabled_Components.KILL_OWNED_ROAD_MOBS | ||||
|                                                 || !(passenger instanceof Player)) && entity.getMetadata("keep").isEmpty()) { | ||||
|                                         if ((Settings.Enabled_Components.KILL_OWNED_ROAD_MOBS || !(passenger instanceof Player)) | ||||
|                                                 && (Settings.Enabled_Components.KILL_NAMED_ROAD_MOBS && entity.getCustomName() != null) | ||||
|                                                 && entity.getMetadata("keep").isEmpty()) { | ||||
|                                             if (entity.hasMetadata("ps-tmp-teleport")) { | ||||
|                                                 continue; | ||||
|                                             } | ||||
|                                             iterator.remove(); | ||||
|                                             entity.remove(); | ||||
|                                             this.removeRoadEntity(entity, iterator); | ||||
|                                         } | ||||
|                                     } | ||||
|                                 } | ||||
| @@ -1005,6 +1054,17 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|         }), TaskTime.seconds(1L)); | ||||
|     } | ||||
|  | ||||
|     private void removeRoadEntity(Entity entity, Iterator<Entity> entityIterator) { | ||||
|         RemoveRoadEntityEvent event = eventDispatcher.callRemoveRoadEntity(BukkitAdapter.adapt(entity)); | ||||
|  | ||||
|         if (event.getEventResult() == Result.DENY) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         entityIterator.remove(); | ||||
|         entity.remove(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public @Nullable | ||||
|     final ChunkGenerator getDefaultWorldGenerator( | ||||
| @@ -1138,7 +1198,9 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|     @Override | ||||
|     public @NonNull String serverNativePackage() { | ||||
|         final String name = Bukkit.getServer().getClass().getPackage().getName(); | ||||
|         return name.substring(name.lastIndexOf('.') + 1); | ||||
|         String ver = name.substring(name.lastIndexOf('.') + 1); | ||||
|         // org.bukkit.craftbukkit is no longer suffixed by a version | ||||
|         return ver.equals("craftbukkit") ? "" : ver; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -1149,6 +1211,7 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|         return new BukkitPlotGenerator(world, generator, this.plotAreaManager); | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("deprecation") // Paper deprecation | ||||
|     @Override | ||||
|     public @NonNull String pluginsFormatted() { | ||||
|         StringBuilder msg = new StringBuilder(); | ||||
| @@ -1165,18 +1228,37 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                     .append("  • Load Before: ").append(p.getDescription().getLoadBefore()).append("\n") | ||||
|                     .append("  • Dependencies: ").append(p.getDescription().getDepend()).append("\n") | ||||
|                     .append("  • Soft Dependencies: ").append(p.getDescription().getSoftDepend()).append("\n"); | ||||
|             List<RegisteredServiceProvider<?>> providers = Bukkit.getServicesManager().getRegistrations(p); | ||||
|             if (!providers.isEmpty()) { | ||||
|                 msg.append("  • Provided Services: \n"); | ||||
|                 for (RegisteredServiceProvider<?> provider : providers) { | ||||
|                     msg.append("    • ") | ||||
|                             .append(provider.getService().getName()).append(" = ") | ||||
|                             .append(provider.getProvider().getClass().getName()) | ||||
|                             .append(" (priority: ").append(provider.getPriority()).append(")") | ||||
|                             .append("\n"); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         return msg.toString(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     @SuppressWarnings("ConstantConditions") | ||||
|     @SuppressWarnings({"ConstantConditions", "deprecation"}) // Paper deprecation | ||||
|     public @NonNull String worldEditImplementations() { | ||||
|         StringBuilder msg = new StringBuilder(); | ||||
|         if (Bukkit.getPluginManager().getPlugin("FastAsyncWorldEdit") != null) { | ||||
|             msg.append("FastAsyncWorldEdit: ").append(Bukkit.getPluginManager().getPlugin("FastAsyncWorldEdit").getDescription().getVersion()); | ||||
|             msg.append("FastAsyncWorldEdit: ").append(Bukkit | ||||
|                     .getPluginManager() | ||||
|                     .getPlugin("FastAsyncWorldEdit") | ||||
|                     .getDescription() | ||||
|                     .getVersion()); | ||||
|         } else if (Bukkit.getPluginManager().getPlugin("AsyncWorldEdit") != null) { | ||||
|             msg.append("AsyncWorldEdit: ").append(Bukkit.getPluginManager().getPlugin("AsyncWorldEdit").getDescription().getVersion()).append("\n"); | ||||
|             msg.append("AsyncWorldEdit: ").append(Bukkit | ||||
|                     .getPluginManager() | ||||
|                     .getPlugin("AsyncWorldEdit") | ||||
|                     .getDescription() | ||||
|                     .getVersion()).append("\n"); | ||||
|             msg.append("WorldEdit: ").append(Bukkit.getPluginManager().getPlugin("WorldEdit").getDescription().getVersion()); | ||||
|         } else { | ||||
|             msg.append("WorldEdit: ").append(Bukkit.getPluginManager().getPlugin("WorldEdit").getDescription().getVersion()); | ||||
| @@ -1226,15 +1308,13 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|  | ||||
|     @Override | ||||
|     public @NonNull PlatformWorldManager<?> worldManager() { | ||||
|         return injector().getInstance(Key.get(new TypeLiteral<PlatformWorldManager<World>>() { | ||||
|         })); | ||||
|         return this.worldManager; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     @NonNull | ||||
|     @SuppressWarnings("unchecked") | ||||
|     public PlayerManager<? extends PlotPlayer<Player>, ? extends Player> playerManager() { | ||||
|         return (PlayerManager<BukkitPlayer, Player>) injector().getInstance(PlayerManager.class); | ||||
|         return this.playerManager; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -32,7 +25,6 @@ import org.bukkit.Art; | ||||
| import org.bukkit.DyeColor; | ||||
| import org.bukkit.Location; | ||||
| import org.bukkit.Rotation; | ||||
| import org.bukkit.TreeSpecies; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.BlockFace; | ||||
| import org.bukkit.entity.AbstractHorse; | ||||
| @@ -40,6 +32,8 @@ import org.bukkit.entity.Ageable; | ||||
| import org.bukkit.entity.ArmorStand; | ||||
| import org.bukkit.entity.Bat; | ||||
| import org.bukkit.entity.Boat; | ||||
| import org.bukkit.entity.Breedable; | ||||
| import org.bukkit.entity.ChestBoat; | ||||
| import org.bukkit.entity.ChestedHorse; | ||||
| import org.bukkit.entity.EnderDragon; | ||||
| import org.bukkit.entity.Entity; | ||||
| @@ -50,7 +44,6 @@ import org.bukkit.entity.LivingEntity; | ||||
| import org.bukkit.entity.Painting; | ||||
| import org.bukkit.entity.Rabbit; | ||||
| import org.bukkit.entity.Sheep; | ||||
| import org.bukkit.entity.Slime; | ||||
| import org.bukkit.entity.Tameable; | ||||
| import org.bukkit.inventory.EntityEquipment; | ||||
| import org.bukkit.inventory.InventoryHolder; | ||||
| @@ -81,6 +74,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|     private HorseStats horse; | ||||
|     private boolean noGravity; | ||||
|  | ||||
|     @SuppressWarnings("deprecation") // Deprecation exists since 1.20, while we support 1.16 onwards | ||||
|     public ReplicatingEntityWrapper(Entity entity, short depth) { | ||||
|         super(entity); | ||||
|  | ||||
| @@ -108,50 +102,34 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|             this.noGravity = true; | ||||
|         } | ||||
|         switch (entity.getType().toString()) { | ||||
|             case "BOAT": | ||||
|             case "BOAT", "ACACIA_BOAT", "BIRCH_BOAT", "CHERRY_BOAT", "DARK_OAK_BOAT", "JUNGLE_BOAT", "MANGROVE_BOAT", | ||||
|                  "OAK_BOAT", "PALE_OAK_BOAT", "SPRUCE_BOAT", "BAMBOO_RAFT" -> { | ||||
|                 Boat boat = (Boat) entity; | ||||
|                 this.dataByte = getOrdinal(TreeSpecies.values(), boat.getWoodType()); | ||||
|                 this.dataByte = getOrdinal(Boat.Type.values(), boat.getBoatType()); | ||||
|                 return; | ||||
|             case "ARROW": | ||||
|             case "EGG": | ||||
|             case "ENDER_CRYSTAL": | ||||
|             case "ENDER_PEARL": | ||||
|             case "ENDER_SIGNAL": | ||||
|             case "EXPERIENCE_ORB": | ||||
|             case "FALLING_BLOCK": | ||||
|             case "FIREBALL": | ||||
|             case "FIREWORK": | ||||
|             case "FISHING_HOOK": | ||||
|             case "LEASH_HITCH": | ||||
|             case "LIGHTNING": | ||||
|             case "MINECART": | ||||
|             case "MINECART_COMMAND": | ||||
|             case "MINECART_MOB_SPAWNER": | ||||
|             case "MINECART_TNT": | ||||
|             case "PLAYER": | ||||
|             case "PRIMED_TNT": | ||||
|             case "SLIME": | ||||
|             case "SMALL_FIREBALL": | ||||
|             case "SNOWBALL": | ||||
|             case "MINECART_FURNACE": | ||||
|             case "SPLASH_POTION": | ||||
|             case "THROWN_EXP_BOTTLE": | ||||
|             case "WITHER_SKULL": | ||||
|             case "UNKNOWN": | ||||
|             case "SPECTRAL_ARROW": | ||||
|             case "SHULKER_BULLET": | ||||
|             case "DRAGON_FIREBALL": | ||||
|             case "AREA_EFFECT_CLOUD": | ||||
|             case "TRIDENT": | ||||
|             case "LLAMA_SPIT": | ||||
|             } | ||||
|             case "ACACIA_CHEST_BOAT", "BIRCH_CHEST_BOAT", "CHERRY_CHEST_BOAT", "DARK_OAK_CHEST_BOAT", | ||||
|                  "JUNGLE_CHEST_BOAT", "MANGROVE_CHEST_BOAT", "OAK_CHEST_BOAT", "PALE_OAK_CHEST_BOAT", | ||||
|                  "SPRUCE_CHEST_BOAT", "BAMBOO_CHEST_RAFT" -> { | ||||
|                 ChestBoat boat = (ChestBoat) entity; | ||||
|                 this.dataByte = getOrdinal(Boat.Type.values(), boat.getBoatType()); | ||||
|                 storeInventory(boat); | ||||
|             } | ||||
|             case "ARROW", "EGG", "END_CRYSTAL", "ENDER_CRYSTAL", "ENDER_PEARL", "ENDER_SIGNAL", "EXPERIENCE_ORB", "FALLING_BLOCK", "FIREBALL", | ||||
|                     "FIREWORK", "FISHING_HOOK", "LEASH_HITCH", "LIGHTNING", "MINECART", "MINECART_COMMAND", "MINECART_MOB_SPAWNER", | ||||
|                     "MINECART_TNT", "PLAYER", "PRIMED_TNT", "SLIME", "SMALL_FIREBALL", "SNOWBALL", "MINECART_FURNACE", "SPLASH_POTION", | ||||
|                     "THROWN_EXP_BOTTLE", "WITHER_SKULL", "UNKNOWN", "SPECTRAL_ARROW", "SHULKER_BULLET", "DRAGON_FIREBALL", "AREA_EFFECT_CLOUD", | ||||
|                     "TRIDENT", "LLAMA_SPIT" -> { | ||||
|                 // Do this stuff later | ||||
|                 return; | ||||
|             } | ||||
|             // MISC // | ||||
|             case "DROPPED_ITEM": | ||||
|             case "DROPPED_ITEM", "ITEM" -> { | ||||
|                 Item item = (Item) entity; | ||||
|                 this.stack = item.getItemStack(); | ||||
|                 return; | ||||
|             case "ITEM_FRAME": | ||||
|             } | ||||
|             case "ITEM_FRAME" -> { | ||||
|                 this.x = Math.floor(this.getX()); | ||||
|                 this.y = Math.floor(this.getY()); | ||||
|                 this.z = Math.floor(this.getZ()); | ||||
| @@ -159,7 +137,8 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|                 this.dataByte = getOrdinal(Rotation.values(), itemFrame.getRotation()); | ||||
|                 this.stack = itemFrame.getItem().clone(); | ||||
|                 return; | ||||
|             case "PAINTING": | ||||
|             } | ||||
|             case "PAINTING" -> { | ||||
|                 this.x = Math.floor(this.getX()); | ||||
|                 this.y = Math.floor(this.getY()); | ||||
|                 this.z = Math.floor(this.getZ()); | ||||
| @@ -172,20 +151,17 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|                 } | ||||
|                 this.dataString = art.name(); | ||||
|                 return; | ||||
|             } | ||||
|             // END MISC // | ||||
|             // INVENTORY HOLDER // | ||||
|             case "MINECART_CHEST": | ||||
|             case "MINECART_HOPPER": | ||||
|             case "MINECART_CHEST", "CHEST_MINECART", "MINECART_HOPPER", "HOPPER_MINECART" -> { | ||||
|                 storeInventory((InventoryHolder) entity); | ||||
|                 return; | ||||
|             } | ||||
|             // START LIVING ENTITY // | ||||
|             // START AGEABLE // | ||||
|             // START TAMEABLE // | ||||
|             case "HORSE": | ||||
|             case "DONKEY": | ||||
|             case "LLAMA": | ||||
|             case "MULE": | ||||
|             case "SKELETON_HORSE": | ||||
|             case "CAMEL", "HORSE", "DONKEY", "LLAMA", "TRADER_LLAMA", "MULE", "SKELETON_HORSE", "ZOMBIE_HORSE" -> { | ||||
|                 AbstractHorse horse = (AbstractHorse) entity; | ||||
|                 this.horse = new HorseStats(); | ||||
|                 this.horse.jump = horse.getJumpStrength(); | ||||
| @@ -197,47 +173,44 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|                 //this.horse.style = horse.getStyle(); | ||||
|                 //this.horse.color = horse.getColor(); | ||||
|                 storeTameable(horse); | ||||
|                 storeAgeable(horse); | ||||
|                 storeBreedable(horse); | ||||
|                 storeLiving(horse); | ||||
|                 storeInventory(horse); | ||||
|                 return; | ||||
|             } | ||||
|             // END INVENTORY HOLDER // | ||||
|             case "WOLF": | ||||
|             case "OCELOT": | ||||
|             case "WOLF", "OCELOT", "CAT", "PARROT" -> { | ||||
|                 storeTameable((Tameable) entity); | ||||
|                 storeAgeable((Ageable) entity); | ||||
|                 storeBreedable((Breedable) entity); | ||||
|                 storeLiving((LivingEntity) entity); | ||||
|                 return; | ||||
|             } | ||||
|             // END TAMEABLE // | ||||
|             //todo fix sheep | ||||
|             case "SHEEP": | ||||
|             case "SHEEP" -> { | ||||
|                 Sheep sheep = (Sheep) entity; | ||||
|                 if (sheep.isSheared()) { | ||||
|                     this.dataByte = (byte) 1; | ||||
|                 } else { | ||||
|                     this.dataByte = (byte) 0; | ||||
|                 } | ||||
|                 this.dataByte2 = sheep.getColor().getDyeData(); | ||||
|                 storeAgeable(sheep); | ||||
|                 this.dataByte2 = getOrdinal(DyeColor.values(), sheep.getColor()); | ||||
|                 storeBreedable(sheep); | ||||
|                 storeLiving(sheep); | ||||
|                 return; | ||||
|             case "VILLAGER": | ||||
|             case "CHICKEN": | ||||
|             case "COW": | ||||
|             case "MUSHROOM_COW": | ||||
|             case "PIG": | ||||
|             case "TURTLE": | ||||
|             case "POLAR_BEAR": | ||||
|                 storeAgeable((Ageable) entity); | ||||
|             } | ||||
|             case "VILLAGER", "CHICKEN", "COW", "MUSHROOM_COW", "PIG", "TURTLE", "POLAR_BEAR" -> { | ||||
|                 storeBreedable((Breedable) entity); | ||||
|                 storeLiving((LivingEntity) entity); | ||||
|                 return; | ||||
|             case "RABBIT": | ||||
|             } | ||||
|             case "RABBIT" -> { | ||||
|                 this.dataByte = getOrdinal(Rabbit.Type.values(), ((Rabbit) entity).getRabbitType()); | ||||
|                 storeAgeable((Ageable) entity); | ||||
|                 storeBreedable((Breedable) entity); | ||||
|                 storeLiving((LivingEntity) entity); | ||||
|                 return; | ||||
|             } | ||||
|             // END AGEABLE // | ||||
|             case "ARMOR_STAND": | ||||
|             case "ARMOR_STAND" -> { | ||||
|                 ArmorStand stand = (ArmorStand) entity; | ||||
|                 this.inventory = | ||||
|                         new ItemStack[]{stand.getItemInHand().clone(), stand.getHelmet().clone(), | ||||
| @@ -245,37 +218,30 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|                                 stand.getBoots().clone()}; | ||||
|                 storeLiving(stand); | ||||
|                 this.stand = new ArmorStandStats(); | ||||
|  | ||||
|                 EulerAngle head = stand.getHeadPose(); | ||||
|                 this.stand.head[0] = (float) head.getX(); | ||||
|                 this.stand.head[1] = (float) head.getY(); | ||||
|                 this.stand.head[2] = (float) head.getZ(); | ||||
|  | ||||
|                 EulerAngle body = stand.getBodyPose(); | ||||
|                 this.stand.body[0] = (float) body.getX(); | ||||
|                 this.stand.body[1] = (float) body.getY(); | ||||
|                 this.stand.body[2] = (float) body.getZ(); | ||||
|  | ||||
|                 EulerAngle leftLeg = stand.getLeftLegPose(); | ||||
|                 this.stand.leftLeg[0] = (float) leftLeg.getX(); | ||||
|                 this.stand.leftLeg[1] = (float) leftLeg.getY(); | ||||
|                 this.stand.leftLeg[2] = (float) leftLeg.getZ(); | ||||
|  | ||||
|                 EulerAngle rightLeg = stand.getRightLegPose(); | ||||
|                 this.stand.rightLeg[0] = (float) rightLeg.getX(); | ||||
|                 this.stand.rightLeg[1] = (float) rightLeg.getY(); | ||||
|                 this.stand.rightLeg[2] = (float) rightLeg.getZ(); | ||||
|  | ||||
|                 EulerAngle leftArm = stand.getLeftArmPose(); | ||||
|                 this.stand.leftArm[0] = (float) leftArm.getX(); | ||||
|                 this.stand.leftArm[1] = (float) leftArm.getY(); | ||||
|                 this.stand.leftArm[2] = (float) leftArm.getZ(); | ||||
|  | ||||
|                 EulerAngle rightArm = stand.getRightArmPose(); | ||||
|                 this.stand.rightArm[0] = (float) rightArm.getX(); | ||||
|                 this.stand.rightArm[1] = (float) rightArm.getY(); | ||||
|                 this.stand.rightArm[2] = (float) rightArm.getZ(); | ||||
|  | ||||
|                 if (stand.hasArms()) { | ||||
|                     this.stand.arms = true; | ||||
|                 } | ||||
| @@ -289,52 +255,37 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|                     this.stand.small = true; | ||||
|                 } | ||||
|                 return; | ||||
|             case "ENDERMITE": | ||||
|             } | ||||
|             case "ENDERMITE" -> { | ||||
|                 return; | ||||
|             case "BAT": | ||||
|             } | ||||
|             case "BAT" -> { | ||||
|                 if (((Bat) entity).isAwake()) { | ||||
|                     this.dataByte = (byte) 1; | ||||
|                 } else { | ||||
|                     this.dataByte = (byte) 0; | ||||
|                 } | ||||
|                 return; | ||||
|             case "ENDER_DRAGON": | ||||
|             } | ||||
|             case "ENDER_DRAGON" -> { | ||||
|                 EnderDragon entity1 = (EnderDragon) entity; | ||||
|                 this.dataByte = (byte) entity1.getPhase().ordinal(); | ||||
|                 return; | ||||
|             case "SKELETON": | ||||
|             case "WITHER_SKELETON": | ||||
|             case "GUARDIAN": | ||||
|             case "ELDER_GUARDIAN": | ||||
|             case "GHAST": | ||||
|             case "MAGMA_CUBE": | ||||
|             case "SQUID": | ||||
|             case "PIG_ZOMBIE": | ||||
|             case "HOGLIN": | ||||
|             case "ZOMBIFIED_PIGLIN": | ||||
|             case "PIGLIN": | ||||
|             case "PIGLIN_BRUTE": | ||||
|             case "ZOMBIE": | ||||
|             case "WITHER": | ||||
|             case "WITCH": | ||||
|             case "SPIDER": | ||||
|             case "CAVE_SPIDER": | ||||
|             case "SILVERFISH": | ||||
|             case "GIANT": | ||||
|             case "ENDERMAN": | ||||
|             case "CREEPER": | ||||
|             case "BLAZE": | ||||
|             case "SHULKER": | ||||
|             case "SNOWMAN": | ||||
|             } | ||||
|             case "SKELETON", "WITHER_SKELETON", "GUARDIAN", "ELDER_GUARDIAN", "GHAST", "HAPPY_GHAST", "GHASTLING", "MAGMA_CUBE", "SQUID", "PIG_ZOMBIE", "HOGLIN", | ||||
|                     "ZOMBIFIED_PIGLIN", "PIGLIN", "PIGLIN_BRUTE", "ZOMBIE", "WITHER", "WITCH", "SPIDER", "CAVE_SPIDER", "SILVERFISH", | ||||
|                     "GIANT", "ENDERMAN", "CREEPER", "BLAZE", "SHULKER", "SNOWMAN", "SNOW_GOLEM" -> { | ||||
|                 storeLiving((LivingEntity) entity); | ||||
|                 return; | ||||
|             case "IRON_GOLEM": | ||||
|             } | ||||
|             case "IRON_GOLEM" -> { | ||||
|                 if (((IronGolem) entity).isPlayerCreated()) { | ||||
|                     this.dataByte = (byte) 1; | ||||
|                 } else { | ||||
|                     this.dataByte = (byte) 0; | ||||
|                 } | ||||
|                 storeLiving((LivingEntity) entity); | ||||
|             } | ||||
|             // END LIVING // | ||||
|         } | ||||
|     } | ||||
| @@ -438,6 +389,11 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @deprecated Use {@link #restoreBreedable(Breedable)} instead | ||||
|      * @since 7.1.0 | ||||
|      */ | ||||
|     @Deprecated(forRemoval = true, since = "7.1.0") | ||||
|     private void restoreAgeable(Ageable entity) { | ||||
|         if (!this.aged.adult) { | ||||
|             entity.setBaby(); | ||||
| @@ -448,6 +404,11 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @deprecated Use {@link #storeBreedable(Breedable)} instead | ||||
|      * @since 7.1.0 | ||||
|      */ | ||||
|     @Deprecated(forRemoval = true, since = "7.1.0") | ||||
|     public void storeAgeable(Ageable aged) { | ||||
|         this.aged = new AgeableStats(); | ||||
|         this.aged.age = aged.getAge(); | ||||
| @@ -455,6 +416,29 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|         this.aged.adult = aged.isAdult(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @since 7.1.0 | ||||
|      */ | ||||
|     private void restoreBreedable(Breedable entity) { | ||||
|         if (!this.aged.adult) { | ||||
|             entity.setBaby(); | ||||
|         } | ||||
|         entity.setAgeLock(this.aged.locked); | ||||
|         if (this.aged.age > 0) { | ||||
|             entity.setAge(this.aged.age); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @since 7.1.0 | ||||
|      */ | ||||
|     private void storeBreedable(Breedable breedable) { | ||||
|         this.aged = new AgeableStats(); | ||||
|         this.aged.age = breedable.getAge(); | ||||
|         this.aged.locked = breedable.getAgeLock(); | ||||
|         this.aged.adult = breedable.isAdult(); | ||||
|     } | ||||
|  | ||||
|     public void storeTameable(Tameable tamed) { | ||||
|         this.tamed = new TameableStats(); | ||||
|         this.tamed.owner = tamed.getOwner(); | ||||
| @@ -472,20 +456,15 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|         } | ||||
|         Entity entity; | ||||
|         switch (this.getType().toString()) { | ||||
|             case "DROPPED_ITEM": | ||||
|             case "DROPPED_ITEM", "ITEM" -> { | ||||
|                 return world.dropItem(location, this.stack); | ||||
|             case "PLAYER": | ||||
|             case "LEASH_HITCH": | ||||
|             } | ||||
|             case "PLAYER", "LEASH_HITCH" -> { | ||||
|                 return null; | ||||
|             case "ITEM_FRAME": | ||||
|                 entity = world.spawn(location, ItemFrame.class); | ||||
|                 break; | ||||
|             case "PAINTING": | ||||
|                 entity = world.spawn(location, Painting.class); | ||||
|                 break; | ||||
|             default: | ||||
|                 entity = world.spawnEntity(location, this.getType()); | ||||
|                 break; | ||||
|             } | ||||
|             case "ITEM_FRAME" -> entity = world.spawn(location, ItemFrame.class); | ||||
|             case "PAINTING" -> entity = world.spawn(location, Painting.class); | ||||
|             default -> entity = world.spawnEntity(location, this.getType()); | ||||
|         } | ||||
|         if (this.depth == 0) { | ||||
|             return entity; | ||||
| @@ -513,72 +492,56 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|             entity.setGravity(false); | ||||
|         } | ||||
|         switch (entity.getType().toString()) { | ||||
|             case "BOAT": | ||||
|             case "BOAT", "ACACIA_BOAT", "BIRCH_BOAT", "CHERRY_BOAT", "DARK_OAK_BOAT", "JUNGLE_BOAT", "MANGROVE_BOAT", | ||||
|                  "OAK_BOAT", "PALE_OAK_BOAT", "SPRUCE_BOAT", "BAMBOO_RAFT" -> { | ||||
|                 Boat boat = (Boat) entity; | ||||
|                 boat.setWoodType(TreeSpecies.values()[dataByte]); | ||||
|                 boat.setBoatType(Boat.Type.values()[dataByte]); | ||||
|                 return entity; | ||||
|             case "SLIME": | ||||
|             } | ||||
|             case "ACACIA_CHEST_BOAT", "BIRCH_CHEST_BOAT", "CHERRY_CHEST_BOAT", "DARK_OAK_CHEST_BOAT", | ||||
|                  "JUNGLE_CHEST_BOAT", "MANGROVE_CHEST_BOAT", "OAK_CHEST_BOAT", "PALE_OAK_CHEST_BOAT", | ||||
|                  "SPRUCE_CHEST_BOAT", "BAMBOO_CHEST_RAFT" -> { | ||||
|                 ChestBoat boat = (ChestBoat) entity; | ||||
|                 boat.setBoatType(Boat.Type.values()[dataByte]); | ||||
|                 restoreInventory(boat); | ||||
|                 return entity; | ||||
|             } | ||||
|             // SLIME is not even stored | ||||
|             /* case "SLIME" -> { | ||||
|                 ((Slime) entity).setSize(this.dataByte); | ||||
|                 return entity; | ||||
|             case "ARROW": | ||||
|             case "EGG": | ||||
|             case "ENDER_CRYSTAL": | ||||
|             case "ENDER_PEARL": | ||||
|             case "ENDER_SIGNAL": | ||||
|             case "DROPPED_ITEM": | ||||
|             case "EXPERIENCE_ORB": | ||||
|             case "FALLING_BLOCK": | ||||
|             case "FIREBALL": | ||||
|             case "FIREWORK": | ||||
|             case "FISHING_HOOK": | ||||
|             case "LEASH_HITCH": | ||||
|             case "LIGHTNING": | ||||
|             case "MINECART": | ||||
|             case "MINECART_COMMAND": | ||||
|             case "MINECART_MOB_SPAWNER": | ||||
|             case "MINECART_TNT": | ||||
|             case "PLAYER": | ||||
|             case "PRIMED_TNT": | ||||
|             case "SMALL_FIREBALL": | ||||
|             case "SNOWBALL": | ||||
|             case "SPLASH_POTION": | ||||
|             case "THROWN_EXP_BOTTLE": | ||||
|             case "SPECTRAL_ARROW": | ||||
|             case "SHULKER_BULLET": | ||||
|             case "AREA_EFFECT_CLOUD": | ||||
|             case "DRAGON_FIREBALL": | ||||
|             case "WITHER_SKULL": | ||||
|             case "MINECART_FURNACE": | ||||
|             case "LLAMA_SPIT": | ||||
|             case "TRIDENT": | ||||
|             case "UNKNOWN": | ||||
|             } */ | ||||
|             case "ARROW", "EGG", "END_CRYSTAL", "ENDER_CRYSTAL", "ENDER_PEARL", "ENDER_SIGNAL", "DROPPED_ITEM", "EXPERIENCE_ORB", "FALLING_BLOCK", | ||||
|                     "FIREBALL", "FIREWORK", "FISHING_HOOK", "LEASH_HITCH", "LIGHTNING", "MINECART", "MINECART_COMMAND", | ||||
|                     "MINECART_MOB_SPAWNER", "MINECART_TNT", "PLAYER", "PRIMED_TNT", "SMALL_FIREBALL", "SNOWBALL", | ||||
|                     "SPLASH_POTION", "THROWN_EXP_BOTTLE", "SPECTRAL_ARROW", "SHULKER_BULLET", "AREA_EFFECT_CLOUD", | ||||
|                     "DRAGON_FIREBALL", "WITHER_SKULL", "MINECART_FURNACE", "LLAMA_SPIT", "TRIDENT", "UNKNOWN" -> { | ||||
|                 // Do this stuff later | ||||
|                 return entity; | ||||
|             } | ||||
|             // MISC // | ||||
|             case "ITEM_FRAME": | ||||
|             case "ITEM_FRAME" -> { | ||||
|                 ItemFrame itemframe = (ItemFrame) entity; | ||||
|                 itemframe.setRotation(Rotation.values()[this.dataByte]); | ||||
|                 itemframe.setItem(this.stack); | ||||
|                 return entity; | ||||
|             case "PAINTING": | ||||
|             } | ||||
|             case "PAINTING" -> { | ||||
|                 Painting painting = (Painting) entity; | ||||
|                 painting.setFacingDirection(BlockFace.values()[this.dataByte], true); | ||||
|                 painting.setArt(Art.getByName(this.dataString), true); | ||||
|                 return entity; | ||||
|             } | ||||
|             // END MISC // | ||||
|             // INVENTORY HOLDER // | ||||
|             case "MINECART_CHEST": | ||||
|             case "MINECART_HOPPER": | ||||
|             case "MINECART_CHEST", "CHEST_MINECART", "MINECART_HOPPER", "HOPPER_MINECART" -> { | ||||
|                 restoreInventory((InventoryHolder) entity); | ||||
|                 return entity; | ||||
|             } | ||||
|             // START LIVING ENTITY // | ||||
|             // START AGEABLE // | ||||
|             // START TAMEABLE // | ||||
|             case "HORSE": | ||||
|             case "LLAMA": | ||||
|             case "SKELETON_HORSE": | ||||
|             case "DONKEY": | ||||
|             case "MULE": | ||||
|             case "CAMEL", "HORSE", "DONKEY", "LLAMA", "TRADER_LLAMA", "MULE", "SKELETON_HORSE", "ZOMBIE_HORSE" -> { | ||||
|                 AbstractHorse horse = (AbstractHorse) entity; | ||||
|                 horse.setJumpStrength(this.horse.jump); | ||||
|                 if (horse instanceof ChestedHorse) { | ||||
| @@ -589,48 +552,46 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|                 //horse.setStyle(this.horse.style); | ||||
|                 //horse.setColor(this.horse.color); | ||||
|                 restoreTameable(horse); | ||||
|                 restoreAgeable(horse); | ||||
|                 restoreBreedable(horse); | ||||
|                 restoreLiving(horse); | ||||
|                 restoreInventory(horse); | ||||
|                 return entity; | ||||
|             } | ||||
|             // END INVENTORY HOLDER // | ||||
|             case "WOLF": | ||||
|             case "OCELOT": | ||||
|             case "WOLF", "OCELOT", "CAT", "PARROT" -> { | ||||
|                 restoreTameable((Tameable) entity); | ||||
|                 restoreAgeable((Ageable) entity); | ||||
|                 restoreBreedable((Breedable) entity); | ||||
|                 restoreLiving((LivingEntity) entity); | ||||
|                 return entity; | ||||
|             } | ||||
|             // END AGEABLE // | ||||
|             case "SHEEP": | ||||
|             case "SHEEP" -> { | ||||
|                 Sheep sheep = (Sheep) entity; | ||||
|                 if (this.dataByte == 1) { | ||||
|                     sheep.setSheared(true); | ||||
|                 } | ||||
|                 if (this.dataByte2 != 0) { | ||||
|                     sheep.setColor(DyeColor.getByDyeData(this.dataByte2)); | ||||
|                     sheep.setColor(DyeColor.values()[this.dataByte2]); | ||||
|                 } | ||||
|                 restoreAgeable(sheep); | ||||
|                 restoreBreedable(sheep); | ||||
|                 restoreLiving(sheep); | ||||
|                 return sheep; | ||||
|             case "VILLAGER": | ||||
|             case "CHICKEN": | ||||
|             case "COW": | ||||
|             case "TURTLE": | ||||
|             case "POLAR_BEAR": | ||||
|             case "MUSHROOM_COW": | ||||
|             case "PIG": | ||||
|                 restoreAgeable((Ageable) entity); | ||||
|             } | ||||
|             case "VILLAGER", "CHICKEN", "COW", "TURTLE", "POLAR_BEAR", "MUSHROOM_COW", "PIG" -> { | ||||
|                 restoreBreedable((Breedable) entity); | ||||
|                 restoreLiving((LivingEntity) entity); | ||||
|                 return entity; | ||||
|             } | ||||
|             // END AGEABLE // | ||||
|             case "RABBIT": | ||||
|             case "RABBIT" -> { | ||||
|                 if (this.dataByte != 0) { | ||||
|                     ((Rabbit) entity).setRabbitType(Rabbit.Type.values()[this.dataByte]); | ||||
|                 } | ||||
|                 restoreAgeable((Ageable) entity); | ||||
|                 restoreBreedable((Breedable) entity); | ||||
|                 restoreLiving((LivingEntity) entity); | ||||
|                 return entity; | ||||
|             case "ARMOR_STAND": | ||||
|             } | ||||
|             case "ARMOR_STAND" -> { | ||||
|                 // CHECK positions | ||||
|                 ArmorStand stand = (ArmorStand) entity; | ||||
|                 if (this.inventory[0] != null) { | ||||
| @@ -700,56 +661,38 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | ||||
|                 } | ||||
|                 restoreLiving(stand); | ||||
|                 return stand; | ||||
|             case "BAT": | ||||
|             } | ||||
|             case "BAT" -> { | ||||
|                 if (this.dataByte != 0) { | ||||
|                     ((Bat) entity).setAwake(true); | ||||
|                 } | ||||
|                 restoreLiving((LivingEntity) entity); | ||||
|                 return entity; | ||||
|             case "ENDER_DRAGON": | ||||
|             } | ||||
|             case "ENDER_DRAGON" -> { | ||||
|                 if (this.dataByte != 0) { | ||||
|                     ((EnderDragon) entity).setPhase(EnderDragon.Phase.values()[this.dataByte]); | ||||
|                 } | ||||
|                 restoreLiving((LivingEntity) entity); | ||||
|                 return entity; | ||||
|             case "ENDERMITE": | ||||
|             case "GHAST": | ||||
|             case "MAGMA_CUBE": | ||||
|             case "SQUID": | ||||
|             case "PIG_ZOMBIE": | ||||
|             case "HOGLIN": | ||||
|             case "PIGLIN": | ||||
|             case "ZOMBIFIED_PIGLIN": | ||||
|             case "PIGLIN_BRUTE": | ||||
|             case "ZOMBIE": | ||||
|             case "WITHER": | ||||
|             case "WITCH": | ||||
|             case "SPIDER": | ||||
|             case "CAVE_SPIDER": | ||||
|             case "SILVERFISH": | ||||
|             case "GIANT": | ||||
|             case "ENDERMAN": | ||||
|             case "CREEPER": | ||||
|             case "BLAZE": | ||||
|             case "SNOWMAN": | ||||
|             case "SHULKER": | ||||
|             case "GUARDIAN": | ||||
|             case "ELDER_GUARDIAN": | ||||
|             case "SKELETON": | ||||
|             case "WITHER_SKELETON": | ||||
|             } | ||||
|             case "ENDERMITE", "GHAST", "HAPPY_GHAST", "GHASTLING", "MAGMA_CUBE", "SQUID", "PIG_ZOMBIE", "HOGLIN", "PIGLIN", "ZOMBIFIED_PIGLIN", "PIGLIN_BRUTE", "ZOMBIE", "WITHER", "WITCH", "SPIDER", "CAVE_SPIDER", "SILVERFISH", "GIANT", "ENDERMAN", "CREEPER", "BLAZE", "SNOWMAN", "SHULKER", "GUARDIAN", "ELDER_GUARDIAN", "SKELETON", "WITHER_SKELETON" -> { | ||||
|                 restoreLiving((LivingEntity) entity); | ||||
|                 return entity; | ||||
|             case "IRON_GOLEM": | ||||
|             } | ||||
|             case "IRON_GOLEM" -> { | ||||
|                 if (this.dataByte != 0) { | ||||
|                     ((IronGolem) entity).setPlayerCreated(true); | ||||
|                 } | ||||
|                 restoreLiving((LivingEntity) entity); | ||||
|                 return entity; | ||||
|             default: | ||||
|             } | ||||
|             default -> { | ||||
|                 if (Settings.DEBUG) { | ||||
|                     LOGGER.info("Could not identify entity: {}", entity.getType()); | ||||
|                 } | ||||
|                 return entity; | ||||
|             } | ||||
|             // END LIVING | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -25,17 +18,17 @@ | ||||
|  */ | ||||
| package com.plotsquared.bukkit.generator; | ||||
|  | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.bukkit.queue.LimitedRegionWrapperQueue; | ||||
| import com.plotsquared.core.generator.HybridPlotWorld; | ||||
| import com.plotsquared.core.generator.IndependentPlotGenerator; | ||||
| import com.plotsquared.core.location.ChunkWrapper; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.location.UncheckedWorldLocation; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.queue.QueueCoordinator; | ||||
| import com.plotsquared.core.queue.ScopedQueueCoordinator; | ||||
| import com.sk89q.worldedit.bukkit.BukkitWorld; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.World; | ||||
| import com.plotsquared.core.plot.world.SinglePlotArea; | ||||
| import com.plotsquared.core.queue.ZeroedDelegateScopedQueueCoordinator; | ||||
| import org.bukkit.generator.BlockPopulator; | ||||
| import org.bukkit.generator.LimitedRegion; | ||||
| import org.bukkit.generator.WorldInfo; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Random; | ||||
| @@ -43,35 +36,40 @@ import java.util.Random; | ||||
| final class BlockStatePopulator extends BlockPopulator { | ||||
|  | ||||
|     private final IndependentPlotGenerator plotGenerator; | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|  | ||||
|     private QueueCoordinator queue; | ||||
|  | ||||
|     /** | ||||
|      * @since 6.9.0 | ||||
|      */ | ||||
|     public BlockStatePopulator( | ||||
|             final @NonNull IndependentPlotGenerator plotGenerator, | ||||
|             final @NonNull PlotAreaManager plotAreaManager | ||||
|             final @NonNull IndependentPlotGenerator plotGenerator | ||||
|     ) { | ||||
|         this.plotGenerator = plotGenerator; | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void populate(final @NonNull World world, final @NonNull Random random, final @NonNull Chunk source) { | ||||
|         if (this.queue == null) { | ||||
|             this.queue = PlotSquared.platform().globalBlockQueue().getNewQueue(new BukkitWorld(world)); | ||||
|         } | ||||
|         final PlotArea area = this.plotAreaManager.getPlotArea(world.getName(), null); | ||||
|         if (area == null) { | ||||
|     public void populate( | ||||
|             @NonNull final WorldInfo worldInfo, | ||||
|             @NonNull final Random random, | ||||
|             final int chunkX, | ||||
|             final int chunkZ, | ||||
|             @NonNull final LimitedRegion limitedRegion | ||||
|     ) { | ||||
|         PlotArea area = UncheckedWorldLocation.at(worldInfo.getName(), chunkX << 4, 0, chunkZ << 4).getPlotArea(); | ||||
|         if (area == null || (area instanceof HybridPlotWorld hpw && !hpw.populationNeeded()) || area instanceof SinglePlotArea) { | ||||
|             return; | ||||
|         } | ||||
|         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 | ||||
|         LimitedRegionWrapperQueue wrapped = new LimitedRegionWrapperQueue(limitedRegion); | ||||
|         // It is possible for the region to be larger than the chunk, but there is no reason for P2 to need to populate | ||||
|         // outside of the actual chunk area. | ||||
|         Location min = UncheckedWorldLocation.at(worldInfo.getName(), chunkX << 4, worldInfo.getMinHeight(), chunkZ << 4); | ||||
|         Location max = UncheckedWorldLocation.at( | ||||
|                 worldInfo.getName(), | ||||
|                 (chunkX << 4) + 15, | ||||
|                 worldInfo.getMaxHeight(), | ||||
|                 (chunkZ << 4) + 15 | ||||
|         ); | ||||
|         if (this.plotGenerator.populateChunk(chunk, area)) { | ||||
|             this.queue.enqueue(); | ||||
|         } | ||||
|         ZeroedDelegateScopedQueueCoordinator offsetChunkQueue = new ZeroedDelegateScopedQueueCoordinator(wrapped, min, max); | ||||
|         this.plotGenerator.populateChunk(offsetChunkQueue, area); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -25,7 +18,11 @@ | ||||
|  */ | ||||
| package com.plotsquared.bukkit.generator; | ||||
|  | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| 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.World; | ||||
| import org.bukkit.generator.BlockPopulator; | ||||
| @@ -52,7 +49,14 @@ public class BukkitAugmentedGenerator extends BlockPopulator { | ||||
|  | ||||
|     @Override | ||||
|     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(); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -29,28 +22,46 @@ import com.plotsquared.bukkit.queue.GenChunk; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.bukkit.util.BukkitWorld; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.generator.ClassicPlotWorld; | ||||
| import com.plotsquared.core.generator.GeneratorWrapper; | ||||
| import com.plotsquared.core.generator.IndependentPlotGenerator; | ||||
| import com.plotsquared.core.generator.SingleWorldGenerator; | ||||
| import com.plotsquared.core.location.ChunkWrapper; | ||||
| import com.plotsquared.core.location.UncheckedWorldLocation; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.queue.ScopedQueueCoordinator; | ||||
| import com.plotsquared.core.queue.ZeroedDelegateScopedQueueCoordinator; | ||||
| import com.plotsquared.core.util.ChunkManager; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.math.BlockVector2; | ||||
| import com.sk89q.worldedit.math.BlockVector3; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.bukkit.HeightMap; | ||||
| import org.bukkit.NamespacedKey; | ||||
| import org.bukkit.Registry; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.Biome; | ||||
| import org.bukkit.generator.BiomeProvider; | ||||
| import org.bukkit.generator.BlockPopulator; | ||||
| import org.bukkit.generator.ChunkGenerator; | ||||
| import org.bukkit.generator.WorldInfo; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Arrays; | ||||
| import java.util.HashSet; | ||||
| import java.util.List; | ||||
| import java.util.Random; | ||||
| import java.util.Set; | ||||
|  | ||||
| public class BukkitPlotGenerator extends ChunkGenerator | ||||
|         implements GeneratorWrapper<ChunkGenerator> { | ||||
| import static java.util.function.Predicate.not; | ||||
|  | ||||
| public class BukkitPlotGenerator extends ChunkGenerator implements GeneratorWrapper<ChunkGenerator> { | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + BukkitPlotGenerator.class.getSimpleName()); | ||||
|  | ||||
|     @SuppressWarnings("unused") | ||||
|     public final boolean PAPER_ASYNC_SAFE = true; | ||||
| @@ -60,9 +71,15 @@ public class BukkitPlotGenerator extends ChunkGenerator | ||||
|     private final ChunkGenerator platformGenerator; | ||||
|     private final boolean full; | ||||
|     private final String levelName; | ||||
|     private final boolean useNewGenerationMethods; | ||||
|     private final BiomeProvider biomeProvider; | ||||
|     private List<BlockPopulator> populators; | ||||
|     private boolean loaded = false; | ||||
|  | ||||
|     private PlotArea lastPlotArea; | ||||
|     private int lastChunkX = Integer.MIN_VALUE; | ||||
|     private int lastChunkZ = Integer.MIN_VALUE; | ||||
|  | ||||
|     public BukkitPlotGenerator( | ||||
|             final @NonNull String name, | ||||
|             final @NonNull IndependentPlotGenerator generator, | ||||
| @@ -73,20 +90,25 @@ public class BukkitPlotGenerator extends ChunkGenerator | ||||
|         this.plotGenerator = generator; | ||||
|         this.platformGenerator = this; | ||||
|         this.populators = new ArrayList<>(); | ||||
|         this.populators.add(new BlockStatePopulator(this.plotGenerator, this.plotAreaManager)); | ||||
|         this.populators.add(new BlockStatePopulator(this.plotGenerator)); | ||||
|         this.full = true; | ||||
|         this.useNewGenerationMethods = PlotSquared.platform().serverVersion()[1] >= 19; | ||||
|         this.biomeProvider = new BukkitPlotBiomeProvider(); | ||||
|     } | ||||
|  | ||||
|     public BukkitPlotGenerator(final String world, final ChunkGenerator cg, final @NonNull PlotAreaManager plotAreaManager) { | ||||
|         if (cg instanceof BukkitPlotGenerator) { | ||||
|             throw new IllegalArgumentException("ChunkGenerator: " + cg.getClass().getName() | ||||
|                     + " is already a BukkitPlotGenerator!"); | ||||
|             throw new IllegalArgumentException("ChunkGenerator: " + cg | ||||
|                     .getClass() | ||||
|                     .getName() + " is already a BukkitPlotGenerator!"); | ||||
|         } | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|         this.levelName = world; | ||||
|         this.full = false; | ||||
|         this.platformGenerator = cg; | ||||
|         this.plotGenerator = new DelegatePlotGenerator(cg, world); | ||||
|         this.useNewGenerationMethods = PlotSquared.platform().serverVersion()[1] >= 19; | ||||
|         this.biomeProvider = null; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -112,32 +134,9 @@ public class BukkitPlotGenerator extends ChunkGenerator | ||||
|     @Override | ||||
|     public @NonNull List<BlockPopulator> getDefaultPopulators(@NonNull World world) { | ||||
|         try { | ||||
|             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); | ||||
|                         } | ||||
|                         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; | ||||
|             } | ||||
|             checkLoaded(world); | ||||
|         } catch (Exception e) { | ||||
|             e.printStackTrace(); | ||||
|             LOGGER.error("Error attempting to load world into PlotSquared.", e); | ||||
|         } | ||||
|         ArrayList<BlockPopulator> toAdd = new ArrayList<>(); | ||||
|         List<BlockPopulator> existing = world.getPopulators(); | ||||
| @@ -154,11 +153,151 @@ public class BukkitPlotGenerator extends ChunkGenerator | ||||
|         return toAdd; | ||||
|     } | ||||
|  | ||||
|     // Extracted to synchronized method for thread-safety, preventing multiple internal world load calls | ||||
|     private synchronized void checkLoaded(@NonNull World world) { | ||||
|         // Do not attempt to load configurations until WorldEdit has a platform ready. | ||||
|         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; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("deprecation") // Kept for compatibility with <=1.17.1 | ||||
|     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 | ||||
|     public void generateNoise( | ||||
|             @NotNull final WorldInfo worldInfo, | ||||
|             @NotNull final Random random, | ||||
|             final int chunkX, | ||||
|             final int chunkZ, | ||||
|             @NotNull final ChunkData chunkData | ||||
|     ) { | ||||
|         if (this.platformGenerator != this) { | ||||
|             this.platformGenerator.generateNoise(worldInfo, random, chunkX, chunkZ, chunkData); | ||||
|             return; | ||||
|         } | ||||
|         int minY = chunkData.getMinHeight(); | ||||
|         int maxY = chunkData.getMaxHeight(); | ||||
|         GenChunk result = new GenChunk(minY, maxY); | ||||
|         // Set the chunk location | ||||
|         result.setChunk(new ChunkWrapper(worldInfo.getName(), chunkX, chunkZ)); | ||||
|         // Set the result data | ||||
|         result.setChunkData(chunkData); | ||||
|         result.result = null; | ||||
|  | ||||
|         // Catch any exceptions (as exceptions usually thrown) | ||||
|         try { | ||||
|             generate(BlockVector2.at(chunkX, chunkZ), worldInfo.getName(), result, false); | ||||
|         } catch (Throwable e) { | ||||
|             LOGGER.error("Error attempting to generate chunk.", e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void generateSurface( | ||||
|             @NotNull final WorldInfo worldInfo, | ||||
|             @NotNull final Random random, | ||||
|             final int chunkX, | ||||
|             final int chunkZ, | ||||
|             @NotNull final ChunkData chunkData | ||||
|     ) { | ||||
|         if (platformGenerator != this) { | ||||
|             platformGenerator.generateSurface(worldInfo, random, chunkX, chunkZ, chunkData); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void generateBedrock( | ||||
|             @NotNull final WorldInfo worldInfo, | ||||
|             @NotNull final Random random, | ||||
|             final int chunkX, | ||||
|             final int chunkZ, | ||||
|             @NotNull final ChunkData chunkData | ||||
|     ) { | ||||
|         if (platformGenerator != this) { | ||||
|             platformGenerator.generateBedrock(worldInfo, random, chunkX, chunkZ, chunkData); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void generateCaves( | ||||
|             @NotNull final WorldInfo worldInfo, | ||||
|             @NotNull final Random random, | ||||
|             final int chunkX, | ||||
|             final int chunkZ, | ||||
|             @NotNull final ChunkData chunkData | ||||
|     ) { | ||||
|         if (platformGenerator != this) { | ||||
|             platformGenerator.generateCaves(worldInfo, random, chunkX, chunkZ, chunkData); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public @Nullable BiomeProvider getDefaultBiomeProvider(@NotNull final WorldInfo worldInfo) { | ||||
|         if (platformGenerator != this) { | ||||
|             return platformGenerator.getDefaultBiomeProvider(worldInfo); | ||||
|         } | ||||
|         return biomeProvider; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public int getBaseHeight( | ||||
|             @NotNull final WorldInfo worldInfo, | ||||
|             @NotNull final Random random, | ||||
|             final int x, | ||||
|             final int z, | ||||
|             @NotNull final HeightMap heightMap | ||||
|     ) { | ||||
|         PlotArea area = getPlotArea(worldInfo.getName(), x, z); | ||||
|         if (area instanceof ClassicPlotWorld cpw) { | ||||
|             // Default to plot height being the heighest point before decoration (i.e. roads, walls etc.) | ||||
|             return cpw.PLOT_HEIGHT; | ||||
|         } | ||||
|         return super.getBaseHeight(worldInfo, random, x, z, heightMap); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * The entire method is deprecated, but kept for compatibility with versions lower than or equal to 1.16.2. | ||||
|      * The method will be removed in future versions, because WorldEdit and FastAsyncWorldEdit only support the latest point | ||||
|      * release. | ||||
|      */ | ||||
|     @SuppressWarnings("deprecation") // The entire method is deprecated, but kept for compatibility with <=1.16.2 | ||||
|     @Override | ||||
|     @Deprecated(since = "7.0.0") | ||||
|     public @NonNull ChunkData generateChunkData( | ||||
|             @NonNull World world, @NonNull Random random, int x, int z, @NonNull BiomeGrid biome | ||||
|     ) { | ||||
|         if (useNewGenerationMethods) { | ||||
|             if (this.platformGenerator != this) { | ||||
|                 return this.platformGenerator.generateChunkData(world, random, x, z, biome); | ||||
|             } else { | ||||
|                 // Throw exception to be caught by the server that indicates the new generation API is being used. | ||||
|                 throw new UnsupportedOperationException("Using new generation methods. This method is unsupported."); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         int minY = BukkitWorld.getMinWorldHeight(world); | ||||
|         int maxY = BukkitWorld.getMaxWorldHeight(world); | ||||
| @@ -169,7 +308,6 @@ public class BukkitPlotGenerator extends ChunkGenerator | ||||
|                     for (int chunkZ = 0; chunkZ < 16; chunkZ++) { | ||||
|                         for (int y = minY; y < maxY; y++) { | ||||
|                             biome.setBiome(chunkX, y, chunkZ, Biome.PLAINS); | ||||
|  | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
| @@ -189,37 +327,32 @@ public class BukkitPlotGenerator extends ChunkGenerator | ||||
|             if (this.platformGenerator != this) { | ||||
|                 return this.platformGenerator.generateChunkData(world, random, x, z, biome); | ||||
|             } else { | ||||
|                 generate(BlockVector2.at(x, z), world, result); | ||||
|                 generate(BlockVector2.at(x, z), world.getName(), result, true); | ||||
|             } | ||||
|         } catch (Throwable e) { | ||||
|             e.printStackTrace(); | ||||
|             LOGGER.error("Error attempting to load world into PlotSquared.", e); | ||||
|         } | ||||
|         // Return the result data | ||||
|         return result.getChunkData(); | ||||
|     } | ||||
|  | ||||
|     private void generate(BlockVector2 loc, World world, ScopedQueueCoordinator result) { | ||||
|     private void generate(BlockVector2 loc, String world, ZeroedDelegateScopedQueueCoordinator result, boolean biomes) { | ||||
|         // Load if improperly loaded | ||||
|         if (!this.loaded) { | ||||
|             String name = world.getName(); | ||||
|             PlotSquared.get().loadWorld(name, this); | ||||
|             this.loaded = true; | ||||
|             synchronized (this) { | ||||
|                 PlotSquared.get().loadWorld(world, this); | ||||
|             } | ||||
|         } | ||||
|         // Process the chunk | ||||
|         if (ChunkManager.preProcessChunk(loc, result)) { | ||||
|             return; | ||||
|         } | ||||
|         PlotArea area = this.plotAreaManager.getPlotArea(world.getName(), null); | ||||
|         if (area == null && (area = this.plotAreaManager.getPlotArea(this.levelName, null)) == null) { | ||||
|             throw new IllegalStateException( | ||||
|                     "Cannot regenerate chunk that does not belong to a plot area." + " Location: " + loc | ||||
|                             + ", world: " + world); | ||||
|         } | ||||
|         PlotArea area = getPlotArea(world, loc.getX(), loc.getZ()); | ||||
|         try { | ||||
|             this.plotGenerator.generateChunk(result, area); | ||||
|             this.plotGenerator.generateChunk(result, area, biomes); | ||||
|         } catch (Throwable e) { | ||||
|             // Recover from generator error | ||||
|             e.printStackTrace(); | ||||
|             LOGGER.error("Error attempting to generate chunk.", e); | ||||
|         } | ||||
|         ChunkManager.postProcessChunk(loc, result); | ||||
|     } | ||||
| @@ -273,4 +406,63 @@ public class BukkitPlotGenerator extends ChunkGenerator | ||||
|         return this.levelName; | ||||
|     } | ||||
|  | ||||
|     private synchronized PlotArea getPlotArea(String name, int chunkX, int chunkZ) { | ||||
|         // Load if improperly loaded | ||||
|         if (!this.loaded) { | ||||
|             PlotSquared.get().loadWorld(name, this); | ||||
|             // Do not set loaded to true as we want to ensure spawn limits are set when "loading" is actually able to be | ||||
|             // completed properly. | ||||
|         } | ||||
|         if (lastPlotArea != null && name.equals(this.levelName) && chunkX == lastChunkX && chunkZ == lastChunkZ) { | ||||
|             return lastPlotArea; | ||||
|         } | ||||
|         BlockVector3 loc = BlockVector3.at(chunkX << 4, 0, chunkZ << 4); | ||||
|         if (lastPlotArea != null && lastPlotArea.getRegion().contains(loc) && lastPlotArea.getRegion().contains(loc)) { | ||||
|             return lastPlotArea; | ||||
|         } | ||||
|         PlotArea area = UncheckedWorldLocation.at(name, loc).getPlotArea(); | ||||
|         if (area == null) { | ||||
|             throw new IllegalStateException(String.format( | ||||
|                     "Cannot generate chunk that does not belong to a plot area. World: %s", | ||||
|                     name | ||||
|             )); | ||||
|         } | ||||
|         this.lastChunkX = chunkX; | ||||
|         this.lastChunkZ = chunkZ; | ||||
|         return this.lastPlotArea = area; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Biome provider should never need to be accessed outside of this class. | ||||
|      */ | ||||
|     private final class BukkitPlotBiomeProvider extends BiomeProvider { | ||||
|  | ||||
|         private static final List<Biome> BIOMES; | ||||
|  | ||||
|         static { | ||||
|             Set<Biome> disabledBiomes = new HashSet<>(List.of(Biome.CUSTOM)); | ||||
|             if (PlotSquared.platform().serverVersion()[1] <= 19) { | ||||
|                 final Biome cherryGrove = Registry.BIOME.get(NamespacedKey.minecraft("cherry_grove")); | ||||
|                 if (cherryGrove != null) { | ||||
|                     disabledBiomes.add(cherryGrove); | ||||
|                 } | ||||
|             } | ||||
|             BIOMES = Arrays.stream(Biome.values()) | ||||
|                     .filter(not(disabledBiomes::contains)) | ||||
|                     .toList(); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public @NotNull Biome getBiome(@NotNull final WorldInfo worldInfo, final int x, final int y, final int z) { | ||||
|             PlotArea area = getPlotArea(worldInfo.getName(), x >> 4, z >> 4); | ||||
|             return BukkitAdapter.adapt(plotGenerator.getBiome(area, x, y, z)); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public @NotNull List<Biome> getBiomes(@NotNull final WorldInfo worldInfo) { | ||||
|             return BIOMES; // Allow all biomes | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -31,9 +24,10 @@ import com.plotsquared.core.generator.IndependentPlotGenerator; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.PlotId; | ||||
| import com.plotsquared.core.queue.ScopedQueueCoordinator; | ||||
| import com.plotsquared.core.queue.ZeroedDelegateScopedQueueCoordinator; | ||||
| import com.plotsquared.core.util.MathMan; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.world.biome.BiomeType; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.Biome; | ||||
| import org.bukkit.generator.BlockPopulator; | ||||
| @@ -56,6 +50,11 @@ final class DelegatePlotGenerator extends IndependentPlotGenerator { | ||||
|     public void initialize(PlotArea area) { | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public BiomeType getBiome(final PlotArea settings, final int x, final int y, final int z) { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public String getName() { | ||||
|         return this.chunkGenerator.getClass().getName(); | ||||
| @@ -67,7 +66,7 @@ final class DelegatePlotGenerator extends IndependentPlotGenerator { | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void generateChunk(final ScopedQueueCoordinator result, PlotArea settings) { | ||||
|     public void generateChunk(final ZeroedDelegateScopedQueueCoordinator result, PlotArea settings, boolean biomes) { | ||||
|         World world = BukkitUtil.getWorld(this.world); | ||||
|         Location min = result.getMin(); | ||||
|         int chunkX = min.getX() >> 4; | ||||
| @@ -80,7 +79,8 @@ final class DelegatePlotGenerator extends IndependentPlotGenerator { | ||||
|                     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) | ||||
|                 @Override | ||||
|                 public void setBiome(int x, int y, int z, @NonNull Biome biome) { | ||||
|                     result.setBiome(x, z, BukkitAdapter.adapt(biome)); | ||||
|  | ||||
|   | ||||
| @@ -0,0 +1,74 @@ | ||||
| /* | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.generator; | ||||
|  | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.generator.HybridPlotWorld; | ||||
| import com.plotsquared.core.generator.IndependentPlotGenerator; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.location.UncheckedWorldLocation; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.world.SinglePlotArea; | ||||
| import com.plotsquared.core.queue.QueueCoordinator; | ||||
| import com.plotsquared.core.queue.ZeroedDelegateScopedQueueCoordinator; | ||||
| import com.sk89q.worldedit.bukkit.BukkitWorld; | ||||
| import com.sk89q.worldedit.util.SideEffectSet; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.generator.BlockPopulator; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Random; | ||||
|  | ||||
| @Deprecated(since = "TODO") | ||||
| final class LegacyBlockStatePopulator extends BlockPopulator { | ||||
|  | ||||
|     private final IndependentPlotGenerator plotGenerator; | ||||
|  | ||||
|     /** | ||||
|      * @since 6.9.0 | ||||
|      */ | ||||
|     public LegacyBlockStatePopulator( | ||||
|             final @NonNull IndependentPlotGenerator plotGenerator | ||||
|     ) { | ||||
|         this.plotGenerator = plotGenerator; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void populate(@NonNull final World world, @NonNull final Random random, @NonNull final Chunk source) { | ||||
|         int chunkMinX = source.getX() << 4; | ||||
|         int chunkMinZ = source.getZ() << 4; | ||||
|         PlotArea area = Location.at(world.getName(), chunkMinX, 0, chunkMinZ).getPlotArea(); | ||||
|         if (area == null || (area instanceof HybridPlotWorld hpw && !hpw.populationNeeded()) || area instanceof SinglePlotArea) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         QueueCoordinator queue = PlotSquared.platform().globalBlockQueue().getNewQueue(new BukkitWorld(world)); | ||||
|         queue.setForceSync(true); | ||||
|         queue.setSideEffectSet(SideEffectSet.none()); | ||||
|         queue.setBiomesEnabled(false); | ||||
|         queue.setChunkObject(source); | ||||
|         Location min = UncheckedWorldLocation.at(world.getName(), chunkMinX, world.getMinHeight(), chunkMinZ); | ||||
|         Location max = UncheckedWorldLocation.at(world.getName(), chunkMinX + 15, world.getMaxHeight(), chunkMinZ + 15); | ||||
|         ZeroedDelegateScopedQueueCoordinator offsetChunkQueue = new ZeroedDelegateScopedQueueCoordinator(queue, min, max); | ||||
|         this.plotGenerator.populateChunk(offsetChunkQueue, area); | ||||
|         queue.enqueue(); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -30,13 +23,13 @@ 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.ServerListener; | ||||
| 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; | ||||
| @@ -54,6 +47,9 @@ 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.player.OfflinePlotPlayer; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.world.DefaultPlotAreaManager; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.plot.world.SinglePlotAreaManager; | ||||
| @@ -79,6 +75,8 @@ import org.bukkit.command.ConsoleCommandSender; | ||||
| import org.bukkit.plugin.java.JavaPlugin; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Objects; | ||||
|  | ||||
| public class BukkitModule extends AbstractModule { | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + BukkitModule.class.getSimpleName()); | ||||
| @@ -135,21 +133,64 @@ public class BukkitModule extends AbstractModule { | ||||
|     @Provides | ||||
|     @Singleton | ||||
|     @NonNull EconHandler provideEconHandler() { | ||||
|         if (!Settings.Enabled_Components.ECONOMY) { | ||||
|         if (!Settings.Enabled_Components.ECONOMY || !Bukkit.getPluginManager().isPluginEnabled("Vault")) { | ||||
|             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(); | ||||
|         // Guice eagerly initializes singletons, so we need to bring the laziness ourselves | ||||
|         return new LazyEconHandler(); | ||||
|     } | ||||
|                 return econHandler; | ||||
|             } catch (final Exception ignored) { | ||||
|  | ||||
|     private static final class LazyEconHandler extends EconHandler implements ServerListener.MutableEconHandler { | ||||
|         private volatile EconHandler implementation; | ||||
|  | ||||
|         public void setImplementation(EconHandler econHandler) { | ||||
|             this.implementation = econHandler; | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public boolean init() { | ||||
|             return get().init(); | ||||
|         } | ||||
|         return EconHandler.nullEconHandler(); | ||||
|  | ||||
|         @Override | ||||
|         public double getBalance(final PlotPlayer<?> player) { | ||||
|             return get().getBalance(player); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public void withdrawMoney(final PlotPlayer<?> player, final double amount) { | ||||
|             get().withdrawMoney(player, amount); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public void depositMoney(final PlotPlayer<?> player, final double amount) { | ||||
|             get().depositMoney(player, amount); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public void depositMoney(final OfflinePlotPlayer player, final double amount) { | ||||
|             get().depositMoney(player, amount); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public boolean isEnabled(final PlotArea plotArea) { | ||||
|             return get().isEnabled(plotArea); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public @NonNull String format(final double balance) { | ||||
|             return get().format(balance); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public boolean isSupported() { | ||||
|             return get().isSupported(); | ||||
|         } | ||||
|  | ||||
|         private EconHandler get() { | ||||
|             return Objects.requireNonNull(this.implementation, "EconHandler not ready yet."); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -31,7 +24,6 @@ import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.database.DBFunc; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| @@ -40,6 +32,7 @@ import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.flag.implementations.BlockBurnFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.BlockIgnitionFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.BreakFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.ConcreteHardenFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.CoralDryFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.CropGrowFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.DisablePhysicsFlag; | ||||
| @@ -54,7 +47,6 @@ import com.plotsquared.core.plot.flag.implementations.LeafDecayFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.LiquidFlowFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.MycelGrowFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.PlaceFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.RedstoneFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.SnowFormFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.SnowMeltFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.SoilDryFlag; | ||||
| @@ -62,13 +54,14 @@ import com.plotsquared.core.plot.flag.implementations.VineGrowFlag; | ||||
| import com.plotsquared.core.plot.flag.types.BlockTypeWrapper; | ||||
| import com.plotsquared.core.plot.flag.types.BooleanFlag; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.PlotFlagUtil; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import com.plotsquared.core.util.task.TaskTime; | ||||
| import com.sk89q.worldedit.WorldEdit; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.world.block.BlockType; | ||||
| import net.kyori.adventure.text.minimessage.Template; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.GameMode; | ||||
| import org.bukkit.Material; | ||||
| @@ -76,6 +69,8 @@ import org.bukkit.block.Block; | ||||
| import org.bukkit.block.BlockFace; | ||||
| import org.bukkit.block.BlockState; | ||||
| import org.bukkit.block.data.BlockData; | ||||
| import org.bukkit.block.data.type.Dispenser; | ||||
| import org.bukkit.block.data.type.Farmland; | ||||
| import org.bukkit.entity.Entity; | ||||
| import org.bukkit.entity.Fireball; | ||||
| import org.bukkit.entity.Player; | ||||
| @@ -94,18 +89,16 @@ import org.bukkit.event.block.BlockFromToEvent; | ||||
| import org.bukkit.event.block.BlockGrowEvent; | ||||
| import org.bukkit.event.block.BlockIgniteEvent; | ||||
| import org.bukkit.event.block.BlockMultiPlaceEvent; | ||||
| import org.bukkit.event.block.BlockPhysicsEvent; | ||||
| import org.bukkit.event.block.BlockPistonExtendEvent; | ||||
| import org.bukkit.event.block.BlockPistonRetractEvent; | ||||
| import org.bukkit.event.block.BlockPlaceEvent; | ||||
| import org.bukkit.event.block.BlockRedstoneEvent; | ||||
| import org.bukkit.event.block.BlockSpreadEvent; | ||||
| import org.bukkit.event.block.CauldronLevelChangeEvent; | ||||
| import org.bukkit.event.block.EntityBlockFormEvent; | ||||
| import org.bukkit.event.block.LeavesDecayEvent; | ||||
| import org.bukkit.event.block.MoistureChangeEvent; | ||||
| import org.bukkit.event.block.SpongeAbsorbEvent; | ||||
| import org.bukkit.event.world.StructureGrowEvent; | ||||
| import org.bukkit.material.Directional; | ||||
| import org.bukkit.projectiles.BlockProjectileSource; | ||||
| import org.bukkit.util.Vector; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| @@ -113,11 +106,13 @@ import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import java.util.Iterator; | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
| import java.util.UUID; | ||||
|  | ||||
| import static org.bukkit.Tag.CORALS; | ||||
| import static org.bukkit.Tag.CORAL_BLOCKS; | ||||
| import static org.bukkit.Tag.WALL_CORALS; | ||||
|  | ||||
| @SuppressWarnings("unused") | ||||
| public class BlockEventListener implements Listener { | ||||
|  | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|     private final WorldEdit worldEdit; | ||||
|  | ||||
| @@ -146,128 +141,6 @@ public class BlockEventListener implements Listener { | ||||
|         }, TaskTime.ticks(3L)); | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onRedstoneEvent(BlockRedstoneEvent event) { | ||||
|         Block block = event.getBlock(); | ||||
|         Location location = BukkitUtil.adapt(block.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot == null) { | ||||
|             if (area.isRoadFlags() && !area.getRoadFlag(RedstoneFlag.class)) { | ||||
|                 event.setNewCurrent(0); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (!plot.getFlag(RedstoneFlag.class)) { | ||||
|             event.setNewCurrent(0); | ||||
|             plot.debug("Redstone event was cancelled because redstone = false"); | ||||
|             return; | ||||
|         } | ||||
|         if (Settings.Redstone.DISABLE_OFFLINE) { | ||||
|             boolean disable = false; | ||||
|             if (!DBFunc.SERVER.equals(plot.getOwner())) { | ||||
|                 if (plot.isMerged()) { | ||||
|                     disable = true; | ||||
|                     for (UUID owner : plot.getOwners()) { | ||||
|                         if (PlotSquared.platform().playerManager().getPlayerIfExists(owner) != null) { | ||||
|                             disable = false; | ||||
|                             break; | ||||
|                         } | ||||
|                     } | ||||
|                 } else { | ||||
|                     disable = PlotSquared.platform().playerManager().getPlayerIfExists(plot.getOwnerAbs()) == null; | ||||
|                 } | ||||
|             } | ||||
|             if (disable) { | ||||
|                 for (UUID trusted : plot.getTrusted()) { | ||||
|                     if (PlotSquared.platform().playerManager().getPlayerIfExists(trusted) != null) { | ||||
|                         disable = false; | ||||
|                         break; | ||||
|                     } | ||||
|                 } | ||||
|                 if (disable) { | ||||
|                     event.setNewCurrent(0); | ||||
|                     plot.debug("Redstone event was cancelled because no trusted player was in the plot"); | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         if (Settings.Redstone.DISABLE_UNOCCUPIED) { | ||||
|             for (final PlotPlayer<?> player : PlotSquared.platform().playerManager().getPlayers()) { | ||||
|                 if (plot.equals(player.getCurrentPlot())) { | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|             event.setNewCurrent(0); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(ignoreCancelled = true, priority = EventPriority.HIGHEST) | ||||
|     public void onPhysicsEvent(BlockPhysicsEvent event) { | ||||
|         Block block = event.getBlock(); | ||||
|         Location location = BukkitUtil.adapt(block.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = area.getOwnedPlotAbs(location); | ||||
|         if (plot == null) { | ||||
|             return; | ||||
|         } | ||||
|         if (event.getChangedType().hasGravity() && plot.getFlag(DisablePhysicsFlag.class)) { | ||||
|             event.setCancelled(true); | ||||
|             sendBlockChange(event.getBlock().getLocation(), event.getBlock().getBlockData()); | ||||
|             plot.debug("Prevented block physics and resent block change because disable-physics = true"); | ||||
|             return; | ||||
|         } | ||||
|         switch (event.getChangedType()) { | ||||
|             case COMPARATOR: { | ||||
|                 if (!plot.getFlag(RedstoneFlag.class)) { | ||||
|                     event.setCancelled(true); | ||||
|                     plot.debug("Prevented comparator update because redstone = false"); | ||||
|                 } | ||||
|                 return; | ||||
|             } | ||||
|             case ANVIL: | ||||
|             case DRAGON_EGG: | ||||
|             case GRAVEL: | ||||
|             case SAND: | ||||
|             case TURTLE_EGG: | ||||
|             case TURTLE_HELMET: | ||||
|             case TURTLE_SPAWN_EGG: { | ||||
|                 if (plot.getFlag(DisablePhysicsFlag.class)) { | ||||
|                     event.setCancelled(true); | ||||
|                     plot.debug("Prevented block physics because disable-physics = true"); | ||||
|                 } | ||||
|                 return; | ||||
|             } | ||||
|             default: | ||||
|                 if (Settings.Redstone.DETECT_INVALID_EDGE_PISTONS) { | ||||
|                     switch (block.getType()) { | ||||
|                         case PISTON, STICKY_PISTON -> { | ||||
|                             org.bukkit.block.data.Directional piston = (org.bukkit.block.data.Directional) block.getBlockData(); | ||||
|                             switch (piston.getFacing()) { | ||||
|                                 case EAST -> location = location.add(1, 0, 0); | ||||
|                                 case SOUTH -> location = location.add(-1, 0, 0); | ||||
|                                 case WEST -> location = location.add(0, 0, 1); | ||||
|                                 case NORTH -> location = location.add(0, 0, -1); | ||||
|                             } | ||||
|                             Plot newPlot = area.getOwnedPlotAbs(location); | ||||
|                             if (!plot.equals(newPlot)) { | ||||
|                                 event.setCancelled(true); | ||||
|                                 plot.debug("Prevented piston update because of invalid edge piston detection"); | ||||
|                                 return; | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void blockCreate(BlockPlaceEvent event) { | ||||
|         Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||
| @@ -279,21 +152,18 @@ public class BlockEventListener implements Listener { | ||||
|         BukkitPlayer pp = BukkitUtil.adapt(player); | ||||
|         Plot plot = area.getPlot(location); | ||||
|         if (plot != null) { | ||||
|             if ((location.getY() >= area.getMaxBuildHeight() || location.getY() < area | ||||
|                     .getMinBuildHeight()) && !Permissions | ||||
|                     .hasPermission(pp, Permission.PERMISSION_ADMIN_BUILD_HEIGHT_LIMIT)) { | ||||
|             if (area.notifyIfOutsideBuildArea(pp, location.getY())) { | ||||
|                 event.setCancelled(true); | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("height.height_limit"), | ||||
|                         Template.of("minHeight", String.valueOf(area.getMinBuildHeight())), | ||||
|                         Template.of("maxHeight", String.valueOf(area.getMaxBuildHeight())) | ||||
|                 ); | ||||
|                 return; | ||||
|             } | ||||
|             if (!plot.hasOwner()) { | ||||
|                 if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_BUILD_UNOWNED)) { | ||||
|                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_BUILD_UNOWNED)) { | ||||
|                     pp.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_BUILD_UNOWNED)) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Permission.PERMISSION_ADMIN_BUILD_UNOWNED) | ||||
|                             ) | ||||
|                     ); | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
| @@ -307,10 +177,13 @@ public class BlockEventListener implements Listener { | ||||
|                         return; | ||||
|                     } | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_BUILD_OTHER)) { | ||||
|                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_BUILD_OTHER)) { | ||||
|                     pp.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_BUILD_OTHER)) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Permission.PERMISSION_ADMIN_BUILD_OTHER) | ||||
|                             ) | ||||
|                     ); | ||||
|                     event.setCancelled(true); | ||||
|                     plot.debug(player.getName() + " could not place " + event.getBlock().getType() | ||||
| @@ -318,7 +191,7 @@ public class BlockEventListener implements Listener { | ||||
|                     return; | ||||
|                 } | ||||
|             } else if (Settings.Done.RESTRICT_BUILDING && DoneFlag.isDone(plot)) { | ||||
|                 if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_BUILD_OTHER)) { | ||||
|                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_BUILD_OTHER)) { | ||||
|                     pp.sendMessage( | ||||
|                             TranslatableCaption.of("done.building_restricted") | ||||
|                     ); | ||||
| @@ -334,10 +207,13 @@ public class BlockEventListener implements Listener { | ||||
|                             + " did not fall because of disable-physics = true"); | ||||
|                 } | ||||
|             } | ||||
|         } else if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_BUILD_ROAD)) { | ||||
|         } else if (!pp.hasPermission(Permission.PERMISSION_ADMIN_BUILD_ROAD)) { | ||||
|             pp.sendMessage( | ||||
|                     TranslatableCaption.of("permission.no_permission_event"), | ||||
|                     Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_BUILD_ROAD)) | ||||
|                     TagResolver.resolver( | ||||
|                             "node", | ||||
|                             Tag.inserting(Permission.PERMISSION_ADMIN_BUILD_ROAD) | ||||
|                     ) | ||||
|             ); | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
| @@ -356,28 +232,23 @@ public class BlockEventListener implements Listener { | ||||
|             BukkitPlayer plotPlayer = BukkitUtil.adapt(player); | ||||
|             // == rather than <= as we only care about the "ground level" not being destroyed | ||||
|             if (event.getBlock().getY() == area.getMinGenHeight()) { | ||||
|                 if (!Permissions | ||||
|                         .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_DESTROY_GROUNDLEVEL)) { | ||||
|                 if (!plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_DESTROY_GROUNDLEVEL)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_DESTROY_GROUNDLEVEL)) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Permission.PERMISSION_ADMIN_DESTROY_GROUNDLEVEL) | ||||
|                             ) | ||||
|                     ); | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|             } else if ((location.getY() >= area.getMaxBuildHeight() || location.getY() < area | ||||
|                     .getMinBuildHeight()) && !Permissions | ||||
|                     .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_BUILD_HEIGHT_LIMIT)) { | ||||
|             } else if (area.notifyIfOutsideBuildArea(plotPlayer, location.getY())) { | ||||
|                 event.setCancelled(true); | ||||
|                 plotPlayer.sendMessage( | ||||
|                         TranslatableCaption.of("height.height_limit"), | ||||
|                         Template.of("minHeight", String.valueOf(area.getMinBuildHeight())), | ||||
|                         Template.of("maxHeight", String.valueOf(area.getMaxBuildHeight())) | ||||
|                 ); | ||||
|                 return; | ||||
|             } | ||||
|             if (!plot.hasOwner()) { | ||||
|                 if (!Permissions | ||||
|                         .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_DESTROY_UNOWNED, true)) { | ||||
|                 if (!plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_DESTROY_UNOWNED, true)) { | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
|                 return; | ||||
| @@ -391,17 +262,19 @@ public class BlockEventListener implements Listener { | ||||
|                         return; | ||||
|                     } | ||||
|                 } | ||||
|                 if (Permissions | ||||
|                         .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_DESTROY_OTHER)) { | ||||
|                 if (plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_DESTROY_OTHER)) { | ||||
|                     return; | ||||
|                 } | ||||
|                 plotPlayer.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_DESTROY_OTHER)) | ||||
|                         TagResolver.resolver( | ||||
|                                 "node", | ||||
|                                 Tag.inserting(Permission.PERMISSION_ADMIN_DESTROY_OTHER) | ||||
|                         ) | ||||
|                 ); | ||||
|                 event.setCancelled(true); | ||||
|             } else if (Settings.Done.RESTRICT_BUILDING && DoneFlag.isDone(plot)) { | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_BUILD_OTHER)) { | ||||
|                 if (!plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_BUILD_OTHER)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("done.building_restricted") | ||||
|                     ); | ||||
| @@ -412,7 +285,7 @@ public class BlockEventListener implements Listener { | ||||
|             return; | ||||
|         } | ||||
|         BukkitPlayer pp = BukkitUtil.adapt(player); | ||||
|         if (Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_DESTROY_ROAD)) { | ||||
|         if (pp.hasPermission(Permission.PERMISSION_ADMIN_DESTROY_ROAD)) { | ||||
|             return; | ||||
|         } | ||||
|         if (this.worldEdit != null && pp.getAttribute("worldedit")) { | ||||
| @@ -423,7 +296,10 @@ public class BlockEventListener implements Listener { | ||||
|         } | ||||
|         pp.sendMessage( | ||||
|                 TranslatableCaption.of("permission.no_permission_event"), | ||||
|                 Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_DESTROY_ROAD)) | ||||
|                 TagResolver.resolver( | ||||
|                         "node", | ||||
|                         Tag.inserting(Permission.PERMISSION_ADMIN_DESTROY_ROAD) | ||||
|                 ) | ||||
|         ); | ||||
|         event.setCancelled(true); | ||||
|     } | ||||
| @@ -461,6 +337,7 @@ public class BlockEventListener implements Listener { | ||||
|             case "TWISTING_VINES": | ||||
|             case "CAVE_VINES": | ||||
|             case "VINE": | ||||
|             case "GLOW_BERRIES": | ||||
|                 if (!plot.getFlag(VineGrowFlag.class)) { | ||||
|                     plot.debug("Vine could not grow because vine-grow = false"); | ||||
|                     event.setCancelled(true); | ||||
| @@ -496,18 +373,18 @@ public class BlockEventListener implements Listener { | ||||
|                     BukkitPlayer plotPlayer = BukkitUtil.adapt(player); | ||||
|                     if (plot != null) { | ||||
|                         if (!plot.hasOwner()) { | ||||
|                             if (Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_INTERACT_UNOWNED)) { | ||||
|                             if (plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_INTERACT_UNOWNED)) { | ||||
|                                 return; | ||||
|                             } | ||||
|                         } else if (!plot.isAdded(plotPlayer.getUUID())) { | ||||
|                             if (Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_INTERACT_OTHER)) { | ||||
|                             if (plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_INTERACT_OTHER)) { | ||||
|                                 return; | ||||
|                             } | ||||
|                         } else { | ||||
|                             return; | ||||
|                         } | ||||
|                     } else { | ||||
|                         if (Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_INTERACT_ROAD)) { | ||||
|                         if (plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_INTERACT_ROAD)) { | ||||
|                             return; | ||||
|                         } | ||||
|                         if (this.worldEdit != null && plotPlayer.getAttribute("worldedit")) { | ||||
| @@ -551,22 +428,29 @@ public class BlockEventListener implements Listener { | ||||
|         if (plot == null) { | ||||
|             return; | ||||
|         } | ||||
|         switch (event.getNewState().getType()) { | ||||
|             case SNOW: | ||||
|             case SNOW_BLOCK: | ||||
|         if (!area.buildRangeContainsY(location.getY())) { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
|         if (org.bukkit.Tag.SNOW.isTagged(event.getNewState().getType())) { | ||||
|             if (!plot.getFlag(SnowFormFlag.class)) { | ||||
|                 plot.debug("Snow could not form because snow-form = false"); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|             return; | ||||
|             case ICE: | ||||
|             case FROSTED_ICE: | ||||
|             case PACKED_ICE: | ||||
|         } | ||||
|         if (org.bukkit.Tag.ICE.isTagged(event.getNewState().getType())) { | ||||
|             if (!plot.getFlag(IceFormFlag.class)) { | ||||
|                 plot.debug("Ice could not form because ice-form = false"); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } | ||||
|         if (event.getNewState().getType().toString().endsWith("CONCRETE")) { | ||||
|             if (!plot.getFlag(ConcreteHardenFlag.class)) { | ||||
|                 plot.debug("Concrete powder could not harden because concrete-harden = false"); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
| @@ -586,18 +470,12 @@ public class BlockEventListener implements Listener { | ||||
|             return; | ||||
|         } | ||||
|         Class<? extends BooleanFlag<?>> flag; | ||||
|         switch (event.getNewState().getType()) { | ||||
|             case SNOW: | ||||
|             case SNOW_BLOCK: | ||||
|         if (org.bukkit.Tag.SNOW.isTagged(event.getNewState().getType())) { | ||||
|             flag = SnowFormFlag.class; | ||||
|                 break; | ||||
|             case ICE: | ||||
|             case FROSTED_ICE: | ||||
|             case PACKED_ICE: | ||||
|         } else if (org.bukkit.Tag.ICE.isTagged(event.getNewState().getType())) { | ||||
|             flag = IceFormFlag.class; | ||||
|                 break; | ||||
|             default: | ||||
|                 return; // other blocks are ignored by this event | ||||
|         } else { | ||||
|             return; | ||||
|         } | ||||
|         boolean allowed = plot.getFlag(flag); | ||||
|         Entity entity = event.getEntity(); | ||||
| @@ -647,7 +525,11 @@ public class BlockEventListener implements Listener { | ||||
|                 BlockBreakEvent call = new BlockBreakEvent(block, player); | ||||
|                 Bukkit.getServer().getPluginManager().callEvent(call); | ||||
|                 if (!call.isCancelled()) { | ||||
|                     event.getBlock().breakNaturally(); | ||||
|                     if (Settings.Flags.INSTABREAK_CONSIDER_TOOL) { | ||||
|                         block.breakNaturally(event.getItemInHand()); | ||||
|                     } else { | ||||
|                         block.breakNaturally(); | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             // == rather than <= as we only care about the "ground level" not being destroyed | ||||
| @@ -657,8 +539,7 @@ public class BlockEventListener implements Listener { | ||||
|             } | ||||
|             if (!plot.hasOwner()) { | ||||
|                 BukkitPlayer plotPlayer = BukkitUtil.adapt(player); | ||||
|                 if (Permissions | ||||
|                         .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_DESTROY_UNOWNED)) { | ||||
|                 if (plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_DESTROY_UNOWNED)) { | ||||
|                     return; | ||||
|                 } | ||||
|                 event.setCancelled(true); | ||||
| @@ -670,8 +551,7 @@ public class BlockEventListener implements Listener { | ||||
|                 Block block = event.getBlock(); | ||||
|                 if (destroy | ||||
|                         .contains(BlockTypeWrapper.get(BukkitAdapter.asBlockType(block.getType()))) | ||||
|                         || Permissions | ||||
|                         .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_DESTROY_OTHER)) { | ||||
|                         || plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_DESTROY_OTHER)) { | ||||
|                     return; | ||||
|                 } | ||||
|                 plot.debug(player.getName() + " could not break " + block.getType() | ||||
| @@ -682,7 +562,7 @@ public class BlockEventListener implements Listener { | ||||
|             return; | ||||
|         } | ||||
|         BukkitPlayer plotPlayer = BukkitUtil.adapt(player); | ||||
|         if (Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_DESTROY_ROAD)) { | ||||
|         if (plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_DESTROY_ROAD)) { | ||||
|             return; | ||||
|         } | ||||
|         event.setCancelled(true); | ||||
| @@ -701,111 +581,141 @@ public class BlockEventListener implements Listener { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
|         switch (block.getType()) { | ||||
|             case ICE: | ||||
|         Material blockType = block.getType(); | ||||
|         if (org.bukkit.Tag.ICE.isTagged(blockType)) { | ||||
|             if (!plot.getFlag(IceMeltFlag.class)) { | ||||
|                 plot.debug("Ice could not melt because ice-melt = false"); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|                 break; | ||||
|             case SNOW: | ||||
|             return; | ||||
|         } | ||||
|         if (org.bukkit.Tag.SNOW.isTagged(blockType)) { | ||||
|             if (!plot.getFlag(SnowMeltFlag.class)) { | ||||
|                 plot.debug("Snow could not melt because snow-melt = false"); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|                 break; | ||||
|             case FARMLAND: | ||||
|             return; | ||||
|         } | ||||
|         if (blockType == Material.FARMLAND) { | ||||
|             if (!plot.getFlag(SoilDryFlag.class)) { | ||||
|                 plot.debug("Soil could not dry because soil-dry = false"); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|                 break; | ||||
|             case TUBE_CORAL_BLOCK: | ||||
|             case BRAIN_CORAL_BLOCK: | ||||
|             case BUBBLE_CORAL_BLOCK: | ||||
|             case FIRE_CORAL_BLOCK: | ||||
|             case HORN_CORAL_BLOCK: | ||||
|             case TUBE_CORAL: | ||||
|             case BRAIN_CORAL: | ||||
|             case BUBBLE_CORAL: | ||||
|             case FIRE_CORAL: | ||||
|             case HORN_CORAL: | ||||
|             case TUBE_CORAL_FAN: | ||||
|             case BRAIN_CORAL_FAN: | ||||
|             case BUBBLE_CORAL_FAN: | ||||
|             case FIRE_CORAL_FAN: | ||||
|             case HORN_CORAL_FAN: | ||||
|             case BRAIN_CORAL_WALL_FAN: | ||||
|             case BUBBLE_CORAL_WALL_FAN: | ||||
|             case FIRE_CORAL_WALL_FAN: | ||||
|             case HORN_CORAL_WALL_FAN: | ||||
|             case TUBE_CORAL_WALL_FAN: | ||||
|             return; | ||||
|         } | ||||
|         if (CORAL_BLOCKS.isTagged(blockType) || CORALS.isTagged(blockType) || WALL_CORALS.isTagged(blockType)) { | ||||
|             if (!plot.getFlag(CoralDryFlag.class)) { | ||||
|                 plot.debug("Coral could not dry because coral-dry = false"); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onMoistureChange(MoistureChangeEvent event) { | ||||
|         Block block = event.getBlock(); | ||||
|         Location location = BukkitUtil.adapt(block.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|  | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         Plot plot = area.getOwnedPlot(location); | ||||
|  | ||||
|         if (plot == null) { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (block.getBlockData() instanceof Farmland farmland && event | ||||
|                 .getNewState() | ||||
|                 .getBlockData() instanceof Farmland newFarmland) { | ||||
|             int currentMoisture = farmland.getMoisture(); | ||||
|             int newMoisture = newFarmland.getMoisture(); | ||||
|  | ||||
|             // farmland gets moisturizes | ||||
|             if (newMoisture > currentMoisture) { | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             if (plot.getFlag(SoilDryFlag.class)) { | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             plot.debug("Soil could not dry because soil-dry = false"); | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onChange(BlockFromToEvent event) { | ||||
|         Block from = event.getBlock(); | ||||
|         Block fromBlock = event.getBlock(); | ||||
|  | ||||
|         // Check liquid flow flag inside of origin plot too | ||||
|         final Location fLocation = BukkitUtil.adapt(from.getLocation()); | ||||
|         final PlotArea fromArea = fLocation.getPlotArea(); | ||||
|         final Location fromLocation = BukkitUtil.adapt(fromBlock.getLocation()); | ||||
|         final PlotArea fromArea = fromLocation.getPlotArea(); | ||||
|         if (fromArea != null) { | ||||
|             final Plot plot = fromArea.getOwnedPlot(fLocation); | ||||
|             if (plot != null && plot.getFlag(LiquidFlowFlag.class) == LiquidFlowFlag.FlowStatus.DISABLED && event | ||||
|             final Plot fromPlot = fromArea.getOwnedPlot(fromLocation); | ||||
|             if (fromPlot != null && fromPlot.getFlag(LiquidFlowFlag.class) == LiquidFlowFlag.FlowStatus.DISABLED && event | ||||
|                     .getBlock() | ||||
|                     .isLiquid()) { | ||||
|                 plot.debug("Liquid could not flow because liquid-flow = disabled"); | ||||
|                 fromPlot.debug("Liquid could not flow because liquid-flow = disabled"); | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         Block to = event.getToBlock(); | ||||
|         Location tLocation = BukkitUtil.adapt(to.getLocation()); | ||||
|         PlotArea area = tLocation.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             if (from.getType() == Material.DRAGON_EGG && fromArea != null) { | ||||
|         Block toBlock = event.getToBlock(); | ||||
|         Location toLocation = BukkitUtil.adapt(toBlock.getLocation()); | ||||
|         PlotArea toArea = toLocation.getPlotArea(); | ||||
|         if (toArea == null) { | ||||
|             if (fromBlock.getType() == Material.DRAGON_EGG && fromArea != null) { | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = area.getOwnedPlot(tLocation); | ||||
|         if (!toArea.buildRangeContainsY(toLocation.getY())) { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
|         Plot toPlot = toArea.getOwnedPlot(toLocation); | ||||
|  | ||||
|         if (from.getType() == Material.DRAGON_EGG && fromArea != null) { | ||||
|             final Plot fromPlot = fromArea.getOwnedPlot(fLocation); | ||||
|         if (fromBlock.getType() == Material.DRAGON_EGG && fromArea != null) { | ||||
|             final Plot fromPlot = fromArea.getOwnedPlot(fromLocation); | ||||
|  | ||||
|             if (fromPlot != null || plot != null) { | ||||
|                 if ((fromPlot == null || !fromPlot.equals(plot)) && (plot == null || !plot.equals(fromPlot))) { | ||||
|             if (fromPlot != null || toPlot != null) { | ||||
|                 if ((fromPlot == null || !fromPlot.equals(toPlot)) && (toPlot == null || !toPlot.equals(fromPlot))) { | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (plot != null) { | ||||
|             if (!area.contains(fLocation.getX(), fLocation.getZ()) || !Objects.equals(plot, area.getOwnedPlot(fLocation))) { | ||||
|         if (toPlot != null) { | ||||
|             if (!toArea.contains(fromLocation.getX(), fromLocation.getZ()) || !Objects.equals( | ||||
|                     toPlot, | ||||
|                     toArea.getOwnedPlot(fromLocation) | ||||
|             )) { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|             if (plot.getFlag(LiquidFlowFlag.class) == LiquidFlowFlag.FlowStatus.ENABLED && event.getBlock().isLiquid()) { | ||||
|             if (toPlot.getFlag(LiquidFlowFlag.class) == LiquidFlowFlag.FlowStatus.ENABLED && event.getBlock().isLiquid()) { | ||||
|                 return; | ||||
|             } | ||||
|             if (plot.getFlag(DisablePhysicsFlag.class)) { | ||||
|                 plot.debug(event.getBlock().getType() + " could not update because disable-physics = true"); | ||||
|             if (toPlot.getFlag(DisablePhysicsFlag.class)) { | ||||
|                 toPlot.debug(event.getBlock().getType() + " could not update because disable-physics = true"); | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|             if (plot.getFlag(LiquidFlowFlag.class) == LiquidFlowFlag.FlowStatus.DISABLED && event.getBlock().isLiquid()) { | ||||
|                 plot.debug("Liquid could not flow because liquid-flow = disabled"); | ||||
|             if (toPlot.getFlag(LiquidFlowFlag.class) == LiquidFlowFlag.FlowStatus.DISABLED && event.getBlock().isLiquid()) { | ||||
|                 toPlot.debug("Liquid could not flow because liquid-flow = disabled"); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } else if (!area.contains(fLocation.getX(), fLocation.getZ()) || !Objects.equals(null, area.getOwnedPlot(fLocation))) { | ||||
|         } else if (!toArea.contains(fromLocation.getX(), fromLocation.getZ()) || !Objects.equals( | ||||
|                 null, | ||||
|                 toArea.getOwnedPlot(fromLocation) | ||||
|         )) { | ||||
|             event.setCancelled(true); | ||||
|         } else if (event.getBlock().isLiquid()) { | ||||
|             final org.bukkit.Location location = event.getBlock().getLocation(); | ||||
| @@ -845,6 +755,11 @@ public class BlockEventListener implements Listener { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (!area.buildRangeContainsY(location.getY())) { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot == null || !plot.getFlag(CropGrowFlag.class)) { | ||||
|             if (plot != null) { | ||||
| @@ -888,15 +803,16 @@ public class BlockEventListener implements Listener { | ||||
|         } | ||||
|         for (Block block1 : event.getBlocks()) { | ||||
|             Location bloc = BukkitUtil.adapt(block1.getLocation()); | ||||
|             if (!area.contains(bloc.getX(), bloc.getZ()) || !area.contains( | ||||
|                     bloc.getX() + relative.getBlockX(), | ||||
|                     bloc.getZ() + relative.getBlockZ() | ||||
|             )) { | ||||
|             Location newLoc = bloc.add(relative.getBlockX(), relative.getBlockY(), relative.getBlockZ()); | ||||
|             if (!area.contains(bloc.getX(), bloc.getZ()) || !area.contains(newLoc)) { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|             if (!plot.equals(area.getOwnedPlot(bloc)) || !plot | ||||
|                     .equals(area.getOwnedPlot(bloc.add(relative.getBlockX(), relative.getBlockY(), relative.getBlockZ())))) { | ||||
|             if (!plot.equals(area.getOwnedPlot(bloc)) || !plot.equals(area.getOwnedPlot(newLoc))) { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|             if (!area.buildRangeContainsY(bloc.getY()) || !area.buildRangeContainsY(newLoc.getY())) { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
| @@ -922,9 +838,8 @@ public class BlockEventListener implements Listener { | ||||
|             } | ||||
|             for (Block block1 : event.getBlocks()) { | ||||
|                 Location bloc = BukkitUtil.adapt(block1.getLocation()); | ||||
|                 if (bloc.isPlotArea() || bloc | ||||
|                         .add(relative.getBlockX(), relative.getBlockY(), relative.getBlockZ()) | ||||
|                         .isPlotArea()) { | ||||
|                 Location newLoc = bloc.add(relative.getBlockX(), relative.getBlockY(), relative.getBlockZ()); | ||||
|                 if (bloc.isPlotArea() || newLoc.isPlotArea()) { | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
| @@ -938,15 +853,16 @@ public class BlockEventListener implements Listener { | ||||
|         } | ||||
|         for (Block block1 : event.getBlocks()) { | ||||
|             Location bloc = BukkitUtil.adapt(block1.getLocation()); | ||||
|             if (!area.contains(bloc.getX(), bloc.getZ()) || !area.contains( | ||||
|                     bloc.getX() + relative.getBlockX(), | ||||
|                     bloc.getZ() + relative.getBlockZ() | ||||
|             )) { | ||||
|             Location newLoc = bloc.add(relative.getBlockX(), relative.getBlockY(), relative.getBlockZ()); | ||||
|             if (!area.contains(bloc.getX(), bloc.getZ()) || !area.contains(newLoc)) { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|             if (!plot.equals(area.getOwnedPlot(bloc)) || !plot | ||||
|                     .equals(area.getOwnedPlot(bloc.add(relative.getBlockX(), relative.getBlockY(), relative.getBlockZ())))) { | ||||
|             if (!plot.equals(area.getOwnedPlot(bloc)) || !plot.equals(area.getOwnedPlot(newLoc))) { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|             if (!area.buildRangeContainsY(bloc.getY()) || !area.buildRangeContainsY(newLoc.getY())) { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
| @@ -955,16 +871,29 @@ public class BlockEventListener implements Listener { | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onBlockDispense(BlockDispenseEvent event) { | ||||
|         if (!this.plotAreaManager.hasPlotArea(event.getBlock().getWorld().getName())) { | ||||
|             return; | ||||
|         } | ||||
|         Material type = event.getItem().getType(); | ||||
|         switch (type.toString()) { | ||||
|             case "SHULKER_BOX", "WHITE_SHULKER_BOX", "ORANGE_SHULKER_BOX", "MAGENTA_SHULKER_BOX", "LIGHT_BLUE_SHULKER_BOX", "YELLOW_SHULKER_BOX", "LIME_SHULKER_BOX", "PINK_SHULKER_BOX", "GRAY_SHULKER_BOX", "LIGHT_GRAY_SHULKER_BOX", "CYAN_SHULKER_BOX", "PURPLE_SHULKER_BOX", "BLUE_SHULKER_BOX", "BROWN_SHULKER_BOX", "GREEN_SHULKER_BOX", "RED_SHULKER_BOX", "BLACK_SHULKER_BOX", "CARVED_PUMPKIN", "WITHER_SKELETON_SKULL", "FLINT_AND_STEEL", "BONE_MEAL", "SHEARS", "GLASS_BOTTLE", "GLOWSTONE", "COD_BUCKET", "PUFFERFISH_BUCKET", "SALMON_BUCKET", "TROPICAL_FISH_BUCKET", "AXOLOTL_BUCKET", "BUCKET", "WATER_BUCKET", "LAVA_BUCKET" -> { | ||||
|             case "SHULKER_BOX", "WHITE_SHULKER_BOX", "ORANGE_SHULKER_BOX", "MAGENTA_SHULKER_BOX", "LIGHT_BLUE_SHULKER_BOX", | ||||
|                     "YELLOW_SHULKER_BOX", "LIME_SHULKER_BOX", "PINK_SHULKER_BOX", "GRAY_SHULKER_BOX", "LIGHT_GRAY_SHULKER_BOX", | ||||
|                     "CYAN_SHULKER_BOX", "PURPLE_SHULKER_BOX", "BLUE_SHULKER_BOX", "BROWN_SHULKER_BOX", "GREEN_SHULKER_BOX", | ||||
|                     "RED_SHULKER_BOX", "BLACK_SHULKER_BOX", "CARVED_PUMPKIN", "WITHER_SKELETON_SKULL", "FLINT_AND_STEEL", | ||||
|                     "BONE_MEAL", "SHEARS", "GLASS_BOTTLE", "GLOWSTONE", "COD_BUCKET", "PUFFERFISH_BUCKET", "SALMON_BUCKET", | ||||
|                     "TROPICAL_FISH_BUCKET", "AXOLOTL_BUCKET", "BUCKET", "WATER_BUCKET", "LAVA_BUCKET", "TADPOLE_BUCKET" -> { | ||||
|                 if (event.getBlock().getType() == Material.DROPPER) { | ||||
|                     return; | ||||
|                 } | ||||
|                 BlockFace targetFace = ((Directional) event.getBlock().getState().getData()).getFacing(); | ||||
|                 BlockFace targetFace = ((Dispenser) event.getBlock().getBlockData()).getFacing(); | ||||
|                 Location location = BukkitUtil.adapt(event.getBlock().getRelative(targetFace).getLocation()); | ||||
|                 if (location.isPlotRoad()) { | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 PlotArea area = location.getPlotArea(); | ||||
|                 if (area != null && !area.buildRangeContainsY(location.getY())) { | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| @@ -1004,6 +933,10 @@ public class BlockEventListener implements Listener { | ||||
|                 Plot plot = area.getOwnedPlot(location); | ||||
|                 if (!Objects.equals(plot, origin)) { | ||||
|                     event.getBlocks().remove(i); | ||||
|                     continue; | ||||
|                 } | ||||
|                 if (!area.buildRangeContainsY(location.getY())) { | ||||
|                     event.getBlocks().remove(i); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| @@ -1052,6 +985,7 @@ public class BlockEventListener implements Listener { | ||||
|             if (plot != null) { | ||||
|                 plot.debug("Explosion was cancelled because explosion = false"); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         event.blockList().removeIf(blox -> !plot.equals(area.getOwnedPlot(BukkitUtil.adapt(blox.getLocation())))); | ||||
|     } | ||||
| @@ -1095,27 +1029,44 @@ public class BlockEventListener implements Listener { | ||||
|         Plot plot = area.getOwnedPlot(location1); | ||||
|         if (player != null) { | ||||
|             BukkitPlayer pp = BukkitUtil.adapt(player); | ||||
|             if (area.notifyIfOutsideBuildArea(pp, location1.getY())) { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|             if (plot == null) { | ||||
|                 if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_BUILD_ROAD)) { | ||||
|                 if (!PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, BlockIgnitionFlag.class, true) && !pp.hasPermission( | ||||
|                         Permission.PERMISSION_ADMIN_BUILD_ROAD | ||||
|                 )) { | ||||
|                     pp.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_BUILD_ROAD)) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Permission.PERMISSION_ADMIN_BUILD_ROAD) | ||||
|                             ) | ||||
|                     ); | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
|             } else if (!plot.hasOwner()) { | ||||
|                 if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_BUILD_UNOWNED)) { | ||||
|                 if (!PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, BlockIgnitionFlag.class, true) && !pp.hasPermission( | ||||
|                         Permission.PERMISSION_ADMIN_BUILD_UNOWNED | ||||
|                 )) { | ||||
|                     pp.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_BUILD_UNOWNED)) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Permission.PERMISSION_ADMIN_BUILD_UNOWNED) | ||||
|                             ) | ||||
|                     ); | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
|             } else if (!plot.isAdded(pp.getUUID())) { | ||||
|                 if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_BUILD_OTHER)) { | ||||
|                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_BUILD_OTHER)) { | ||||
|                     pp.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_BUILD_OTHER)) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Permission.PERMISSION_ADMIN_BUILD_OTHER) | ||||
|                             ) | ||||
|                     ); | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
| @@ -1202,7 +1153,10 @@ public class BlockEventListener implements Listener { | ||||
|                     return true; | ||||
|                 } | ||||
|                 Plot plot = area.getOwnedPlot(blockLocation); | ||||
|                 return !Objects.equals(plot, origin); | ||||
|                 if (!Objects.equals(plot, origin)) { | ||||
|                     return true; | ||||
|                 } | ||||
|                 return !area.buildRangeContainsY(location.getY()); | ||||
|             }); | ||||
|         } | ||||
|         if (blocks.isEmpty()) { | ||||
| @@ -1233,30 +1187,25 @@ public class BlockEventListener implements Listener { | ||||
|  | ||||
|         for (final BlockState state : event.getReplacedBlockStates()) { | ||||
|             Location currentLocation = BukkitUtil.adapt(state.getLocation()); | ||||
|             if (!Permissions.hasPermission( | ||||
|                     pp, | ||||
|             if (!pp.hasPermission( | ||||
|                     Permission.PERMISSION_ADMIN_BUILD_ROAD | ||||
|             ) && !(Objects.equals(currentLocation.getPlot(), plot))) { | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_BUILD_ROAD)) | ||||
|                         TagResolver.resolver("node", Tag.inserting(Permission.PERMISSION_ADMIN_BUILD_ROAD)) | ||||
|                 ); | ||||
|                 event.setCancelled(true); | ||||
|                 break; | ||||
|             } | ||||
|             if (Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_BUILD_HEIGHT_LIMIT)) { | ||||
|             if (pp.hasPermission(Permission.PERMISSION_ADMIN_BUILD_HEIGHT_LIMIT)) { | ||||
|                 continue; | ||||
|             } | ||||
|             if (currentLocation.getY() >= area.getMaxBuildHeight() || currentLocation.getY() < area.getMinBuildHeight()) { | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("height.height_limit"), | ||||
|                         Template.of("minHeight", String.valueOf(area.getMinBuildHeight())), | ||||
|                         Template.of("maxHeight", String.valueOf(area.getMaxBuildHeight())) | ||||
|                 ); | ||||
|             if (area.notifyIfOutsideBuildArea(pp, currentLocation.getY())) { | ||||
|                 event.setCancelled(true); | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -33,6 +26,8 @@ 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 com.plotsquared.core.plot.flag.implementations.SculkSensorInteractFlag; | ||||
| import com.plotsquared.core.util.PlotFlagUtil; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.entity.Entity; | ||||
| import org.bukkit.entity.Item; | ||||
| @@ -66,13 +61,27 @@ public class BlockEventListener117 implements Listener { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot == null || !plot.getFlag(MiscInteractFlag.class)) { | ||||
|         BukkitPlayer plotPlayer = null; | ||||
|  | ||||
|         if (entity instanceof Player player) { | ||||
|                 BukkitPlayer plotPlayer = BukkitUtil.adapt(player); | ||||
|             plotPlayer = BukkitUtil.adapt(player); | ||||
|             if (area.notifyIfOutsideBuildArea(plotPlayer, location.getY())) { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot == null && !PlotFlagUtil.isAreaRoadFlagsAndFlagEquals( | ||||
|                 area, | ||||
|                 MiscInteractFlag.class, | ||||
|                 true | ||||
|         ) || plot != null && (!plot.getFlag(MiscInteractFlag.class) || !plot.getFlag(SculkSensorInteractFlag.class))) { | ||||
|             if (plotPlayer != null) { | ||||
|                 if (plot != null) { | ||||
|                     if (!plot.isAdded(plotPlayer.getUUID())) { | ||||
|                         plot.debug(plotPlayer.getName() + " couldn't trigger sculk sensors because misc-interact = false"); | ||||
|                         plot.debug(plotPlayer.getName() + " couldn't trigger sculk sensors because both " + | ||||
|                                 "sculk-sensor-interact and misc-interact = false"); | ||||
|                         event.setCancelled(true); | ||||
|                     } | ||||
|                 } | ||||
| @@ -81,9 +90,17 @@ public class BlockEventListener117 implements Listener { | ||||
|             if (entity instanceof Item item) { | ||||
|                 UUID itemThrower = item.getThrower(); | ||||
|                 if (plot != null) { | ||||
|                     if (itemThrower == null && (itemThrower = item.getOwner()) == null) { | ||||
|                         plot.debug( | ||||
|                                 "A thrown item couldn't trigger sculk sensors because both sculk-sensor-interact and " + | ||||
|                                         "misc-interact = false and the item's owner could not be resolved."); | ||||
|                         event.setCancelled(true); | ||||
|                         return; | ||||
|                     } | ||||
|                     if (!plot.isAdded(itemThrower)) { | ||||
|                         if (!plot.isAdded(itemThrower)) { | ||||
|                             plot.debug("A thrown item couldn't trigger sculk sensors because misc-interact = false"); | ||||
|                             plot.debug("A thrown item couldn't trigger sculk sensors because both sculk-sensor-interact and " + | ||||
|                                     "misc-interact = false"); | ||||
|                             event.setCancelled(true); | ||||
|                         } | ||||
|                     } | ||||
| @@ -96,17 +113,16 @@ public class BlockEventListener117 implements Listener { | ||||
|     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()); | ||||
|         Location location = BukkitUtil.adapt(block.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()) { | ||||
|                 Location blockLocation = BukkitUtil.adapt(blocks.get(i).getLocation()); | ||||
|                 if (blockLocation.isPlotArea()) { | ||||
|                     blocks.remove(i); | ||||
|                 } | ||||
|             } | ||||
|             return; | ||||
|         } else { | ||||
|             Plot origin = area.getOwnedPlot(location); | ||||
|             if (origin == null) { | ||||
| @@ -114,30 +130,22 @@ public class BlockEventListener117 implements Listener { | ||||
|                 return; | ||||
|             } | ||||
|             for (int i = blocks.size() - 1; i >= 0; i--) { | ||||
|                 location = BukkitUtil.adapt(blocks.get(i).getLocation()); | ||||
|                 if (!area.contains(location.getX(), location.getZ())) { | ||||
|                 Location blockLocation = BukkitUtil.adapt(blocks.get(i).getLocation()); | ||||
|                 if (!area.contains(blockLocation.getX(), blockLocation.getZ())) { | ||||
|                     blocks.remove(i); | ||||
|                     continue; | ||||
|                 } | ||||
|                 Plot plot = area.getOwnedPlot(location); | ||||
|                 Plot plot = area.getOwnedPlot(blockLocation); | ||||
|                 if (!Objects.equals(plot, origin)) { | ||||
|                     event.getBlocks().remove(i); | ||||
|                     continue; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         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())) { | ||||
|                 if (!area.buildRangeContainsY(location.getY())) { | ||||
|                     event.getBlocks().remove(i); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onBlockForm(BlockFormEvent event) { | ||||
| @@ -155,23 +163,7 @@ public class BlockEventListener117 implements Listener { | ||||
|         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 (event.getNewState().getType().name().contains("COPPER")) { | ||||
|             if (!plot.getFlag(CopperOxideFlag.class)) { | ||||
|                 plot.debug("Copper could not oxide because copper-oxide = false"); | ||||
|                 event.setCancelled(true); | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -33,6 +26,7 @@ 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; | ||||
| import com.plotsquared.core.util.ReflectionUtils.RefClass; | ||||
| import com.plotsquared.core.util.ReflectionUtils.RefField; | ||||
| import com.plotsquared.core.util.ReflectionUtils.RefMethod; | ||||
| @@ -71,9 +65,11 @@ public class ChunkListener implements Listener { | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|     private final int version; | ||||
|  | ||||
|     private RefMethod methodSetUnsaved; | ||||
|     private RefMethod methodGetHandleChunk; | ||||
|     private RefMethod methodGetHandleWorld; | ||||
|     private RefField mustSave; | ||||
|     private RefField mustNotSave; | ||||
|     private Object objChunkStatusFull = null; | ||||
|     /* | ||||
|     private RefMethod methodGetFullChunk; | ||||
|     private RefMethod methodGetBukkitChunk; | ||||
| @@ -86,7 +82,6 @@ public class ChunkListener implements Listener { | ||||
|     */ | ||||
|     private Chunk lastChunk; | ||||
|     private boolean ignoreUnload = false; | ||||
|     private boolean isTrueForNotSave = true; | ||||
|  | ||||
|     @Inject | ||||
|     public ChunkListener(final @NonNull PlotAreaManager plotAreaManager) { | ||||
| @@ -97,22 +92,27 @@ public class ChunkListener implements Listener { | ||||
|         } | ||||
|         try { | ||||
|             RefClass classCraftWorld = getRefClass("{cb}.CraftWorld"); | ||||
|             this.methodGetHandleWorld = classCraftWorld.getMethod("getHandle"); | ||||
|             RefClass classCraftChunk = getRefClass("{cb}.CraftChunk"); | ||||
|             ReflectionUtils.RefClass classChunkAccess = getRefClass("net.minecraft.world.level.chunk.IChunkAccess"); | ||||
|             this.methodSetUnsaved = classChunkAccess.getMethod("a", boolean.class); | ||||
|             try { | ||||
|                 this.methodGetHandleChunk = classCraftChunk.getMethod("getHandle"); | ||||
|             } catch (NoSuchMethodException ignored) { | ||||
|                 try { | ||||
|                     RefClass classChunkStatus = getRefClass("net.minecraft.world.level.chunk.ChunkStatus"); | ||||
|                     this.objChunkStatusFull = classChunkStatus.getRealClass().getField("n").get(null); | ||||
|                     this.methodGetHandleChunk = classCraftChunk.getMethod("getHandle", classChunkStatus.getRealClass()); | ||||
|                 } catch (NoSuchMethodException ex) { | ||||
|                     throw new RuntimeException(ex); | ||||
|                 } | ||||
|             } | ||||
|             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"); | ||||
|                     } | ||||
|                     this.mustNotSave = classChunk.getField("mustNotSave"); | ||||
|                 } else { | ||||
|                     RefClass classChunk = getRefClass("net.minecraft.world.level.chunk.Chunk"); | ||||
|                     this.mustSave = classChunk.getField("mustNotSave"); | ||||
|  | ||||
|                     this.mustNotSave = classChunk.getField("mustNotSave"); | ||||
|                 } | ||||
|             } catch (NoSuchFieldException e) { | ||||
|                 e.printStackTrace(); | ||||
| @@ -174,10 +174,13 @@ public class ChunkListener implements Listener { | ||||
|         if (safe && shouldSave(world, chunk.getX(), chunk.getZ())) { | ||||
|             return false; | ||||
|         } | ||||
|         Object c = this.methodGetHandleChunk.of(chunk).call(); | ||||
|         RefField.RefExecutor field = this.mustSave.of(c); | ||||
|         if ((Boolean) field.get() != isTrueForNotSave) { | ||||
|             field.set(isTrueForNotSave); | ||||
|         Object c = objChunkStatusFull != null | ||||
|                 ? this.methodGetHandleChunk.of(chunk).call(objChunkStatusFull) | ||||
|                 : this.methodGetHandleChunk.of(chunk).call(); | ||||
|         RefField.RefExecutor field = this.mustNotSave.of(c); | ||||
|         methodSetUnsaved.of(c).call(false); | ||||
|         if (!((Boolean) field.get())) { | ||||
|             field.set(true); | ||||
|             if (chunk.isLoaded()) { | ||||
|                 ignoreUnload = true; | ||||
|                 chunk.unload(false); | ||||
| @@ -241,7 +244,8 @@ public class ChunkListener implements Listener { | ||||
|         Chunk chunk = event.getChunk(); | ||||
|         if (Settings.Chunk_Processor.AUTO_TRIM) { | ||||
|             String world = chunk.getWorld().getName(); | ||||
|             if ((!Settings.Enabled_Components.WORLDS || !SinglePlotArea.isSinglePlotWorld(world)) && this.plotAreaManager.hasPlotArea(world)) { | ||||
|             if ((!Settings.Enabled_Components.WORLDS || !SinglePlotArea.isSinglePlotWorld(world)) && this.plotAreaManager.hasPlotArea( | ||||
|                     world)) { | ||||
|                 if (unloadChunk(world, chunk, true)) { | ||||
|                     return; | ||||
|                 } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -26,6 +19,7 @@ | ||||
| package com.plotsquared.bukkit.listener; | ||||
|  | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.bukkit.BukkitPlatform; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | ||||
| import com.plotsquared.bukkit.util.BukkitEntityUtil; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| @@ -42,11 +36,15 @@ import com.plotsquared.core.plot.flag.implementations.DisablePhysicsFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.EntityChangeBlockFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.ExplosionFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.InvincibleFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.ProjectileChangeBlockFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.WeavingDeathPlace; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.PlotFlagUtil; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.util.Enums; | ||||
| import com.sk89q.worldedit.world.block.BlockType; | ||||
| import io.papermc.lib.PaperLib; | ||||
| import org.bukkit.Material; | ||||
| import org.bukkit.Particle; | ||||
| import org.bukkit.World; | ||||
| @@ -60,6 +58,8 @@ import org.bukkit.entity.Player; | ||||
| import org.bukkit.entity.Projectile; | ||||
| import org.bukkit.entity.TNTPrimed; | ||||
| import org.bukkit.entity.Vehicle; | ||||
| import org.bukkit.entity.minecart.ExplosiveMinecart; | ||||
| import org.bukkit.event.Cancellable; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.EventPriority; | ||||
| import org.bukkit.event.Listener; | ||||
| @@ -80,56 +80,54 @@ import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Iterator; | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
|  | ||||
| @SuppressWarnings("unused") | ||||
| public class EntityEventListener implements Listener { | ||||
|  | ||||
|     private static final Particle EXPLOSION_HUGE = Objects.requireNonNull(Enums.findByValue( | ||||
|             Particle.class, | ||||
|             "EXPLOSION_EMITTER", | ||||
|             "EXPLOSION_HUGE" | ||||
|     )); | ||||
|  | ||||
|     private final BukkitPlatform platform; | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|     private final EventDispatcher eventDispatcher; | ||||
|     private float lastRadius; | ||||
|  | ||||
|     @Inject | ||||
|     public EntityEventListener( | ||||
|             final @NonNull BukkitPlatform platform, | ||||
|             final @NonNull PlotAreaManager plotAreaManager, | ||||
|             final @NonNull EventDispatcher eventDispatcher | ||||
|     ) { | ||||
|         this.platform = platform; | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|         this.eventDispatcher = eventDispatcher; | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST) | ||||
|     public void onEntityCombustByEntity(EntityCombustByEntityEvent event) { | ||||
|         EntityDamageByEntityEvent eventChange = | ||||
|                 new EntityDamageByEntityEvent( | ||||
|                         event.getCombuster(), | ||||
|                         event.getEntity(), | ||||
|                         EntityDamageEvent.DamageCause.FIRE_TICK, | ||||
|                         event.getDuration() | ||||
|                 ); | ||||
|         onEntityDamageByEntityEvent(eventChange); | ||||
|         if (eventChange.isCancelled()) { | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|         onEntityDamageByEntityCommon(event.getCombuster(), event.getEntity(), EntityDamageEvent.DamageCause.FIRE_TICK, event); | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST) | ||||
|     public void onEntityDamageByEntityEvent(EntityDamageByEntityEvent event) { | ||||
|         Entity damager = event.getDamager(); | ||||
|         onEntityDamageByEntityCommon(event.getDamager(), event.getEntity(), event.getCause(), event); | ||||
|     } | ||||
|  | ||||
|     private void onEntityDamageByEntityCommon( | ||||
|             final Entity damager, | ||||
|             final Entity victim, | ||||
|             final EntityDamageEvent.DamageCause cause, | ||||
|             final Cancellable event | ||||
|     ) { | ||||
|         Location location = BukkitUtil.adapt(damager.getLocation()); | ||||
|         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||
|             return; | ||||
|         } | ||||
|         Entity victim = event.getEntity(); | ||||
| /* | ||||
|         if (victim.getType().equals(EntityType.ITEM_FRAME)) { | ||||
|             Plot plot = BukkitUtil.getLocation(victim).getPlot(); | ||||
|             if (plot != null && !plot.isAdded(damager.getUniqueId())) { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
| */ | ||||
|         if (!BukkitEntityUtil.entityDamage(damager, victim, event.getCause())) { | ||||
|         if (!BukkitEntityUtil.entityDamage(damager, victim, cause)) { | ||||
|             if (event.isCancelled()) { | ||||
|                 if (victim instanceof Ageable ageable) { | ||||
|                     if (ageable.getAge() == -24000) { | ||||
| @@ -150,55 +148,55 @@ public class EntityEventListener implements Listener { | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         // Armour-stands are handled elsewhere and should not be handled by area-wide entity-spawn options | ||||
|         if (entity.getType() == EntityType.ARMOR_STAND) { | ||||
|             return; | ||||
|         } | ||||
|         CreatureSpawnEvent.SpawnReason reason = event.getSpawnReason(); | ||||
|         switch (reason.toString()) { | ||||
|             case "DISPENSE_EGG": | ||||
|             case "EGG": | ||||
|             case "OCELOT_BABY": | ||||
|             case "SPAWNER_EGG": | ||||
|             case "DISPENSE_EGG", "EGG", "OCELOT_BABY", "SPAWNER_EGG" -> { | ||||
|                 if (!area.isSpawnEggs()) { | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 break; | ||||
|             case "REINFORCEMENTS": | ||||
|             case "NATURAL": | ||||
|             case "MOUNT": | ||||
|             case "PATROL": | ||||
|             case "RAID": | ||||
|             case "SHEARED": | ||||
|             case "SILVERFISH_BLOCK": | ||||
|             case "TRAP": | ||||
|             case "VILLAGE_DEFENSE": | ||||
|             case "VILLAGE_INVASION": | ||||
|             case "BEEHIVE": | ||||
|             case "CHUNK_GEN": | ||||
|             } | ||||
|             case "REINFORCEMENTS", "NATURAL", "MOUNT", "PATROL", "RAID", "SILVERFISH_BLOCK", "ENDER_PEARL", | ||||
|                  "TRAP", "VILLAGE_DEFENSE", "VILLAGE_INVASION", "BEEHIVE", "CHUNK_GEN", "NETHER_PORTAL", | ||||
|                  "FROZEN", "SPELL", "DEFAULT" -> { | ||||
|                 if (!area.isMobSpawning()) { | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 break; | ||||
|             case "BREEDING": | ||||
|             } | ||||
|             case "BREEDING", "DUPLICATION" -> { | ||||
|                 if (!area.isSpawnBreeding()) { | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 break; | ||||
|             case "BUILD_IRONGOLEM": | ||||
|             case "BUILD_SNOWMAN": | ||||
|             case "BUILD_WITHER": | ||||
|             case "CUSTOM": | ||||
|                 if (!area.isSpawnCustom() && entity.getType() != EntityType.ARMOR_STAND) { | ||||
|             } | ||||
|             case "CUSTOM" -> { | ||||
|                 if (!area.isSpawnCustom()) { | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 break; | ||||
|             case "SPAWNER": | ||||
|                 // No need to clutter metadata if running paper | ||||
|                 if (!PaperLib.isPaper()) { | ||||
|                     entity.setMetadata("ps_custom_spawned", new FixedMetadataValue(this.platform, true)); | ||||
|                 } | ||||
|                 return; // Don't cancel if mob spawning is disabled | ||||
|             } | ||||
|             case "BUILD_IRONGOLEM", "BUILD_SNOWMAN", "BUILD_WITHER" -> { | ||||
|                 if (!area.isSpawnCustom()) { | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|             case "SPAWNER" -> { | ||||
|                 if (!area.isMobSpawnerSpawning()) { | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|         Plot plot = area.getOwnedPlotAbs(location); | ||||
|         if (plot == null) { | ||||
| @@ -207,7 +205,7 @@ public class EntityEventListener implements Listener { | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (BukkitEntityUtil.checkEntity(entity, plot)) { | ||||
|         if (BukkitEntityUtil.checkEntity(entity, plot.getBasePlot(false))) { | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|     } | ||||
| @@ -255,6 +253,29 @@ public class EntityEventListener implements Listener { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(ignoreCancelled = true, priority = EventPriority.HIGHEST) | ||||
|     public void onWeavingEffect(EntityChangeBlockEvent event) { | ||||
|         if (event.getTo() != Material.COBWEB) { | ||||
|             return; | ||||
|         } | ||||
|         Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot == null) { | ||||
|             if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, WeavingDeathPlace.class, false)) { | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (!plot.getFlag(WeavingDeathPlace.class)) { | ||||
|             plot.debug(event.getTo() + " could not spawn because weaving-death-place = false"); | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGH) | ||||
|     public void onDamage(EntityDamageEvent event) { | ||||
|         if (event.getEntityType() != EntityType.PLAYER) { | ||||
| @@ -267,7 +288,7 @@ public class EntityEventListener implements Listener { | ||||
|         } | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot == null) { | ||||
|             if (area.isRoadFlags() && area.getRoadFlag(InvincibleFlag.class)) { | ||||
|             if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, InvincibleFlag.class, true)) { | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|             return; | ||||
| @@ -302,7 +323,7 @@ public class EntityEventListener implements Listener { | ||||
|                 if (this.lastRadius != 0) { | ||||
|                     List<Entity> nearby = event.getEntity().getNearbyEntities(this.lastRadius, this.lastRadius, this.lastRadius); | ||||
|                     for (Entity near : nearby) { | ||||
|                         if (near instanceof TNTPrimed || near.getType().equals(EntityType.MINECART_TNT)) { | ||||
|                         if (near instanceof TNTPrimed || near instanceof ExplosiveMinecart) { | ||||
|                             if (!near.hasMetadata("plot")) { | ||||
|                                 near.setMetadata("plot", new FixedMetadataValue((Plugin) PlotSquared.platform(), plot)); | ||||
|                             } | ||||
| @@ -326,7 +347,7 @@ public class EntityEventListener implements Listener { | ||||
|         event.setCancelled(true); | ||||
|         //Spawn Explosion Particles when enabled in settings | ||||
|         if (Settings.General.ALWAYS_SHOW_EXPLOSIONS) { | ||||
|             event.getLocation().getWorld().spawnParticle(Particle.EXPLOSION_HUGE, event.getLocation(), 0); | ||||
|             event.getLocation().getWorld().spawnParticle(EXPLOSION_HUGE, event.getLocation(), 0); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -376,14 +397,13 @@ public class EntityEventListener implements Listener { | ||||
|             if (shooter instanceof Player) { | ||||
|                 PlotPlayer<?> pp = BukkitUtil.adapt((Player) shooter); | ||||
|                 if (plot == null) { | ||||
|                     if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { | ||||
|                     if (area.isRoadFlags() && !area.getRoadFlag(ProjectileChangeBlockFlag.class) && !pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { | ||||
|                         entity.remove(); | ||||
|                         event.setCancelled(true); | ||||
|                     } | ||||
|                     return; | ||||
|                 } | ||||
|                 if (plot.isAdded(pp.getUUID()) || Permissions | ||||
|                         .hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|                 if (plot.isAdded(pp.getUUID()) || plot.getFlag(ProjectileChangeBlockFlag.class) || pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|                     return; | ||||
|                 } | ||||
|                 entity.remove(); | ||||
| @@ -414,7 +434,13 @@ public class EntityEventListener implements Listener { | ||||
|         } | ||||
|  | ||||
|         Plot plot = area.getOwnedPlot(location); | ||||
|         if (plot != null && !plot.getFlag(EntityChangeBlockFlag.class)) { | ||||
|         if (plot == null) { | ||||
|             if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, EntityChangeBlockFlag.class, false)) { | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (!plot.getFlag(EntityChangeBlockFlag.class)) { | ||||
|             plot.debug(e.getType() + " could not change block because entity-change-block = false"); | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -38,8 +31,10 @@ import org.bukkit.Chunk; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.entity.ArmorStand; | ||||
| import org.bukkit.entity.EnderCrystal; | ||||
| import org.bukkit.entity.Entity; | ||||
| import org.bukkit.entity.EntityType; | ||||
| import org.bukkit.entity.Item; | ||||
| import org.bukkit.entity.Vehicle; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.EventPriority; | ||||
| @@ -127,20 +122,26 @@ public class EntitySpawnListener implements Listener { | ||||
|         Entity entity = event.getEntity(); | ||||
|         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (!location.isPlotArea()) { | ||||
|         if (!location.isPlotArea() || area == null) { | ||||
|             return; | ||||
|         } | ||||
|         if (PaperLib.isPaper()) { | ||||
|             //noinspection ConstantValue - getEntitySpawnReason annotated as NotNull, but is not NotNull. lol. | ||||
|             if (area.isSpawnCustom() && entity.getEntitySpawnReason() != null && "CUSTOM".equals(entity.getEntitySpawnReason().name())) { | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|         Plot plot = location.getOwnedPlotAbs(); | ||||
|         if (plot == null) { | ||||
|         EntityType type = entity.getType(); | ||||
|             if (!area.isMobSpawning()) { | ||||
|                 switch (type) { | ||||
|                     case DROPPED_ITEM: | ||||
|         if (plot == null) { | ||||
|             if (entity instanceof Item) { | ||||
|                 if (Settings.Enabled_Components.KILL_ROAD_ITEMS) { | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
|                 return; | ||||
|             } | ||||
|                     case PLAYER: | ||||
|             if (!area.isMobSpawning()) { | ||||
|                 if (type == EntityType.PLAYER) { | ||||
|                     return; | ||||
|                 } | ||||
|                 if (type.isAlive()) { | ||||
| @@ -155,12 +156,13 @@ public class EntitySpawnListener implements Listener { | ||||
|         if (Settings.Done.RESTRICT_BUILDING && DoneFlag.isDone(plot)) { | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|         switch (entity.getType()) { | ||||
|             case ENDER_CRYSTAL: | ||||
|         if (entity instanceof EnderCrystal || type == EntityType.ARMOR_STAND) { | ||||
|             if (BukkitEntityUtil.checkEntity(entity, plot)) { | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|             case SHULKER: | ||||
|             return; | ||||
|         } | ||||
|         if (type == EntityType.SHULKER) { | ||||
|             if (!entity.hasMetadata("shulkerPlot")) { | ||||
|                 entity.setMetadata("shulkerPlot", new FixedMetadataValue((Plugin) PlotSquared.platform(), plot.getId())); | ||||
|             } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -25,7 +18,6 @@ | ||||
|  */ | ||||
| package com.plotsquared.bukkit.listener; | ||||
|  | ||||
| import com.google.common.collect.Iterables; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.location.Location; | ||||
| @@ -33,7 +25,6 @@ import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.flag.implementations.ForcefieldFlag; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.util.Vector; | ||||
|  | ||||
| @@ -46,8 +37,11 @@ public class ForceFieldListener { | ||||
|  | ||||
|     private static Set<PlotPlayer<?>> getNearbyPlayers(Player player, Plot plot) { | ||||
|         Set<PlotPlayer<?>> players = new HashSet<>(); | ||||
|         for (Player nearPlayer : Iterables | ||||
|                 .filter(player.getNearbyEntities(5d, 5d, 5d), Player.class)) { | ||||
|         for (Player nearPlayer : player.getNearbyEntities(5d, 5d, 5d).stream() | ||||
|                 .filter(entity -> entity instanceof Player) | ||||
|                 .map(entity -> (Player) entity) | ||||
|                 .toList() | ||||
|         ) { | ||||
|             PlotPlayer<?> plotPlayer; | ||||
|             if ((plotPlayer = BukkitUtil.adapt(nearPlayer)) == null || !plot | ||||
|                     .equals(plotPlayer.getCurrentPlot())) { | ||||
| @@ -61,8 +55,11 @@ public class ForceFieldListener { | ||||
|     } | ||||
|  | ||||
|     private static PlotPlayer<?> hasNearbyPermitted(Player player, Plot plot) { | ||||
|         for (Player nearPlayer : Iterables | ||||
|                 .filter(player.getNearbyEntities(5d, 5d, 5d), Player.class)) { | ||||
|         for (Player nearPlayer : player.getNearbyEntities(5d, 5d, 5d).stream() | ||||
|                 .filter(entity -> entity instanceof Player) | ||||
|                 .map(entity -> (Player) entity) | ||||
|                 .toList() | ||||
|         ) { | ||||
|             PlotPlayer<?> plotPlayer; | ||||
|             if ((plotPlayer = BukkitUtil.adapt(nearPlayer)) == null || !plot | ||||
|                     .equals(plotPlayer.getCurrentPlot())) { | ||||
| @@ -111,8 +108,7 @@ public class ForceFieldListener { | ||||
|             if (plot.isAdded(uuid)) { | ||||
|                 Set<PlotPlayer<?>> players = getNearbyPlayers(player, plot); | ||||
|                 for (PlotPlayer<?> oPlayer : players) { | ||||
|                     if (!Permissions | ||||
|                             .hasPermission(oPlayer, Permission.PERMISSION_ADMIN_ENTRY_FORCEFIELD)) { | ||||
|                     if (!oPlayer.hasPermission(Permission.PERMISSION_ADMIN_ENTRY_FORCEFIELD)) { | ||||
|                         ((BukkitPlayer) oPlayer).player | ||||
|                                 .setVelocity(calculateVelocity(plotPlayer, oPlayer)); | ||||
|                     } | ||||
| @@ -122,8 +118,7 @@ public class ForceFieldListener { | ||||
|                 if (oPlayer == null) { | ||||
|                     return; | ||||
|                 } | ||||
|                 if (!Permissions | ||||
|                         .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_ENTRY_FORCEFIELD)) { | ||||
|                 if (!plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_ENTRY_FORCEFIELD)) { | ||||
|                     player.setVelocity(calculateVelocity(oPlayer, plotPlayer)); | ||||
|                 } | ||||
|             } | ||||
|   | ||||
| @@ -0,0 +1,201 @@ | ||||
| /* | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.listener; | ||||
|  | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.database.DBFunc; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.flag.implementations.DisablePhysicsFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.RedstoneFlag; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.PlotFlagUtil; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import com.plotsquared.core.util.task.TaskTime; | ||||
| import com.sk89q.worldedit.WorldEdit; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.Material; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.block.BlockFace; | ||||
| import org.bukkit.block.data.BlockData; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.EventPriority; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.event.block.BlockPhysicsEvent; | ||||
| import org.bukkit.event.block.BlockRedstoneEvent; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Set; | ||||
| import java.util.UUID; | ||||
|  | ||||
| @SuppressWarnings("unused") | ||||
| public class HighFreqBlockEventListener implements Listener { | ||||
|  | ||||
|     private static final Set<Material> PISTONS = Set.of( | ||||
|             Material.PISTON, | ||||
|             Material.STICKY_PISTON | ||||
|     ); | ||||
|     private static final Set<Material> PHYSICS_BLOCKS = Set.of( | ||||
|             Material.TURTLE_EGG, | ||||
|             Material.TURTLE_SPAWN_EGG | ||||
|     ); | ||||
|  | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|     private final WorldEdit worldEdit; | ||||
|  | ||||
|     @Inject | ||||
|     public HighFreqBlockEventListener(final @NonNull PlotAreaManager plotAreaManager, final @NonNull WorldEdit worldEdit) { | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|         this.worldEdit = worldEdit; | ||||
|     } | ||||
|  | ||||
|     public static void sendBlockChange(final org.bukkit.Location bloc, final BlockData data) { | ||||
|         TaskManager.runTaskLater(() -> { | ||||
|             String world = bloc.getWorld().getName(); | ||||
|             int x = bloc.getBlockX(); | ||||
|             int z = bloc.getBlockZ(); | ||||
|             int distance = Bukkit.getViewDistance() * 16; | ||||
|  | ||||
|             for (final PlotPlayer<?> player : PlotSquared.platform().playerManager().getPlayers()) { | ||||
|                 Location location = player.getLocation(); | ||||
|                 if (location.getWorldName().equals(world)) { | ||||
|                     if (16 * Math.abs(location.getX() - x) / 16 > distance || 16 * Math.abs(location.getZ() - z) / 16 > distance) { | ||||
|                         continue; | ||||
|                     } | ||||
|                     ((BukkitPlayer) player).player.sendBlockChange(bloc, data); | ||||
|                 } | ||||
|             } | ||||
|         }, TaskTime.ticks(3L)); | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onRedstoneEvent(BlockRedstoneEvent event) { | ||||
|         Block block = event.getBlock(); | ||||
|         Location location = BukkitUtil.adapt(block.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot == null) { | ||||
|             if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, RedstoneFlag.class, false)) { | ||||
|                 event.setNewCurrent(0); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (!plot.getFlag(RedstoneFlag.class)) { | ||||
|             event.setNewCurrent(0); | ||||
|             plot.debug("Redstone event was cancelled because redstone = false"); | ||||
|             return; | ||||
|         } | ||||
|         if (Settings.Redstone.DISABLE_OFFLINE) { | ||||
|             boolean disable = false; | ||||
|             if (!DBFunc.SERVER.equals(plot.getOwner())) { | ||||
|                 if (plot.isMerged()) { | ||||
|                     disable = true; | ||||
|                     for (UUID owner : plot.getOwners()) { | ||||
|                         if (PlotSquared.platform().playerManager().getPlayerIfExists(owner) != null) { | ||||
|                             disable = false; | ||||
|                             break; | ||||
|                         } | ||||
|                     } | ||||
|                 } else { | ||||
|                     disable = PlotSquared.platform().playerManager().getPlayerIfExists(plot.getOwnerAbs()) == null; | ||||
|                 } | ||||
|             } | ||||
|             if (disable) { | ||||
|                 for (UUID trusted : plot.getTrusted()) { | ||||
|                     if (PlotSquared.platform().playerManager().getPlayerIfExists(trusted) != null) { | ||||
|                         disable = false; | ||||
|                         break; | ||||
|                     } | ||||
|                 } | ||||
|                 if (disable) { | ||||
|                     event.setNewCurrent(0); | ||||
|                     plot.debug("Redstone event was cancelled because no trusted player was in the plot"); | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         if (Settings.Redstone.DISABLE_UNOCCUPIED) { | ||||
|             for (final PlotPlayer<?> player : PlotSquared.platform().playerManager().getPlayers()) { | ||||
|                 if (plot.equals(player.getCurrentPlot())) { | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|             event.setNewCurrent(0); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(ignoreCancelled = true, priority = EventPriority.HIGHEST) | ||||
|     public void onPhysicsEvent(BlockPhysicsEvent event) { | ||||
|         Block block = event.getBlock(); | ||||
|         Location location = BukkitUtil.adapt(block.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = area.getOwnedPlotAbs(location); | ||||
|         if (plot == null) { | ||||
|             return; | ||||
|         } | ||||
|         if (event.getChangedType().hasGravity() && plot.getFlag(DisablePhysicsFlag.class)) { | ||||
|             event.setCancelled(true); | ||||
|             sendBlockChange(event.getBlock().getLocation(), event.getBlock().getBlockData()); | ||||
|             plot.debug("Prevented block physics and resent block change because disable-physics = true"); | ||||
|             return; | ||||
|         } | ||||
|         if (event.getChangedType() == Material.COMPARATOR) { | ||||
|             if (!plot.getFlag(RedstoneFlag.class)) { | ||||
|                 event.setCancelled(true); | ||||
|                 plot.debug("Prevented comparator update because redstone = false"); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (PHYSICS_BLOCKS.contains(event.getChangedType())) { | ||||
|             if (plot.getFlag(DisablePhysicsFlag.class)) { | ||||
|                 event.setCancelled(true); | ||||
|                 plot.debug("Prevented block physics because disable-physics = true"); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (Settings.Redstone.DETECT_INVALID_EDGE_PISTONS) { | ||||
|             if (PISTONS.contains(block.getType())) { | ||||
|                 org.bukkit.block.data.Directional piston = (org.bukkit.block.data.Directional) block.getBlockData(); | ||||
|                 final BlockFace facing = piston.getFacing(); | ||||
|                 location = location.add(facing.getModX(), facing.getModY(), facing.getModZ()); | ||||
|                 Plot newPlot = area.getOwnedPlotAbs(location); | ||||
|                 if (plot.equals(newPlot)) { | ||||
|                     return; | ||||
|                 } | ||||
|                 if (!plot.isMerged() || !plot.getConnectedPlots().contains(newPlot)) { | ||||
|                     event.setCancelled(true); | ||||
|                     plot.debug("Prevented piston update because of invalid edge piston detection"); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -25,6 +18,8 @@ | ||||
|  */ | ||||
| package com.plotsquared.bukkit.listener; | ||||
|  | ||||
| import com.destroystokyo.paper.event.block.BeaconEffectEvent; | ||||
| import com.destroystokyo.paper.event.block.BlockDestroyEvent; | ||||
| import com.destroystokyo.paper.event.entity.EntityPathfindEvent; | ||||
| import com.destroystokyo.paper.event.entity.PlayerNaturallySpawnCreaturesEvent; | ||||
| import com.destroystokyo.paper.event.entity.PreCreatureSpawnEvent; | ||||
| @@ -34,6 +29,7 @@ import com.destroystokyo.paper.event.player.PlayerLaunchProjectileEvent; | ||||
| import com.destroystokyo.paper.event.server.AsyncTabCompleteEvent; | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.command.Command; | ||||
| import com.plotsquared.core.command.MainCommand; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| @@ -43,11 +39,23 @@ import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.PlotAreaType; | ||||
| import com.plotsquared.core.plot.flag.FlagContainer; | ||||
| import com.plotsquared.core.plot.flag.implementations.BeaconEffectsFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.DoneFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.FishingFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.ProjectilesFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.TileDropFlag; | ||||
| import com.plotsquared.core.plot.flag.types.BooleanFlag; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import net.kyori.adventure.text.minimessage.Template; | ||||
| import com.plotsquared.core.util.PlotFlagUtil; | ||||
| import io.papermc.paper.event.entity.EntityMoveEvent; | ||||
| import io.papermc.paper.event.world.StructuresLocateEvent; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.NamespacedKey; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.block.TileState; | ||||
| import org.bukkit.entity.Entity; | ||||
| @@ -55,6 +63,7 @@ import org.bukkit.entity.EntityType; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.entity.Projectile; | ||||
| import org.bukkit.entity.Slime; | ||||
| import org.bukkit.event.Cancellable; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.EventPriority; | ||||
| import org.bukkit.event.Listener; | ||||
| @@ -75,6 +84,9 @@ import java.util.regex.Pattern; | ||||
| @SuppressWarnings("unused") | ||||
| public class PaperListener implements Listener { | ||||
|  | ||||
|     private static final NamespacedKey ITEM = NamespacedKey.minecraft("item"); | ||||
|     private static final NamespacedKey FISHING_BOBBER = NamespacedKey.minecraft("fishing_bobber"); | ||||
|  | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|     private Chunk lastChunk; | ||||
|  | ||||
| @@ -83,38 +95,25 @@ public class PaperListener implements Listener { | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|     } | ||||
|  | ||||
|     @EventHandler(ignoreCancelled = true, priority = EventPriority.LOWEST) | ||||
|     public void onBlockDestroy(final BlockDestroyEvent event) { | ||||
|         Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = area.getPlot(location); | ||||
|         if (plot != null) { | ||||
|             event.setWillDrop(plot.getFlag(TileDropFlag.class)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onEntityPathfind(EntityPathfindEvent event) { | ||||
|         if (!Settings.Paper_Components.ENTITY_PATHING) { | ||||
|             return; | ||||
|         } | ||||
|         Location toLoc = BukkitUtil.adapt(event.getLoc()); | ||||
|         Location fromLoc = BukkitUtil.adapt(event.getEntity().getLocation()); | ||||
|         PlotArea tarea = toLoc.getPlotArea(); | ||||
|         if (tarea == null) { | ||||
|             return; | ||||
|         } | ||||
|         PlotArea farea = fromLoc.getPlotArea(); | ||||
|         if (farea == null) { | ||||
|             return; | ||||
|         } | ||||
|         if (tarea != farea) { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
|         Plot tplot = toLoc.getPlot(); | ||||
|         Plot fplot = fromLoc.getPlot(); | ||||
|         if (tplot == null ^ fplot == null) { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
|         if (tplot == null || tplot.getId().hashCode() == fplot.getId().hashCode()) { | ||||
|             return; | ||||
|         } | ||||
|         if (fplot.isMerged() && fplot.getConnectedPlots().contains(fplot)) { | ||||
|             return; | ||||
|         } | ||||
|         event.setCancelled(true); | ||||
|         handleEntityMovement(event, event.getEntity().getLocation(), event.getLoc()); | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
| @@ -124,13 +123,28 @@ public class PaperListener implements Listener { | ||||
|         } | ||||
|         Slime slime = event.getEntity(); | ||||
|  | ||||
|         Block b = slime.getTargetBlock(4); | ||||
|         Block b = slime.getTargetBlockExact(4); | ||||
|         if (b == null) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         Location toLoc = BukkitUtil.adapt(b.getLocation()); | ||||
|         Location fromLoc = BukkitUtil.adapt(event.getEntity().getLocation()); | ||||
|         handleEntityMovement(event, event.getEntity().getLocation(),  b.getLocation()); | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onEntityMove(EntityMoveEvent event) { | ||||
|         if (!Settings.Paper_Components.ENTITY_MOVEMENT) { | ||||
|             return; | ||||
|         } | ||||
|         if (!event.hasExplicitlyChangedBlock()) { | ||||
|             return; | ||||
|         } | ||||
|         handleEntityMovement(event, event.getFrom(), event.getTo()); | ||||
|     } | ||||
|  | ||||
|     private static void handleEntityMovement(Cancellable event, org.bukkit.Location from, org.bukkit.Location target) { | ||||
|         Location toLoc = BukkitUtil.adapt(target); | ||||
|         Location fromLoc = BukkitUtil.adapt(from); | ||||
|         PlotArea tarea = toLoc.getPlotArea(); | ||||
|         if (tarea == null) { | ||||
|             return; | ||||
| @@ -139,7 +153,6 @@ public class PaperListener implements Listener { | ||||
|         if (farea == null) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (tarea != farea) { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
| @@ -150,10 +163,10 @@ public class PaperListener implements Listener { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
|         if (tplot == null || tplot.getId().hashCode() == fplot.getId().hashCode()) { | ||||
|         if (tplot == null || tplot.getId().equals(fplot.getId())) { | ||||
|             return; | ||||
|         } | ||||
|         if (fplot.isMerged() && fplot.getConnectedPlots().contains(fplot)) { | ||||
|         if (fplot.isMerged() && fplot.getConnectedPlots().contains(tplot)) { | ||||
|             return; | ||||
|         } | ||||
|         event.setCancelled(true); | ||||
| @@ -166,83 +179,70 @@ public class PaperListener implements Listener { | ||||
|         } | ||||
|         Location location = BukkitUtil.adapt(event.getSpawnLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (!location.isPlotArea()) { | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         // Armour-stands are handled elsewhere and should not be handled by area-wide entity-spawn options | ||||
|         if (event.getType() == EntityType.ARMOR_STAND) { | ||||
|             return; | ||||
|         } | ||||
|         // If entities are spawning... the chunk should be loaded? | ||||
|         Entity[] entities = event.getSpawnLocation().getChunk().getEntities(); | ||||
|         if (entities.length > Settings.Chunk_Processor.MAX_ENTITIES) { | ||||
|         if (entities.length >= Settings.Chunk_Processor.MAX_ENTITIES) { | ||||
|             event.setShouldAbortSpawn(true); | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
|         CreatureSpawnEvent.SpawnReason reason = event.getReason(); | ||||
|         switch (reason.toString()) { | ||||
|             case "DISPENSE_EGG": | ||||
|             case "EGG": | ||||
|             case "OCELOT_BABY": | ||||
|             case "SPAWNER_EGG": | ||||
|             case "DISPENSE_EGG", "EGG", "OCELOT_BABY", "SPAWNER_EGG" -> { | ||||
|                 if (!area.isSpawnEggs()) { | ||||
|                     event.setShouldAbortSpawn(true); | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 break; | ||||
|             case "REINFORCEMENTS": | ||||
|             case "NATURAL": | ||||
|             case "MOUNT": | ||||
|             case "PATROL": | ||||
|             case "RAID": | ||||
|             case "SHEARED": | ||||
|             case "SILVERFISH_BLOCK": | ||||
|             case "TRAP": | ||||
|             case "VILLAGE_DEFENSE": | ||||
|             case "VILLAGE_INVASION": | ||||
|             case "BEEHIVE": | ||||
|             case "CHUNK_GEN": | ||||
|             } | ||||
|             case "REINFORCEMENTS", "NATURAL", "MOUNT", "PATROL", "RAID", "SILVERFISH_BLOCK", "ENDER_PEARL", "TRAP", "VILLAGE_DEFENSE", "VILLAGE_INVASION", "BEEHIVE", "CHUNK_GEN" -> { | ||||
|                 if (!area.isMobSpawning()) { | ||||
|                     event.setShouldAbortSpawn(true); | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 break; | ||||
|             case "BREEDING": | ||||
|             } | ||||
|             case "BREEDING" -> { | ||||
|                 if (!area.isSpawnBreeding()) { | ||||
|                     event.setShouldAbortSpawn(true); | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 break; | ||||
|             case "BUILD_IRONGOLEM": | ||||
|             case "BUILD_SNOWMAN": | ||||
|             case "BUILD_WITHER": | ||||
|             case "CUSTOM": | ||||
|                 if (!area.isSpawnCustom() && event.getType() != EntityType.ARMOR_STAND) { | ||||
|             } | ||||
|             case "BUILD_IRONGOLEM", "BUILD_SNOWMAN", "BUILD_WITHER", "CUSTOM" -> { | ||||
|                 if (!area.isSpawnCustom()) { | ||||
|                     event.setShouldAbortSpawn(true); | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 break; | ||||
|             case "SPAWNER": | ||||
|             } | ||||
|             case "SPAWNER" -> { | ||||
|                 if (!area.isMobSpawnerSpawning()) { | ||||
|                     event.setShouldAbortSpawn(true); | ||||
|                     event.setCancelled(true); | ||||
|                     return; | ||||
|                 } | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|         Plot plot = location.getOwnedPlotAbs(); | ||||
|         if (plot == null) { | ||||
|             EntityType type = event.getType(); | ||||
|             if (!area.isMobSpawning()) { | ||||
|                 switch (type) { | ||||
|                     case DROPPED_ITEM: | ||||
|             // PreCreatureSpawnEvent **should** not be called for DROPPED_ITEM, just for the sake of consistency | ||||
|             if (type.getKey().equals(ITEM)) { | ||||
|                 if (Settings.Enabled_Components.KILL_ROAD_ITEMS) { | ||||
|                             event.setShouldAbortSpawn(true); | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
|                 return; | ||||
|             } | ||||
|                     case PLAYER: | ||||
|             if (!area.isMobSpawning()) { | ||||
|                 if (type == EntityType.PLAYER) { | ||||
|                     return; | ||||
|                 } | ||||
|                 if (type.isAlive()) { | ||||
| @@ -303,7 +303,7 @@ public class PaperListener implements Listener { | ||||
|             final PlotPlayer<?> plotPlayer = BukkitUtil.adapt(event.getPlayer()); | ||||
|             plotPlayer.sendMessage( | ||||
|                     TranslatableCaption.of("errors.tile_entity_cap_reached"), | ||||
|                     Template.of("amount", String.valueOf(Settings.Chunk_Processor.MAX_TILES)) | ||||
|                     TagResolver.resolver("amount", Tag.inserting(Component.text(Settings.Chunk_Processor.MAX_TILES))) | ||||
|             ); | ||||
|             event.setCancelled(true); | ||||
|             event.setBuild(false); | ||||
| @@ -327,41 +327,60 @@ public class PaperListener implements Listener { | ||||
|             return; | ||||
|         } | ||||
|         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         PlotPlayer<Player> pp = BukkitUtil.adapt((Player) shooter); | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|  | ||||
|         if (plot == null) { | ||||
|             if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_ROAD)) { | ||||
|             if (!PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, ProjectilesFlag.class, true) && !pp.hasPermission( | ||||
|                     Permission.PERMISSION_ADMIN_PROJECTILE_ROAD | ||||
|             )) { | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_PROJECTILE_ROAD)) | ||||
|                         TagResolver.resolver( | ||||
|                                 "node", | ||||
|                                 Tag.inserting(Permission.PERMISSION_ADMIN_PROJECTILE_ROAD) | ||||
|                         ) | ||||
|                 ); | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } else if (!plot.hasOwner()) { | ||||
|             if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { | ||||
|             if (!pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) | ||||
|                         TagResolver.resolver( | ||||
|                                 "node", | ||||
|                                 Tag.inserting(Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED) | ||||
|                         ) | ||||
|                 ); | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } else if (!plot.isAdded(pp.getUUID())) { | ||||
|             if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|             if (entity.getType().getKey().equals(FISHING_BOBBER)) { | ||||
|                 if (plot.getFlag(FishingFlag.class)) { | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|             if (!plot.getFlag(ProjectilesFlag.class)) { | ||||
|                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|                     pp.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER) | ||||
|                             ) | ||||
|                     ); | ||||
|                     entity.remove(); | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onAsyncTabCompletion(final AsyncTabCompleteEvent event) { | ||||
| @@ -403,4 +422,67 @@ public class PaperListener implements Listener { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(ignoreCancelled = true) | ||||
|     public void onBeaconEffect(final BeaconEffectEvent event) { | ||||
|         Block block = event.getBlock(); | ||||
|         Location beaconLocation = BukkitUtil.adapt(block.getLocation()); | ||||
|         Plot beaconPlot = beaconLocation.getPlot(); | ||||
|  | ||||
|         PlotArea area = beaconLocation.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         Player player = event.getPlayer(); | ||||
|         Location playerLocation = BukkitUtil.adapt(player.getLocation()); | ||||
|  | ||||
|         PlotPlayer<Player> plotPlayer = BukkitUtil.adapt(player); | ||||
|         Plot playerStandingPlot = playerLocation.getPlot(); | ||||
|         if (playerStandingPlot == null) { | ||||
|             FlagContainer container = area.getRoadFlagContainer(); | ||||
|             if (!getBooleanFlagValue(container, BeaconEffectsFlag.class, true) || | ||||
|                     (beaconPlot != null && Settings.Enabled_Components.DISABLE_BEACON_EFFECT_OVERFLOW)) { | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         FlagContainer container = playerStandingPlot.getFlagContainer(); | ||||
|         boolean plotBeaconEffects = getBooleanFlagValue(container, BeaconEffectsFlag.class, true); | ||||
|         if (playerStandingPlot.equals(beaconPlot)) { | ||||
|             if (!plotBeaconEffects) { | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (!plotBeaconEffects || Settings.Enabled_Components.DISABLE_BEACON_EFFECT_OVERFLOW) { | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Don't let the server die when populating cartographers (villager offering maps) in classic plot worlds | ||||
|      * (as those don't generate POIs) | ||||
|      */ | ||||
|     @EventHandler(ignoreCancelled = true, priority = EventPriority.HIGHEST) | ||||
|     public void onStructuresLocate(StructuresLocateEvent event) { | ||||
|         if (!PlotSquared.get().getPlotAreaManager().hasPlotArea(event.getWorld().getName())) { | ||||
|             return; | ||||
|         } | ||||
|         final PlotArea area = PlotSquared.get().getPlotAreaManager().getPlotAreaByString(event.getWorld().getName()); | ||||
|         if (area != null && area.getType() == PlotAreaType.NORMAL) { | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private boolean getBooleanFlagValue( | ||||
|             @NonNull FlagContainer container, | ||||
|             @NonNull Class<? extends BooleanFlag<?>> flagClass, | ||||
|             boolean defaultValue | ||||
|     ) { | ||||
|         BooleanFlag<?> flag = container.getFlag(flagClass); | ||||
|         return flag == null ? defaultValue : flag.getValue(); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,94 +0,0 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            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.util.BukkitUtil; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import net.kyori.adventure.text.minimessage.Template; | ||||
| import org.bukkit.block.Banner; | ||||
| import org.bukkit.block.Beacon; | ||||
| import org.bukkit.block.BlockState; | ||||
| import org.bukkit.block.CommandBlock; | ||||
| import org.bukkit.block.Comparator; | ||||
| import org.bukkit.block.Conduit; | ||||
| import org.bukkit.block.Container; | ||||
| import org.bukkit.block.CreatureSpawner; | ||||
| import org.bukkit.block.DaylightDetector; | ||||
| import org.bukkit.block.EnchantingTable; | ||||
| import org.bukkit.block.EndGateway; | ||||
| import org.bukkit.block.EnderChest; | ||||
| import org.bukkit.block.Jukebox; | ||||
| import org.bukkit.block.Sign; | ||||
| import org.bukkit.block.Skull; | ||||
| import org.bukkit.block.Structure; | ||||
| import org.bukkit.block.data.type.Bed; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.block.BlockPlaceEvent; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| public class PaperListener113 extends PaperListener { | ||||
|  | ||||
|     @Inject | ||||
|     public PaperListener113(@NonNull PlotAreaManager plotAreaManager) { | ||||
|         super(plotAreaManager); | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onBlockPlace(BlockPlaceEvent event) { | ||||
|         if (!Settings.Paper_Components.TILE_ENTITY_CHECK || !Settings.Enabled_Components.CHUNK_PROCESSOR) { | ||||
|             return; | ||||
|         } | ||||
|         BlockState state = event.getBlock().getState(false); | ||||
|         if (!(state instanceof Banner || state instanceof Beacon || state instanceof Bed || state instanceof CommandBlock | ||||
|                 || state instanceof Comparator || state instanceof Conduit || state instanceof Container || state instanceof CreatureSpawner | ||||
|                 || state instanceof DaylightDetector || state instanceof EnchantingTable || state instanceof EnderChest || state instanceof EndGateway | ||||
|                 || state instanceof Jukebox || state instanceof Sign || state instanceof Skull || state instanceof Structure)) { | ||||
|             return; | ||||
|         } | ||||
|         final Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||
|         final PlotArea plotArea = location.getPlotArea(); | ||||
|         if (plotArea == null) { | ||||
|             return; | ||||
|         } | ||||
|         final int tileEntityCount = event.getBlock().getChunk().getTileEntities(false).length; | ||||
|         if (tileEntityCount >= Settings.Chunk_Processor.MAX_TILES) { | ||||
|             final PlotPlayer<?> plotPlayer = BukkitUtil.adapt(event.getPlayer()); | ||||
|             plotPlayer.sendMessage( | ||||
|                     TranslatableCaption.of("errors.tile_entity_cap_reached"), | ||||
|                     Template.of("amount", String.valueOf(Settings.Chunk_Processor.MAX_TILES)) | ||||
|             ); | ||||
|             event.setCancelled(true); | ||||
|             event.setBuild(false); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -0,0 +1,66 @@ | ||||
| /* | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.listener; | ||||
|  | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.flag.implementations.EditSignFlag; | ||||
| import com.plotsquared.core.util.PlotFlagUtil; | ||||
| import org.bukkit.block.Sign; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.event.player.PlayerSignOpenEvent; | ||||
|  | ||||
| /** | ||||
|  * For events since 1.20.1 | ||||
|  * @since 7.2.1 | ||||
|  */ | ||||
| public class PlayerEventListener1201 implements Listener { | ||||
|  | ||||
|     @EventHandler(ignoreCancelled = true) | ||||
|     @SuppressWarnings({"removal", "UnstableApiUsage"}) // thanks Paper, thanks Spigot | ||||
|     public void onPlayerSignOpenEvent(PlayerSignOpenEvent event) { | ||||
|         Sign sign = event.getSign(); | ||||
|         Location location = BukkitUtil.adapt(sign.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot == null) { | ||||
|             if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, EditSignFlag.class, false) | ||||
|                     && !event.getPlayer().hasPermission(Permission.PERMISSION_ADMIN_INTERACT_ROAD.toString())) { | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (plot.isAdded(event.getPlayer().getUniqueId())) { | ||||
|             return; // allow for added players | ||||
|         } | ||||
|         if (!plot.getFlag(EditSignFlag.class) | ||||
|                 && !event.getPlayer().hasPermission(Permission.PERMISSION_ADMIN_INTERACT_OTHER.toString())) { | ||||
|             plot.debug(event.getPlayer().getName() + " could not edit the sign because of edit-sign = false"); | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -35,10 +28,14 @@ import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.PlotHandler; | ||||
| import com.plotsquared.core.plot.flag.implementations.FishingFlag; | ||||
| 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 com.plotsquared.core.util.PlotFlagUtil; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
| import org.bukkit.entity.Entity; | ||||
| import org.bukkit.entity.FishHook; | ||||
| import org.bukkit.entity.LivingEntity; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.entity.Projectile; | ||||
| @@ -50,6 +47,7 @@ import org.bukkit.event.entity.LingeringPotionSplashEvent; | ||||
| import org.bukkit.event.entity.PotionSplashEvent; | ||||
| import org.bukkit.event.entity.ProjectileHitEvent; | ||||
| import org.bukkit.event.entity.ProjectileLaunchEvent; | ||||
| import org.bukkit.event.player.PlayerEggThrowEvent; | ||||
| import org.bukkit.projectiles.BlockProjectileSource; | ||||
| import org.bukkit.projectiles.ProjectileSource; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| @@ -95,7 +93,7 @@ public class ProjectileEventListener implements Listener { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     @EventHandler(ignoreCancelled = true) | ||||
|     public void onProjectileLaunch(ProjectileLaunchEvent event) { | ||||
|         Projectile entity = event.getEntity(); | ||||
|         ProjectileSource shooter = entity.getShooter(); | ||||
| @@ -103,92 +101,136 @@ public class ProjectileEventListener implements Listener { | ||||
|             return; | ||||
|         } | ||||
|         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         PlotPlayer<Player> pp = BukkitUtil.adapt((Player) shooter); | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|  | ||||
|         if (plot == null) { | ||||
|             if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_ROAD)) { | ||||
|             if (!PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, ProjectilesFlag.class, true) && !pp.hasPermission( | ||||
|                     Permission.PERMISSION_ADMIN_PROJECTILE_ROAD | ||||
|             )) { | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_PROJECTILE_ROAD)) | ||||
|                         TagResolver.resolver( | ||||
|                                 "node", | ||||
|                                 Tag.inserting(Permission.PERMISSION_ADMIN_PROJECTILE_ROAD) | ||||
|                         ) | ||||
|                 ); | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } else if (!plot.hasOwner()) { | ||||
|             if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { | ||||
|             if (!pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) | ||||
|                         TagResolver.resolver( | ||||
|                                 "node", | ||||
|                                 Tag.inserting(Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED) | ||||
|                         ) | ||||
|                 ); | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } else if (!plot.isAdded(pp.getUUID())) { | ||||
|             if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|             if (entity instanceof FishHook) { | ||||
|                 if (plot.getFlag(FishingFlag.class)) { | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|             if (!plot.getFlag(ProjectilesFlag.class)) { | ||||
|                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|                     pp.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                         Template.of("node", String.valueOf(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER) | ||||
|                             ) | ||||
|                     ); | ||||
|                     entity.remove(); | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onProjectileHit(ProjectileHitEvent event) { | ||||
|         Projectile entity = event.getEntity(); | ||||
|         if (cancelProjectileHit(event.getEntity())) { | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onPlayerEggThrow(PlayerEggThrowEvent event) { | ||||
|         if (cancelProjectileHit(event.getEgg())) { | ||||
|             event.setHatching(false); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private boolean cancelProjectileHit(Projectile entity) { | ||||
|         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||
|             return; | ||||
|             return false; | ||||
|         } | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|             return false; | ||||
|         } | ||||
|         Plot plot = area.getPlot(location); | ||||
|         ProjectileSource shooter = entity.getShooter(); | ||||
|         if (shooter instanceof Player) { | ||||
|             if (!((Player) shooter).isOnline()) { | ||||
|                 if (plot != null) { | ||||
|                     if (plot.isAdded(((Player) shooter).getUniqueId()) || plot.getFlag(ProjectilesFlag.class)) { | ||||
|                         return false; | ||||
|                     } | ||||
|                 } else if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, ProjectilesFlag.class, true)) { | ||||
|                     return false; | ||||
|                 } | ||||
|  | ||||
|                 entity.remove(); | ||||
|                 return true; | ||||
|             } | ||||
|  | ||||
|             PlotPlayer<?> pp = BukkitUtil.adapt((Player) shooter); | ||||
|             if (plot == null) { | ||||
|                 if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED)) { | ||||
|                 if (!PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, ProjectilesFlag.class, true) && !pp.hasPermission( | ||||
|                         Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED | ||||
|                 )) { | ||||
|                     entity.remove(); | ||||
|                     event.setCancelled(true); | ||||
|                     return true; | ||||
|                 } | ||||
|                 return; | ||||
|                 return false; | ||||
|             } | ||||
|             if (plot.isAdded(pp.getUUID()) || Permissions | ||||
|                     .hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|                 return; | ||||
|             if (plot.isAdded(pp.getUUID()) || pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER) || plot.getFlag( | ||||
|                     ProjectilesFlag.class) || (entity instanceof FishHook && plot.getFlag( | ||||
|                     FishingFlag.class))) { | ||||
|                 return false; | ||||
|             } | ||||
|             entity.remove(); | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|             return true; | ||||
|         } | ||||
|         if (!(shooter instanceof Entity) && shooter != null) { | ||||
|             if (plot == null) { | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|                 return true; | ||||
|             } | ||||
|             Location sLoc = | ||||
|                     BukkitUtil.adapt(((BlockProjectileSource) shooter).getBlock().getLocation()); | ||||
|             if (!area.contains(sLoc.getX(), sLoc.getZ())) { | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|                 return true; | ||||
|             } | ||||
|             Plot sPlot = area.getOwnedPlotAbs(sLoc); | ||||
|             if (sPlot == null || !PlotHandler.sameOwners(plot, sPlot)) { | ||||
|                 entity.remove(); | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -28,9 +21,14 @@ package com.plotsquared.bukkit.listener; | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.bukkit.BukkitPlatform; | ||||
| import com.plotsquared.bukkit.placeholder.MVdWPlaceholders; | ||||
| import com.plotsquared.bukkit.util.BukkitEconHandler; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.player.ConsolePlayer; | ||||
| import com.plotsquared.core.util.EconHandler; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.Listener; | ||||
| @@ -39,6 +37,8 @@ import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| public class ServerListener implements Listener { | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + ServerListener.class.getSimpleName()); | ||||
|  | ||||
|     private final BukkitPlatform plugin; | ||||
|  | ||||
|     @Inject | ||||
| @@ -52,6 +52,29 @@ public class ServerListener implements Listener { | ||||
|             new MVdWPlaceholders(this.plugin, this.plugin.placeholderRegistry()); | ||||
|             ConsolePlayer.getConsole().sendMessage(TranslatableCaption.of("placeholder.hooked")); | ||||
|         } | ||||
|         if (Settings.Enabled_Components.ECONOMY && Bukkit.getPluginManager().isPluginEnabled("Vault")) { | ||||
|             EconHandler econHandler = new BukkitEconHandler(); | ||||
|             try { | ||||
|                 if (!econHandler.init()) { | ||||
|                     LOGGER.warn("Economy is enabled but no plugin is providing an economy service. Falling back..."); | ||||
|                     econHandler = EconHandler.nullEconHandler(); | ||||
|                 } | ||||
|             } catch (final Exception ignored) { | ||||
|                 econHandler = EconHandler.nullEconHandler(); | ||||
|             } | ||||
|             if (PlotSquared.platform().econHandler() instanceof MutableEconHandler meh) { | ||||
|                 meh.setImplementation(econHandler); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Internal use only. Required to implement lazy econ loading using Guice. | ||||
|      * | ||||
|      * @since 7.2.0 | ||||
|      */ | ||||
|     public interface MutableEconHandler { | ||||
|         void setImplementation(EconHandler econHandler); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -38,45 +31,44 @@ import org.bukkit.event.Listener; | ||||
| import org.bukkit.event.world.ChunkEvent; | ||||
| import org.bukkit.event.world.ChunkLoadEvent; | ||||
|  | ||||
| import java.lang.reflect.Field; | ||||
| import java.lang.reflect.Method; | ||||
|  | ||||
| import static com.plotsquared.core.util.ReflectionUtils.getRefClass; | ||||
|  | ||||
| public class SingleWorldListener implements Listener { | ||||
|  | ||||
|     private final Method methodGetHandleChunk; | ||||
|     private Field shouldSave = null; | ||||
|     private final Method methodSetUnsaved; | ||||
|     private Method methodGetHandleChunk; | ||||
|     private Object objChunkStatusFull = null; | ||||
|  | ||||
|     public SingleWorldListener() throws Exception { | ||||
|         ReflectionUtils.RefClass classCraftChunk = getRefClass("{cb}.CraftChunk"); | ||||
|         this.methodGetHandleChunk = classCraftChunk.getMethod("getHandle").getRealMethod(); | ||||
|         ReflectionUtils.RefClass classChunkAccess = getRefClass("net.minecraft.world.level.chunk.IChunkAccess"); | ||||
|         this.methodSetUnsaved = classChunkAccess.getMethod("a", boolean.class).getRealMethod(); | ||||
|         try { | ||||
|             if (PlotSquared.platform().serverVersion()[1] < 17) { | ||||
|                 ReflectionUtils.RefClass classChunk = getRefClass("{nms}.Chunk"); | ||||
|                 if (PlotSquared.platform().serverVersion()[1] == 13) { | ||||
|                     this.shouldSave = classChunk.getField("mustSave").getRealField(); | ||||
|                 } else { | ||||
|                     this.shouldSave = classChunk.getField("s").getRealField(); | ||||
|             this.methodGetHandleChunk = classCraftChunk.getMethod("getHandle").getRealMethod(); | ||||
|         } catch (NoSuchMethodException ignored) { | ||||
|             try { | ||||
|                 String chunkStatus = PlotSquared.platform().serverVersion()[1] < 21 | ||||
|                         ? "net.minecraft.world.level.chunk" + ".ChunkStatus" | ||||
|                         : "net.minecraft.world.level.chunk.status.ChunkStatus"; | ||||
|                 ReflectionUtils.RefClass classChunkStatus = getRefClass(chunkStatus); | ||||
|                 this.objChunkStatusFull = classChunkStatus.getRealClass().getField("n").get(null); | ||||
|                 this.methodGetHandleChunk = classCraftChunk | ||||
|                         .getMethod("getHandle", classChunkStatus.getRealClass()) | ||||
|                         .getRealMethod(); | ||||
|             } catch (NoSuchMethodException ex) { | ||||
|                 throw new RuntimeException(ex); | ||||
|             } | ||||
|             } 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) { | ||||
|             e.printStackTrace(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public void markChunkAsClean(Chunk chunk) { | ||||
|         try { | ||||
|             Object nmsChunk = methodGetHandleChunk.invoke(chunk); | ||||
|             if (shouldSave != null) { | ||||
|                 this.shouldSave.set(nmsChunk, false); | ||||
|             } | ||||
|             Object nmsChunk = objChunkStatusFull != null | ||||
|                     ? this.methodGetHandleChunk.invoke(chunk, objChunkStatusFull) | ||||
|                     : this.methodGetHandleChunk.invoke(chunk); | ||||
|             methodSetUnsaved.invoke(nmsChunk, false); | ||||
|         } catch (Throwable e) { | ||||
|             e.printStackTrace(); | ||||
|         } | ||||
| @@ -92,7 +84,12 @@ public class SingleWorldListener implements Listener { | ||||
|         if (!SinglePlotArea.isSinglePlotWorld(name)) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         int x = event.getChunk().getX(); | ||||
|         int z = event.getChunk().getZ(); | ||||
|         if (x < 16 && x > -16 && z < 16 && z > -16) { | ||||
|             // Allow spawn to generate | ||||
|             return; | ||||
|         } | ||||
|         markChunkAsClean(event.getChunk()); | ||||
|     } | ||||
|  | ||||
| @@ -103,7 +100,8 @@ public class SingleWorldListener implements Listener { | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.LOWEST) | ||||
|     public void onChunkLoad(ChunkLoadEvent event) { | ||||
|         handle(event); | ||||
|         // disable this for now, should address https://github.com/IntellectualSites/PlotSquared/issues/4413 | ||||
|         // handle(event); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,57 @@ | ||||
| /* | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.listener; | ||||
|  | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.flag.FlagContainer; | ||||
| import com.plotsquared.core.plot.flag.implementations.BeaconEffectsFlag; | ||||
| import org.bukkit.entity.Entity; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.event.entity.EntityPotionEffectEvent; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| /** | ||||
|  * Fallback listener for paper events on spigot | ||||
|  */ | ||||
| public class SpigotListener implements Listener { | ||||
|  | ||||
|     @EventHandler(ignoreCancelled = true) | ||||
|     public void onEffect(@NonNull EntityPotionEffectEvent event) { | ||||
|         if (event.getCause() != EntityPotionEffectEvent.Cause.BEACON) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         Entity entity = event.getEntity(); | ||||
|         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||
|         Plot plot = location.getPlot(); | ||||
|         if (plot == null) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         FlagContainer container = plot.getFlagContainer(); | ||||
|         BeaconEffectsFlag effectsEnabled = container.getFlag(BeaconEffectsFlag.class); | ||||
|         if (effectsEnabled != null && !effectsEnabled.getValue()) { | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,70 +0,0 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            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.managers; | ||||
|  | ||||
| /* | ||||
| import com.google.inject.Singleton; | ||||
| import org.bukkit.World; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
| import se.hyperver.hyperverse.Hyperverse; | ||||
| import se.hyperver.hyperverse.world.WorldConfiguration; | ||||
| import se.hyperver.hyperverse.world.WorldConfigurationBuilder; | ||||
| import se.hyperver.hyperverse.world.WorldFeatures; | ||||
| import se.hyperver.hyperverse.world.WorldType; | ||||
|  | ||||
| Hyperverse implementation is currently put on ice until Hyperverse is released on a stable line and deployed to the central | ||||
| repository. | ||||
|  | ||||
| @Singleton | ||||
| public class HyperverseWorldManager extends BukkitWorldManager { | ||||
|  | ||||
|     @Override | ||||
|     public @Nullable World handleWorldCreation(@NonNull String worldName, @Nullable String generator) { | ||||
|         // First let Bukkit register the world | ||||
|         this.setGenerator(worldName, generator); | ||||
|         // Create the world | ||||
|         final WorldConfigurationBuilder worldConfigurationBuilder = WorldConfiguration.builder() | ||||
|                 .setName(worldName).setType(WorldType.OVER_WORLD); | ||||
|         if (generator != null) { | ||||
|             worldConfigurationBuilder.setGenerator(generator).setWorldFeatures(WorldFeatures.FLATLAND); | ||||
|         } | ||||
|         try { | ||||
|             return Hyperverse.getApi().createWorld(worldConfigurationBuilder.createWorldConfiguration()) | ||||
|                     .getBukkitWorld(); | ||||
|         } catch (final Exception e) { | ||||
|             e.printStackTrace(); | ||||
|         } | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public String getName() { | ||||
|         return "bukkit-hyperverse"; | ||||
|     } | ||||
|  | ||||
| } | ||||
|  */ | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -37,8 +30,9 @@ import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  * | ||||
|  * @deprecated Deprecated and scheduled for removal without replacement | ||||
|  *         in favor of the build in setup wizard. | ||||
|  *         However, this class will be kept around for a while, given it's not a maintenance burden. | ||||
|  */ | ||||
| @Deprecated(forRemoval = true, since = "6.0.0") | ||||
| @Deprecated | ||||
| @Singleton | ||||
| public class MultiverseWorldManager extends BukkitWorldManager { | ||||
|  | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -59,7 +52,7 @@ public class MVdWPlaceholders { | ||||
|  | ||||
|     @Subscribe | ||||
|     public void onNewPlaceholder(final PlaceholderRegistry.@NonNull PlaceholderAddedEvent event) { | ||||
|         this.addPlaceholder(event.getPlaceholder()); | ||||
|         this.addPlaceholder(event.placeholder()); | ||||
|     } | ||||
|  | ||||
|     private void addPlaceholder(final @NonNull Placeholder placeholder) { | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -27,6 +20,8 @@ package com.plotsquared.bukkit.placeholder; | ||||
|  | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.flag.implementations.DoneFlag; | ||||
| import com.plotsquared.core.util.query.PlotQuery; | ||||
| import me.clip.placeholderapi.PlaceholderAPIPlugin; | ||||
| import me.clip.placeholderapi.expansion.PlaceholderExpansion; | ||||
| import org.bukkit.entity.Player; | ||||
| @@ -90,6 +85,20 @@ public class PAPIPlaceholders extends PlaceholderExpansion { | ||||
|             return String.valueOf(pl.getPlotCount(identifier)); | ||||
|         } | ||||
|  | ||||
|         if (identifier.startsWith("base_plot_count_")) { | ||||
|             identifier = identifier.substring("base_plot_count_".length()); | ||||
|             if (identifier.isEmpty()) { | ||||
|                 return ""; | ||||
|             } | ||||
|  | ||||
|             return String.valueOf(PlotQuery.newQuery() | ||||
|                     .ownedBy(pl) | ||||
|                     .inWorld(identifier) | ||||
|                     .whereBasePlot() | ||||
|                     .thatPasses(plot -> !DoneFlag.isDone(plot)) | ||||
|                     .count()); | ||||
|         } | ||||
|  | ||||
|         // PlotSquared placeholders | ||||
|         return PlotSquared.platform().placeholderRegistry().getPlaceholderValue(identifier, pl); | ||||
|     } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -94,4 +87,9 @@ public class BukkitOfflinePlayer implements OfflinePlotPlayer { | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean hasPermission(@NonNull final String permission, final boolean notify) { | ||||
|         return hasPermission(permission); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -39,6 +32,7 @@ import com.plotsquared.core.plot.PlotWeather; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.MathMan; | ||||
| import com.plotsquared.core.util.WorldUtil; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.extension.platform.Actor; | ||||
| import com.sk89q.worldedit.world.item.ItemType; | ||||
| @@ -47,6 +41,7 @@ import io.papermc.lib.PaperLib; | ||||
| import net.kyori.adventure.audience.Audience; | ||||
| import org.bukkit.GameMode; | ||||
| import org.bukkit.Sound; | ||||
| import org.bukkit.SoundCategory; | ||||
| import org.bukkit.WeatherType; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.event.Event; | ||||
| @@ -54,13 +49,12 @@ import org.bukkit.event.EventException; | ||||
| import org.bukkit.event.player.PlayerTeleportEvent; | ||||
| import org.bukkit.permissions.PermissionAttachmentInfo; | ||||
| import org.bukkit.plugin.RegisteredListener; | ||||
| import org.bukkit.potion.PotionEffectType; | ||||
| import org.checkerframework.checker.index.qual.NonNegative; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Arrays; | ||||
| import java.util.Set; | ||||
| import java.util.UUID; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| import static com.sk89q.worldedit.world.gamemode.GameModes.ADVENTURE; | ||||
| import static com.sk89q.worldedit.world.gamemode.GameModes.CREATIVE; | ||||
| @@ -74,24 +68,15 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|     private String name; | ||||
|  | ||||
|     /** | ||||
|      * <p>Please do not use this method. Instead use | ||||
|      * BukkitUtil.getPlayer(Player), as it caches player objects.</p> | ||||
|      * | ||||
|      * @param plotAreaManager   PlotAreaManager instance | ||||
|      * @param eventDispatcher   EventDispatcher instance | ||||
|      * @param player            Bukkit player instance | ||||
|      * @param permissionHandler PermissionHandler instance | ||||
|      */ | ||||
|     public BukkitPlayer( | ||||
|             final @NonNull PlotAreaManager plotAreaManager, final @NonNull EventDispatcher eventDispatcher, | ||||
|             final @NonNull Player player, final @NonNull PermissionHandler permissionHandler | ||||
|     ) { | ||||
|         this(plotAreaManager, eventDispatcher, player, false, permissionHandler); | ||||
|     } | ||||
|  | ||||
|     public BukkitPlayer( | ||||
|             final @NonNull PlotAreaManager plotAreaManager, final @NonNull | ||||
|             EventDispatcher eventDispatcher, final @NonNull Player player, | ||||
|     BukkitPlayer( | ||||
|             final @NonNull PlotAreaManager plotAreaManager, | ||||
|             final @NonNull EventDispatcher eventDispatcher, | ||||
|             final @NonNull Player player, | ||||
|             final boolean realPlayer, | ||||
|             final @NonNull PermissionHandler permissionHandler | ||||
|     ) { | ||||
| @@ -136,6 +121,9 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|  | ||||
|     @Override | ||||
|     public boolean canTeleport(final @NonNull Location location) { | ||||
|         if (!WorldUtil.isValidLocation(location)) { | ||||
|             return false; | ||||
|         } | ||||
|         final org.bukkit.Location to = BukkitUtil.adapt(location); | ||||
|         final org.bukkit.Location from = player.getLocation(); | ||||
|         PlayerTeleportEvent event = new PlayerTeleportEvent(player, from, to); | ||||
| @@ -174,6 +162,7 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|         } | ||||
|         final String[] nodes = stub.split("\\."); | ||||
|         final StringBuilder n = new StringBuilder(); | ||||
|         // Wildcard check from less specific permission to more specific permission | ||||
|         for (int i = 0; i < (nodes.length - 1); i++) { | ||||
|             n.append(nodes[i]).append("."); | ||||
|             if (!stub.equals(n + Permission.PERMISSION_STAR.toString())) { | ||||
| @@ -182,9 +171,11 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         // Wildcard check for the full permission | ||||
|         if (hasPermission(stub + ".*")) { | ||||
|             return Integer.MAX_VALUE; | ||||
|         } | ||||
|         // Permission value cache for iterative check | ||||
|         int max = 0; | ||||
|         if (CHECK_EFFECTIVE) { | ||||
|             boolean hasAny = false; | ||||
| @@ -192,6 +183,10 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|             final Set<PermissionAttachmentInfo> effective = player.getEffectivePermissions(); | ||||
|             if (!effective.isEmpty()) { | ||||
|                 for (PermissionAttachmentInfo attach : effective) { | ||||
|                     // Ignore all "false" permissions | ||||
|                     if (!attach.getValue()) { | ||||
|                         continue; | ||||
|                     } | ||||
|                     String permStr = attach.getPermission(); | ||||
|                     if (permStr.startsWith(stubPlus)) { | ||||
|                         hasAny = true; | ||||
| @@ -230,7 +225,7 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|  | ||||
|     @Override | ||||
|     public void teleport(final @NonNull Location location, final @NonNull TeleportCause cause) { | ||||
|         if (Math.abs(location.getX()) >= 30000000 || Math.abs(location.getZ()) >= 30000000) { | ||||
|         if (!WorldUtil.isValidLocation(location)) { | ||||
|             return; | ||||
|         } | ||||
|         final org.bukkit.Location bukkitLocation = | ||||
| @@ -318,19 +313,22 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|     @Override | ||||
|     public void playMusic(final @NonNull Location location, final @NonNull ItemType id) { | ||||
|         if (id == ItemTypes.AIR) { | ||||
|             // Let's just stop all the discs because why not? | ||||
|             for (final Sound sound : Arrays.stream(Sound.values()) | ||||
|                     .filter(sound -> sound.name().contains("DISC")).toList()) { | ||||
|                 player.stopSound(sound); | ||||
|             if (PlotSquared.platform().serverVersion()[1] >= 19) { | ||||
|                 player.stopSound(SoundCategory.MUSIC); | ||||
|                 return; | ||||
|             } | ||||
|             // this.player.playEffect(BukkitUtil.getLocation(location), Effect.RECORD_PLAY, Material.AIR); | ||||
|         } else { | ||||
|             // this.player.playEffect(BukkitUtil.getLocation(location), Effect.RECORD_PLAY, id.to(Material.class)); | ||||
|             this.player.playSound(BukkitUtil.adapt(location), | ||||
|                     Sound.valueOf(BukkitAdapter.adapt(id).name()), Float.MAX_VALUE, 1f | ||||
|             // 1.18 and downwards require a specific Sound to stop (even tho the packet does not??) | ||||
|             for (final Sound sound : Sound.values()) { | ||||
|                 if (sound.name().startsWith("MUSIC_DISC")) { | ||||
|                     this.player.stopSound(sound, SoundCategory.MUSIC); | ||||
|                 } | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         this.player.playSound(BukkitUtil.adapt(location), Sound.valueOf(BukkitAdapter.adapt(id).name()), | ||||
|                 SoundCategory.MUSIC, Float.MAX_VALUE, 1f | ||||
|         ); | ||||
|     } | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("deprecation") // Needed for Spigot compatibility | ||||
|     @Override | ||||
| @@ -355,6 +353,14 @@ public class BukkitPlayer extends PlotPlayer<Player> { | ||||
|         return BukkitUtil.BUKKIT_AUDIENCES.player(this.player); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void removeEffect(@NonNull String name) { | ||||
|         PotionEffectType type = PotionEffectType.getByName(name); | ||||
|         if (type != null) { | ||||
|             player.removePotionEffect(type); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean canSee(final PlotPlayer<?> other) { | ||||
|         if (other instanceof ConsolePlayer) { | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -77,7 +70,7 @@ public class BukkitPlayerManager extends PlayerManager<BukkitPlayer, Player> { | ||||
|         if (player == null || !player.isOnline()) { | ||||
|             throw new NoSuchPlayerException(uuid); | ||||
|         } | ||||
|         return new BukkitPlayer(this.plotAreaManager, this.eventDispatcher, player, this.permissionHandler); | ||||
|         return new BukkitPlayer(this.plotAreaManager, this.eventDispatcher, player, false, this.permissionHandler); | ||||
|     } | ||||
|  | ||||
|     @Nullable | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -37,6 +30,8 @@ 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.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.Chunk; | ||||
| import org.bukkit.plugin.Plugin; | ||||
| @@ -48,6 +43,8 @@ import java.util.LinkedList; | ||||
| import java.util.List; | ||||
| import java.util.Queue; | ||||
| import java.util.concurrent.LinkedBlockingQueue; | ||||
| import java.util.concurrent.TimeUnit; | ||||
| import java.util.concurrent.TimeoutException; | ||||
| import java.util.concurrent.atomic.AtomicInteger; | ||||
| import java.util.function.Consumer; | ||||
|  | ||||
| @@ -62,6 +59,8 @@ import java.util.function.Consumer; | ||||
|  **/ | ||||
| public final class BukkitChunkCoordinator extends ChunkCoordinator { | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + BukkitChunkCoordinator.class.getSimpleName()); | ||||
|  | ||||
|     private final List<ProgressSubscriber> progressSubscribers = new LinkedList<>(); | ||||
|  | ||||
|     private final Queue<BlockVector2> requestedChunks; | ||||
| @@ -76,10 +75,12 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | ||||
|     private final int totalSize; | ||||
|     private final AtomicInteger expectedSize; | ||||
|     private final AtomicInteger loadingChunks = new AtomicInteger(); | ||||
|     private final boolean forceSync; | ||||
|     private final boolean shouldGen; | ||||
|  | ||||
|     private int batchSize; | ||||
|     private PlotSquaredTask task; | ||||
|     private boolean shouldCancel; | ||||
|     private volatile boolean shouldCancel; | ||||
|     private boolean finished; | ||||
|  | ||||
|     @Inject | ||||
| @@ -91,8 +92,10 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | ||||
|             @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("unloadAfter") final boolean unloadAfter, | ||||
|             @Assisted final @NonNull Collection<ProgressSubscriber> progressSubscribers, | ||||
|             @Assisted("forceSync") final boolean forceSync, | ||||
|             @Assisted("shouldGen") final boolean shouldGen | ||||
|     ) { | ||||
|         this.requestedChunks = new LinkedBlockingQueue<>(requestedChunks); | ||||
|         this.availableChunks = new LinkedBlockingQueue<>(); | ||||
| @@ -107,14 +110,28 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | ||||
|         this.plugin = JavaPlugin.getPlugin(BukkitPlatform.class); | ||||
|         this.bukkitWorld = Bukkit.getWorld(world.getName()); | ||||
|         this.progressSubscribers.addAll(progressSubscribers); | ||||
|         this.forceSync = forceSync; | ||||
|         this.shouldGen = shouldGen; | ||||
|     } | ||||
|  | ||||
|     @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 | ||||
| @@ -131,7 +148,9 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | ||||
|             for (final ProgressSubscriber subscriber : this.progressSubscribers) { | ||||
|                 subscriber.notifyEnd(); | ||||
|             } | ||||
|             if (task != null) { | ||||
|                 task.cancel(); | ||||
|             } | ||||
|             finished = true; | ||||
|         } | ||||
|     } | ||||
| @@ -202,18 +221,28 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | ||||
|      * 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++) { | ||||
|         for (int i = 0; i < this.batchSize && this.requestedChunks.peek() != null; i++) { | ||||
|             // This required PaperLib to be bumped to version 1.0.4 to mark the request as urgent | ||||
|             final BlockVector2 chunk = this.requestedChunks.poll(); | ||||
|             loadingChunks.incrementAndGet(); | ||||
|             PaperLib | ||||
|                     .getChunkAtAsync(this.bukkitWorld, chunk.getX(), chunk.getZ(), true, true) | ||||
|                     .getChunkAtAsync(this.bukkitWorld, chunk.getX(), chunk.getZ(), shouldGen, true) | ||||
|                     .orTimeout(10L, TimeUnit.SECONDS) | ||||
|                     .whenComplete((chunkObject, throwable) -> { | ||||
|                         loadingChunks.decrementAndGet(); | ||||
|                         if (throwable != null) { | ||||
|                             throwable.printStackTrace(); | ||||
|                             if (throwable instanceof TimeoutException) { | ||||
|                                 LOGGER.warn("Timed out awaiting chunk load {}", chunk); | ||||
|                                 this.requestedChunks.offer(chunk); | ||||
|                             } else { | ||||
|                                 LOGGER.error("Failed to load chunk {}", chunk, throwable); | ||||
|                                 // We want one less because this couldn't be processed | ||||
|                                 this.expectedSize.decrementAndGet(); | ||||
|                             } | ||||
|                         } else if (chunkObject == null) { | ||||
|                             if (shouldGen) { | ||||
|                                 LOGGER.error("Null chunk returned for chunk at {}", chunk); | ||||
|                             } | ||||
|                         } else if (PlotSquared.get().isMainThread(Thread.currentThread())) { | ||||
|                             this.processChunk(chunkObject); | ||||
|                         } else { | ||||
| @@ -229,9 +258,11 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | ||||
|      * server's main thread. | ||||
|      */ | ||||
|     private void processChunk(final @NonNull Chunk chunk) { | ||||
|         /* Chunk#isLoaded does not necessarily return true shortly after PaperLib#getChunkAtAsync completes, but the chunk is | ||||
|         still loaded. | ||||
|         if (!chunk.isLoaded()) { | ||||
|             throw new IllegalArgumentException(String.format("Chunk %d;%d is is not loaded", chunk.getX(), chunk.getZ())); | ||||
|         } | ||||
|             throw new IllegalArgumentException(String.format("Chunk %d;%d is is not loaded", chunk.getX(), chunk.getZ()); | ||||
|         }*/ | ||||
|         if (finished) { | ||||
|             return; | ||||
|         } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -51,6 +44,7 @@ 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; | ||||
| @@ -58,14 +52,41 @@ import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collection; | ||||
| import java.util.Objects; | ||||
| import java.util.function.Consumer; | ||||
|  | ||||
| public class BukkitQueueCoordinator extends BasicQueueCoordinator { | ||||
|  | ||||
|     private final SideEffectSet noSideEffectSet; | ||||
|     private final SideEffectSet lightingSideEffectSet; | ||||
|     private final SideEffectSet edgeSideEffectSet; | ||||
|     private final SideEffectSet edgeLightingSideEffectSet; | ||||
|     private 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 = enableNetworkIfNeeded() | ||||
|                 .with(SideEffect.LIGHTING, SideEffect.State.OFF) | ||||
|                 .with(SideEffect.NEIGHBORS, SideEffect.State.OFF); | ||||
|         EDGE_SIDE_EFFECT_SET = NO_SIDE_EFFECT_SET | ||||
|                 .with(SideEffect.UPDATE, SideEffect.State.ON) | ||||
|                 .with(SideEffect.NEIGHBORS, SideEffect.State.ON); | ||||
|         LIGHTING_SIDE_EFFECT_SET = NO_SIDE_EFFECT_SET | ||||
|                 .with(SideEffect.NEIGHBORS, SideEffect.State.OFF); | ||||
|         EDGE_LIGHTING_SIDE_EFFECT_SET = NO_SIDE_EFFECT_SET | ||||
|                 .with(SideEffect.UPDATE, SideEffect.State.ON) | ||||
|                 .with(SideEffect.NEIGHBORS, SideEffect.State.ON); | ||||
|     } | ||||
|  | ||||
|     // make sure block changes are sent | ||||
|     private static SideEffectSet enableNetworkIfNeeded() { | ||||
|         SideEffect network; | ||||
|         try { | ||||
|             network = SideEffect.valueOf("NETWORK"); | ||||
|         } catch (IllegalArgumentException ignored) { | ||||
|             return SideEffectSet.none(); | ||||
|         } | ||||
|         return SideEffectSet.none().with(network, SideEffect.State.ON); | ||||
|     } | ||||
|  | ||||
|     private org.bukkit.World bukkitWorld; | ||||
|     @Inject | ||||
|     private ChunkCoordinatorBuilderFactory chunkCoordinatorBuilderFactory; | ||||
| @@ -76,19 +97,6 @@ public class BukkitQueueCoordinator extends BasicQueueCoordinator { | ||||
|     @Inject | ||||
|     public BukkitQueueCoordinator(@NonNull World world) { | ||||
|         super(world); | ||||
|         noSideEffectSet = SideEffectSet.none().with(SideEffect.LIGHTING, SideEffect.State.OFF).with( | ||||
|                 SideEffect.NEIGHBORS, | ||||
|                 SideEffect.State.OFF | ||||
|         ); | ||||
|         lightingSideEffectSet = SideEffectSet.none().with(SideEffect.NEIGHBORS, SideEffect.State.OFF); | ||||
|         edgeSideEffectSet = noSideEffectSet.with(SideEffect.UPDATE, SideEffect.State.ON).with( | ||||
|                 SideEffect.NEIGHBORS, | ||||
|                 SideEffect.State.ON | ||||
|         ); | ||||
|         edgeLightingSideEffectSet = noSideEffectSet.with(SideEffect.UPDATE, SideEffect.State.ON).with( | ||||
|                 SideEffect.NEIGHBORS, | ||||
|                 SideEffect.State.ON | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -201,10 +209,15 @@ public class BukkitQueueCoordinator extends BasicQueueCoordinator { | ||||
|                     localChunk.getTiles().forEach((blockVector3, tag) -> { | ||||
|                         try { | ||||
|                             BaseBlock block = getWorld().getBlock(blockVector3).toBaseBlock(tag); | ||||
|                             getWorld().setBlock(blockVector3, block, noSideEffectSet); | ||||
|                             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()); | ||||
|                             StateWrapper.INSTANCE.restore( | ||||
|                                     getWorld().getName(), | ||||
|                                     blockVector3.getX(), | ||||
|                                     blockVector3.getY(), | ||||
|                                     blockVector3.getZ(), | ||||
|                                     tag | ||||
|                             ); | ||||
|                         } | ||||
|                     }); | ||||
|                 } | ||||
| @@ -230,6 +243,8 @@ public class BukkitQueueCoordinator extends BasicQueueCoordinator { | ||||
|                         .withConsumer(consumer) | ||||
|                         .unloadAfter(isUnloadAfter()) | ||||
|                         .withProgressSubscribers(getProgressSubscribers()) | ||||
|                         .forceSync(isForceSync()) | ||||
|                         .shouldGen(isShouldGen()) | ||||
|                         .build(); | ||||
|         return super.enqueue(); | ||||
|     } | ||||
| @@ -258,15 +273,21 @@ public class BukkitQueueCoordinator extends BasicQueueCoordinator { | ||||
|             } | ||||
|             SideEffectSet sideEffectSet; | ||||
|             if (lighting) { | ||||
|                 sideEffectSet = edge ? edgeLightingSideEffectSet : lightingSideEffectSet; | ||||
|                 sideEffectSet = getSideEffectSet(edge ? SideEffectState.EDGE_LIGHTING : SideEffectState.LIGHTING); | ||||
|             } else { | ||||
|                 sideEffectSet = edge ? edgeSideEffectSet : noSideEffectSet; | ||||
|                 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 = getBukkitWorld().getBlockAt(x, y, z); | ||||
|             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; | ||||
| @@ -280,9 +301,7 @@ public class BukkitQueueCoordinator extends BasicQueueCoordinator { | ||||
|             existing.setBlockData(blockData, false); | ||||
|             if (block.hasNbtData()) { | ||||
|                 CompoundTag tag = block.getNbtData(); | ||||
|                 StateWrapper sw = new StateWrapper(tag); | ||||
|  | ||||
|                 sw.restoreTag(getWorld().getName(), existing.getX(), existing.getY(), existing.getZ()); | ||||
|                 StateWrapper.INSTANCE.restore(existing, Objects.requireNonNull(tag)); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| @@ -375,4 +394,23 @@ public class BukkitQueueCoordinator extends BasicQueueCoordinator { | ||||
|         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,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -26,12 +19,12 @@ | ||||
| package com.plotsquared.bukkit.queue; | ||||
|  | ||||
| import com.google.common.base.Preconditions; | ||||
| import com.intellectualsites.annotations.DoNotUse; | ||||
| import com.plotsquared.bukkit.util.BukkitBlockUtil; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.location.ChunkWrapper; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.queue.ScopedQueueCoordinator; | ||||
| import com.plotsquared.core.util.AnnotationHelper; | ||||
| import com.plotsquared.core.queue.ZeroedDelegateScopedQueueCoordinator; | ||||
| import com.plotsquared.core.util.ChunkUtil; | ||||
| import com.plotsquared.core.util.PatternUtil; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| @@ -51,8 +44,11 @@ import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| import java.util.Arrays; | ||||
|  | ||||
| @AnnotationHelper.ApiDescription(info = "Internal use only. Subject to changes at any time.") | ||||
| public class GenChunk extends ScopedQueueCoordinator { | ||||
| /** | ||||
|  * Internal use only. Subject to changes at any time. | ||||
|  */ | ||||
| @DoNotUse | ||||
| public class GenChunk extends ZeroedDelegateScopedQueueCoordinator { | ||||
|  | ||||
|     public final Biome[] biomes; | ||||
|     public BlockState[][] result; | ||||
| @@ -66,7 +62,6 @@ public class GenChunk extends ScopedQueueCoordinator { | ||||
|     /** | ||||
|      * @param minY minimum world Y, inclusive | ||||
|      * @param maxY maximum world Y, inclusive | ||||
|      * | ||||
|      * @since 6.6.0 | ||||
|      */ | ||||
|     public GenChunk(int minY, int maxY) { | ||||
| @@ -110,13 +105,13 @@ public class GenChunk extends ScopedQueueCoordinator { | ||||
|     /** | ||||
|      * Set the world and XZ of the chunk being represented via {@link ChunkWrapper} | ||||
|      * | ||||
|      * @param wrap P2 ChunkWrapper | ||||
|      * @param wrap PlotSquared ChunkWrapper | ||||
|      */ | ||||
|     public void setChunk(@NonNull ChunkWrapper wrap) { | ||||
|         chunk = null; | ||||
|         world = wrap.world; | ||||
|         chunkX = wrap.x; | ||||
|         chunkZ = wrap.z; | ||||
|         world = wrap.world(); | ||||
|         chunkX = wrap.x(); | ||||
|         chunkZ = wrap.z(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -190,7 +185,11 @@ public class GenChunk extends ScopedQueueCoordinator { | ||||
|  | ||||
|     @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)); | ||||
|         final BaseBlock block = PatternUtil.apply(Preconditions.checkNotNull( | ||||
|                 pattern, | ||||
|                 "Pattern may not be null" | ||||
|         ), x + (chunkX << 4), y, z + (chunkZ << 4)); | ||||
|         return setBlock(x, y, z, block); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|   | ||||
| @@ -0,0 +1,121 @@ | ||||
| /* | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.queue; | ||||
|  | ||||
| import com.plotsquared.bukkit.schematic.StateWrapper; | ||||
| import com.plotsquared.core.queue.DelegateQueueCoordinator; | ||||
| import com.sk89q.jnbt.CompoundTag; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.entity.Entity; | ||||
| import com.sk89q.worldedit.function.pattern.Pattern; | ||||
| import com.sk89q.worldedit.math.BlockVector3; | ||||
| import com.sk89q.worldedit.world.block.BaseBlock; | ||||
| import com.sk89q.worldedit.world.block.BlockState; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.bukkit.Location; | ||||
| import org.bukkit.entity.EntityType; | ||||
| import org.bukkit.generator.LimitedRegion; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Objects; | ||||
|  | ||||
| /** | ||||
|  * Wraps a {@link LimitedRegion} inside a {@link com.plotsquared.core.queue.QueueCoordinator} so it can be written to. | ||||
|  * | ||||
|  * @since 6.9.0 | ||||
|  */ | ||||
| public class LimitedRegionWrapperQueue extends DelegateQueueCoordinator { | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + LimitedRegionWrapperQueue.class.getSimpleName()); | ||||
|  | ||||
|     private final LimitedRegion limitedRegion; | ||||
|  | ||||
|     /** | ||||
|      * @since 6.9.0 | ||||
|      */ | ||||
|     public LimitedRegionWrapperQueue(LimitedRegion limitedRegion) { | ||||
|         super(null); | ||||
|         this.limitedRegion = limitedRegion; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean setBlock(final int x, final int y, final int z, @NonNull final Pattern pattern) { | ||||
|         return setBlock(x, y, z, pattern.applyBlock(BlockVector3.at(x, y, z))); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean setBlock(final int x, final int y, final int z, @NonNull final BaseBlock id) { | ||||
|         boolean result = setBlock(x, y, z, id.toImmutableState()); | ||||
|         if (result && id.hasNbtData()) { | ||||
|             CompoundTag tag = id.getNbtData(); | ||||
|             try { | ||||
|                 StateWrapper.INSTANCE.restore(limitedRegion.getBlockState(x, y, z).getBlock(), Objects.requireNonNull(tag)); | ||||
|             } catch (IllegalArgumentException e) { | ||||
|                 LOGGER.error("Error attempting to populate tile entity into the world at location {},{},{}", x, y, z, e); | ||||
|                 return false; | ||||
|             } | ||||
|         } | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean setBlock(final int x, final int y, final int z, @NonNull final BlockState id) { | ||||
|         try { | ||||
|             limitedRegion.setType(x, y, z, BukkitAdapter.adapt(id.getBlockType())); | ||||
|             limitedRegion.setBlockData(x, y, z, BukkitAdapter.adapt(id)); | ||||
|         } catch (IllegalArgumentException e) { | ||||
|             LOGGER.error("Error attempting to populate block into the world at location {},{},{}", x, y, z, e); | ||||
|             return false; | ||||
|         } | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean setEntity(@NonNull final Entity entity) { | ||||
|         EntityType type = BukkitAdapter.adapt(entity.getState().getType()); | ||||
|         double x = entity.getLocation().getX(); | ||||
|         double y = entity.getLocation().getY(); | ||||
|         double z = entity.getLocation().getZ(); | ||||
|         Location location = new Location(limitedRegion.getWorld(), x, y, z); | ||||
|         try { | ||||
|             limitedRegion.spawnEntity(location, type); | ||||
|         } catch (IllegalArgumentException e) { | ||||
|             LOGGER.error("Error attempting to populate entity into the world at location {},{},{}", (int) x, (int) y, (int) z, e); | ||||
|             return false; | ||||
|         } | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean setTile(final int x, final int y, final int z, @NonNull final CompoundTag tag) { | ||||
|         try { | ||||
|             return StateWrapper.INSTANCE.restore(limitedRegion.getBlockState(x, y, z).getBlock(), tag); | ||||
|         } catch (IllegalArgumentException e) { | ||||
|             LOGGER.error("Error attempting to populate tile entity into the world at location {},{},{}", x, y, z, e); | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean isSettingTiles() { | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -34,6 +27,8 @@ import com.plotsquared.core.util.WorldUtil; | ||||
| import com.sk89q.jnbt.CompoundTag; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Objects; | ||||
|  | ||||
| /** | ||||
|  * Schematic Handler. | ||||
|  */ | ||||
| @@ -46,8 +41,8 @@ public class BukkitSchematicHandler extends SchematicHandler { | ||||
|     } | ||||
|  | ||||
|     @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); | ||||
|     public boolean restoreTile(QueueCoordinator queue, CompoundTag tag, int x, int y, int z) { | ||||
|         return StateWrapper.INSTANCE.restore(Objects.requireNonNull(queue.getWorld()).getName(), x, y, z, tag); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -26,246 +19,34 @@ | ||||
| package com.plotsquared.bukkit.schematic; | ||||
|  | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.sk89q.jnbt.ByteTag; | ||||
| import com.sk89q.jnbt.CompoundTag; | ||||
| import com.sk89q.jnbt.ListTag; | ||||
| import com.sk89q.jnbt.ShortTag; | ||||
| import com.sk89q.jnbt.StringTag; | ||||
| import com.sk89q.jnbt.Tag; | ||||
| import com.sk89q.worldedit.blocks.BaseItemStack; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.world.item.ItemType; | ||||
| import org.bukkit.ChatColor; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.block.Container; | ||||
| import org.bukkit.block.Sign; | ||||
| import org.bukkit.enchantments.Enchantment; | ||||
| import org.bukkit.inventory.Inventory; | ||||
| import org.bukkit.inventory.InventoryHolder; | ||||
| import org.bukkit.inventory.ItemStack; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.jetbrains.annotations.ApiStatus; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.Map.Entry; | ||||
| @ApiStatus.Internal | ||||
| public sealed interface StateWrapper permits StateWrapperSpigot { | ||||
|  | ||||
| public class StateWrapper { | ||||
|     StateWrapper INSTANCE = Factory.createStateWrapper(); | ||||
|  | ||||
|     public org.bukkit.block.BlockState state = null; | ||||
|     public CompoundTag tag = null; | ||||
|     boolean restore(final @NonNull Block block, final @NonNull CompoundTag data); | ||||
|  | ||||
|     public StateWrapper(org.bukkit.block.BlockState state) { | ||||
|         this.state = state; | ||||
|     } | ||||
|  | ||||
|     public StateWrapper(CompoundTag tag) { | ||||
|         this.tag = tag; | ||||
|     } | ||||
|  | ||||
|     public static String jsonToColourCode(String str) { | ||||
|         str = str.replace("{\"extra\":", "").replace("],\"text\":\"\"}", "]") | ||||
|                 .replace("[{\"color\":\"black\",\"text\":\"", "&0") | ||||
|                 .replace("[{\"color\":\"dark_blue\",\"text\":\"", "&1") | ||||
|                 .replace("[{\"color\":\"dark_green\",\"text\":\"", "&2") | ||||
|                 .replace("[{\"color\":\"dark_aqua\",\"text\":\"", "&3") | ||||
|                 .replace("[{\"color\":\"dark_red\",\"text\":\"", "&4") | ||||
|                 .replace("[{\"color\":\"dark_purple\",\"text\":\"", "&5") | ||||
|                 .replace("[{\"color\":\"gold\",\"text\":\"", "&6") | ||||
|                 .replace("[{\"color\":\"gray\",\"text\":\"", "&7") | ||||
|                 .replace("[{\"color\":\"dark_gray\",\"text\":\"", "&8") | ||||
|                 .replace("[{\"color\":\"blue\",\"text\":\"", "&9") | ||||
|                 .replace("[{\"color\":\"green\",\"text\":\"", "&a") | ||||
|                 .replace("[{\"color\":\"aqua\",\"text\":\"", "&b") | ||||
|                 .replace("[{\"color\":\"red\",\"text\":\"", "&c") | ||||
|                 .replace("[{\"color\":\"light_purple\",\"text\":\"", "&d") | ||||
|                 .replace("[{\"color\":\"yellow\",\"text\":\"", "&e") | ||||
|                 .replace("[{\"color\":\"white\",\"text\":\"", "&f") | ||||
|                 .replace("[{\"obfuscated\":true,\"text\":\"", "&k") | ||||
|                 .replace("[{\"bold\":true,\"text\":\"", "&l") | ||||
|                 .replace("[{\"strikethrough\":true,\"text\":\"", "&m") | ||||
|                 .replace("[{\"underlined\":true,\"text\":\"", "&n") | ||||
|                 .replace("[{\"italic\":true,\"text\":\"", "&o").replace("[{\"color\":\"black\",", "&0") | ||||
|                 .replace("[{\"color\":\"dark_blue\",", "&1") | ||||
|                 .replace("[{\"color\":\"dark_green\",", "&2") | ||||
|                 .replace("[{\"color\":\"dark_aqua\",", "&3").replace("[{\"color\":\"dark_red\",", "&4") | ||||
|                 .replace("[{\"color\":\"dark_purple\",", "&5").replace("[{\"color\":\"gold\",", "&6") | ||||
|                 .replace("[{\"color\":\"gray\",", "&7").replace("[{\"color\":\"dark_gray\",", "&8") | ||||
|                 .replace("[{\"color\":\"blue\",", "&9").replace("[{\"color\":\"green\",", "&a") | ||||
|                 .replace("[{\"color\":\"aqua\",", "&b").replace("[{\"color\":\"red\",", "&c") | ||||
|                 .replace("[{\"color\":\"light_purple\",", "&d").replace("[{\"color\":\"yellow\",", "&e") | ||||
|                 .replace("[{\"color\":\"white\",", "&f").replace("[{\"obfuscated\":true,", "&k") | ||||
|                 .replace("[{\"bold\":true,", "&l").replace("[{\"strikethrough\":true,", "&m") | ||||
|                 .replace("[{\"underlined\":true,", "&n").replace("[{\"italic\":true,", "&o") | ||||
|                 .replace("{\"color\":\"black\",\"text\":\"", "&0") | ||||
|                 .replace("{\"color\":\"dark_blue\",\"text\":\"", "&1") | ||||
|                 .replace("{\"color\":\"dark_green\",\"text\":\"", "&2") | ||||
|                 .replace("{\"color\":\"dark_aqua\",\"text\":\"", "&3") | ||||
|                 .replace("{\"color\":\"dark_red\",\"text\":\"", "&4") | ||||
|                 .replace("{\"color\":\"dark_purple\",\"text\":\"", "&5") | ||||
|                 .replace("{\"color\":\"gold\",\"text\":\"", "&6") | ||||
|                 .replace("{\"color\":\"gray\",\"text\":\"", "&7") | ||||
|                 .replace("{\"color\":\"dark_gray\",\"text\":\"", "&8") | ||||
|                 .replace("{\"color\":\"blue\",\"text\":\"", "&9") | ||||
|                 .replace("{\"color\":\"green\",\"text\":\"", "&a") | ||||
|                 .replace("{\"color\":\"aqua\",\"text\":\"", "&b") | ||||
|                 .replace("{\"color\":\"red\",\"text\":\"", "&c") | ||||
|                 .replace("{\"color\":\"light_purple\",\"text\":\"", "&d") | ||||
|                 .replace("{\"color\":\"yellow\",\"text\":\"", "&e") | ||||
|                 .replace("{\"color\":\"white\",\"text\":\"", "&f") | ||||
|                 .replace("{\"obfuscated\":true,\"text\":\"", "&k") | ||||
|                 .replace("{\"bold\":true,\"text\":\"", "&l") | ||||
|                 .replace("{\"strikethrough\":true,\"text\":\"", "&m") | ||||
|                 .replace("{\"underlined\":true,\"text\":\"", "&n") | ||||
|                 .replace("{\"italic\":true,\"text\":\"", "&o").replace("{\"color\":\"black\",", "&0") | ||||
|                 .replace("{\"color\":\"dark_blue\",", "&1").replace("{\"color\":\"dark_green\",", "&2") | ||||
|                 .replace("{\"color\":\"dark_aqua\",", "&3").replace("{\"color\":\"dark_red\",", "&4") | ||||
|                 .replace("{\"color\":\"dark_purple\",", "&5").replace("{\"color\":\"gold\",", "&6") | ||||
|                 .replace("{\"color\":\"gray\",", "&7").replace("{\"color\":\"dark_gray\",", "&8") | ||||
|                 .replace("{\"color\":\"blue\",", "&9").replace("{\"color\":\"green\",", "&a") | ||||
|                 .replace("{\"color\":\"aqua\",", "&b").replace("{\"color\":\"red\",", "&c") | ||||
|                 .replace("{\"color\":\"light_purple\",", "&d").replace("{\"color\":\"yellow\",", "&e") | ||||
|                 .replace("{\"color\":\"white\",", "&f").replace("{\"obfuscated\":true,", "&k") | ||||
|                 .replace("{\"bold\":true,", "&l").replace("{\"strikethrough\":true,", "&m") | ||||
|                 .replace("{\"underlined\":true,", "&n").replace("{\"italic\":true,", "&o") | ||||
|                 .replace("\"color\":\"black\",\"text\":\"", "&0") | ||||
|                 .replace("\"color\":\"dark_blue\",\"text\":\"", "&1") | ||||
|                 .replace("\"color\":\"dark_green\",\"text\":\"", "&2") | ||||
|                 .replace("\"color\":\"dark_aqua\",\"text\":\"", "&3") | ||||
|                 .replace("\"color\":\"dark_red\",\"text\":\"", "&4") | ||||
|                 .replace("\"color\":\"dark_purple\",\"text\":\"", "&5") | ||||
|                 .replace("\"color\":\"gold\",\"text\":\"", "&6") | ||||
|                 .replace("\"color\":\"gray\",\"text\":\"", "&7") | ||||
|                 .replace("\"color\":\"dark_gray\",\"text\":\"", "&8") | ||||
|                 .replace("\"color\":\"blue\",\"text\":\"", "&9") | ||||
|                 .replace("\"color\":\"green\",\"text\":\"", "&a") | ||||
|                 .replace("\"color\":\"aqua\",\"text\":\"", "&b") | ||||
|                 .replace("\"color\":\"red\",\"text\":\"", "&c") | ||||
|                 .replace("\"color\":\"light_purple\",\"text\":\"", "&d") | ||||
|                 .replace("\"color\":\"yellow\",\"text\":\"", "&e") | ||||
|                 .replace("\"color\":\"white\",\"text\":\"", "&f") | ||||
|                 .replace("\"obfuscated\":true,\"text\":\"", "&k") | ||||
|                 .replace("\"bold\":true,\"text\":\"", "&l") | ||||
|                 .replace("\"strikethrough\":true,\"text\":\"", "&m") | ||||
|                 .replace("\"underlined\":true,\"text\":\"", "&n") | ||||
|                 .replace("\"italic\":true,\"text\":\"", "&o").replace("\"color\":\"black\",", "&0") | ||||
|                 .replace("\"color\":\"dark_blue\",", "&1").replace("\"color\":\"dark_green\",", "&2") | ||||
|                 .replace("\"color\":\"dark_aqua\",", "&3").replace("\"color\":\"dark_red\",", "&4") | ||||
|                 .replace("\"color\":\"dark_purple\",", "&5").replace("\"color\":\"gold\",", "&6") | ||||
|                 .replace("\"color\":\"gray\",", "&7").replace("\"color\":\"dark_gray\",", "&8") | ||||
|                 .replace("\"color\":\"blue\",", "&9").replace("\"color\":\"green\",", "&a") | ||||
|                 .replace("\"color\":\"aqua\",", "&b").replace("\"color\":\"red\",", "&c") | ||||
|                 .replace("\"color\":\"light_purple\",", "&d").replace("\"color\":\"yellow\",", "&e") | ||||
|                 .replace("\"color\":\"white\",", "&f").replace("\"obfuscated\":true,", "&k") | ||||
|                 .replace("\"bold\":true,", "&l").replace("\"strikethrough\":true,", "&m") | ||||
|                 .replace("\"underlined\":true,", "&n").replace("\"italic\":true,", "&o") | ||||
|                 .replace("[{\"text\":\"", "&0").replace("{\"text\":\"", "&0").replace("\"},", "") | ||||
|                 .replace("\"}]", "").replace("\"}", ""); | ||||
|         str = ChatColor.translateAlternateColorCodes('&', str); | ||||
|         return str; | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("deprecation") // #setLine is needed for Spigot compatibility | ||||
|     public boolean restoreTag(String worldName, int x, int y, int z) { | ||||
|         if (this.tag == null) { | ||||
|     default boolean restore(final String worldName, final int x, final int y, final int z, final CompoundTag data) { | ||||
|         final World world = BukkitUtil.getWorld(worldName); | ||||
|         if (world == null) { | ||||
|             return false; | ||||
|         } | ||||
|         World world = BukkitUtil.getWorld(worldName); | ||||
|         Block block = world.getBlockAt(x, y, z); | ||||
|         if (block == null) { | ||||
|             return false; | ||||
|         } | ||||
|         org.bukkit.block.BlockState state = block.getState(); | ||||
|         switch (getId()) { | ||||
|             case "chest", "beacon", "brewingstand", "dispenser", "dropper", "furnace", "hopper", "shulkerbox" -> { | ||||
|                 if (!(state instanceof Container container)) { | ||||
|                     return false; | ||||
|                 } | ||||
|                 List<Tag> itemsTag = this.tag.getListTag("Items").getValue(); | ||||
|                 Inventory inv = container.getSnapshotInventory(); | ||||
|                 for (Tag itemTag : itemsTag) { | ||||
|                     CompoundTag itemComp = (CompoundTag) itemTag; | ||||
|                     ItemType type = ItemType.REGISTRY.get(itemComp.getString("id").toLowerCase()); | ||||
|                     if (type == null) { | ||||
|                         continue; | ||||
|                     } | ||||
|                     int count = itemComp.getByte("Count"); | ||||
|                     int slot = itemComp.getByte("Slot"); | ||||
|                     CompoundTag tag = (CompoundTag) itemComp.getValue().get("tag"); | ||||
|                     BaseItemStack baseItemStack = new BaseItemStack(type, tag, count); | ||||
|                     ItemStack itemStack = BukkitAdapter.adapt(baseItemStack); | ||||
|                     inv.setItem(slot, itemStack); | ||||
|                 } | ||||
|                 container.update(true, false); | ||||
|                 return true; | ||||
|             } | ||||
|             case "sign" -> { | ||||
|                 if (state instanceof Sign sign) { | ||||
|                     sign.setLine(0, jsonToColourCode(tag.getString("Text1"))); | ||||
|                     sign.setLine(1, jsonToColourCode(tag.getString("Text2"))); | ||||
|                     sign.setLine(2, jsonToColourCode(tag.getString("Text3"))); | ||||
|                     sign.setLine(3, jsonToColourCode(tag.getString("Text4"))); | ||||
|                     state.update(true); | ||||
|                     return true; | ||||
|                 } | ||||
|                 return false; | ||||
|             } | ||||
|         } | ||||
|         return false; | ||||
|         return this.restore(world.getBlockAt(x, y, z), data); | ||||
|     } | ||||
|  | ||||
|     public CompoundTag getTag() { | ||||
|         if (this.tag != null) { | ||||
|             return this.tag; | ||||
|         } | ||||
|         if (this.state instanceof InventoryHolder inv) { | ||||
|             ItemStack[] contents = inv.getInventory().getContents(); | ||||
|             Map<String, Tag> values = new HashMap<>(); | ||||
|             values.put("Items", new ListTag(CompoundTag.class, serializeInventory(contents))); | ||||
|             return new CompoundTag(values); | ||||
|         } | ||||
|         return null; | ||||
|     } | ||||
|     @ApiStatus.Internal | ||||
|     final class Factory { | ||||
|  | ||||
|         private static StateWrapper createStateWrapper() { | ||||
|             return new StateWrapperSpigot(); | ||||
|         } | ||||
|  | ||||
|     public String getId() { | ||||
|         String tileid = this.tag.getString("id").toLowerCase(); | ||||
|         if (tileid.startsWith("minecraft:")) { | ||||
|             tileid = tileid.replace("minecraft:", ""); | ||||
|         } | ||||
|         return tileid; | ||||
|     } | ||||
|  | ||||
|     public List<CompoundTag> serializeInventory(ItemStack[] items) { | ||||
|         List<CompoundTag> tags = new ArrayList<>(); | ||||
|         for (int i = 0; i < items.length; ++i) { | ||||
|             if (items[i] != null) { | ||||
|                 Map<String, Tag> tagData = serializeItem(items[i]); | ||||
|                 tagData.put("Slot", new ByteTag((byte) i)); | ||||
|                 tags.add(new CompoundTag(tagData)); | ||||
|             } | ||||
|         } | ||||
|         return tags; | ||||
|     } | ||||
|  | ||||
|     public Map<String, Tag> serializeItem(ItemStack item) { | ||||
|         Map<String, Tag> data = new HashMap<>(); | ||||
|         data.put("id", new StringTag(item.getType().name())); | ||||
|         data.put("Damage", new ShortTag(item.getDurability())); | ||||
|         data.put("Count", new ByteTag((byte) item.getAmount())); | ||||
|         if (!item.getEnchantments().isEmpty()) { | ||||
|             List<CompoundTag> enchantmentList = new ArrayList<>(); | ||||
|             for (Entry<Enchantment, Integer> entry : item.getEnchantments().entrySet()) { | ||||
|                 Map<String, Tag> enchantment = new HashMap<>(); | ||||
|                 enchantment.put("id", new StringTag(entry.getKey().toString())); | ||||
|                 enchantment.put("lvl", new ShortTag(entry.getValue().shortValue())); | ||||
|                 enchantmentList.add(new CompoundTag(enchantment)); | ||||
|             } | ||||
|             Map<String, Tag> auxData = new HashMap<>(); | ||||
|             auxData.put("ench", new ListTag(CompoundTag.class, enchantmentList)); | ||||
|             data.put("tag", new CompoundTag(auxData)); | ||||
|         } | ||||
|         return data; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,263 @@ | ||||
| /* | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.schematic; | ||||
|  | ||||
| import com.plotsquared.core.util.ReflectionHelper; | ||||
| import com.plotsquared.core.util.ReflectionUtils; | ||||
| import com.sk89q.jnbt.CompoundTag; | ||||
| import com.sk89q.worldedit.bukkit.WorldEditPlugin; | ||||
| import com.sk89q.worldedit.bukkit.adapter.BukkitImplAdapter; | ||||
| import com.sk89q.worldedit.bukkit.adapter.Refraction; | ||||
| import com.sk89q.worldedit.extension.platform.NoCapablePlatformException; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.block.BlockState; | ||||
| import org.bukkit.block.Sign; | ||||
| import org.bukkit.block.sign.Side; | ||||
| import org.bukkit.block.sign.SignSide; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.lang.invoke.MethodHandle; | ||||
| import java.lang.invoke.MethodHandles; | ||||
| import java.lang.invoke.MethodType; | ||||
| import java.lang.reflect.Field; | ||||
| import java.lang.reflect.Method; | ||||
| import java.lang.reflect.Modifier; | ||||
| import java.util.Arrays; | ||||
| import java.util.Optional; | ||||
|  | ||||
| final class StateWrapperSpigot implements StateWrapper { | ||||
|  | ||||
|     private static final boolean FORCE_UPDATE_STATE = true; | ||||
|     private static final boolean UPDATE_TRIGGER_PHYSICS = false; | ||||
|     private static final String CRAFTBUKKIT_PACKAGE = Bukkit.getServer().getClass().getPackageName(); | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + StateWrapperSpigot.class.getSimpleName()); | ||||
|     private static final MethodHandles.Lookup LOOKUP = MethodHandles.lookup(); | ||||
|  | ||||
|     private static BukkitImplAdapter ADAPTER = null; | ||||
|     private static Class<?> LIN_TAG_CLASS = null; | ||||
|     private static Class<?> CRAFT_BLOCK_ENTITY_STATE_CLASS = null; | ||||
|     private static Field CRAFT_SIGN_SIDE_SIGN_TEXT = null; | ||||
|     private static Field CRAFT_SIGN_SIDE_LINES = null; | ||||
|     private static MethodHandle PAPERWEIGHT_ADAPTER_FROM_NATIVE = null; | ||||
|     private static MethodHandle CRAFT_BLOCK_ENTITY_STATE_LOAD_DATA = null; | ||||
|     private static MethodHandle CRAFT_BLOCK_ENTITY_STATE_UPDATE = null; | ||||
|     private static MethodHandle CRAFT_BLOCK_ENTITY_STATE_GET_SNAPSHOT = null; | ||||
|     private static MethodHandle SIGN_BLOCK_ENTITY_SET_TEXT = null; | ||||
|     private static MethodHandle DECODER_PARSE = null; | ||||
|     private static MethodHandle DATA_RESULT_RESULT = null; | ||||
|     private static MethodHandle TO_LIN_TAG = null; | ||||
|  | ||||
|     private static Object SIGN_TEXT_DIRECT_CODEC = null; | ||||
|     private static Object NBT_OPS_INSTANCE = null; | ||||
|  | ||||
|     public StateWrapperSpigot() { | ||||
|         try { | ||||
|             ReflectionUtils.RefClass worldEditPluginRefClass = ReflectionUtils.getRefClass(WorldEditPlugin.class); | ||||
|             WorldEditPlugin worldEditPlugin = (WorldEditPlugin) worldEditPluginRefClass | ||||
|                     .getMethod("getInstance") | ||||
|                     .of(null) | ||||
|                     .call(); | ||||
|             ADAPTER = (BukkitImplAdapter) worldEditPluginRefClass | ||||
|                     .getMethod("getBukkitImplAdapter") | ||||
|                     .of(worldEditPlugin) | ||||
|                     .call(); | ||||
|             LIN_TAG_CLASS = Class.forName("org.enginehub.linbus.tree.LinTag"); // provided WE / FAWE version is too old | ||||
|             PAPERWEIGHT_ADAPTER_FROM_NATIVE = findPaperweightAdapterFromNativeMethodHandle(ADAPTER.getClass()); | ||||
|             TO_LIN_TAG = findToLinTagMethodHandle(); | ||||
|         } catch (NoSuchMethodException | ClassNotFoundException | IllegalAccessException | NoCapablePlatformException e) { | ||||
|             throw new RuntimeException("Failed to access required WorldEdit classes or methods", e); | ||||
|         } | ||||
|         try { | ||||
|             final Class<?> SIGN_TEXT_CLASS = Class.forName("net.minecraft.world.level.block.entity.SignText"); | ||||
|             final Class<?> CRAFT_SIGN_SIDE_CLASS = Class.forName(CRAFTBUKKIT_PACKAGE + ".block.sign.CraftSignSide"); | ||||
|             CRAFT_SIGN_SIDE_SIGN_TEXT = CRAFT_SIGN_SIDE_CLASS.getDeclaredField("signText"); | ||||
|             CRAFT_SIGN_SIDE_SIGN_TEXT.setAccessible(true); | ||||
|             CRAFT_SIGN_SIDE_LINES = CRAFT_SIGN_SIDE_CLASS.getDeclaredField("lines"); | ||||
|             CRAFT_SIGN_SIDE_LINES.setAccessible(true); | ||||
|             CRAFT_BLOCK_ENTITY_STATE_CLASS = Class.forName(CRAFTBUKKIT_PACKAGE + ".block.CraftBlockEntityState"); | ||||
|             CRAFT_BLOCK_ENTITY_STATE_LOAD_DATA = findCraftBlockEntityStateLoadDataMethodHandle(CRAFT_BLOCK_ENTITY_STATE_CLASS); | ||||
|             CRAFT_BLOCK_ENTITY_STATE_UPDATE = findCraftBlockEntityStateUpdateMethodHandle(CRAFT_BLOCK_ENTITY_STATE_CLASS); | ||||
|             CRAFT_BLOCK_ENTITY_STATE_GET_SNAPSHOT = findCraftBlockEntityStateSnapshotMethodHandle(CRAFT_BLOCK_ENTITY_STATE_CLASS); | ||||
|             SIGN_BLOCK_ENTITY_SET_TEXT = findSignBlockEntitySetTextMethodHandle( | ||||
|                     Class.forName(Refraction.pickName( | ||||
|                             "net.minecraft.world.level.block.entity.SignBlockEntity", | ||||
|                             "net.minecraft.world.level.block.entity.TileEntitySign" | ||||
|                     )), | ||||
|                     SIGN_TEXT_CLASS | ||||
|             ); | ||||
|             final Class<?> CODEC_CLASS = Class.forName("com.mojang.serialization.Codec"); | ||||
|             final Class<?> DECODER_CLASS = Class.forName("com.mojang.serialization.Decoder"); | ||||
|             final Class<?> DATA_RESULT_CLASS = Class.forName("com.mojang.serialization.DataResult"); | ||||
|             final Class<?> DYNAMIC_OPS_CLASS = Class.forName("com.mojang.serialization.DynamicOps"); | ||||
|             final Class<?> NBT_OPS_CLASS = Class.forName(Refraction.pickName( | ||||
|                     "net.minecraft.nbt.NbtOps", | ||||
|                     "net.minecraft.nbt.DynamicOpsNBT" | ||||
|             )); | ||||
|             SIGN_TEXT_DIRECT_CODEC = Arrays.stream(SIGN_TEXT_CLASS.getFields()) | ||||
|                     .filter(field -> Modifier.isStatic(field.getModifiers()) && Modifier.isPublic(field.getModifiers())) | ||||
|                     .filter(field -> field.getType() == CODEC_CLASS) | ||||
|                     .findFirst().orElseThrow().get(null); | ||||
|             DECODER_PARSE = LOOKUP.findVirtual( | ||||
|                     DECODER_CLASS, "parse", MethodType.methodType( | ||||
|                             DATA_RESULT_CLASS, DYNAMIC_OPS_CLASS, Object.class | ||||
|                     ) | ||||
|             ); | ||||
|             NBT_OPS_INSTANCE = Arrays.stream(NBT_OPS_CLASS.getFields()) | ||||
|                     .filter(field -> Modifier.isStatic(field.getModifiers()) && Modifier.isPublic(field.getModifiers())) | ||||
|                     .filter(field -> field.getType() == NBT_OPS_CLASS) | ||||
|                     .findFirst().orElseThrow().get(null); | ||||
|             DATA_RESULT_RESULT = LOOKUP.findVirtual( | ||||
|                     DATA_RESULT_CLASS, "result", | ||||
|                     MethodType.methodType(Optional.class) | ||||
|             ); | ||||
|         } catch (Throwable e) { | ||||
|             throw new RuntimeException("Failed to initialize required native method accessors", e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean restore(final @NonNull Block block, final @NonNull CompoundTag data) { | ||||
|         try { | ||||
|             final BlockState blockState = block.getState(); | ||||
|             if (!CRAFT_BLOCK_ENTITY_STATE_CLASS.isAssignableFrom(blockState.getClass())) { | ||||
|                 return false; | ||||
|             } | ||||
|             // get native tag | ||||
|             Object nativeTag = PAPERWEIGHT_ADAPTER_FROM_NATIVE.invoke(ADAPTER, TO_LIN_TAG.invoke(data)); | ||||
|             // load block entity data | ||||
|             CRAFT_BLOCK_ENTITY_STATE_LOAD_DATA.invoke(blockState, nativeTag); | ||||
|  | ||||
|             // signs need to be handled explicitly (at least during worldgen) | ||||
|             if (blockState instanceof Sign sign) { | ||||
|                 if (data.getValue().get("front_text") instanceof CompoundTag textTag) { | ||||
|                     setSignContents(true, sign.getSide(Side.FRONT), blockState, textTag); | ||||
|                 } | ||||
|                 if (data.getValue().get("back_text") instanceof CompoundTag textTag) { | ||||
|                     setSignContents(false, sign.getSide(Side.BACK), blockState, textTag); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             CRAFT_BLOCK_ENTITY_STATE_UPDATE.invoke(blockState, FORCE_UPDATE_STATE, UPDATE_TRIGGER_PHYSICS); | ||||
|         } catch (Throwable e) { | ||||
|             LOGGER.error("Failed to update tile entity", e); | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     private static void setSignContents(boolean front, SignSide side, BlockState blockState, CompoundTag data) throws Throwable { | ||||
|         Object nativeTag = PAPERWEIGHT_ADAPTER_FROM_NATIVE.invoke(ADAPTER, TO_LIN_TAG.invoke(data)); | ||||
|         Object dataResult = DECODER_PARSE.invoke(SIGN_TEXT_DIRECT_CODEC, NBT_OPS_INSTANCE, nativeTag); | ||||
|         //noinspection rawtypes | ||||
|         Object signText = ((Optional) DATA_RESULT_RESULT.invoke(dataResult)).orElseThrow(); | ||||
|  | ||||
|         // set the SignText on the underlying tile entity snapshot (SignBlockEntity) | ||||
|         SIGN_BLOCK_ENTITY_SET_TEXT.invoke(CRAFT_BLOCK_ENTITY_STATE_GET_SNAPSHOT.invoke(blockState), signText, front); | ||||
|         // and update the SignText field on the CraftSignSide - changes are otherwise not reflected | ||||
|         CRAFT_SIGN_SIDE_SIGN_TEXT.set(side, signText); | ||||
|  | ||||
|         // reset cached lines to null, so it can be re-retrieved from SignText (for API access etc.) | ||||
|         CRAFT_SIGN_SIDE_LINES.set(side, null); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Finds the {@code toLinTag} method on the {@code ToLinTag} interface, if lin-bus is available in the classpath. | ||||
|      * <br /> | ||||
|      * Required to access the underlying lin tag of the used JNBT tag by PlotSquared, so it can be converted into the platforms | ||||
|      * native tag later. | ||||
|      * | ||||
|      * @return the MethodHandle for {@code toLinTag}, or {@code null} if lin-bus is not available in the classpath. | ||||
|      * @throws ClassNotFoundException if the {@code ToLinTag} class could not be found. | ||||
|      * @throws NoSuchMethodException  if no {@code toLinTag} method exists. | ||||
|      * @throws IllegalAccessException shouldn't happen. | ||||
|      */ | ||||
|     private static MethodHandle findToLinTagMethodHandle() throws ClassNotFoundException, | ||||
|             NoSuchMethodException, IllegalAccessException { | ||||
|         return LOOKUP.findVirtual( | ||||
|                 Class.forName("org.enginehub.linbus.tree.ToLinTag"), | ||||
|                 "toLinTag", | ||||
|                 MethodType.methodType(LIN_TAG_CLASS) | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Find the method (handle) to convert from native (= WE/FAWE) NBT tags to minecraft NBT tags. | ||||
|      * <br /> | ||||
|      * Depending on the used version of WE/FAWE, this differs: | ||||
|      * <ul> | ||||
|      *     <li>On WE versions post LinBus introduction: {@code fromNative(org.enginehub.linbus.tree.LinTag)}</li> | ||||
|      *     <li>On FAWE versions post LinBus introduction: {@code fromNativeLin(org.enginehub.linbus.tree.LinTag)}</li> | ||||
|      * </ul> | ||||
|      * | ||||
|      * @param adapterClass The bukkit adapter implementation class | ||||
|      * @return the method. | ||||
|      * @throws IllegalAccessException shouldn't happen as private lookup is used. | ||||
|      * @throws NoSuchMethodException  if the method couldn't be found. | ||||
|      */ | ||||
|     private static MethodHandle findPaperweightAdapterFromNativeMethodHandle(Class<?> adapterClass) throws | ||||
|             IllegalAccessException, NoSuchMethodException { | ||||
|         final MethodHandles.Lookup lookup = MethodHandles.privateLookupIn(adapterClass, LOOKUP); | ||||
|         try { | ||||
|             // FAWE | ||||
|             return lookup.findVirtual(adapterClass, "fromNativeLin", MethodType.methodType(Object.class, LIN_TAG_CLASS)); | ||||
|         } catch (NoSuchMethodException e) { | ||||
|             // WE | ||||
|             return lookup.findVirtual(adapterClass, "fromNative", MethodType.methodType(Object.class, LIN_TAG_CLASS)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private static MethodHandle findCraftBlockEntityStateLoadDataMethodHandle(Class<?> craftBlockEntityStateClass) throws | ||||
|             NoSuchMethodException, IllegalAccessException { | ||||
|         for (final Method method : craftBlockEntityStateClass.getMethods()) { | ||||
|             if (method.getName().equals("loadData") && method.getParameterCount() == 1) { | ||||
|                 return LOOKUP.unreflect(method); | ||||
|             } | ||||
|         } | ||||
|         throw new NoSuchMethodException("Couldn't find #loadData(CompoundTag) in " + craftBlockEntityStateClass.getName()); | ||||
|     } | ||||
|  | ||||
|     private static MethodHandle findCraftBlockEntityStateUpdateMethodHandle(Class<?> craftBlockEntityStateClass) throws | ||||
|             NoSuchMethodException, IllegalAccessException { | ||||
|         return LOOKUP.unreflect(ReflectionHelper.findMethod( | ||||
|                 craftBlockEntityStateClass, | ||||
|                 MethodType.methodType(Boolean.TYPE, Boolean.TYPE, Boolean.TYPE), | ||||
|                 Modifier.PUBLIC | ||||
|         ).orElseThrow(() -> new NoSuchMethodException("Couldn't lookup CraftBlockEntityState#update(boolean, boolean) boolean"))); | ||||
|     } | ||||
|  | ||||
|     private static MethodHandle findCraftBlockEntityStateSnapshotMethodHandle(Class<?> craftBlockEntityStateClass) throws | ||||
|             IllegalAccessException, NoSuchMethodException { | ||||
|         // doesn't seem to be obfuscated, but protected | ||||
|         final MethodHandles.Lookup lookup = MethodHandles.privateLookupIn(craftBlockEntityStateClass, LOOKUP); | ||||
|         return lookup.unreflect(craftBlockEntityStateClass.getDeclaredMethod("getSnapshot")); | ||||
|     } | ||||
|  | ||||
|     private static MethodHandle findSignBlockEntitySetTextMethodHandle(Class<?> signBlockEntity, Class<?> signText) throws | ||||
|             NoSuchMethodException, IllegalAccessException { | ||||
|         return LOOKUP.unreflect(ReflectionHelper.findMethod( | ||||
|                 signBlockEntity, | ||||
|                 MethodType.methodType(Boolean.TYPE, signText, Boolean.TYPE), | ||||
|                 Modifier.PUBLIC | ||||
|         ).orElseThrow(() -> new NoSuchMethodException("Couldn't lookup SignBlockEntity#setText(SignText, boolean) boolean"))); | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -47,14 +40,16 @@ import com.plotsquared.core.plot.flag.implementations.PvpFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.TamedAttackFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.VehicleCapFlag; | ||||
| import com.plotsquared.core.util.EntityUtil; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.entity.EntityCategories; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import net.kyori.adventure.text.minimessage.Template; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
| import org.bukkit.entity.Arrow; | ||||
| import org.bukkit.entity.Creature; | ||||
| import org.bukkit.entity.Entity; | ||||
| import org.bukkit.entity.EntityType; | ||||
| import org.bukkit.entity.Firework; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.entity.Projectile; | ||||
| import org.bukkit.event.entity.EntityDamageEvent; | ||||
| @@ -175,8 +170,7 @@ public class BukkitEntityUtil { | ||||
|                 if (plot != null && (plot.getFlag(HangingBreakFlag.class) || plot | ||||
|                         .isAdded(plotPlayer.getUUID()))) { | ||||
|                     if (Settings.Done.RESTRICT_BUILDING && DoneFlag.isDone(plot)) { | ||||
|                         if (!Permissions | ||||
|                                 .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_BUILD_OTHER)) { | ||||
|                         if (!plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_BUILD_OTHER)) { | ||||
|                             plotPlayer.sendMessage( | ||||
|                                     TranslatableCaption.of("done.building_restricted") | ||||
|                             ); | ||||
| @@ -185,10 +179,13 @@ public class BukkitEntityUtil { | ||||
|                     } | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_DESTROY + "." + stub)) { | ||||
|                 if (!plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_DESTROY + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_DESTROY + "." + stub) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Component.text(Permission.PERMISSION_ADMIN_DESTROY + "." + stub)) | ||||
|                             ) | ||||
|                     ); | ||||
|                     return false; | ||||
|                 } | ||||
| @@ -197,10 +194,13 @@ public class BukkitEntityUtil { | ||||
|                         .isAdded(plotPlayer.getUUID()))) { | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_DESTROY + "." + stub)) { | ||||
|                 if (!plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_DESTROY + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_DESTROY + "." + stub) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Component.text(Permission.PERMISSION_ADMIN_DESTROY + "." + stub)) | ||||
|                             ) | ||||
|                     ); | ||||
|                     if (plot != null) { | ||||
|                         plot.debug(player.getName() | ||||
| @@ -218,10 +218,13 @@ public class BukkitEntityUtil { | ||||
|                         .getFlag(PveFlag.class))) { | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||
|                 if (!plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_PVE + "." + stub) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Component.text(Permission.PERMISSION_ADMIN_PVE + "." + stub)) | ||||
|                             ) | ||||
|                     ); | ||||
|                     if (plot != null) { | ||||
|                         plot.debug(player.getName() + " could not attack " + entityType | ||||
| @@ -239,10 +242,13 @@ public class BukkitEntityUtil { | ||||
|                         .getFlag(PveFlag.class))) { | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||
|                 if (!plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_PVE + "." + stub) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Component.text(Permission.PERMISSION_ADMIN_PVE + "." + stub)) | ||||
|                             ) | ||||
|                     ); | ||||
|                     if (plot != null) { | ||||
|                         plot.debug(player.getName() + " could not attack " + entityType | ||||
| @@ -252,11 +258,13 @@ public class BukkitEntityUtil { | ||||
|                 } | ||||
|             } else if (EntityCategories.PLAYER.contains(entityType)) { | ||||
|                 if (isPlot) { | ||||
|                     if (!plot.getFlag(PvpFlag.class) && !Permissions | ||||
|                             .hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVP + "." + stub)) { | ||||
|                     if (!plot.getFlag(PvpFlag.class) && !plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_PVP + "." + stub)) { | ||||
|                         plotPlayer.sendMessage( | ||||
|                                 TranslatableCaption.of("permission.no_permission_event"), | ||||
|                                 Template.of("node", Permission.PERMISSION_ADMIN_PVP + "." + stub) | ||||
|                                 TagResolver.resolver( | ||||
|                                         "node", | ||||
|                                         Tag.inserting(Component.text(Permission.PERMISSION_ADMIN_PVP + "." + stub)) | ||||
|                                 ) | ||||
|                         ); | ||||
|                         plot.debug(player.getName() + " could not attack " + entityType | ||||
|                                 + " because pve = false"); | ||||
| @@ -267,10 +275,13 @@ public class BukkitEntityUtil { | ||||
|                 } else if (roadFlags && area.getRoadFlag(PvpFlag.class)) { | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVP + "." + stub)) { | ||||
|                 if (!plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_PVP + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_PVP + "." + stub) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Component.text(Permission.PERMISSION_ADMIN_PVP + "." + stub)) | ||||
|                             ) | ||||
|                     ); | ||||
|                     return false; | ||||
|                 } | ||||
| @@ -284,10 +295,13 @@ public class BukkitEntityUtil { | ||||
|                         .getFlag(PveFlag.class))) { | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||
|                 if (!plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_PVE + "." + stub) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Component.text(Permission.PERMISSION_ADMIN_PVE + "." + stub)) | ||||
|                             ) | ||||
|                     ); | ||||
|                     if (plot != null) { | ||||
|                         plot.debug(player.getName() + " could not attack " + entityType | ||||
| @@ -306,10 +320,13 @@ public class BukkitEntityUtil { | ||||
|                 } else if (roadFlags && area.getRoadFlag(PveFlag.class)) { | ||||
|                     return true; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(plotPlayer, Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||
|                 if (!plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_PVE + "." + stub)) { | ||||
|                     plotPlayer.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             Template.of("node", Permission.PERMISSION_ADMIN_PVE + "." + stub) | ||||
|                             TagResolver.resolver( | ||||
|                                     "node", | ||||
|                                     Tag.inserting(Component.text(Permission.PERMISSION_ADMIN_PVE + "." + stub)) | ||||
|                             ) | ||||
|                     ); | ||||
|                     if (plot != null) { | ||||
|                         plot.debug(player.getName() + " could not attack " + entityType | ||||
| @@ -325,8 +342,7 @@ public class BukkitEntityUtil { | ||||
|         } | ||||
|         //disable the firework damage. too much of a headache to support at the moment. | ||||
|         if (vplot != null) { | ||||
|             if (EntityDamageEvent.DamageCause.ENTITY_EXPLOSION == cause | ||||
|                     && damager.getType() == EntityType.FIREWORK) { | ||||
|             if (EntityDamageEvent.DamageCause.ENTITY_EXPLOSION == cause && damager instanceof Firework) { | ||||
|                 return false; | ||||
|             } | ||||
|         } | ||||
| @@ -338,13 +354,17 @@ public class BukkitEntityUtil { | ||||
|     } | ||||
|  | ||||
|     public static boolean checkEntity(Entity entity, Plot plot) { | ||||
|         return checkEntity(entity.getType(), plot); | ||||
|     } | ||||
|  | ||||
|     public static boolean checkEntity(EntityType type, Plot plot) { | ||||
|         if (plot == null || !plot.hasOwner() || plot.getFlags().isEmpty() && plot.getArea() | ||||
|                 .getFlagContainer().getFlagMap().isEmpty()) { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         final com.sk89q.worldedit.world.entity.EntityType entityType = | ||||
|                 BukkitAdapter.adapt(entity.getType()); | ||||
|                 BukkitAdapter.adapt(type); | ||||
|  | ||||
|         if (EntityCategories.PLAYER.contains(entityType)) { | ||||
|             return false; | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -51,6 +44,7 @@ import java.util.stream.IntStream; | ||||
| @Singleton | ||||
| public class BukkitInventoryUtil extends InventoryUtil { | ||||
|  | ||||
|     @SuppressWarnings("deprecation") // Paper deprecation | ||||
|     private static @Nullable ItemStack getItem(PlotItemStack item) { | ||||
|         if (item == null) { | ||||
|             return null; | ||||
| @@ -63,7 +57,7 @@ public class BukkitInventoryUtil extends InventoryUtil { | ||||
|         ItemMeta meta = null; | ||||
|         if (item.getName() != null) { | ||||
|             meta = stack.getItemMeta(); | ||||
|             Component nameComponent = BukkitUtil.MINI_MESSAGE.parse(item.getName()); | ||||
|             Component nameComponent = BukkitUtil.MINI_MESSAGE.deserialize(item.getName()); | ||||
|             meta.setDisplayName(BukkitUtil.LEGACY_COMPONENT_SERIALIZER.serialize(nameComponent)); | ||||
|         } | ||||
|         if (item.getLore() != null) { | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -37,7 +30,7 @@ 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.queue.ScopedQueueCoordinator; | ||||
| import com.plotsquared.core.queue.ZeroedDelegateScopedQueueCoordinator; | ||||
| import com.plotsquared.core.util.ChunkManager; | ||||
| import com.plotsquared.core.util.RegionManager; | ||||
| import com.plotsquared.core.util.WorldUtil; | ||||
| @@ -264,9 +257,9 @@ public class BukkitRegionManager extends RegionManager { | ||||
|             map.saveEntitiesOut(Bukkit.getWorld(world.getName()).getChunkAt(x, z), currentPlotClear); | ||||
|             AugmentedUtils.bypass( | ||||
|                     ignoreAugment, | ||||
|                     () -> ChunkManager.setChunkInPlotArea(null, new RunnableVal<ScopedQueueCoordinator>() { | ||||
|                     () -> ChunkManager.setChunkInPlotArea(null, new RunnableVal<ZeroedDelegateScopedQueueCoordinator>() { | ||||
|                         @Override | ||||
|                         public void run(ScopedQueueCoordinator value) { | ||||
|                         public void run(ZeroedDelegateScopedQueueCoordinator value) { | ||||
|                             Location min = value.getMin(); | ||||
|                             int bx = min.getX(); | ||||
|                             int bz = min.getZ(); | ||||
| @@ -278,7 +271,7 @@ public class BukkitRegionManager extends RegionManager { | ||||
|                                         int minY = value.getMin().getY(); | ||||
|                                         for (int yIndex = 0; yIndex < ids.length; yIndex++) { | ||||
|                                             int y = yIndex + minY; | ||||
|                                             BaseBlock id = ids[y]; | ||||
|                                             BaseBlock id = ids[yIndex]; | ||||
|                                             if (id != null) { | ||||
|                                                 value.setBlock(x1, y, z1, id); | ||||
|                                             } else { | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -74,9 +67,10 @@ public class BukkitSetupUtils extends SetupUtils { | ||||
|         this.worldFile = worldFile; | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("deprecation") // Paper deprecation | ||||
|     @Override | ||||
|     public void updateGenerators(final boolean force) { | ||||
|         if (!SetupUtils.generators.isEmpty() && !force) { | ||||
|         if (loaded && !SetupUtils.generators.isEmpty() && !force) { | ||||
|             return; | ||||
|         } | ||||
|         String testWorld = "CheckingPlotSquaredGenerator"; | ||||
| @@ -100,6 +94,7 @@ public class BukkitSetupUtils extends SetupUtils { | ||||
|                 e.printStackTrace(); | ||||
|             } | ||||
|         } | ||||
|         loaded = true; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -130,7 +125,7 @@ public class BukkitSetupUtils extends SetupUtils { | ||||
|     public String setupWorld(PlotAreaBuilder builder) { | ||||
|         this.updateGenerators(false); | ||||
|         ConfigurationNode[] steps = builder.settingsNodesWrapper() == null ? | ||||
|                 new ConfigurationNode[0] : builder.settingsNodesWrapper().getSettingsNodes(); | ||||
|                 new ConfigurationNode[0] : builder.settingsNodesWrapper().settingsNodes(); | ||||
|         String world = builder.worldName(); | ||||
|         PlotAreaType type = builder.plotAreaType(); | ||||
|         String worldPath = "worlds." + builder.worldName(); | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -52,7 +45,7 @@ import com.sk89q.worldedit.world.block.BlockTypes; | ||||
| import io.papermc.lib.PaperLib; | ||||
| 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.minimessage.tag.resolver.TagResolver; | ||||
| import net.kyori.adventure.text.serializer.legacy.LegacyComponentSerializer; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| @@ -65,6 +58,7 @@ import org.bukkit.block.BlockFace; | ||||
| import org.bukkit.block.Sign; | ||||
| import org.bukkit.block.data.Directional; | ||||
| import org.bukkit.block.data.type.WallSign; | ||||
| import org.bukkit.entity.Allay; | ||||
| import org.bukkit.entity.Ambient; | ||||
| import org.bukkit.entity.Animals; | ||||
| import org.bukkit.entity.AreaEffectCloud; | ||||
| @@ -81,6 +75,7 @@ import org.bukkit.entity.FallingBlock; | ||||
| import org.bukkit.entity.Firework; | ||||
| import org.bukkit.entity.Ghast; | ||||
| import org.bukkit.entity.Hanging; | ||||
| import org.bukkit.entity.Interaction; | ||||
| import org.bukkit.entity.IronGolem; | ||||
| import org.bukkit.entity.Item; | ||||
| import org.bukkit.entity.LightningStrike; | ||||
| @@ -267,6 +262,11 @@ public class BukkitUtil extends WorldUtil { | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean isSmallBlock(Location location) { | ||||
|         return adapt(location).getBlock().getBoundingBox().getHeight() < 0.25; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     @NonNegative | ||||
|     public int getHighestBlockSynchronous(final @NonNull String world, final int x, final int z) { | ||||
| @@ -339,7 +339,7 @@ public class BukkitUtil extends WorldUtil { | ||||
|     @SuppressWarnings("deprecation") | ||||
|     public void setSign( | ||||
|             final @NonNull Location location, final @NonNull Caption[] lines, | ||||
|             final @NonNull Template... replacements | ||||
|             final @NonNull TagResolver... replacements | ||||
|     ) { | ||||
|         ensureLoaded(location.getWorldName(), location.getX(), location.getZ(), chunk -> { | ||||
|             PlotArea area = location.getPlotArea(); | ||||
| @@ -372,8 +372,9 @@ public class BukkitUtil extends WorldUtil { | ||||
|             final org.bukkit.block.BlockState blockstate = block.getState(); | ||||
|             if (blockstate instanceof final Sign sign) { | ||||
|                 for (int i = 0; i < lines.length; i++) { | ||||
|                     sign.setLine(i, LEGACY_COMPONENT_SERIALIZER | ||||
|                             .serialize(MINI_MESSAGE.parse(lines[i].getComponent(LocaleHolder.console()), replacements))); | ||||
|                     sign.setLine(i, LEGACY_COMPONENT_SERIALIZER.serialize( | ||||
|                             MINI_MESSAGE.deserialize(lines[i].getComponent(LocaleHolder.console()), replacements) | ||||
|                     )); | ||||
|                 } | ||||
|                 sign.update(true, false); | ||||
|             } | ||||
| @@ -437,6 +438,7 @@ public class BukkitUtil extends WorldUtil { | ||||
|     @Override | ||||
|     public @NonNull Set<com.sk89q.worldedit.world.entity.EntityType> getTypesInCategory(final @NonNull String category) { | ||||
|         final Collection<Class<?>> allowedInterfaces = new HashSet<>(); | ||||
|         final int[] version = PlotSquared.platform().serverVersion(); | ||||
|         switch (category) { | ||||
|             case "animal" -> { | ||||
|                 allowedInterfaces.add(IronGolem.class); | ||||
| @@ -444,6 +446,9 @@ public class BukkitUtil extends WorldUtil { | ||||
|                 allowedInterfaces.add(Animals.class); | ||||
|                 allowedInterfaces.add(WaterMob.class); | ||||
|                 allowedInterfaces.add(Ambient.class); | ||||
|                 if (version[1] >= 19) { | ||||
|                     allowedInterfaces.add(Allay.class); | ||||
|                 } | ||||
|             } | ||||
|             case "tameable" -> allowedInterfaces.add(Tameable.class); | ||||
|             case "vehicle" -> allowedInterfaces.add(Vehicle.class); | ||||
| @@ -472,6 +477,11 @@ public class BukkitUtil extends WorldUtil { | ||||
|                 allowedInterfaces.add(Firework.class); | ||||
|             } | ||||
|             case "player" -> allowedInterfaces.add(Player.class); | ||||
|             case "interaction" -> { | ||||
|                 if ((version[1] > 19) || (version[1] == 19 && version[2] >= 4)) { | ||||
|                     allowedInterfaces.add(Interaction.class); | ||||
|                 } | ||||
|             } | ||||
|             default -> LOGGER.error("Unknown entity category requested: {}", category); | ||||
|         } | ||||
|         final Set<com.sk89q.worldedit.world.entity.EntityType> types = new HashSet<>(); | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -30,12 +23,15 @@ import com.plotsquared.core.location.World; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.lang.ref.SoftReference; | ||||
| import java.lang.ref.WeakReference; | ||||
| 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(); | ||||
|     // Upon world unload we should probably have the P2 BukkitWorld be GCed relatively eagerly, thus freeing the bukkit world. | ||||
|     //  We also want to avoid circumstances where a bukkit world has been GCed, but a P2 BukkitWorld has not | ||||
|     private static final Map<String, WeakReference<BukkitWorld>> worldMap = Maps.newHashMap(); | ||||
|     private static final boolean HAS_MIN_Y; | ||||
|  | ||||
|     static { | ||||
| @@ -49,10 +45,11 @@ public class BukkitWorld implements World<org.bukkit.World> { | ||||
|         HAS_MIN_Y = temp; | ||||
|     } | ||||
|  | ||||
|     private final org.bukkit.World world; | ||||
|     // We want to allow GC to remove bukkit worlds, but not too eagerly | ||||
|     private final SoftReference<org.bukkit.World> world; | ||||
|  | ||||
|     private BukkitWorld(final org.bukkit.World world) { | ||||
|         this.world = world; | ||||
|         this.world = new SoftReference<>(world); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -76,12 +73,13 @@ public class BukkitWorld implements World<org.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)) { | ||||
|         WeakReference<BukkitWorld> bukkitWorldRef = worldMap.get(world.getName()); | ||||
|         BukkitWorld bukkitWorld; | ||||
|         if (bukkitWorldRef != null && (bukkitWorld = bukkitWorldRef.get()) != null && world.equals(bukkitWorld.world.get())) { | ||||
|             return bukkitWorld; | ||||
|         } | ||||
|         bukkitWorld = new BukkitWorld(world); | ||||
|         worldMap.put(world.getName(), bukkitWorld); | ||||
|         worldMap.put(world.getName(), new WeakReference<>(bukkitWorld)); | ||||
|         return bukkitWorld; | ||||
|     } | ||||
|  | ||||
| @@ -105,22 +103,26 @@ public class BukkitWorld implements World<org.bukkit.World> { | ||||
|  | ||||
|     @Override | ||||
|     public org.bukkit.World getPlatformWorld() { | ||||
|         return this.world; | ||||
|         org.bukkit.World world = this.world.get(); | ||||
|         if (world == null) { | ||||
|             throw new IllegalStateException("Bukkit platform world was unloaded from memory"); | ||||
|         } | ||||
|         return world; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public @NonNull String getName() { | ||||
|         return this.world.getName(); | ||||
|         return this.getPlatformWorld().getName(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public int getMinHeight() { | ||||
|         return getMinWorldHeight(world); | ||||
|         return getMinWorldHeight(getPlatformWorld()); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public int getMaxHeight() { | ||||
|         return getMaxWorldHeight(world) - 1; | ||||
|         return getMaxWorldHeight(getPlatformWorld()) - 1; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -140,14 +142,6 @@ public class BukkitWorld implements World<org.bukkit.World> { | ||||
|         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 + ")"; | ||||
|     } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -0,0 +1,98 @@ | ||||
| /* | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.util; | ||||
|  | ||||
| import com.intellectualsites.annotations.NotPublic; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.nio.file.Files; | ||||
| import java.nio.file.Path; | ||||
| import java.nio.file.Paths; | ||||
| import java.util.stream.Stream; | ||||
|  | ||||
| /** | ||||
|  * This is a helper class which replaces older syntax no longer supported by MiniMessage with replacements in messages_%.json. | ||||
|  * MiniMessage changed the syntax between major releases. To warrant a smooth upgrade, we attempt to replace any occurrences | ||||
|  * while loading PlotSquared. | ||||
|  * | ||||
|  * @since 7.0.0 | ||||
|  */ | ||||
| @NotPublic | ||||
| public class TranslationUpdateManager { | ||||
|  | ||||
|     public static void upgradeTranslationFile() throws IOException { | ||||
|         String suggestCommand = "suggest_command"; | ||||
|         String suggestCommandReplacement = "run_command"; | ||||
|         String minHeight = "minHeight"; | ||||
|         String minheightReplacement = "minheight"; | ||||
|         String maxHeight = "maxHeight"; | ||||
|         String maxheightReplacement = "maxheight"; | ||||
|         String usedGrants = "usedGrants"; | ||||
|         String usedGrantsReplacement = "used_grants"; | ||||
|         String remainingGrants = "remainingGrants"; | ||||
|         String remainingGrantsReplacement = "remaining_grants"; | ||||
|         String minimumRadius = "minimumRadius"; | ||||
|         String minimumRadiusReplacement = "minimum_radius"; | ||||
|         String maximumMoves = "maximumMoves"; | ||||
|         String maximumMovesReplacement = "maximum_moves"; | ||||
|         String userMove = "userMove"; | ||||
|         String userMoveReplacement = "user_move"; | ||||
|  | ||||
|         // tag opening / closing characters are important, as the locale keys exist as well, which should not be replaced | ||||
|         String listInfoUnknown = "<info.unknown>"; | ||||
|         String listInfoUnknownReplacement = "<unknown>"; | ||||
|         String listInfoServer = "<info.server>"; | ||||
|         String listInfoServerReplacement = "<server>"; | ||||
|         String listInfoEveryone = "<info.everyone>"; | ||||
|         String listInfoEveryoneReplacement = "<everyone>"; | ||||
|  | ||||
|         try (Stream<Path> paths = Files.walk(Paths.get(PlotSquared.platform().getDirectory().toPath().resolve("lang").toUri()))) { | ||||
|             paths | ||||
|                     .filter(Files::isRegularFile) | ||||
|                     .filter(p -> p.getFileName().toString().matches("messages_[a-z]{2}\\.json")) | ||||
|                     .forEach(p -> { | ||||
|                         replaceInFile(p, suggestCommand, suggestCommandReplacement); | ||||
|                         replaceInFile(p, minHeight, minheightReplacement); | ||||
|                         replaceInFile(p, maxHeight, maxheightReplacement); | ||||
|                         replaceInFile(p, usedGrants, usedGrantsReplacement); | ||||
|                         replaceInFile(p, remainingGrants, remainingGrantsReplacement); | ||||
|                         replaceInFile(p, minimumRadius, minimumRadiusReplacement); | ||||
|                         replaceInFile(p, maximumMoves, maximumMovesReplacement); | ||||
|                         replaceInFile(p, userMove, userMoveReplacement); | ||||
|                         replaceInFile(p, listInfoUnknown, listInfoUnknownReplacement); | ||||
|                         replaceInFile(p, listInfoServer, listInfoServerReplacement); | ||||
|                         replaceInFile(p, listInfoEveryone, listInfoEveryoneReplacement); | ||||
|                     }); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private static void replaceInFile(Path path, String searchText, String replacementText) { | ||||
|         try { | ||||
|             String content = Files.readString(path); | ||||
|             if (content.contains(searchText)) { | ||||
|                 content = content.replaceAll(searchText, replacementText); | ||||
|                 Files.writeString(path, content); | ||||
|             } | ||||
|         } catch (IOException e) { | ||||
|             e.printStackTrace(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -42,7 +35,7 @@ import org.bukkit.scheduler.BukkitTask; | ||||
| import javax.net.ssl.HttpsURLConnection; | ||||
| import java.io.IOException; | ||||
| import java.io.InputStreamReader; | ||||
| import java.net.URL; | ||||
| import java.net.URI; | ||||
|  | ||||
| public class UpdateUtility implements Listener { | ||||
|  | ||||
| @@ -61,12 +54,14 @@ public class UpdateUtility implements Listener { | ||||
|         internalVersion = PlotSquared.get().getVersion(); | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings({"deprecation", "DefaultCharset"}) // Suppress Json deprecation, we can't use features from gson 2.8.1 and newer yet | ||||
|     @SuppressWarnings({"deprecation", "DefaultCharset"}) | ||||
|     // Suppress Json deprecation, we can't use features from gson 2.8.1 and newer yet | ||||
|     public void updateChecker() { | ||||
|         task = Bukkit.getScheduler().runTaskTimerAsynchronously(this.javaPlugin, () -> { | ||||
|             try { | ||||
|                 HttpsURLConnection connection = (HttpsURLConnection) new URL( | ||||
|                         "https://api.spigotmc.org/simple/0.1/index.php?action=getResource&id=77506") | ||||
|                 HttpsURLConnection connection = (HttpsURLConnection) URI.create( | ||||
|                         "https://api.spigotmc.org/simple/0.2/index.php?action=getResource&id=77506") | ||||
|                         .toURL() | ||||
|                         .openConnection(); | ||||
|                 connection.setRequestMethod("GET"); | ||||
|                 JsonObject result = new JsonParser() | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -43,9 +36,10 @@ 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.checkerframework.checker.nullness.qual.Nullable; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
|  | ||||
| import java.util.Objects; | ||||
| import java.util.Set; | ||||
|  | ||||
| public class FaweRegionManager extends BukkitRegionManager { | ||||
| @@ -53,10 +47,7 @@ public class FaweRegionManager extends BukkitRegionManager { | ||||
|     private final FaweDelegateRegionManager delegate = new FaweDelegateRegionManager(); | ||||
|  | ||||
|     @Inject | ||||
|     public FaweRegionManager( | ||||
|             @NonNull WorldUtil worldUtil, @NonNull GlobalBlockQueue blockQueue, @NonNull | ||||
|             ProgressSubscriberFactory subscriberFactory | ||||
|     ) { | ||||
|     public FaweRegionManager(WorldUtil worldUtil, GlobalBlockQueue blockQueue, ProgressSubscriberFactory subscriberFactory) { | ||||
|         super(worldUtil, blockQueue, subscriberFactory); | ||||
|     } | ||||
|  | ||||
| @@ -70,7 +61,10 @@ public class FaweRegionManager extends BukkitRegionManager { | ||||
|             @Nullable PlotPlayer<?> actor, | ||||
|             @Nullable QueueCoordinator queue | ||||
|     ) { | ||||
|         return delegate.setCuboids(area, regions, blocks, minY, maxY, queue.getCompleteTask()); | ||||
|         return delegate.setCuboids( | ||||
|                 area, regions, blocks, minY, maxY, | ||||
|                 Objects.requireNonNullElseGet(queue, area::getQueue).getCompleteTask() | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -83,9 +77,9 @@ public class FaweRegionManager extends BukkitRegionManager { | ||||
|  | ||||
|     @Override | ||||
|     public boolean handleClear( | ||||
|             @NotNull Plot plot, | ||||
|             @NonNull Plot plot, | ||||
|             @Nullable Runnable whenDone, | ||||
|             @NotNull PlotManager manager, | ||||
|             @NonNull PlotManager manager, | ||||
|             final @Nullable PlotPlayer<?> player | ||||
|     ) { | ||||
|         if (!Settings.FAWE_Components.CLEAR || !(manager instanceof HybridPlotManager)) { | ||||
| @@ -105,11 +99,6 @@ public class FaweRegionManager extends BukkitRegionManager { | ||||
|         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); | ||||
| @@ -127,7 +116,7 @@ public class FaweRegionManager extends BukkitRegionManager { | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean regenerateRegion(final Location pos1, final Location pos2, boolean ignore, final Runnable whenDone) { | ||||
|     public boolean regenerateRegion(final @NotNull Location pos1, final @NotNull Location pos2, boolean ignore, final Runnable whenDone) { | ||||
|         return delegate.regenerateRegion(pos1, pos2, ignore, whenDone); | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -36,7 +29,7 @@ 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 org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.io.InputStream; | ||||
| import java.net.URL; | ||||
| @@ -47,7 +40,7 @@ public class FaweSchematicHandler extends SchematicHandler { | ||||
|     private final FaweDelegateSchematicHandler delegate = new FaweDelegateSchematicHandler(); | ||||
|  | ||||
|     @Inject | ||||
|     public FaweSchematicHandler(@NotNull WorldUtil worldUtil, @NotNull ProgressSubscriberFactory subscriberFactory) { | ||||
|     public FaweSchematicHandler(WorldUtil worldUtil, ProgressSubscriberFactory subscriberFactory) { | ||||
|         super(worldUtil, subscriberFactory); | ||||
|     } | ||||
|  | ||||
| @@ -82,9 +75,8 @@ public class FaweSchematicHandler extends SchematicHandler { | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public Schematic getSchematic(@NotNull InputStream is) { | ||||
|     public Schematic getSchematic(@NonNull InputStream is) { | ||||
|         return delegate.getSchematic(is); | ||||
|     } | ||||
|  | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -118,8 +111,8 @@ public class SQLiteUUIDService implements UUIDService, Consumer<List<UUIDMapping | ||||
|         try (final PreparedStatement statement = getConnection() | ||||
|                 .prepareStatement("INSERT OR REPLACE INTO `usercache` (`uuid`, `username`) VALUES(?, ?)")) { | ||||
|             for (final UUIDMapping mapping : uuidWrappers) { | ||||
|                 statement.setString(1, mapping.getUuid().toString()); | ||||
|                 statement.setString(2, mapping.getUsername()); | ||||
|                 statement.setString(1, mapping.uuid().toString()); | ||||
|                 statement.setString(2, mapping.username()); | ||||
|                 statement.executeUpdate(); | ||||
|             } | ||||
|         } catch (SQLException e) { | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,76 +0,0 @@ | ||||
| # Contributor Covenant Code of Conduct | ||||
|  | ||||
| ## Our Pledge | ||||
|  | ||||
| In the interest of fostering an open and welcoming environment, we as | ||||
| contributors and maintainers pledge to making participation in our project and | ||||
| our community a harassment-free experience for everyone, regardless of age, body | ||||
| size, disability, ethnicity, sex characteristics, gender identity and expression, | ||||
| level of experience, education, socio-economic status, nationality, personal | ||||
| appearance, race, religion, or sexual identity and orientation. | ||||
|  | ||||
| ## Our Standards | ||||
|  | ||||
| Examples of behavior that contributes to creating a positive environment | ||||
| include: | ||||
|  | ||||
| * Using welcoming and inclusive language | ||||
| * Being respectful of differing viewpoints and experiences | ||||
| * Gracefully accepting constructive criticism | ||||
| * Focusing on what is best for the community | ||||
| * Showing empathy towards other community members | ||||
|  | ||||
| Examples of unacceptable behavior by participants include: | ||||
|  | ||||
| * The use of sexualized language or imagery and unwelcome sexual attention or | ||||
|  advances | ||||
| * Trolling, insulting/derogatory comments, and personal or political attacks | ||||
| * Public or private harassment | ||||
| * Publishing others' private information, such as a physical or electronic | ||||
|  address, without explicit permission | ||||
| * Other conduct which could reasonably be considered inappropriate in a | ||||
|  professional setting | ||||
|  | ||||
| ## Our Responsibilities | ||||
|  | ||||
| Project maintainers are responsible for clarifying the standards of acceptable | ||||
| behavior and are expected to take appropriate and fair corrective action in | ||||
| response to any instances of unacceptable behavior. | ||||
|  | ||||
| Project maintainers have the right and responsibility to remove, edit, or | ||||
| reject comments, commits, code, wiki edits, issues, and other contributions | ||||
| that are not aligned to this Code of Conduct, or to ban temporarily or | ||||
| permanently any contributor for other behaviors that they deem inappropriate, | ||||
| threatening, offensive, or harmful. | ||||
|  | ||||
| ## Scope | ||||
|  | ||||
| This Code of Conduct applies both within project spaces and in public spaces | ||||
| when an individual is representing the project or its community. Examples of | ||||
| representing a project or community include using an official project e-mail | ||||
| address, posting via an official social media account, or acting as an appointed | ||||
| representative at an online or offline event. Representation of a project may be | ||||
| further defined and clarified by project maintainers. | ||||
|  | ||||
| ## Enforcement | ||||
|  | ||||
| Instances of abusive, harassing, or otherwise unacceptable behavior may be | ||||
| reported by contacting the project team at contact@alexander-soderberg.com. All | ||||
| complaints will be reviewed and investigated and will result in a response that | ||||
| is deemed necessary and appropriate to the circumstances. The project team is | ||||
| obligated to maintain confidentiality with regard to the reporter of an incident. | ||||
| Further details of specific enforcement policies may be posted separately. | ||||
|  | ||||
| Project maintainers who do not follow or enforce the Code of Conduct in good | ||||
| faith may face temporary or permanent repercussions as determined by other | ||||
| members of the project's leadership. | ||||
|  | ||||
| ## Attribution | ||||
|  | ||||
| This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4, | ||||
| available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html | ||||
|  | ||||
| [homepage]: https://www.contributor-covenant.org | ||||
|  | ||||
| For answers to common questions about this code of conduct, see | ||||
| https://www.contributor-covenant.org/faq | ||||
| @@ -12,8 +12,8 @@ dependencies { | ||||
|     compileOnlyApi(libs.snakeyaml) | ||||
|  | ||||
|     // Adventure | ||||
|     api(libs.adventure) | ||||
|     api(libs.minimessage) | ||||
|     api(libs.adventureApi) | ||||
|     api(libs.adventureMiniMessage) | ||||
|  | ||||
|     // Guice | ||||
|     api(libs.guice) { | ||||
| @@ -22,7 +22,7 @@ dependencies { | ||||
|     api(libs.guiceassistedinject) { | ||||
|         exclude("com.google.inject", "guice") | ||||
|     } | ||||
|     api(libs.findbugs) | ||||
|     api(libs.spotbugs) | ||||
|  | ||||
|     // Plugins | ||||
|     compileOnly(libs.worldeditCore) { | ||||
| @@ -31,8 +31,8 @@ dependencies { | ||||
|         exclude(group = "dummypermscompat") | ||||
|     } | ||||
|     testImplementation(libs.worldeditCore) | ||||
|     compileOnly(libs.fastasyncworldeditCore) { isTransitive = false } | ||||
|     testImplementation(libs.fastasyncworldeditCore) { isTransitive = false } | ||||
|     compileOnly(libs.faweBukkit) { isTransitive = false } | ||||
|     testImplementation(libs.faweCore) { isTransitive = false } | ||||
|  | ||||
|     // Logging | ||||
|     compileOnlyApi(libs.log4j) | ||||
| @@ -43,6 +43,7 @@ dependencies { | ||||
|     api(libs.cloudServices) | ||||
|     api(libs.arkitektonika) | ||||
|     api(libs.paster) | ||||
|     api(libs.informativeAnnotations) | ||||
| } | ||||
|  | ||||
| tasks.processResources { | ||||
| @@ -53,14 +54,30 @@ tasks.processResources { | ||||
|                 "date" to rootProject.grgit.head().dateTime.format(DateTimeFormatter.ofPattern("yy.MM.dd")) | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     doLast { | ||||
|         copy { | ||||
|             from(layout.buildDirectory.file("$rootDir/LICENSE")) | ||||
|             into(layout.buildDirectory.dir("resources/main")) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| tasks { | ||||
|     withType<Javadoc> { | ||||
|         val isRelease = if (rootProject.version.toString().endsWith("-SNAPSHOT")) "TODO" else rootProject.version.toString() | ||||
|         val opt = options as StandardJavadocDocletOptions | ||||
|         opt.links("https://docs.enginehub.org/javadoc/com.sk89q.worldedit/worldedit-core/" + libs.worldeditCore.get().versionConstraint.toString()) | ||||
|         opt.links("https://jd.adventure.kyori.net/api/" + libs.adventure.get().versionConstraint.toString()) | ||||
|         opt.links("https://jd.advntr.dev/api/" + libs.adventureApi.get().versionConstraint.toString()) | ||||
|         opt.links("https://jd.advntr.dev/text-minimessage/" + libs.adventureApi.get().versionConstraint.toString()) | ||||
|         opt.links("https://google.github.io/guice/api-docs/" + libs.guice.get().versionConstraint.toString() + "/javadoc/") | ||||
|         opt.links("https://checkerframework.org/api/") | ||||
|         opt.isLinkSource = true | ||||
|         opt.bottom(File("$rootDir/javadocfooter.html").readText()) | ||||
|         opt.isUse = true | ||||
|         opt.encoding("UTF-8") | ||||
|         opt.keyWords() | ||||
|         opt.addStringOption("-since", isRelease) | ||||
|         opt.noTimestamp() | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -37,7 +30,7 @@ import com.plotsquared.core.util.ChunkManager; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.SchematicHandler; | ||||
| import com.plotsquared.core.util.query.PlotQuery; | ||||
| import net.kyori.adventure.text.minimessage.Template; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| @@ -152,7 +145,7 @@ public class PlotAPI { | ||||
|      */ | ||||
|     public void sendConsoleMessage( | ||||
|             final @NonNull String message, | ||||
|             final @NonNull Template @NonNull ... replacements | ||||
|             final @NonNull TagResolver @NonNull ... replacements | ||||
|     ) { | ||||
|         ConsolePlayer.getConsole().sendMessage(StaticCaption.of(message), replacements); | ||||
|     } | ||||
| @@ -165,7 +158,7 @@ public class PlotAPI { | ||||
|      */ | ||||
|     public void sendConsoleMessage( | ||||
|             final @NonNull Caption caption, | ||||
|             final @NonNull Template @NonNull ... replacements | ||||
|             final @NonNull TagResolver @NonNull ... replacements | ||||
|     ) { | ||||
|         ConsolePlayer.getConsole().sendMessage(caption, replacements); | ||||
|     } | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -29,6 +22,7 @@ import cloud.commandframework.services.ServicePipeline; | ||||
| import com.google.inject.Injector; | ||||
| import com.google.inject.Key; | ||||
| import com.google.inject.TypeLiteral; | ||||
| import com.intellectualsites.annotations.DoNotUse; | ||||
| import com.plotsquared.core.backup.BackupManager; | ||||
| import com.plotsquared.core.configuration.caption.LocaleHolder; | ||||
| import com.plotsquared.core.generator.GeneratorWrapper; | ||||
| @@ -38,9 +32,9 @@ import com.plotsquared.core.inject.annotations.DefaultGenerator; | ||||
| import com.plotsquared.core.location.World; | ||||
| import com.plotsquared.core.permissions.PermissionHandler; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.expiration.ExpireManager; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.queue.GlobalBlockQueue; | ||||
| import com.plotsquared.core.util.AnnotationHelper; | ||||
| import com.plotsquared.core.util.ChunkManager; | ||||
| import com.plotsquared.core.util.EconHandler; | ||||
| import com.plotsquared.core.util.PlatformWorldManager; | ||||
| @@ -82,6 +76,11 @@ public interface PlotPlatform<P> extends LocaleHolder { | ||||
|      */ | ||||
|     void shutdown(); | ||||
|  | ||||
|     /** | ||||
|      * Completely shuts down the server. | ||||
|      */ | ||||
|     void shutdownServer(); | ||||
|  | ||||
|     /** | ||||
|      * Get the name of the plugin | ||||
|      * | ||||
| @@ -121,6 +120,14 @@ public interface PlotPlatform<P> extends LocaleHolder { | ||||
|      */ | ||||
|     @NonNull String serverImplementation(); | ||||
|  | ||||
|     /** | ||||
|      * Gets the server brand name | ||||
|      * | ||||
|      * @return server brand | ||||
|      * @since 7.5.3 | ||||
|      */ | ||||
|     @NonNull String serverBrand(); | ||||
|  | ||||
|     /** | ||||
|      * Gets the native server code package prefix. | ||||
|      * | ||||
| @@ -286,6 +293,16 @@ public interface PlotPlatform<P> extends LocaleHolder { | ||||
|         return injector().getInstance(ChunkManager.class); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get the {@link ExpireManager} implementation for the platform | ||||
|      * | ||||
|      * @return Expire manager | ||||
|      * @since 6.10.2 | ||||
|      */ | ||||
|     default @NonNull ExpireManager expireManager() { | ||||
|         return injector().getInstance(ExpireManager.class); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get the {@link PlotAreaManager} implementation. | ||||
|      * | ||||
| @@ -315,7 +332,7 @@ public interface PlotPlatform<P> extends LocaleHolder { | ||||
|      * @return worldedit implementations | ||||
|      * @since 6.3.0 | ||||
|      */ | ||||
|     @AnnotationHelper.ApiDescription(info = "Internal use only") | ||||
|     @DoNotUse | ||||
|     @NonNull String worldEditImplementations(); | ||||
|  | ||||
|     /** | ||||
| @@ -359,9 +376,9 @@ public interface PlotPlatform<P> extends LocaleHolder { | ||||
|     @NonNull String toLegacyPlatformString(@NonNull Component component); | ||||
|  | ||||
|     /** | ||||
|      * Returns if the FAWE-P2 hook is active/enabled | ||||
|      * Returns if the FastAsyncWorldEdit-PlotSquared hook is active/enabled | ||||
|      * | ||||
|      * @return status of FAWE-P2 hook | ||||
|      * @return status of FastAsyncWorldEdit-PlotSquared hook | ||||
|      */ | ||||
|     default boolean isFaweHooking() { | ||||
|         return false; | ||||
|   | ||||
| @@ -1,14 +1,7 @@ | ||||
| /* | ||||
|  *       _____  _       _    _____                                _ | ||||
|  *      |  __ \| |     | |  / ____|                              | | | ||||
|  *      | |__) | | ___ | |_| (___   __ _ _   _  __ _ _ __ ___  __| | | ||||
|  *      |  ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` | | ||||
|  *      | |    | | (_) | |_ ____) | (_| | |_| | (_| | | |  __/ (_| | | ||||
|  *      |_|    |_|\___/ \__|_____/ \__, |\__,_|\__,_|_|  \___|\__,_| | ||||
|  *                                    | | | ||||
|  *                                    |_| | ||||
|  *            PlotSquared plot management system for Minecraft | ||||
|  *               Copyright (C) 2014 - 2022 IntellectualSites | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
| @@ -72,7 +65,10 @@ import com.plotsquared.core.util.ReflectionUtils; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import com.plotsquared.core.uuid.UUIDPipeline; | ||||
| import com.sk89q.worldedit.WorldEdit; | ||||
| import com.sk89q.worldedit.event.platform.PlatformReadyEvent; | ||||
| import com.sk89q.worldedit.math.BlockVector2; | ||||
| import com.sk89q.worldedit.util.eventbus.EventHandler; | ||||
| import com.sk89q.worldedit.util.eventbus.Subscribe; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.checkerframework.checker.nullness.qual.MonotonicNonNull; | ||||
| @@ -88,7 +84,7 @@ import java.io.InputStream; | ||||
| import java.io.InputStreamReader; | ||||
| import java.io.ObjectInputStream; | ||||
| import java.io.ObjectOutputStream; | ||||
| import java.net.MalformedURLException; | ||||
| import java.net.URI; | ||||
| import java.net.URISyntaxException; | ||||
| import java.net.URL; | ||||
| import java.nio.file.Files; | ||||
| @@ -153,6 +149,8 @@ public class PlotSquared { | ||||
|     private EventDispatcher eventDispatcher; | ||||
|     private PlotListener plotListener; | ||||
|  | ||||
|     private boolean weInitialised; | ||||
|  | ||||
|     /** | ||||
|      * Initialize PlotSquared with the desired Implementation class. | ||||
|      * | ||||
| @@ -199,20 +197,25 @@ public class PlotSquared { | ||||
|             this.loadCaptionMap(); | ||||
|         } catch (final Exception e) { | ||||
|             LOGGER.error("Failed to load caption map", e); | ||||
|             LOGGER.error("Shutting down server to prevent further issues"); | ||||
|             this.platform.shutdownServer(); | ||||
|             throw new RuntimeException("Abort loading PlotSquared"); | ||||
|         } | ||||
|  | ||||
|         // Setup the global flag container | ||||
|         GlobalFlagContainer.setup(); | ||||
|  | ||||
|         try { | ||||
|             new ReflectionUtils(this.platform.serverNativePackage()); | ||||
|             String ver = this.platform.serverNativePackage(); | ||||
|             new ReflectionUtils(ver.isEmpty() ? null : ver); | ||||
|             try { | ||||
|                 URL logurl = PlotSquared.class.getProtectionDomain().getCodeSource().getLocation(); | ||||
|                 this.jarFile = new File( | ||||
|                         new URL(logurl.toURI().toString().split("\\!")[0].replaceAll("jar:file", "file")) | ||||
|                                 .toURI().getPath()); | ||||
|             } catch (MalformedURLException | URISyntaxException | SecurityException e) { | ||||
|                 e.printStackTrace(); | ||||
|                         URI.create( | ||||
|                                 logurl.toURI().toString().split("\\!")[0].replaceAll("jar:file", "file")) | ||||
|                                 .getPath()); | ||||
|             } catch (URISyntaxException | SecurityException e) { | ||||
|                 LOGGER.error(e); | ||||
|                 this.jarFile = new File(this.platform.getDirectory().getParentFile(), "PlotSquared.jar"); | ||||
|                 if (!this.jarFile.exists()) { | ||||
|                     this.jarFile = new File( | ||||
| @@ -223,6 +226,7 @@ public class PlotSquared { | ||||
|             } | ||||
|  | ||||
|             this.worldedit = WorldEdit.getInstance(); | ||||
|             WorldEdit.getInstance().getEventBus().register(new WEPlatformReadyListener()); | ||||
|  | ||||
|             // Create Event utility class | ||||
|             this.eventDispatcher = new EventDispatcher(this.worldedit); | ||||
| @@ -235,7 +239,7 @@ public class PlotSquared { | ||||
|             copyFile("skyblock.template", Settings.Paths.TEMPLATES); | ||||
|             showDebug(); | ||||
|         } catch (Throwable e) { | ||||
|             e.printStackTrace(); | ||||
|             LOGGER.error(e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -268,7 +272,11 @@ public class PlotSquared { | ||||
|             captionMap = this.captionLoader.loadAll(this.platform.getDirectory().toPath().resolve("lang")); | ||||
|         } else { | ||||
|             String fileName = "messages_" + Settings.Enabled_Components.DEFAULT_LOCALE + ".json"; | ||||
|             captionMap = this.captionLoader.loadSingle(this.platform.getDirectory().toPath().resolve("lang").resolve(fileName)); | ||||
|             captionMap = this.captionLoader.loadOrCreateSingle(this.platform | ||||
|                     .getDirectory() | ||||
|                     .toPath() | ||||
|                     .resolve("lang") | ||||
|                     .resolve(fileName)); | ||||
|         } | ||||
|         this.captionMaps.put(TranslatableCaption.DEFAULT_NAMESPACE, captionMap); | ||||
|         LOGGER.info( | ||||
| @@ -288,11 +296,11 @@ public class PlotSquared { | ||||
|  | ||||
|     public void startExpiryTasks() { | ||||
|         if (Settings.Enabled_Components.PLOT_EXPIRY) { | ||||
|             ExpireManager.IMP = new ExpireManager(this.eventDispatcher); | ||||
|             ExpireManager.IMP.runAutomatedTask(); | ||||
|             ExpireManager expireManager = PlotSquared.platform().expireManager(); | ||||
|             expireManager.runAutomatedTask(); | ||||
|             for (Settings.Auto_Clear settings : Settings.AUTO_CLEAR.getInstances()) { | ||||
|                 ExpiryTask task = new ExpiryTask(settings, this.getPlotAreaManager()); | ||||
|                 ExpireManager.IMP.addTask(task); | ||||
|                 expireManager.addTask(task); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| @@ -643,7 +651,8 @@ public class PlotSquared { | ||||
|         } else { | ||||
|             list = new ArrayList<>(input); | ||||
|         } | ||||
|         list.sort(Comparator.comparingLong(a -> ExpireManager.IMP.getTimestamp(a.getOwnerAbs()))); | ||||
|         ExpireManager expireManager = PlotSquared.platform().expireManager(); | ||||
|         list.sort(Comparator.comparingLong(a -> expireManager.getTimestamp(a.getOwnerAbs()))); | ||||
|         return list; | ||||
|     } | ||||
|  | ||||
| @@ -787,6 +796,7 @@ public class PlotSquared { | ||||
|         if (world.equals("CheckingPlotSquaredGenerator")) { | ||||
|             return; | ||||
|         } | ||||
|         // Don't check the return result -> breaks runtime loading of single plot areas on creation | ||||
|         this.getPlotAreaManager().addWorld(world); | ||||
|         Set<String> worlds; | ||||
|         if (this.worldConfiguration.contains("worlds")) { | ||||
| @@ -889,8 +899,8 @@ public class PlotSquared { | ||||
|                             e.printStackTrace(); | ||||
|                         } | ||||
|                         LOGGER.info("| generator: {}>{}", baseGenerator, areaGen); | ||||
|                         LOGGER.info("| plot world: {}", pa); | ||||
|                         LOGGER.info("| manager: {}", pa); | ||||
|                         LOGGER.info("| plot world: {}", pa.getClass().getCanonicalName()); | ||||
|                         LOGGER.info("| manager: {}", pa.getPlotManager().getClass().getCanonicalName()); | ||||
|                         LOGGER.info("Note: Area created for cluster '{}' (invalid or old configuration?)", name); | ||||
|                         areaGen.getPlotGenerator().initialize(pa); | ||||
|                         areaGen.augment(pa); | ||||
| @@ -906,6 +916,13 @@ public class PlotSquared { | ||||
|                     throw new IllegalArgumentException("Invalid Generator: " + gen_string); | ||||
|                 } | ||||
|                 PlotArea pa = areaGen.getPlotGenerator().getNewPlotArea(world, null, null, null); | ||||
|                 LOGGER.info("- generator: {}>{}", baseGenerator, areaGen); | ||||
|                 LOGGER.info("- plot world: {}", pa.getClass().getCanonicalName()); | ||||
|                 LOGGER.info("- plot area manager: {}", pa.getPlotManager().getClass().getCanonicalName()); | ||||
|                 if (!this.worldConfiguration.contains(path)) { | ||||
|                     this.worldConfiguration.createSection(path); | ||||
|                     worldSection = this.worldConfiguration.getConfigurationSection(path); | ||||
|                 } | ||||
|                 pa.saveConfiguration(worldSection); | ||||
|                 pa.loadDefaultConfiguration(worldSection); | ||||
|                 try { | ||||
| @@ -913,9 +930,6 @@ public class PlotSquared { | ||||
|                 } catch (IOException e) { | ||||
|                     e.printStackTrace(); | ||||
|                 } | ||||
|                 LOGGER.info("- generator: {}>{}", baseGenerator, areaGen); | ||||
|                 LOGGER.info("- plot world: {}", pa); | ||||
|                 LOGGER.info("- plot area manager: {}", pa.getPlotManager()); | ||||
|                 areaGen.getPlotGenerator().initialize(pa); | ||||
|                 areaGen.augment(pa); | ||||
|                 addPlotArea(pa); | ||||
| @@ -1002,7 +1016,7 @@ public class PlotSquared { | ||||
|  | ||||
|     /** | ||||
|      * Setup the configuration for a plot world based on world arguments. | ||||
|      * | ||||
|      * <p> | ||||
|      * | ||||
|      * <i>e.g. /mv create <world> normal -g PlotSquared:<args></i> | ||||
|      * | ||||
| @@ -1570,6 +1584,13 @@ public class PlotSquared { | ||||
|         return this.plotListener; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get if the {@link PlatformReadyEvent} has been sent by WorldEdit. There is no way to query this within WorldEdit itself. | ||||
|      */ | ||||
|     public boolean isWeInitialised() { | ||||
|         return weInitialised; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Different ways of sorting {@link Plot plots} | ||||
|      */ | ||||
| @@ -1592,4 +1613,15 @@ public class PlotSquared { | ||||
|         DISTANCE_FROM_ORIGIN | ||||
|     } | ||||
|  | ||||
|     private final class WEPlatformReadyListener { | ||||
|  | ||||
|         @SuppressWarnings("unused") | ||||
|         @Subscribe(priority = EventHandler.Priority.VERY_EARLY) | ||||
|         public void onPlatformReady(PlatformReadyEvent event) { | ||||
|             weInitialised = true; | ||||
|             WorldEdit.getInstance().getEventBus().unregister(WEPlatformReadyListener.this); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user