mirror of
				https://github.com/IntellectualSites/PlotSquared.git
				synced 2025-10-23 06:33:43 +02:00 
			
		
		
		
	Compare commits
	
		
			694 Commits
		
	
	
		
			refactor/v
			...
			gh-readonl
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ![renovate[bot]](/assets/img/avatar_default.png)  | 45da6adf72 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 00cda005ce | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 05ba04cf99 | ||
| ![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 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e90fd231d9 | ||
|   | cd9f0789de | ||
|   | 5f4c8d92df | ||
|   | 263cb47a21 | ||
|   | 005600c99e | ||
|   | 26bec7fe2f | ||
|   | f4b886d977 | ||
|   | c09d0d882e | ||
|   | 312cb2996c | ||
|   | f218902581 | ||
|   | f27009216c | 
							
								
								
									
										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 | ||||
							
								
								
									
										19
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										19
									
								
								.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,15 +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.19' | ||||
|         - '1.18.2' | ||||
|         - '1.18.1' | ||||
|         - '1.18' | ||||
|         - '1.17.1' | ||||
|         - '1.16.5' | ||||
|         - '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@v4 | ||||
|       - name: Setup Java | ||||
|         uses: actions/setup-java@v5 | ||||
|         with: | ||||
|           distribution: temurin | ||||
|           java-version: 21 | ||||
|       - name: Clean Build | ||||
|         run: ./gradlew clean build | ||||
							
								
								
									
										43
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										43
									
								
								.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 | ||||
|       - name: Validate Gradle Wrapper" | ||||
|         uses: gradle/wrapper-validation-action@v1 | ||||
|         uses: actions/checkout@v5 | ||||
|       - name: Validate Gradle Wrapper | ||||
|         uses: gradle/actions/wrapper-validation@v4 | ||||
|       - name: Setup Java | ||||
|         uses: actions/setup-java@v3 | ||||
|         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,21 +28,21 @@ 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/v6'}} | ||||
|         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 }} | ||||
| @@ -51,9 +52,9 @@ jobs: | ||||
|           destination-repository-name: 'plotsquared-javadocs' | ||||
|           user-email: ${{ secrets.USER_EMAIL }} | ||||
|           target-branch: main | ||||
|           target-directory: core | ||||
|           target-directory: v7/core | ||||
|       - name: Publish bukkit javadoc | ||||
|         if: ${{ runner.os == 'Linux' && env.STATUS == 'release' && github.event_name == 'push' && github.ref == 'refs/heads/v6'}} | ||||
|         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 }} | ||||
| @@ -63,4 +64,4 @@ jobs: | ||||
|           destination-repository-name: 'plotsquared-javadocs' | ||||
|           user-email: ${{ secrets.USER_EMAIL }} | ||||
|           target-branch: main | ||||
|           target-directory: bukkit | ||||
|           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@v2 | ||||
|         uses: github/codeql-action/init@v3 | ||||
|         with: | ||||
|           languages: ${{ matrix.language }} | ||||
|  | ||||
|       - name: Autobuild | ||||
|         uses: github/codeql-action/autobuild@v2 | ||||
|  | ||||
|         uses: github/codeql-action/autobuild@v3 | ||||
|       - name: Perform CodeQL Analysis | ||||
|         uses: github/codeql-action/analyze@v2 | ||||
|         uses: github/codeql-action/analyze@v3 | ||||
|   | ||||
							
								
								
									
										23
									
								
								.github/workflows/label-merge-conflicts.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								.github/workflows/label-merge-conflicts.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| name: "Label conflicting PRs" | ||||
| on: | ||||
|   push: | ||||
|   pull_request_target: | ||||
|     types: [ synchronize ] | ||||
|   pull_request: | ||||
|     types: [ synchronize ] | ||||
|  | ||||
| permissions: | ||||
|   pull-requests: write | ||||
|  | ||||
| jobs: | ||||
|   main: | ||||
|     if: github.event.pull_request.user.login != 'dependabot[bot]' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Label conflicting PRs | ||||
|         uses: eps1lon/actions-label-merge-conflict@v3.0.3 | ||||
|         with: | ||||
|           dirtyLabel: "unresolved-merge-conflict" | ||||
|           repoToken: "${{ secrets.GITHUB_TOKEN }}" | ||||
|           commentOnDirty: "Please take a moment and address the merge conflicts of your pull request. Thanks!" | ||||
|           continueOnMissingPermissions: true | ||||
							
								
								
									
										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 | ||||
|       - uses: release-drafter/release-drafter@v6 | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|   | ||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -137,3 +137,6 @@ docs/ | ||||
| build/ | ||||
|  | ||||
| .DS_Store | ||||
| # Ignore run folders | ||||
| run-[0-9].[0-9][0-9]/ | ||||
| run-[0-9].[0-9][0-9].[0-9]/ | ||||
|   | ||||
							
								
								
									
										154
									
								
								.idea/icon.svg
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										154
									
								
								.idea/icon.svg
									
									
									
										generated
									
									
									
								
							| @@ -1,48 +1,144 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <svg xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:cc="http://creativecommons.org/ns#" xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns:svg="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg" xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" version="1.1" id="svg2" xml:space="preserve" width="4000" height="3333.3333" viewBox="0 0 4000 3333.3333" sodipodi:docname="PlotSquared.ai"> | ||||
| 		<metadata id="metadata8"> | ||||
| <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=""> | ||||
| 				<cc:Work | ||||
|    rdf:about=""> | ||||
| 					<dc:format>image/svg+xml</dc:format> | ||||
|                     <dc:type rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> | ||||
|                     <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" /> | ||||
|     <defs | ||||
|    id="defs6"> | ||||
| 			<clipPath | ||||
|    clipPathUnits="userSpaceOnUse" | ||||
|    id="clipPath18"> | ||||
| 				<path | ||||
|    d="M 0,2500 H 3000 V 0 H 0 Z" | ||||
|    id="path16" /> | ||||
| 			</clipPath> | ||||
| 		</defs> | ||||
|     <sodipodi:namedview pagecolor="#ffffff" bordercolor="#666666" borderopacity="1" objecttolerance="10" gridtolerance="10" guidetolerance="10" inkscape:pageopacity="0" inkscape:pageshadow="2" inkscape:window-width="640" inkscape:window-height="480" id="namedview4" /> | ||||
|     <g id="g10" inkscape:groupmode="layer" inkscape:label="PlotSquared" transform="matrix(1.3333333,0,0,-1.3333333,0,3333.3333)"> | ||||
| 			<g id="g12"> | ||||
| 				<g id="g14" clip-path="url(#clipPath18)"> | ||||
| 					<g id="g20" transform="translate(1486.1511,2242.6453)"> | ||||
| 						<path d="m 0,0 c 16.533,10.846 33.211,21.453 50.104,31.699 78.972,-48.281 153.985,-102.704 227.269,-159.144 148.61,-115.422 287.884,-243.01 414.393,-382.333 111.39,-122.861 212.751,-255.152 298.898,-396.971 52.744,-87.322 100.544,-177.884 139.514,-272.214 -11.638,-3.551 -23.108,-7.655 -34.362,-12.286 l -0.24,0.288 c -11.135,12.982 -24.141,24.212 -34.915,37.506 -22.557,23.013 -45.425,45.737 -68.03,68.678 -19.725,20.253 -40.601,39.45 -58.958,60.974 -36.355,36.451 -72.517,73.093 -108.944,109.471 -22.628,26.013 -48.064,49.385 -71.965,74.197 -19.029,19.485 -38.706,38.346 -57.519,57.999 -12.166,14.998 -26.684,27.716 -39.93,41.658 -27.668,27.524 -54.903,55.479 -82.571,82.979 -23.924,27.956 -51.664,52.264 -76.692,79.164 -4.68,4.487 -8.855,10.774 -15.886,11.326 -22.34,34.027 -58.311,57.327 -97.377,67.502 -104.312,99.153 -215.487,191.202 -332.661,274.782 -117.942,-83.94 -229.476,-176.781 -334.484,-276.39 -26.684,-0.024 -53.368,0.024 -80.076,-0.024 0.024,-26.564 0.048,-53.104 0,-79.668 -72.229,-73.021 -139.491,-150.937 -202.385,-232.092 -63.758,-82.619 -121.973,-169.51 -173.541,-260.264 131.932,-69.061 257.864,-149.521 375.926,-240.275 0.096,-26.444 -0.12,-52.888 0.096,-79.332 l 0.744,-0.984 c 20.109,-24.14 43.409,-45.233 65.126,-67.861 15.118,-15.382 30.571,-30.404 45.569,-45.881 17.565,-20.733 37.698,-39.042 56.607,-58.503 19.917,-20.781 41.25,-40.218 59.967,-62.151 29.156,-29.299 58.167,-58.815 87.515,-87.922 29.155,-33.043 61.502,-63.111 92.169,-94.738 13.726,-12.67 25.124,-27.571 38.634,-40.457 25.029,-25.365 50.129,-50.657 75.325,-75.853 -37.914,-51.208 -73.741,-103.952 -107.192,-158.183 -167.83,273.317 -397.235,507.305 -662.37,687.158 -81.875,55.335 -167.23,105.584 -255.681,149.641 -52.815,26.276 -106.831,50.248 -162.239,70.381 99.393,233.628 242.795,446.715 410.289,636.79 93.562,106.088 194.634,205.433 301.466,298.13 C -217.335,-155.808 -111.439,-73.789 0,0" style="fill:#062f4c;fill-opacity:1;fill-rule:nonzero;stroke:none" id="path22" /> | ||||
|     <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 | ||||
|    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 | ||||
|    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 | ||||
|    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 | ||||
|    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 | ||||
|    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 | ||||
|    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 | ||||
|    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 | ||||
|    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> | ||||
|   | ||||
| Before Width: | Height: | Size: 11 KiB After Width: | Height: | Size: 12 KiB | 
| @@ -3,7 +3,7 @@ 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 { | ||||
| @@ -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) | ||||
|  | ||||
|     // Metrics | ||||
|     implementation("org.bstats:bstats-bukkit") | ||||
|     implementation(libs.bstatsBukkit) | ||||
|  | ||||
|     // Paper | ||||
|     compileOnly(libs.paper) | ||||
|     implementation("io.papermc:paperlib") | ||||
|     implementation(libs.paperlib) | ||||
|  | ||||
|     // Plugins | ||||
|     compileOnly(libs.worldeditBukkit) { | ||||
|         exclude(group = "org.bukkit") | ||||
|         exclude(group = "org.spigotmc") | ||||
|     } | ||||
|     compileOnly("com.fastasyncworldedit:FastAsyncWorldEdit-Bukkit") { isTransitive = false } | ||||
|     testImplementation("com.fastasyncworldedit:FastAsyncWorldEdit-Bukkit") { isTransitive = false } | ||||
|     compileOnly("com.github.MilkBowl:VaultAPI") { | ||||
|         exclude(group = "org.bukkit") | ||||
|     } | ||||
|     compileOnly(libs.worldeditBukkit) | ||||
|     compileOnly(libs.faweBukkit) { isTransitive = false } | ||||
|     testImplementation(libs.faweBukkit) { isTransitive = false } | ||||
|     compileOnly(libs.vault) | ||||
|     compileOnly(libs.placeholderapi) | ||||
|     compileOnly(libs.luckperms) | ||||
|     compileOnly(libs.essentialsx) | ||||
| @@ -44,15 +52,15 @@ dependencies { | ||||
|  | ||||
|     // Other libraries | ||||
|     implementation(libs.squirrelid) { isTransitive = false } | ||||
|     implementation("dev.notmyfault.serverlib:ServerLib") | ||||
|     implementation(libs.serverlib) | ||||
|  | ||||
|     // Our libraries | ||||
|     implementation(libs.arkitektonika) | ||||
|     implementation(libs.http4j) | ||||
|     implementation("com.intellectualsites.paster:Paster") | ||||
|     implementation(libs.paster) | ||||
|     implementation(libs.informativeAnnotations) | ||||
|  | ||||
|     // Adventure | ||||
|     implementation("net.kyori:adventure-platform-bukkit") | ||||
|     implementation(libs.adventureBukkit) | ||||
| } | ||||
|  | ||||
| tasks.processResources { | ||||
| @@ -62,10 +70,12 @@ 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") | ||||
| @@ -87,6 +97,7 @@ tasks.named<ShadowJar>("shadowJar") { | ||||
|     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() | ||||
| @@ -96,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://jd.papermc.io/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://intellectualsites.github.io/plotsquared-javadocs/core/") | ||||
|         opt.links("https://jd.adventure.kyori.net/api/4.9.3/") | ||||
|         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() | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -24,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; | ||||
| @@ -35,20 +34,22 @@ 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.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; | ||||
| @@ -71,6 +72,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; | ||||
| @@ -109,6 +112,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; | ||||
| @@ -131,12 +135,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; | ||||
| @@ -247,6 +253,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(); | ||||
|  | ||||
| @@ -290,6 +302,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) { | ||||
| @@ -346,7 +364,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); | ||||
|             } | ||||
| @@ -355,11 +379,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); | ||||
|                 } | ||||
|                 getServer().getPluginManager().registerEvents(injector().getInstance(PaperListener.class), this); | ||||
|             } else { | ||||
|                 getServer().getPluginManager().registerEvents(injector().getInstance(SpigotListener.class), this); | ||||
|             } | ||||
|             this.plotListener.startRunnable(); | ||||
|         } | ||||
| @@ -541,7 +563,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); | ||||
| @@ -656,20 +678,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()); | ||||
|  | ||||
| @@ -732,6 +749,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"); | ||||
| @@ -764,22 +786,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": | ||||
| @@ -797,13 +828,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(); | ||||
| @@ -812,8 +856,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; | ||||
|                                 } | ||||
| @@ -826,35 +869,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; | ||||
|                                     } | ||||
|  | ||||
| @@ -863,15 +904,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); | ||||
|                                             } | ||||
|                                         } | ||||
|                                     } | ||||
| @@ -880,11 +918,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()) | ||||
|                                             ); | ||||
|                                         } | ||||
|                                     } | ||||
| @@ -918,12 +956,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": | ||||
| @@ -932,7 +972,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": | ||||
| @@ -970,23 +1010,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); | ||||
|                                         } | ||||
|                                     } | ||||
|                                 } | ||||
| @@ -1000,6 +1041,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( | ||||
| @@ -1133,7 +1185,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 | ||||
| @@ -1144,6 +1198,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(); | ||||
| @@ -1160,18 +1215,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()); | ||||
| @@ -1221,15 +1295,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 | ||||
|   | ||||
| @@ -25,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; | ||||
| @@ -33,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; | ||||
| @@ -43,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; | ||||
| @@ -74,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); | ||||
|  | ||||
| @@ -101,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()); | ||||
| @@ -152,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()); | ||||
| @@ -165,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(); | ||||
| @@ -190,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(), | ||||
| @@ -238,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; | ||||
|                 } | ||||
| @@ -282,53 +255,38 @@ 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 // | ||||
|             } | ||||
|             // END LIVING // | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -431,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(); | ||||
| @@ -441,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(); | ||||
| @@ -448,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(); | ||||
| @@ -465,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; | ||||
| @@ -506,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) { | ||||
| @@ -582,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) { | ||||
| @@ -693,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 | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -24,7 +24,6 @@ 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.PlotAreaManager; | ||||
| import com.plotsquared.core.plot.world.SinglePlotArea; | ||||
| import com.plotsquared.core.queue.ZeroedDelegateScopedQueueCoordinator; | ||||
| import org.bukkit.generator.BlockPopulator; | ||||
| @@ -47,17 +46,6 @@ final class BlockStatePopulator extends BlockPopulator { | ||||
|         this.plotGenerator = plotGenerator; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @deprecated Use {@link BlockStatePopulator#BlockStatePopulator(IndependentPlotGenerator)} as plotAreManager is unused | ||||
|      */ | ||||
|     @Deprecated(forRemoval = true, since = "6.9.0") | ||||
|     public BlockStatePopulator( | ||||
|             final @NonNull IndependentPlotGenerator plotGenerator, | ||||
|             final @NonNull PlotAreaManager plotAreaManager | ||||
|     ) { | ||||
|         this.plotGenerator = plotGenerator; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void populate( | ||||
|             @NonNull final WorldInfo worldInfo, | ||||
|   | ||||
| @@ -34,9 +34,12 @@ 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; | ||||
| @@ -48,13 +51,17 @@ 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; | ||||
|  | ||||
| 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()); | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + BukkitPlotGenerator.class.getSimpleName()); | ||||
|  | ||||
|     @SuppressWarnings("unused") | ||||
|     public final boolean PAPER_ASYNC_SAFE = true; | ||||
| @@ -277,9 +284,14 @@ public class BukkitPlotGenerator extends ChunkGenerator implements GeneratorWrap | ||||
|         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 = "TODO") | ||||
|     @Deprecated(since = "7.0.0") | ||||
|     public @NonNull ChunkData generateChunkData( | ||||
|             @NonNull World world, @NonNull Random random, int x, int z, @NonNull BiomeGrid biome | ||||
|     ) { | ||||
| @@ -287,8 +299,8 @@ public class BukkitPlotGenerator extends ChunkGenerator implements GeneratorWrap | ||||
|             if (this.platformGenerator != this) { | ||||
|                 return this.platformGenerator.generateChunkData(world, random, x, z, biome); | ||||
|             } else { | ||||
|                 // Return super as it will throw an exception caught by the server that will mean this method is no longer used. | ||||
|                 return super.generateChunkData(world, random, x, z, biome); | ||||
|                 // 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."); | ||||
|             } | ||||
|         } | ||||
|  | ||||
| @@ -409,9 +421,14 @@ public class BukkitPlotGenerator extends ChunkGenerator implements GeneratorWrap | ||||
|         if (lastPlotArea != null && name.equals(this.levelName) && chunkX == lastChunkX && chunkZ == lastChunkZ) { | ||||
|             return lastPlotArea; | ||||
|         } | ||||
|         PlotArea area = UncheckedWorldLocation.at(name, chunkX << 4, 0, chunkZ << 4).getPlotArea(); | ||||
|         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", | ||||
|             throw new IllegalStateException(String.format( | ||||
|                     "Cannot generate chunk that does not belong to a plot area. World: %s", | ||||
|                     name | ||||
|             )); | ||||
|         } | ||||
| @@ -428,9 +445,16 @@ public class BukkitPlotGenerator extends ChunkGenerator implements GeneratorWrap | ||||
|         private static final List<Biome> BIOMES; | ||||
|  | ||||
|         static { | ||||
|             ArrayList<Biome> biomes = new ArrayList<>(List.of(Biome.values())); | ||||
|             biomes.remove(Biome.CUSTOM); | ||||
|             BIOMES = List.copyOf(biomes); | ||||
|             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 | ||||
|   | ||||
| @@ -33,7 +33,6 @@ import org.bukkit.Chunk; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.generator.BlockPopulator; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
|  | ||||
| import java.util.Random; | ||||
|  | ||||
| @@ -51,7 +50,7 @@ final class LegacyBlockStatePopulator extends BlockPopulator { | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void populate(@NotNull final World world, @NotNull final Random random, @NotNull final Chunk source) { | ||||
|     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(); | ||||
|   | ||||
| @@ -23,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; | ||||
| @@ -47,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; | ||||
| @@ -72,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()); | ||||
| @@ -128,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(); | ||||
|                 } | ||||
|                 return econHandler; | ||||
|             } catch (final Exception ignored) { | ||||
|             } | ||||
|         // Guice eagerly initializes singletons, so we need to bring the laziness ourselves | ||||
|         return new LazyEconHandler(); | ||||
|     } | ||||
|  | ||||
|     private static final class LazyEconHandler extends EconHandler implements ServerListener.MutableEconHandler { | ||||
|         private volatile EconHandler implementation; | ||||
|  | ||||
|         public void setImplementation(EconHandler econHandler) { | ||||
|             this.implementation = econHandler; | ||||
|         } | ||||
|         return EconHandler.nullEconHandler(); | ||||
|  | ||||
|         @Override | ||||
|         public boolean init() { | ||||
|             return get().init(); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public double getBalance(final PlotPlayer<?> player) { | ||||
|             return get().getBalance(player); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public void withdrawMoney(final PlotPlayer<?> player, final double amount) { | ||||
|             get().withdrawMoney(player, amount); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public void depositMoney(final PlotPlayer<?> player, final double amount) { | ||||
|             get().depositMoney(player, amount); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public void depositMoney(final OfflinePlotPlayer player, final double amount) { | ||||
|             get().depositMoney(player, amount); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public boolean isEnabled(final PlotArea plotArea) { | ||||
|             return get().isEnabled(plotArea); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public @NonNull String format(final double balance) { | ||||
|             return get().format(balance); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public boolean isSupported() { | ||||
|             return get().isSupported(); | ||||
|         } | ||||
|  | ||||
|         private EconHandler get() { | ||||
|             return Objects.requireNonNull(this.implementation, "EconHandler not ready yet."); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -24,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; | ||||
| @@ -33,6 +32,7 @@ import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.flag.implementations.BlockBurnFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.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; | ||||
| @@ -47,7 +47,6 @@ import com.plotsquared.core.plot.flag.implementations.LeafDecayFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.LiquidFlowFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.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; | ||||
| @@ -55,13 +54,12 @@ 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.Component; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
| import org.bukkit.Bukkit; | ||||
| @@ -71,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; | ||||
| @@ -89,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; | ||||
| @@ -108,20 +106,13 @@ import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import java.util.Iterator; | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
| import java.util.Set; | ||||
| 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 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; | ||||
|  | ||||
| @@ -150,111 +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; | ||||
|         } | ||||
|         if (event.getChangedType() == Material.COMPARATOR) { | ||||
|             if (!plot.getFlag(RedstoneFlag.class)) { | ||||
|                 event.setCancelled(true); | ||||
|                 plot.debug("Prevented comparator update because redstone = false"); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (PHYSICS_BLOCKS.contains(event.getChangedType())) { | ||||
|             if (plot.getFlag(DisablePhysicsFlag.class)) { | ||||
|                 event.setCancelled(true); | ||||
|                 plot.debug("Prevented block physics because disable-physics = true"); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (Settings.Redstone.DETECT_INVALID_EDGE_PISTONS) { | ||||
|             if (PISTONS.contains(block.getType())) { | ||||
|                 org.bukkit.block.data.Directional piston = (org.bukkit.block.data.Directional) block.getBlockData(); | ||||
|                 final BlockFace facing = piston.getFacing(); | ||||
|                 location = location.add(facing.getModX(), facing.getModY(), facing.getModZ()); | ||||
|                 Plot newPlot = area.getOwnedPlotAbs(location); | ||||
|                 if (!plot.equals(newPlot)) { | ||||
|                     event.setCancelled(true); | ||||
|                     plot.debug("Prevented piston update because of invalid edge piston detection"); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void blockCreate(BlockPlaceEvent event) { | ||||
|         Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||
| @@ -266,20 +152,12 @@ 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"), | ||||
|                         TagResolver.builder() | ||||
|                                 .tag("minheight", Tag.inserting(Component.text(area.getMinBuildHeight()))) | ||||
|                                 .tag("maxheight", Tag.inserting(Component.text(area.getMaxBuildHeight()))) | ||||
|                                 .build() | ||||
|                 ); | ||||
|                 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"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -299,7 +177,7 @@ 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"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -313,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") | ||||
|                     ); | ||||
| @@ -329,7 +207,7 @@ 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"), | ||||
|                     TagResolver.resolver( | ||||
| @@ -354,8 +232,7 @@ 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"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -366,21 +243,12 @@ public class BlockEventListener implements Listener { | ||||
|                     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"), | ||||
|                         TagResolver.builder() | ||||
|                                 .tag("minheight", Tag.inserting(Component.text(area.getMinBuildHeight()))) | ||||
|                                 .tag("maxheight", Tag.inserting(Component.text(area.getMaxBuildHeight()))) | ||||
|                                 .build() | ||||
|                 ); | ||||
|                 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; | ||||
| @@ -394,8 +262,7 @@ 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( | ||||
| @@ -407,7 +274,7 @@ public class BlockEventListener implements Listener { | ||||
|                 ); | ||||
|                 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") | ||||
|                     ); | ||||
| @@ -418,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")) { | ||||
| @@ -470,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); | ||||
| @@ -505,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")) { | ||||
| @@ -560,7 +428,7 @@ public class BlockEventListener implements Listener { | ||||
|         if (plot == null) { | ||||
|             return; | ||||
|         } | ||||
|         if (location.getY() >= area.getMaxBuildHeight() || location.getY() < area.getMinBuildHeight()) { | ||||
|         if (!area.buildRangeContainsY(location.getY())) { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
| @@ -577,6 +445,12 @@ public class BlockEventListener implements Listener { | ||||
|                 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) | ||||
| @@ -651,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 | ||||
| @@ -661,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); | ||||
| @@ -674,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() | ||||
| @@ -686,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); | ||||
| @@ -727,7 +603,7 @@ public class BlockEventListener implements Listener { | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (org.bukkit.Tag.CORAL_BLOCKS.isTagged(blockType) || org.bukkit.Tag.CORALS.isTagged(blockType)) { | ||||
|         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); | ||||
| @@ -735,6 +611,43 @@ public class BlockEventListener implements Listener { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @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 fromBlock = event.getBlock(); | ||||
| @@ -762,7 +675,7 @@ public class BlockEventListener implements Listener { | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (toLocation.getY() >= toArea.getMaxBuildHeight() || toLocation.getY() < toArea.getMinBuildHeight()) { | ||||
|         if (!toArea.buildRangeContainsY(toLocation.getY())) { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
| @@ -780,7 +693,10 @@ public class BlockEventListener implements Listener { | ||||
|         } | ||||
|  | ||||
|         if (toPlot != null) { | ||||
|             if (!toArea.contains(fromLocation.getX(), fromLocation.getZ()) || !Objects.equals(toPlot, toArea.getOwnedPlot(fromLocation))) { | ||||
|             if (!toArea.contains(fromLocation.getX(), fromLocation.getZ()) || !Objects.equals( | ||||
|                     toPlot, | ||||
|                     toArea.getOwnedPlot(fromLocation) | ||||
|             )) { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
| @@ -796,7 +712,10 @@ public class BlockEventListener implements Listener { | ||||
|                 toPlot.debug("Liquid could not flow because liquid-flow = disabled"); | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } else if (!toArea.contains(fromLocation.getX(), fromLocation.getZ()) || !Objects.equals(null, toArea.getOwnedPlot(fromLocation))) { | ||||
|         } 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(); | ||||
| @@ -836,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) { | ||||
| @@ -879,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; | ||||
|             } | ||||
| @@ -913,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; | ||||
|                 } | ||||
| @@ -929,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; | ||||
|             } | ||||
| @@ -946,6 +871,9 @@ 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", | ||||
| @@ -957,10 +885,15 @@ public class BlockEventListener implements Listener { | ||||
|                 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); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| @@ -1000,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); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| @@ -1048,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())))); | ||||
|     } | ||||
| @@ -1091,8 +1029,14 @@ 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"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -1103,7 +1047,9 @@ public class BlockEventListener implements Listener { | ||||
|                     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"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -1114,7 +1060,7 @@ public class BlockEventListener implements Listener { | ||||
|                     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"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -1207,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()) { | ||||
| @@ -1238,8 +1187,7 @@ 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( | ||||
| @@ -1249,21 +1197,15 @@ public class BlockEventListener implements Listener { | ||||
|                 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"), | ||||
|                         TagResolver.builder() | ||||
|                                 .tag("minheight", Tag.inserting(Component.text(area.getMinBuildHeight()))) | ||||
|                                 .tag("maxheight", Tag.inserting(Component.text(area.getMaxBuildHeight()))) | ||||
|                                 .build() | ||||
|                 ); | ||||
|             if (area.notifyIfOutsideBuildArea(pp, currentLocation.getY())) { | ||||
|                 event.setCancelled(true); | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -26,7 +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 org.bukkit.Material; | ||||
| 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; | ||||
| @@ -40,31 +41,11 @@ import org.bukkit.event.block.BlockReceiveGameEvent; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
| import java.util.Set; | ||||
| import java.util.UUID; | ||||
|  | ||||
| @SuppressWarnings("unused") | ||||
| public class BlockEventListener117 implements Listener { | ||||
|  | ||||
|     private static final Set<Material> COPPER_OXIDIZING = Set.of( | ||||
|             Material.COPPER_BLOCK, | ||||
|             Material.EXPOSED_COPPER, | ||||
|             Material.WEATHERED_COPPER, | ||||
|             Material.OXIDIZED_COPPER, | ||||
|             Material.CUT_COPPER, | ||||
|             Material.EXPOSED_CUT_COPPER, | ||||
|             Material.WEATHERED_CUT_COPPER, | ||||
|             Material.OXIDIZED_CUT_COPPER, | ||||
|             Material.CUT_COPPER_STAIRS, | ||||
|             Material.EXPOSED_CUT_COPPER_STAIRS, | ||||
|             Material.WEATHERED_CUT_COPPER_STAIRS, | ||||
|             Material.OXIDIZED_CUT_COPPER_STAIRS, | ||||
|             Material.CUT_COPPER_SLAB, | ||||
|             Material.EXPOSED_CUT_COPPER_SLAB, | ||||
|             Material.WEATHERED_CUT_COPPER_SLAB, | ||||
|             Material.OXIDIZED_CUT_COPPER_SLAB | ||||
|     ); | ||||
|  | ||||
|     @Inject | ||||
|     public BlockEventListener117() { | ||||
|     } | ||||
| @@ -80,13 +61,27 @@ public class BlockEventListener117 implements Listener { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         BukkitPlayer plotPlayer = null; | ||||
|  | ||||
|         if (entity instanceof Player player) { | ||||
|             plotPlayer = BukkitUtil.adapt(player); | ||||
|             if (area.notifyIfOutsideBuildArea(plotPlayer, location.getY())) { | ||||
|                 event.setCancelled(true); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot == null || !plot.getFlag(MiscInteractFlag.class)) { | ||||
|             if (entity instanceof Player player) { | ||||
|                 BukkitPlayer plotPlayer = BukkitUtil.adapt(player); | ||||
|         if (plot == null && !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); | ||||
|                     } | ||||
|                 } | ||||
| @@ -95,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); | ||||
|                         } | ||||
|                     } | ||||
| @@ -110,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) { | ||||
| @@ -128,27 +130,19 @@ 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; | ||||
|                 } | ||||
|                 if (!area.buildRangeContainsY(location.getY())) { | ||||
|                     event.getBlocks().remove(i); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         Plot origin = area.getPlot(location); | ||||
|         if (origin == null) { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
|         } | ||||
|         for (int i = blocks.size() - 1; i >= 0; i--) { | ||||
|             location = BukkitUtil.adapt(blocks.get(i).getLocation()); | ||||
|             Plot plot = area.getOwnedPlot(location); | ||||
|             if (!Objects.equals(plot, origin) && (!plot.isMerged() && !origin.isMerged())) { | ||||
|                 event.getBlocks().remove(i); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| @@ -169,7 +163,7 @@ public class BlockEventListener117 implements Listener { | ||||
|         if (plot == null) { | ||||
|             return; | ||||
|         } | ||||
|         if (COPPER_OXIDIZING.contains(event.getNewState().getType())) { | ||||
|         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); | ||||
|   | ||||
| @@ -26,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; | ||||
| @@ -64,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; | ||||
| @@ -79,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) { | ||||
| @@ -90,22 +92,27 @@ public class ChunkListener implements Listener { | ||||
|         } | ||||
|         try { | ||||
|             RefClass classCraftWorld = getRefClass("{cb}.CraftWorld"); | ||||
|             this.methodGetHandleWorld = classCraftWorld.getMethod("getHandle"); | ||||
|             RefClass classCraftChunk = getRefClass("{cb}.CraftChunk"); | ||||
|             this.methodGetHandleChunk = classCraftChunk.getMethod("getHandle"); | ||||
|             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(); | ||||
| @@ -167,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); | ||||
| @@ -234,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; | ||||
|                 } | ||||
|   | ||||
| @@ -19,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; | ||||
| @@ -35,11 +36,15 @@ import com.plotsquared.core.plot.flag.implementations.DisablePhysicsFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.EntityChangeBlockFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.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; | ||||
| @@ -53,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; | ||||
| @@ -73,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) { | ||||
| @@ -143,56 +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 "ENDER_PEARL": | ||||
|             case "TRAP": | ||||
|             case "VILLAGE_DEFENSE": | ||||
|             case "VILLAGE_INVASION": | ||||
|             case "BEEHIVE": | ||||
|             case "CHUNK_GEN": | ||||
|             } | ||||
|             case "REINFORCEMENTS", "NATURAL", "MOUNT", "PATROL", "RAID", "SHEARED", "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) { | ||||
| @@ -249,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) { | ||||
| @@ -261,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; | ||||
| @@ -296,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)); | ||||
|                             } | ||||
| @@ -320,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); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -370,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(); | ||||
| @@ -408,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); | ||||
|         } | ||||
|   | ||||
| @@ -31,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; | ||||
| @@ -120,22 +122,28 @@ 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(); | ||||
|         EntityType type = entity.getType(); | ||||
|         if (plot == null) { | ||||
|             if (entity instanceof Item) { | ||||
|                 if (Settings.Enabled_Components.KILL_ROAD_ITEMS) { | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
|                 return; | ||||
|             } | ||||
|             if (!area.isMobSpawning()) { | ||||
|                 if (type == EntityType.PLAYER) { | ||||
|                     return; | ||||
|                 } | ||||
|                 if (type == EntityType.DROPPED_ITEM) { | ||||
|                     if (Settings.Enabled_Components.KILL_ROAD_ITEMS) { | ||||
|                         event.setCancelled(true); | ||||
|                     } | ||||
|                     return; | ||||
|                 } | ||||
|                 if (type.isAlive()) { | ||||
|                     event.setCancelled(true); | ||||
|                 } | ||||
| @@ -148,7 +156,7 @@ public class EntitySpawnListener implements Listener { | ||||
|         if (Settings.Done.RESTRICT_BUILDING && DoneFlag.isDone(plot)) { | ||||
|             event.setCancelled(true); | ||||
|         } | ||||
|         if (type == EntityType.ENDER_CRYSTAL) { | ||||
|         if (entity instanceof EnderCrystal || type == EntityType.ARMOR_STAND) { | ||||
|             if (BukkitEntityUtil.checkEntity(entity, plot)) { | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|   | ||||
| @@ -18,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; | ||||
| @@ -26,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; | ||||
|  | ||||
| @@ -39,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())) { | ||||
| @@ -54,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())) { | ||||
| @@ -104,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)); | ||||
|                     } | ||||
| @@ -115,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"); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -18,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; | ||||
| @@ -27,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; | ||||
| @@ -36,14 +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 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; | ||||
| @@ -51,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; | ||||
| @@ -71,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; | ||||
|  | ||||
| @@ -79,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 | ||||
| @@ -120,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; | ||||
| @@ -135,7 +153,6 @@ public class PaperListener implements Listener { | ||||
|         if (farea == null) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (tarea != farea) { | ||||
|             event.setCancelled(true); | ||||
|             return; | ||||
| @@ -146,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); | ||||
| @@ -162,86 +179,72 @@ public class PaperListener implements Listener { | ||||
|         } | ||||
|         Location location = BukkitUtil.adapt(event.getSpawnLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (!location.isPlotArea()) { | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         //If entities are spawning... the chunk should be loaded? | ||||
|         // Armour-stands are handled elsewhere and should not be handled by area-wide entity-spawn options | ||||
|         if (event.getType() == EntityType.ARMOR_STAND) { | ||||
|             return; | ||||
|         } | ||||
|         // If entities are spawning... the chunk should be loaded? | ||||
|         Entity[] entities = event.getSpawnLocation().getChunk().getEntities(); | ||||
|         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 "ENDER_PEARL": | ||||
|             case "TRAP": | ||||
|             case "VILLAGE_DEFENSE": | ||||
|             case "VILLAGE_INVASION": | ||||
|             case "BEEHIVE": | ||||
|             case "CHUNK_GEN": | ||||
|             } | ||||
|             case "REINFORCEMENTS", "NATURAL", "MOUNT", "PATROL", "RAID", "SHEARED", "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(); | ||||
|             // 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.setCancelled(true); | ||||
|                 } | ||||
|                 return; | ||||
|             } | ||||
|             if (!area.isMobSpawning()) { | ||||
|                 if (type == EntityType.PLAYER) { | ||||
|                     return; | ||||
|                 } | ||||
|                 if (type == EntityType.DROPPED_ITEM) { | ||||
|                     if (Settings.Enabled_Components.KILL_ROAD_ITEMS) { | ||||
|                         event.setShouldAbortSpawn(true); | ||||
|                         event.setCancelled(true); | ||||
|                     } | ||||
|                     return; | ||||
|                 } | ||||
|                 if (type.isAlive()) { | ||||
|                     event.setShouldAbortSpawn(true); | ||||
|                     event.setCancelled(true); | ||||
| @@ -324,14 +327,17 @@ 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"), | ||||
|                         TagResolver.resolver( | ||||
| @@ -343,7 +349,7 @@ public class PaperListener implements Listener { | ||||
|                 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"), | ||||
|                         TagResolver.resolver( | ||||
| @@ -355,8 +361,13 @@ public class PaperListener implements Listener { | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } else if (!plot.isAdded(pp.getUUID())) { | ||||
|             if (entity.getType().getKey().equals(FISHING_BOBBER)) { | ||||
|                 if (plot.getFlag(FishingFlag.class)) { | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|             if (!plot.getFlag(ProjectilesFlag.class)) { | ||||
|                 if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|                     pp.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -411,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,89 +0,0 @@ | ||||
| /* | ||||
|  * 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.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.Component; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
| 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"), | ||||
|                     TagResolver.resolver("amount", Tag.inserting(Component.text(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); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -28,13 +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.Component; | ||||
| 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; | ||||
| @@ -46,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; | ||||
| @@ -91,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(); | ||||
| @@ -99,14 +101,17 @@ 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"), | ||||
|                         TagResolver.resolver( | ||||
| @@ -118,7 +123,7 @@ public class ProjectileEventListener implements Listener { | ||||
|                 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"), | ||||
|                         TagResolver.resolver( | ||||
| @@ -130,8 +135,13 @@ public class ProjectileEventListener implements Listener { | ||||
|                 event.setCancelled(true); | ||||
|             } | ||||
|         } else if (!plot.isAdded(pp.getUUID())) { | ||||
|             if (entity instanceof FishHook) { | ||||
|                 if (plot.getFlag(FishingFlag.class)) { | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|             if (!plot.getFlag(ProjectilesFlag.class)) { | ||||
|                 if (!Permissions.hasPermission(pp, Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|                 if (!pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER)) { | ||||
|                     pp.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission_event"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -148,54 +158,79 @@ public class ProjectileEventListener implements Listener { | ||||
|  | ||||
|     @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) || plot.getFlag(ProjectilesFlag.class)) { | ||||
|                 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; | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -21,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; | ||||
| @@ -32,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 | ||||
| @@ -45,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); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -31,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(); | ||||
|                 } | ||||
|             } 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(); | ||||
|             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); | ||||
|             } | ||||
|         } 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(); | ||||
|         } | ||||
| @@ -85,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()); | ||||
|     } | ||||
|  | ||||
| @@ -96,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,63 +0,0 @@ | ||||
| /* | ||||
|  * 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.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"; | ||||
|     } | ||||
|  | ||||
| } | ||||
|  */ | ||||
| @@ -30,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 { | ||||
|  | ||||
|   | ||||
| @@ -52,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) { | ||||
|   | ||||
| @@ -20,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; | ||||
| @@ -83,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); | ||||
|     } | ||||
|   | ||||
| @@ -43,7 +43,7 @@ public class BukkitOfflinePlayer implements OfflinePlotPlayer { | ||||
|      */ | ||||
|     public BukkitOfflinePlayer( | ||||
|             final @NonNull OfflinePlayer player, final @NonNull | ||||
|             PermissionHandler permissionHandler | ||||
|     PermissionHandler permissionHandler | ||||
|     ) { | ||||
|         this.player = player; | ||||
|         this.permissionProfile = permissionHandler.getPermissionProfile(this) | ||||
| @@ -87,4 +87,9 @@ public class BukkitOfflinePlayer implements OfflinePlotPlayer { | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean hasPermission(@NonNull final String permission, final boolean notify) { | ||||
|         return hasPermission(permission); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -32,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; | ||||
| @@ -40,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; | ||||
| @@ -47,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; | ||||
| @@ -67,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 | ||||
|     ) { | ||||
| @@ -129,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); | ||||
| @@ -167,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())) { | ||||
| @@ -175,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; | ||||
| @@ -185,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; | ||||
| @@ -223,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 = | ||||
| @@ -311,18 +313,21 @@ 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 | ||||
| @@ -348,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) { | ||||
|   | ||||
| @@ -70,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 | ||||
|   | ||||
| @@ -30,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; | ||||
| @@ -41,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; | ||||
|  | ||||
| @@ -55,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; | ||||
| @@ -70,6 +76,7 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | ||||
|     private final AtomicInteger expectedSize; | ||||
|     private final AtomicInteger loadingChunks = new AtomicInteger(); | ||||
|     private final boolean forceSync; | ||||
|     private final boolean shouldGen; | ||||
|  | ||||
|     private int batchSize; | ||||
|     private PlotSquaredTask task; | ||||
| @@ -87,7 +94,8 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | ||||
|             @Assisted final @NonNull Consumer<Throwable> throwableConsumer, | ||||
|             @Assisted("unloadAfter") final boolean unloadAfter, | ||||
|             @Assisted final @NonNull Collection<ProgressSubscriber> progressSubscribers, | ||||
|             @Assisted("forceSync") final boolean forceSync | ||||
|             @Assisted("forceSync") final boolean forceSync, | ||||
|             @Assisted("shouldGen") final boolean shouldGen | ||||
|     ) { | ||||
|         this.requestedChunks = new LinkedBlockingQueue<>(requestedChunks); | ||||
|         this.availableChunks = new LinkedBlockingQueue<>(); | ||||
| @@ -103,6 +111,7 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | ||||
|         this.bukkitWorld = Bukkit.getWorld(world.getName()); | ||||
|         this.progressSubscribers.addAll(progressSubscribers); | ||||
|         this.forceSync = forceSync; | ||||
|         this.shouldGen = shouldGen; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -212,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(); | ||||
|                             // We want one less because this couldn't be processed | ||||
|                             this.expectedSize.decrementAndGet(); | ||||
|                             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 { | ||||
| @@ -239,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; | ||||
|         } | ||||
|   | ||||
| @@ -62,19 +62,28 @@ public class BukkitQueueCoordinator extends BasicQueueCoordinator { | ||||
|     private static final SideEffectSet EDGE_LIGHTING_SIDE_EFFECT_SET; | ||||
|  | ||||
|     static { | ||||
|         NO_SIDE_EFFECT_SET = SideEffectSet.none().with(SideEffect.LIGHTING, SideEffect.State.OFF).with( | ||||
|                 SideEffect.NEIGHBORS, | ||||
|                 SideEffect.State.OFF | ||||
|         ); | ||||
|         EDGE_SIDE_EFFECT_SET = SideEffectSet.none().with(SideEffect.UPDATE, SideEffect.State.ON).with( | ||||
|                 SideEffect.NEIGHBORS, | ||||
|                 SideEffect.State.ON | ||||
|         ); | ||||
|         LIGHTING_SIDE_EFFECT_SET = SideEffectSet.none().with(SideEffect.NEIGHBORS, SideEffect.State.OFF); | ||||
|         EDGE_LIGHTING_SIDE_EFFECT_SET = SideEffectSet.none().with(SideEffect.UPDATE, SideEffect.State.ON).with( | ||||
|                 SideEffect.NEIGHBORS, | ||||
|                 SideEffect.State.ON | ||||
|         ); | ||||
|         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; | ||||
| @@ -229,6 +238,7 @@ public class BukkitQueueCoordinator extends BasicQueueCoordinator { | ||||
|                         .unloadAfter(isUnloadAfter()) | ||||
|                         .withProgressSubscribers(getProgressSubscribers()) | ||||
|                         .forceSync(isForceSync()) | ||||
|                         .shouldGen(isShouldGen()) | ||||
|                         .build(); | ||||
|         return super.enqueue(); | ||||
|     } | ||||
| @@ -270,7 +280,7 @@ public class BukkitQueueCoordinator extends BasicQueueCoordinator { | ||||
|             if (getChunkObject() instanceof Chunk chunkObject) { | ||||
|                 existing = chunkObject.getBlock(x & 15, y, z & 15); | ||||
|             } else { | ||||
|                  existing = getBukkitWorld().getBlockAt(x, y, z); | ||||
|                 existing = getBukkitWorld().getBlockAt(x, y, z); | ||||
|             } | ||||
|             final BlockState existingBaseBlock = BukkitAdapter.adapt(existing.getBlockData()); | ||||
|             if (BukkitBlockUtil.get(existing).equals(existingBaseBlock) && existing.getBlockData().matches(blockData)) { | ||||
|   | ||||
| @@ -19,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.ZeroedDelegateScopedQueueCoordinator; | ||||
| import com.plotsquared.core.util.AnnotationHelper; | ||||
| import com.plotsquared.core.util.ChunkUtil; | ||||
| import com.plotsquared.core.util.PatternUtil; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| @@ -44,7 +44,10 @@ import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| import java.util.Arrays; | ||||
|  | ||||
| @AnnotationHelper.ApiDescription(info = "Internal use only. Subject to changes at any time.") | ||||
| /** | ||||
|  * Internal use only. Subject to changes at any time. | ||||
|  */ | ||||
| @DoNotUse | ||||
| public class GenChunk extends ZeroedDelegateScopedQueueCoordinator { | ||||
|  | ||||
|     public final Biome[] biomes; | ||||
| @@ -59,7 +62,6 @@ public class GenChunk extends ZeroedDelegateScopedQueueCoordinator { | ||||
|     /** | ||||
|      * @param minY minimum world Y, inclusive | ||||
|      * @param maxY maximum world Y, inclusive | ||||
|      * | ||||
|      * @since 6.6.0 | ||||
|      */ | ||||
|     public GenChunk(int minY, int maxY) { | ||||
| @@ -107,9 +109,9 @@ public class GenChunk extends ZeroedDelegateScopedQueueCoordinator { | ||||
|      */ | ||||
|     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 | ||||
| @@ -183,7 +185,11 @@ public class GenChunk extends ZeroedDelegateScopedQueueCoordinator { | ||||
|  | ||||
|     @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 | ||||
|   | ||||
| @@ -44,6 +44,7 @@ public class LimitedRegionWrapperQueue extends DelegateQueueCoordinator { | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + LimitedRegionWrapperQueue.class.getSimpleName()); | ||||
|  | ||||
|     private final LimitedRegion limitedRegion; | ||||
|     private boolean useOtherRestoreTagMethod = false; | ||||
|  | ||||
|     /** | ||||
|      * @since 6.9.0 | ||||
| @@ -65,10 +66,18 @@ public class LimitedRegionWrapperQueue extends DelegateQueueCoordinator { | ||||
|             CompoundTag tag = id.getNbtData(); | ||||
|             StateWrapper sw = new StateWrapper(tag); | ||||
|             try { | ||||
|                 sw.restoreTag(limitedRegion.getBlockState(x, y, z).getBlock()); | ||||
|                 if (useOtherRestoreTagMethod && getWorld() != null) { | ||||
|                     sw.restoreTag(getWorld().getName(), x, y, z); | ||||
|                 } else { | ||||
|                     sw.restoreTag(limitedRegion.getBlockState(x, y, z).getBlock()); | ||||
|                 } | ||||
|             } catch (IllegalArgumentException e) { | ||||
|                 LOGGER.error("Error attempting to populate tile entity into the world at location {},{},{}", x, y, z, e); | ||||
|                 return false; | ||||
|             } catch (IllegalStateException e) { | ||||
|                 useOtherRestoreTagMethod = true; | ||||
|                 LOGGER.warn("IllegalStateException attempting to populate tile entity into the world at location {},{},{}. " + | ||||
|                         "Possibly on <=1.17.1, switching to secondary method.", x, y, z, e); | ||||
|             } | ||||
|         } | ||||
|         return result; | ||||
|   | ||||
| @@ -18,6 +18,8 @@ | ||||
|  */ | ||||
| package com.plotsquared.bukkit.schematic; | ||||
|  | ||||
| import com.destroystokyo.paper.profile.PlayerProfile; | ||||
| import com.destroystokyo.paper.profile.ProfileProperty; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.sk89q.jnbt.ByteTag; | ||||
| import com.sk89q.jnbt.CompoundTag; | ||||
| @@ -28,16 +30,22 @@ 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 io.papermc.lib.PaperLib; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.ChatColor; | ||||
| import org.bukkit.DyeColor; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.block.Banner; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.block.Container; | ||||
| import org.bukkit.block.Sign; | ||||
| import org.bukkit.block.Skull; | ||||
| import org.bukkit.block.banner.Pattern; | ||||
| import org.bukkit.block.banner.PatternType; | ||||
| 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; | ||||
|  | ||||
| @@ -46,20 +54,15 @@ import java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.Map.Entry; | ||||
| import java.util.Objects; | ||||
| import java.util.UUID; | ||||
|  | ||||
| public class StateWrapper { | ||||
|  | ||||
|     public org.bukkit.block.BlockState state = null; | ||||
|     public CompoundTag tag = null; | ||||
|     public CompoundTag tag; | ||||
|  | ||||
|     /** | ||||
|      * @deprecated in favour of using WE methods for obtaining NBT, specifically by obtaining a | ||||
|      *         {@link com.sk89q.worldedit.world.block.BaseBlock} and then using {@link com.sk89q.worldedit.world.block.BaseBlock#getNbtData()} | ||||
|      */ | ||||
|     @Deprecated(forRemoval = true, since = "6.9.0") | ||||
|     public StateWrapper(org.bukkit.block.BlockState state) { | ||||
|         this.state = state; | ||||
|     } | ||||
|     private boolean paperErrorTextureSent = false; | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + StateWrapper.class.getSimpleName()); | ||||
|  | ||||
|     public StateWrapper(CompoundTag tag) { | ||||
|         this.tag = tag; | ||||
| @@ -237,15 +240,64 @@ public class StateWrapper { | ||||
|                         return true; | ||||
|                     } | ||||
|                     String player = skullOwner.getString("Name"); | ||||
|                     if (player == null || player.isEmpty()) { | ||||
|  | ||||
|                     if (player != null && !player.isEmpty()) { | ||||
|                         try { | ||||
|                             skull.setOwningPlayer(Bukkit.getOfflinePlayer(player)); | ||||
|                             skull.update(true); | ||||
|                         } catch (Exception e) { | ||||
|                             e.printStackTrace(); | ||||
|                         } | ||||
|                         return true; | ||||
|                     } | ||||
|  | ||||
|                     final CompoundTag properties = (CompoundTag) skullOwner.getValue().get("Properties"); | ||||
|                     if (properties == null) { | ||||
|                         return false; | ||||
|                     } | ||||
|                     try { | ||||
|                         skull.setOwningPlayer(Bukkit.getOfflinePlayer(player)); | ||||
|                         skull.update(true); | ||||
|                     } catch (Exception e) { | ||||
|                         e.printStackTrace(); | ||||
|                     final ListTag textures = properties.getListTag("textures"); | ||||
|                     if (textures.getValue().isEmpty()) { | ||||
|                         return false; | ||||
|                     } | ||||
|                     final CompoundTag textureCompound = (CompoundTag) textures.getValue().get(0); | ||||
|                     if (textureCompound == null) { | ||||
|                         return false; | ||||
|                     } | ||||
|                     String textureValue = textureCompound.getString("Value"); | ||||
|                     if (textureValue == null) { | ||||
|                         return false; | ||||
|                     } | ||||
|                     if (!PaperLib.isPaper()) { | ||||
|                         if (!paperErrorTextureSent) { | ||||
|                             paperErrorTextureSent = true; | ||||
|                             LOGGER.error("Failed to populate skull data in your road schematic - This is a Spigot limitation."); | ||||
|                         } | ||||
|                         return false; | ||||
|                     } | ||||
|                     final PlayerProfile profile = Bukkit.createProfile(UUID.randomUUID()); | ||||
|                     profile.setProperty(new ProfileProperty("textures", textureValue)); | ||||
|                     skull.setPlayerProfile(profile); | ||||
|                     skull.update(true); | ||||
|                     return true; | ||||
|  | ||||
|                 } | ||||
|                 return false; | ||||
|             } | ||||
|             case "banner" -> { | ||||
|                 if (state instanceof Banner banner) { | ||||
|                     List<Tag> patterns = this.tag.getListTag("Patterns").getValue(); | ||||
|                     if (patterns == null || patterns.isEmpty()) { | ||||
|                         return false; | ||||
|                     } | ||||
|                     banner.setPatterns(patterns.stream().map(t -> (CompoundTag) t).map(compoundTag -> { | ||||
|                         DyeColor color = DyeColor.getByWoolData((byte) compoundTag.getInt("Color")); | ||||
|                         PatternType patternType = PatternType.getByIdentifier(compoundTag.getString("Pattern")); | ||||
|                         if (color == null || patternType == null) { | ||||
|                             return null; | ||||
|                         } | ||||
|                         return new Pattern(color, patternType); | ||||
|                     }).filter(Objects::nonNull).toList()); | ||||
|                     banner.update(true); | ||||
|                     return true; | ||||
|                 } | ||||
|                 return false; | ||||
| @@ -254,26 +306,6 @@ public class StateWrapper { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get a CompoundTag of the contents of a block's inventory (chest, furnace, etc.). | ||||
|      * | ||||
|      * @deprecated in favour of using WorldEdit methods for obtaining NBT, specifically by obtaining a | ||||
|      *         {@link com.sk89q.worldedit.world.block.BaseBlock} and then using {@link com.sk89q.worldedit.world.block.BaseBlock#getNbtData()} | ||||
|      */ | ||||
|     @Deprecated(forRemoval = true, since = "6.9.0") | ||||
|     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; | ||||
|     } | ||||
|  | ||||
|     public String getId() { | ||||
|         String tileid = this.tag.getString("id").toLowerCase(); | ||||
|         if (tileid.startsWith("minecraft:")) { | ||||
|   | ||||
| @@ -40,7 +40,6 @@ 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.Component; | ||||
| @@ -50,6 +49,7 @@ 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; | ||||
| @@ -170,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") | ||||
|                             ); | ||||
| @@ -180,7 +179,7 @@ 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"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -195,7 +194,7 @@ 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"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -219,7 +218,7 @@ 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"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -243,7 +242,7 @@ 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"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -259,8 +258,7 @@ 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"), | ||||
|                                 TagResolver.resolver( | ||||
| @@ -277,7 +275,7 @@ 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"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -297,7 +295,7 @@ 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"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -322,7 +320,7 @@ 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"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -344,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; | ||||
|             } | ||||
|         } | ||||
| @@ -357,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; | ||||
|   | ||||
| @@ -44,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; | ||||
|   | ||||
| @@ -70,7 +70,7 @@ public class BukkitRegionManager extends RegionManager { | ||||
|     @Inject | ||||
|     public BukkitRegionManager( | ||||
|             @NonNull WorldUtil worldUtil, @NonNull GlobalBlockQueue blockQueue, @NonNull | ||||
|             ProgressSubscriberFactory subscriberFactory | ||||
|     ProgressSubscriberFactory subscriberFactory | ||||
|     ) { | ||||
|         super(worldUtil, blockQueue, subscriberFactory); | ||||
|         this.blockQueue = blockQueue; | ||||
|   | ||||
| @@ -67,6 +67,7 @@ public class BukkitSetupUtils extends SetupUtils { | ||||
|         this.worldFile = worldFile; | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("deprecation") // Paper deprecation | ||||
|     @Override | ||||
|     public void updateGenerators(final boolean force) { | ||||
|         if (loaded && !SetupUtils.generators.isEmpty() && !force) { | ||||
| @@ -124,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(); | ||||
|   | ||||
| @@ -58,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; | ||||
| @@ -74,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; | ||||
| @@ -260,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) { | ||||
| @@ -431,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); | ||||
| @@ -438,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); | ||||
| @@ -466,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<>(); | ||||
|   | ||||
| @@ -23,11 +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; | ||||
|  | ||||
| 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 { | ||||
| @@ -41,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); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -68,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; | ||||
|     } | ||||
|  | ||||
| @@ -97,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 | ||||
|   | ||||
| @@ -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(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -35,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 { | ||||
|  | ||||
| @@ -54,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() | ||||
|   | ||||
| @@ -36,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 { | ||||
| @@ -46,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); | ||||
|     } | ||||
|  | ||||
| @@ -63,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 | ||||
| @@ -76,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)) { | ||||
| @@ -115,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); | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -29,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; | ||||
| @@ -40,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); | ||||
|     } | ||||
|  | ||||
| @@ -75,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); | ||||
|     } | ||||
|  | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -111,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) { | ||||
|   | ||||
| @@ -4,16 +4,16 @@ api-version: "1.13" | ||||
| version: "${version}" | ||||
| load: STARTUP | ||||
| description: "Easy, yet powerful Plot World generation and management." | ||||
| authors: [Citymonstret, Empire92, MattBDev, dordsor21, NotMyFault, SirYwell] | ||||
| authors: [ Citymonstret, Empire92, MattBDev, dordsor21, NotMyFault, SirYwell ] | ||||
| website: https://www.spigotmc.org/resources/77506/ | ||||
| softdepend: [Vault, PlaceholderAPI, Essentials, LuckPerms, BungeePerms, MVdWPlaceholderAPI] | ||||
| loadbefore: [MultiWorld, Multiverse-Core] | ||||
| depend: [WorldEdit] | ||||
| softdepend: [ Vault, PlaceholderAPI, Essentials, LuckPerms, BungeePerms, MVdWPlaceholderAPI ] | ||||
| loadbefore: [ MultiWorld, Multiverse-Core ] | ||||
| depend: [ WorldEdit ] | ||||
| database: false | ||||
| commands: | ||||
|   plots: | ||||
|     description: Plot command. | ||||
|     aliases: [p,plot,ps,plotsquared,p2,2,plotme] | ||||
|     aliases: [ p,plot,ps,plotsquared,p2,2,plotme ] | ||||
|     permission: plots.use | ||||
|     permission-message: "You are lacking the permission node 'plots.use'" | ||||
| permissions: | ||||
|   | ||||
| @@ -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@intellectualsites.com. All | ||||
| complaints will be reviewed and investigated and will result in a response that | ||||
| is deemed necessary and appropriate to the circumstances. The project team is | ||||
| obligated to maintain confidentiality with regard to the reporter of an incident. | ||||
| Further details of specific enforcement policies may be posted separately. | ||||
|  | ||||
| Project maintainers who do not follow or enforce the Code of Conduct in good | ||||
| faith may face temporary or permanent repercussions as determined by other | ||||
| members of the project's leadership. | ||||
|  | ||||
| ## Attribution | ||||
|  | ||||
| This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4, | ||||
| available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html | ||||
|  | ||||
| [homepage]: https://www.contributor-covenant.org | ||||
|  | ||||
| For answers to common questions about this code of conduct, see | ||||
| https://www.contributor-covenant.org/faq | ||||
| @@ -2,18 +2,18 @@ import java.time.format.DateTimeFormatter | ||||
|  | ||||
| dependencies { | ||||
|     // Expected everywhere. | ||||
|     compileOnlyApi("org.checkerframework:checker-qual") | ||||
|     compileOnlyApi(libs.checkerqual) | ||||
|  | ||||
|     // Minecraft expectations | ||||
|     compileOnlyApi("com.google.code.gson:gson") | ||||
|     compileOnly("com.google.guava:guava") | ||||
|     compileOnlyApi(libs.gson) | ||||
|     compileOnly(libs.guava) | ||||
|  | ||||
|     // Platform expectations | ||||
|     compileOnlyApi("org.yaml:snakeyaml") | ||||
|     compileOnlyApi(libs.snakeyaml) | ||||
|  | ||||
|     // Adventure | ||||
|     api("net.kyori:adventure-api") | ||||
|     api("net.kyori:adventure-text-minimessage") | ||||
|     api(libs.adventureApi) | ||||
|     api(libs.adventureMiniMessage) | ||||
|  | ||||
|     // Guice | ||||
|     api(libs.guice) { | ||||
| @@ -31,18 +31,19 @@ dependencies { | ||||
|         exclude(group = "dummypermscompat") | ||||
|     } | ||||
|     testImplementation(libs.worldeditCore) | ||||
|     compileOnly("com.fastasyncworldedit:FastAsyncWorldEdit-Core") { isTransitive = false } | ||||
|     testImplementation("com.fastasyncworldedit:FastAsyncWorldEdit-Core") { isTransitive = false } | ||||
|     compileOnly(libs.faweBukkit) { isTransitive = false } | ||||
|     testImplementation(libs.faweCore) { isTransitive = false } | ||||
|  | ||||
|     // Logging | ||||
|     compileOnlyApi("org.apache.logging.log4j:log4j-api") | ||||
|     compileOnlyApi(libs.log4j) | ||||
|  | ||||
|     // Other libraries | ||||
|     api(libs.prtree) | ||||
|     api(libs.aopalliance) | ||||
|     api(libs.cloudServices) | ||||
|     api(libs.arkitektonika) | ||||
|     api("com.intellectualsites.paster:Paster") | ||||
|     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/4.9.3/") | ||||
|         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() | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -22,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; | ||||
| @@ -31,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; | ||||
| @@ -75,6 +76,11 @@ public interface PlotPlatform<P> extends LocaleHolder { | ||||
|      */ | ||||
|     void shutdown(); | ||||
|  | ||||
|     /** | ||||
|      * Completely shuts down the server. | ||||
|      */ | ||||
|     void shutdownServer(); | ||||
|  | ||||
|     /** | ||||
|      * Get the name of the plugin | ||||
|      * | ||||
| @@ -114,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. | ||||
|      * | ||||
| @@ -279,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. | ||||
|      * | ||||
| @@ -308,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(); | ||||
|  | ||||
|     /** | ||||
|   | ||||
| @@ -84,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; | ||||
| @@ -197,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( | ||||
| @@ -234,7 +239,7 @@ public class PlotSquared { | ||||
|             copyFile("skyblock.template", Settings.Paths.TEMPLATES); | ||||
|             showDebug(); | ||||
|         } catch (Throwable e) { | ||||
|             e.printStackTrace(); | ||||
|             LOGGER.error(e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -267,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( | ||||
| @@ -287,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); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| @@ -642,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; | ||||
|     } | ||||
|  | ||||
| @@ -786,9 +796,8 @@ public class PlotSquared { | ||||
|         if (world.equals("CheckingPlotSquaredGenerator")) { | ||||
|             return; | ||||
|         } | ||||
|         if (!this.getPlotAreaManager().addWorld(world)) { | ||||
|             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")) { | ||||
|             worlds = this.worldConfiguration.getConfigurationSection("worlds").getKeys(false); | ||||
| @@ -1007,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> | ||||
|      * | ||||
| @@ -1273,7 +1282,7 @@ public class PlotSquared { | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Setup the database connection. | ||||
|      * Set up the database connection. | ||||
|      */ | ||||
|     public void setupDatabase() { | ||||
|         try { | ||||
|   | ||||
| @@ -18,6 +18,8 @@ | ||||
|  */ | ||||
| package com.plotsquared.core.backup; | ||||
|  | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| import java.io.IOException; | ||||
| @@ -30,12 +32,14 @@ import java.nio.file.Path; | ||||
|  */ | ||||
| public class Backup { | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + Backup.class.getSimpleName()); | ||||
|  | ||||
|     private final BackupProfile owner; | ||||
|     private final long creationTime; | ||||
|     @Nullable | ||||
|     private final Path file; | ||||
|  | ||||
|     Backup(final BackupProfile owner, final long creationTime, final Path file) { | ||||
|     Backup(final BackupProfile owner, final long creationTime, @Nullable final Path file) { | ||||
|         this.owner = owner; | ||||
|         this.creationTime = creationTime; | ||||
|         this.file = file; | ||||
| @@ -49,7 +53,7 @@ public class Backup { | ||||
|             try { | ||||
|                 Files.deleteIfExists(file); | ||||
|             } catch (final IOException e) { | ||||
|                 e.printStackTrace(); | ||||
|                 LOGGER.error("Error deleting backup at {}", file, e); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -21,14 +21,15 @@ package com.plotsquared.core.backup; | ||||
| import com.google.inject.Inject; | ||||
| import com.google.inject.assistedinject.Assisted; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.player.ConsolePlayer; | ||||
| import com.plotsquared.core.exception.PlotSquaredException; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.schematic.Schematic; | ||||
| import com.plotsquared.core.util.SchematicHandler; | ||||
| import com.plotsquared.core.util.task.RunnableVal; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import net.kyori.adventure.text.minimessage.MiniMessage; | ||||
| import org.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| @@ -51,7 +52,7 @@ import java.util.concurrent.CompletableFuture; | ||||
|  */ | ||||
| public class PlayerBackupProfile implements BackupProfile { | ||||
|  | ||||
|     static final MiniMessage MINI_MESSAGE = MiniMessage.builder().build(); | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + PlayerBackupProfile.class.getSimpleName()); | ||||
|  | ||||
|     private final UUID owner; | ||||
|     private final Plot plot; | ||||
| @@ -87,7 +88,7 @@ public class PlayerBackupProfile implements BackupProfile { | ||||
|                 Files.createDirectory(path); | ||||
|             } | ||||
|         } catch (final Exception e) { | ||||
|             e.printStackTrace(); | ||||
|             LOGGER.error("Error resolving {} from {}", child, parent, e); | ||||
|         } | ||||
|         return path; | ||||
|     } | ||||
| @@ -104,7 +105,7 @@ public class PlayerBackupProfile implements BackupProfile { | ||||
|                     try { | ||||
|                         Files.createDirectories(path); | ||||
|                     } catch (IOException e) { | ||||
|                         e.printStackTrace(); | ||||
|                         LOGGER.error("Error creating directory {}", path, e); | ||||
|                         return Collections.emptyList(); | ||||
|                     } | ||||
|                 } | ||||
| @@ -117,11 +118,11 @@ public class PlayerBackupProfile implements BackupProfile { | ||||
|                             backups.add( | ||||
|                                     new Backup(this, basicFileAttributes.creationTime().toMillis(), file)); | ||||
|                         } catch (IOException e) { | ||||
|                             e.printStackTrace(); | ||||
|                             LOGGER.error("Error getting attributes for file {} to create backup", file, e); | ||||
|                         } | ||||
|                     }); | ||||
|                 } catch (IOException e) { | ||||
|                     e.printStackTrace(); | ||||
|                     LOGGER.error("Error walking files from {}", path, e); | ||||
|                 } | ||||
|                 backups.sort(Comparator.comparingLong(Backup::getCreationTime).reversed()); | ||||
|                 return (this.backupCache = backups); | ||||
| @@ -133,7 +134,7 @@ public class PlayerBackupProfile implements BackupProfile { | ||||
|     public void destroy() { | ||||
|         this.listBackups().whenCompleteAsync((backups, error) -> { | ||||
|             if (error != null) { | ||||
|                 error.printStackTrace(); | ||||
|                 LOGGER.error("Error while listing backups", error); | ||||
|             } | ||||
|             backups.forEach(Backup::delete); | ||||
|             this.backupCache = null; | ||||
| @@ -141,10 +142,12 @@ public class PlayerBackupProfile implements BackupProfile { | ||||
|     } | ||||
|  | ||||
|     public @NonNull Path getBackupDirectory() { | ||||
|         return resolve(resolve( | ||||
|                 resolve(backupManager.getBackupPath(), Objects.requireNonNull(plot.getArea().toString(), "plot area id")), | ||||
|                 Objects.requireNonNull(plot.getId().toDashSeparatedString(), "plot id") | ||||
|         ), Objects.requireNonNull(owner.toString(), "owner")); | ||||
|         return resolve( | ||||
|                 resolve( | ||||
|                         resolve(backupManager.getBackupPath(), Objects.requireNonNull(plot.getArea().toString(), "plot area id")), | ||||
|                         Objects.requireNonNull(plot.getId().toDashSeparatedString(), "plot id") | ||||
|                 ), Objects.requireNonNull(owner.toString(), "owner") | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -156,7 +159,8 @@ public class PlayerBackupProfile implements BackupProfile { | ||||
|                     backups.get(backups.size() - 1).delete(); | ||||
|                 } | ||||
|                 final List<Plot> plots = Collections.singletonList(plot); | ||||
|                 final boolean result = this.schematicHandler.exportAll(plots, getBackupDirectory().toFile(), | ||||
|                 final boolean result = this.schematicHandler.exportAll( | ||||
|                         plots, getBackupDirectory().toFile(), | ||||
|                         "%world%-%id%-" + System.currentTimeMillis(), () -> | ||||
|                                 future.complete(new Backup(this, System.currentTimeMillis(), null)) | ||||
|                 ); | ||||
| @@ -180,7 +184,7 @@ public class PlayerBackupProfile implements BackupProfile { | ||||
|                 try { | ||||
|                     schematic = this.schematicHandler.getSchematic(backup.getFile().toFile()); | ||||
|                 } catch (SchematicHandler.UnsupportedFormatException e) { | ||||
|                     e.printStackTrace(); | ||||
|                     LOGGER.error("Unsupported format for backup {}", backup.getFile(), e); | ||||
|                 } | ||||
|                 if (schematic == null) { | ||||
|                     future.completeExceptionally(new IllegalArgumentException( | ||||
| @@ -200,10 +204,9 @@ public class PlayerBackupProfile implements BackupProfile { | ||||
|                                     if (value) { | ||||
|                                         future.complete(null); | ||||
|                                     } else { | ||||
|                                         future.completeExceptionally(new RuntimeException(MINI_MESSAGE.escapeTags( | ||||
|                                         future.completeExceptionally(new PlotSquaredException( | ||||
|                                                 TranslatableCaption | ||||
|                                                         .of("schematics.schematic_paste_failed") | ||||
|                                                         .getComponent(ConsolePlayer.getConsole())))); | ||||
|                                                         .of("schematics.schematic_paste_failed"))); | ||||
|                                     } | ||||
|                                 } | ||||
|                             } | ||||
|   | ||||
| @@ -32,6 +32,8 @@ import com.plotsquared.core.util.task.TaskManager; | ||||
| 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.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| @@ -47,6 +49,7 @@ import java.util.concurrent.TimeUnit; | ||||
| @Singleton | ||||
| public class SimpleBackupManager implements BackupManager { | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + SimpleBackupManager.class.getSimpleName()); | ||||
|     private final Path backupPath; | ||||
|     private final boolean automaticBackup; | ||||
|     private final int backupLimit; | ||||
| @@ -112,7 +115,12 @@ public class SimpleBackupManager implements BackupManager { | ||||
|                                 TagResolver.resolver("reason", Tag.inserting(Component.text(throwable.getMessage()))) | ||||
|                         ); | ||||
|                     } | ||||
|                     throwable.printStackTrace(); | ||||
|                     LOGGER.error( | ||||
|                             "Error creating backup for plot {};{} and player {}", | ||||
|                             plot.getArea(), | ||||
|                             plot.getId(), | ||||
|                             player == null ? "null" : player.getName(), throwable | ||||
|                     ); | ||||
|                 } else { | ||||
|                     if (player != null) { | ||||
|                         player.sendMessage(TranslatableCaption.of("backups.backup_automatic_finished")); | ||||
| @@ -128,6 +136,7 @@ public class SimpleBackupManager implements BackupManager { | ||||
|         return this.automaticBackup; | ||||
|     } | ||||
|  | ||||
|     @NonNull | ||||
|     public Path getBackupPath() { | ||||
|         return this.backupPath; | ||||
|     } | ||||
| @@ -136,13 +145,9 @@ public class SimpleBackupManager implements BackupManager { | ||||
|         return this.backupLimit; | ||||
|     } | ||||
|  | ||||
|     private static final class PlotCacheKey { | ||||
|  | ||||
|         private final Plot plot; | ||||
|  | ||||
|         private PlotCacheKey(Plot plot) { | ||||
|             this.plot = plot; | ||||
|         } | ||||
|     private record PlotCacheKey( | ||||
|             Plot plot | ||||
|     ) { | ||||
|  | ||||
|         @Override | ||||
|         public boolean equals(final Object o) { | ||||
|   | ||||
| @@ -19,6 +19,7 @@ | ||||
| package com.plotsquared.core.command; | ||||
|  | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.database.DBFunc; | ||||
| @@ -26,7 +27,6 @@ import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.PlayerManager; | ||||
| import com.plotsquared.core.util.TabCompletions; | ||||
| import com.plotsquared.core.util.task.RunnableVal2; | ||||
| @@ -60,18 +60,20 @@ public class Add extends Command { | ||||
|  | ||||
|     @Override | ||||
|     public CompletableFuture<Boolean> execute( | ||||
|             final PlotPlayer<?> player, String[] args, | ||||
|             final PlotPlayer<?> player, | ||||
|             String[] args, | ||||
|             RunnableVal3<Command, Runnable, Runnable> confirm, | ||||
|             RunnableVal2<Command, CommandResult> whenDone | ||||
|     ) throws CommandException { | ||||
|         final Plot plot = check(player.getCurrentPlot(), TranslatableCaption.of("errors.not_in_plot")); | ||||
|         checkTrue(plot.hasOwner(), TranslatableCaption.of("info.plot_unowned")); | ||||
|         checkTrue( | ||||
|                 plot.isOwner(player.getUUID()) || Permissions | ||||
|                         .hasPermission(player, Permission.PERMISSION_ADMIN_COMMAND_TRUST), | ||||
|                 plot.isOwner(player.getUUID()) || player.hasPermission(Permission.PERMISSION_ADMIN_COMMAND_TRUST), | ||||
|                 TranslatableCaption.of("permission.no_plot_perms") | ||||
|         ); | ||||
|         checkTrue(args.length == 1, TranslatableCaption.of("commandconfig.command_syntax"), | ||||
|         checkTrue( | ||||
|                 args.length == 1, | ||||
|                 TranslatableCaption.of("commandconfig.command_syntax"), | ||||
|                 TagResolver.resolver("value", Tag.inserting(Component.text("/plot add <player | *>"))) | ||||
|         ); | ||||
|         final CompletableFuture<Boolean> future = new CompletableFuture<>(); | ||||
| @@ -96,14 +98,18 @@ public class Add extends Command { | ||||
|                     int size = plot.getTrusted().size() + plot.getMembers().size(); | ||||
|                     while (iterator.hasNext()) { | ||||
|                         UUID uuid = iterator.next(); | ||||
|                         if (uuid == DBFunc.EVERYONE && !( | ||||
|                                 Permissions.hasPermission(player, Permission.PERMISSION_TRUST_EVERYONE) || Permissions | ||||
|                                         .hasPermission(player, Permission.PERMISSION_ADMIN_COMMAND_TRUST))) { | ||||
|                         if (uuid == DBFunc.EVERYONE && !(player.hasPermission(Permission.PERMISSION_TRUST_EVERYONE) || player.hasPermission( | ||||
|                                 Permission.PERMISSION_ADMIN_COMMAND_TRUST))) { | ||||
|                             player.sendMessage( | ||||
|                                     TranslatableCaption.of("errors.invalid_player"), | ||||
|                                     TagResolver.resolver("value", Tag.inserting( | ||||
|                                             PlayerManager.resolveName(uuid).toComponent(player) | ||||
|                                     )) | ||||
|                                     PlotSquared | ||||
|                                             .platform() | ||||
|                                             .playerManager() | ||||
|                                             .getUsernameCaption(uuid) | ||||
|                                             .thenApply(caption -> TagResolver.resolver( | ||||
|                                                     "value", | ||||
|                                                     Tag.inserting(caption.toComponent(player)) | ||||
|                                             )) | ||||
|                             ); | ||||
|                             iterator.remove(); | ||||
|                             continue; | ||||
| @@ -111,9 +117,11 @@ public class Add extends Command { | ||||
|                         if (plot.isOwner(uuid)) { | ||||
|                             player.sendMessage( | ||||
|                                     TranslatableCaption.of("member.already_added"), | ||||
|                                     TagResolver.resolver("player", Tag.inserting( | ||||
|                                             PlayerManager.resolveName(uuid).toComponent(player) | ||||
|                                     )) | ||||
|                                     PlotSquared.platform().playerManager().getUsernameCaption(uuid) | ||||
|                                             .thenApply(caption -> TagResolver.resolver( | ||||
|                                                     "player", | ||||
|                                                     Tag.inserting(caption.toComponent(player)) | ||||
|                                             )) | ||||
|                             ); | ||||
|                             iterator.remove(); | ||||
|                             continue; | ||||
| @@ -121,9 +129,11 @@ public class Add extends Command { | ||||
|                         if (plot.getMembers().contains(uuid)) { | ||||
|                             player.sendMessage( | ||||
|                                     TranslatableCaption.of("member.already_added"), | ||||
|                                     TagResolver.resolver("player", Tag.inserting( | ||||
|                                             PlayerManager.resolveName(uuid).toComponent(player) | ||||
|                                     )) | ||||
|                                     PlotSquared.platform().playerManager().getUsernameCaption(uuid) | ||||
|                                             .thenApply(caption -> TagResolver.resolver( | ||||
|                                                     "player", | ||||
|                                                     Tag.inserting(caption.toComponent(player)) | ||||
|                                             )) | ||||
|                             ); | ||||
|                             iterator.remove(); | ||||
|                             continue; | ||||
| @@ -132,7 +142,7 @@ public class Add extends Command { | ||||
|                     } | ||||
|                     checkTrue(!uuids.isEmpty(), null); | ||||
|                     int localAddSize = plot.getMembers().size(); | ||||
|                     int maxAddSize = Permissions.hasPermissionRange(player, Permission.PERMISSION_ADD, Settings.Limit.MAX_PLOTS); | ||||
|                     int maxAddSize = player.hasPermissionRange(Permission.PERMISSION_ADD, Settings.Limit.MAX_PLOTS); | ||||
|                     if (localAddSize >= maxAddSize) { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("members.plot_max_members_added"), | ||||
|   | ||||
| @@ -21,12 +21,10 @@ package com.plotsquared.core.command; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.util.MathMan; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.query.PlotQuery; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| @@ -59,8 +57,7 @@ public class Alias extends SubCommand { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         Location location = player.getLocation(); | ||||
|         Plot plot = location.getPlotAbs(); | ||||
|         Plot plot = player.getCurrentPlot(); | ||||
|         if (plot == null) { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||
|             return false; | ||||
| @@ -205,7 +202,7 @@ public class Alias extends SubCommand { | ||||
|     } | ||||
|  | ||||
|     private boolean isPermitted(PlotPlayer<?> player, Permission permission) { | ||||
|         return Permissions.hasPermission(player, permission); | ||||
|         return player.hasPermission(permission); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -47,7 +47,6 @@ import com.plotsquared.core.queue.QueueCoordinator; | ||||
| import com.plotsquared.core.setup.PlotAreaBuilder; | ||||
| import com.plotsquared.core.util.FileUtils; | ||||
| import com.plotsquared.core.util.MathMan; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.RegionUtil; | ||||
| import com.plotsquared.core.util.SchematicHandler; | ||||
| import com.plotsquared.core.util.SetupUtils; | ||||
| @@ -56,6 +55,7 @@ import com.plotsquared.core.util.TabCompletions; | ||||
| import com.plotsquared.core.util.WorldUtil; | ||||
| import com.plotsquared.core.util.task.RunnableVal3; | ||||
| import com.sk89q.worldedit.EditSession; | ||||
| import com.sk89q.worldedit.EditSessionBuilder; | ||||
| import com.sk89q.worldedit.LocalSession; | ||||
| import com.sk89q.worldedit.WorldEdit; | ||||
| import com.sk89q.worldedit.entity.Player; | ||||
| @@ -139,7 +139,7 @@ public class Area extends SubCommand { | ||||
|                     player.sendMessage(RequiredType.CONSOLE.getErrorMessage()); | ||||
|                     return false; | ||||
|                 } | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_AREA_CREATE)) { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_AREA_CREATE)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -184,6 +184,7 @@ public class Area extends SubCommand { | ||||
|                         CuboidRegion.makeCuboid(playerSelectedRegion) | ||||
|                 ).length != 0) { | ||||
|                     player.sendMessage(TranslatableCaption.of("single.single_area_overlapping")); | ||||
|                     return false; | ||||
|                 } | ||||
|                 // Alter the region | ||||
|                 final BlockVector3 playerSelectionMin = playerSelectedRegion.getMinimumPoint(); | ||||
| @@ -233,10 +234,9 @@ public class Area extends SubCommand { | ||||
|                 try (final ClipboardWriter clipboardWriter = BuiltInClipboardFormat.SPONGE_SCHEMATIC.getWriter(new FileOutputStream( | ||||
|                         file))) { | ||||
|                     final BlockArrayClipboard clipboard = new BlockArrayClipboard(selectedRegion); | ||||
|                     final EditSession editSession = WorldEdit | ||||
|                             .getInstance() | ||||
|                             .getEditSessionFactory() | ||||
|                             .getEditSession(selectedRegion.getWorld(), -1); | ||||
|                     EditSessionBuilder editSessionBuilder = WorldEdit.getInstance().newEditSessionBuilder(); | ||||
|                     editSessionBuilder.world(selectedRegion.getWorld()); | ||||
|                     final EditSession editSession = editSessionBuilder.build(); | ||||
|                     final ForwardExtentCopy forwardExtentCopy = | ||||
|                             new ForwardExtentCopy(editSession, selectedRegion, clipboard, selectedRegion.getMinimumPoint()); | ||||
|                     forwardExtentCopy.setCopyingBiomes(true); | ||||
| @@ -292,7 +292,7 @@ public class Area extends SubCommand { | ||||
|                 return true; | ||||
|             } | ||||
|             case "c", "setup", "create" -> { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_AREA_CREATE)) { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_AREA_CREATE)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -618,7 +618,7 @@ public class Area extends SubCommand { | ||||
|                 return true; | ||||
|             } | ||||
|             case "i", "info" -> { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_AREA_INFO)) { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_AREA_INFO)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -694,7 +694,7 @@ public class Area extends SubCommand { | ||||
|                 return true; | ||||
|             } | ||||
|             case "l", "list" -> { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_AREA_LIST)) { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_AREA_LIST)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -772,7 +772,7 @@ public class Area extends SubCommand { | ||||
|                 return true; | ||||
|             } | ||||
|             case "regen", "clear", "reset", "regenerate" -> { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_AREA_REGEN)) { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_AREA_REGEN)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -807,7 +807,7 @@ public class Area extends SubCommand { | ||||
|                 return true; | ||||
|             } | ||||
|             case "goto", "v", "teleport", "visit", "tp" -> { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_AREA_TP)) { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_AREA_TP)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver("node", Tag.inserting(Permission.PERMISSION_AREA_TP)) | ||||
| @@ -866,19 +866,19 @@ public class Area extends SubCommand { | ||||
|     public Collection<Command> tab(final PlotPlayer<?> player, final String[] args, final boolean space) { | ||||
|         if (args.length == 1) { | ||||
|             final List<String> completions = new LinkedList<>(); | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_AREA_CREATE)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_AREA_CREATE)) { | ||||
|                 completions.add("create"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_AREA_CREATE)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_AREA_CREATE)) { | ||||
|                 completions.add("single"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_AREA_LIST)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_AREA_LIST)) { | ||||
|                 completions.add("list"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_AREA_INFO)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_AREA_INFO)) { | ||||
|                 completions.add("info"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_AREA_TP)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_AREA_TP)) { | ||||
|                 completions.add("tp"); | ||||
|             } | ||||
|             final List<Command> commands = completions.stream().filter(completion -> completion | ||||
| @@ -893,7 +893,7 @@ public class Area extends SubCommand { | ||||
|                             CommandCategory.ADMINISTRATION | ||||
|                     ) { | ||||
|                     }).collect(Collectors.toCollection(LinkedList::new)); | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_AREA) && args[0].length() > 0) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_AREA) && args[0].length() > 0) { | ||||
|                 commands.addAll(TabCompletions.completePlayers(player, args[0], Collections.emptyList())); | ||||
|             } | ||||
|             return commands; | ||||
|   | ||||
| @@ -35,10 +35,10 @@ import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.services.plots.AutoQuery; | ||||
| import com.plotsquared.core.services.plots.AutoService; | ||||
| import com.plotsquared.core.util.EconHandler; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.PlotExpression; | ||||
| import com.plotsquared.core.util.task.AutoClaimFinishTask; | ||||
| import com.plotsquared.core.util.task.RunnableVal; | ||||
| @@ -131,8 +131,8 @@ public class Auto extends SubCommand { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("economy.removed_granted_plot"), | ||||
|                                 TagResolver.builder() | ||||
|                                         .tag("usedGrants", Tag.inserting(Component.text(grantedPlots - left))) | ||||
|                                         .tag("remainingGrants", Tag.inserting(Component.text(left))) | ||||
|                                         .tag("used_grants", Tag.inserting(Component.text(grantedPlots - left))) | ||||
|                                         .tag("remaining_grants", Tag.inserting(Component.text(left))) | ||||
|                                         .build() | ||||
|                         ); | ||||
|                     } | ||||
| @@ -249,11 +249,12 @@ public class Auto extends SubCommand { | ||||
|         sizeX = event.getSizeX(); | ||||
|         sizeZ = event.getSizeZ(); | ||||
|         schematic = event.getSchematic(); | ||||
|         if (!force && mega && !Permissions.hasPermission(player, Permission.PERMISSION_AUTO_MEGA)) { | ||||
|         if (!force && mega && !player.hasPermission(Permission.PERMISSION_AUTO_MEGA)) { | ||||
|             player.sendMessage( | ||||
|                     TranslatableCaption.of("permission.no_permission"), | ||||
|                     TagResolver.resolver("node", Tag.inserting(Permission.PERMISSION_AUTO_MEGA)) | ||||
|             ); | ||||
|             return false; | ||||
|         } | ||||
|         if (!force && sizeX * sizeZ > Settings.Claim.MAX_AUTO_AREA) { | ||||
|             player.sendMessage( | ||||
| @@ -283,11 +284,9 @@ public class Auto extends SubCommand { | ||||
|                 ); | ||||
|                 return true; | ||||
|             } | ||||
|             if (!force && !Permissions.hasPermission( | ||||
|                     player, | ||||
|             if (!force && !player.hasPermission( | ||||
|                     Permission.PERMISSION_CLAIM_SCHEMATIC.format(schematic) | ||||
|             ) && !Permissions | ||||
|                     .hasPermission(player, "plots.admin.command.schematic")) { | ||||
|             ) && !player.hasPermission("plots.admin.command.schematic")) { | ||||
|                 player.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission"), | ||||
|                         TagResolver.resolver("node", Tag.inserting(Component.text("plots.claim.%s0"))) | ||||
| @@ -295,12 +294,15 @@ public class Auto extends SubCommand { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         if (this.econHandler != null && plotarea.useEconomy()) { | ||||
|         if (this.econHandler != null && plotarea.useEconomy() && !player.hasPermission(Permission.PERMISSION_ADMIN_BYPASS_ECON)) { | ||||
|             PlotExpression costExp = plotarea.getPrices().get("claim"); | ||||
|             PlotExpression mergeCostExp = plotarea.getPrices().get("merge"); | ||||
|             int size = sizeX * sizeZ; | ||||
|             double mergeCost = size <= 1 || mergeCostExp == null ? 0d : mergeCostExp.evaluate(size); | ||||
|             double cost = costExp.evaluate(Settings.Limit.GLOBAL ? | ||||
|                     player.getPlotCount() : | ||||
|                     player.getPlotCount(plotarea.getWorldName())); | ||||
|             cost = (sizeX * sizeZ) * cost; | ||||
|             cost = size * cost + mergeCost; | ||||
|             if (cost > 0d) { | ||||
|                 if (!this.econHandler.isSupported()) { | ||||
|                     player.sendMessage(TranslatableCaption.of("economy.vault_or_consumer_null")); | ||||
| @@ -328,7 +330,7 @@ public class Auto extends SubCommand { | ||||
|         } | ||||
|  | ||||
|         List<Plot> plots = this.servicePipeline | ||||
|                 .pump(new AutoService.AutoQuery(player, null, sizeX, sizeZ, plotarea)) | ||||
|                 .pump(new AutoQuery(player, null, sizeX, sizeZ, plotarea)) | ||||
|                 .through(AutoService.class) | ||||
|                 .getResult(); | ||||
|  | ||||
|   | ||||
| @@ -24,15 +24,17 @@ import com.plotsquared.core.backup.BackupProfile; | ||||
| import com.plotsquared.core.backup.NullBackupProfile; | ||||
| import com.plotsquared.core.backup.PlayerBackupProfile; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.exception.PlotSquaredException; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.task.RunnableVal2; | ||||
| import com.plotsquared.core.util.task.RunnableVal3; | ||||
| 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.apache.logging.log4j.LogManager; | ||||
| import org.apache.logging.log4j.Logger; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.nio.file.Files; | ||||
| @@ -58,6 +60,8 @@ import java.util.stream.Stream; | ||||
|         permission = "plots.backup") | ||||
| public final class Backup extends Command { | ||||
|  | ||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + Backup.class.getSimpleName()); | ||||
|  | ||||
|     private final BackupManager backupManager; | ||||
|  | ||||
|     @Inject | ||||
| @@ -150,8 +154,7 @@ public final class Backup extends Command { | ||||
|                             TranslatableCaption.of("generic.generic_merged").toComponent(player) | ||||
|                     )) | ||||
|             ); | ||||
|         } else if (!plot.isOwner(player.getUUID()) && !Permissions | ||||
|                 .hasPermission(player, Permission.PERMISSION_ADMIN_BACKUP_OTHER)) { | ||||
|         } else if (!plot.isOwner(player.getUUID()) && !player.hasPermission(Permission.PERMISSION_ADMIN_BACKUP_OTHER)) { | ||||
|             player.sendMessage( | ||||
|                     TranslatableCaption.of("permission.no_permission"), | ||||
|                     TagResolver.resolver( | ||||
| @@ -215,8 +218,7 @@ public final class Backup extends Command { | ||||
|             ); | ||||
|         } else if (plot.getVolume() > Integer.MAX_VALUE) { | ||||
|             player.sendMessage(TranslatableCaption.of("schematics.schematic_too_large")); | ||||
|         } else if (!plot.isOwner(player.getUUID()) && !Permissions | ||||
|                 .hasPermission(player, Permission.PERMISSION_ADMIN_BACKUP_OTHER)) { | ||||
|         } else if (!plot.isOwner(player.getUUID()) && !player.hasPermission(Permission.PERMISSION_ADMIN_BACKUP_OTHER)) { | ||||
|             player.sendMessage( | ||||
|                     TranslatableCaption.of("permission.no_permission"), | ||||
|                     TagResolver.resolver( | ||||
| @@ -301,8 +303,7 @@ public final class Backup extends Command { | ||||
|             ); | ||||
|         } else if (plot.getVolume() > Integer.MAX_VALUE) { | ||||
|             player.sendMessage(TranslatableCaption.of("schematics.schematic_too_large")); | ||||
|         } else if (!plot.isOwner(player.getUUID()) && !Permissions | ||||
|                 .hasPermission(player, Permission.PERMISSION_ADMIN_BACKUP_OTHER)) { | ||||
|         } else if (!plot.isOwner(player.getUUID()) && !player.hasPermission(Permission.PERMISSION_ADMIN_BACKUP_OTHER)) { | ||||
|             player.sendMessage( | ||||
|                     TranslatableCaption.of("permission.no_permission"), | ||||
|                     TagResolver.resolver( | ||||
| @@ -330,20 +331,43 @@ public final class Backup extends Command { | ||||
|             if (backupProfile instanceof NullBackupProfile) { | ||||
|                 player.sendMessage( | ||||
|                         TranslatableCaption.of("backups.backup_impossible"), | ||||
|                         TagResolver.resolver("plot", Tag.inserting( | ||||
|                                 TranslatableCaption.of("generic.generic_other").toComponent(player) | ||||
|                         )) | ||||
|                         TagResolver.resolver( | ||||
|                                 "plot", Tag.inserting( | ||||
|                                         TranslatableCaption.of("generic.generic_other").toComponent(player) | ||||
|                                 ) | ||||
|                         ) | ||||
|                 ); | ||||
|             } else { | ||||
|                 backupProfile.listBackups().whenComplete((backups, throwable) -> { | ||||
|                     if (throwable != null) { | ||||
|                         Component reason; | ||||
|                         if (throwable instanceof PlotSquaredException pe) { | ||||
|                             reason = pe.getCaption().toComponent(player); | ||||
|                         } else { | ||||
|                             reason = Component.text(throwable.getMessage()); | ||||
|                         } | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("backups.backup_load_failure"), | ||||
|                                 TagResolver.resolver("reason", Tag.inserting(Component.text(throwable.getMessage()))) | ||||
|                                 TagResolver.resolver("reason", Tag.inserting(reason)) | ||||
|                         ); | ||||
|                         LOGGER.error("Error loading player ({}) backup", player.getName(), throwable); | ||||
|                         return; | ||||
|                     } | ||||
|                     if (number < 1 || number > backups.size()) { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("backups.backup_impossible"), | ||||
|                                 TagResolver.resolver( | ||||
|                                         "plot", | ||||
|                                         Tag.inserting(TranslatableCaption | ||||
|                                                 .of("generic.generic_invalid_choice") | ||||
|                                                 .toComponent(player)) | ||||
|                                 ) | ||||
|                         ); | ||||
|                         throwable.printStackTrace(); | ||||
|                     } else { | ||||
|                         if (number < 1 || number > backups.size()) { | ||||
|                         final com.plotsquared.core.backup.Backup backup = | ||||
|                                 backups.get(number - 1); | ||||
|                         if (backup == null || backup.getFile() == null || !Files | ||||
|                                 .exists(backup.getFile())) { | ||||
|                             player.sendMessage( | ||||
|                                     TranslatableCaption.of("backups.backup_impossible"), | ||||
|                                     TagResolver.resolver( | ||||
| @@ -354,37 +378,23 @@ public final class Backup extends Command { | ||||
|                                     ) | ||||
|                             ); | ||||
|                         } else { | ||||
|                             final com.plotsquared.core.backup.Backup backup = | ||||
|                                     backups.get(number - 1); | ||||
|                             if (backup == null || backup.getFile() == null || !Files | ||||
|                                     .exists(backup.getFile())) { | ||||
|                                 player.sendMessage( | ||||
|                                         TranslatableCaption.of("backups.backup_impossible"), | ||||
|                                         TagResolver.resolver( | ||||
|                                                 "plot", | ||||
|                                                 Tag.inserting(TranslatableCaption | ||||
|                                                         .of("generic.generic_invalid_choice") | ||||
|                                                         .toComponent(player)) | ||||
|                                         ) | ||||
|                                 ); | ||||
|                             } else { | ||||
|                                 CmdConfirm.addPending(player, "/plot backup load " + number, | ||||
|                                         () -> backupProfile.restoreBackup(backup, player) | ||||
|                                                 .whenComplete((n, error) -> { | ||||
|                                                     if (error != null) { | ||||
|                                                         player.sendMessage( | ||||
|                                                                 TranslatableCaption.of("backups.backup_load_failure"), | ||||
|                                                                 TagResolver.resolver( | ||||
|                                                                         "reason", | ||||
|                                                                         Tag.inserting(Component.text(error.getMessage())) | ||||
|                                                                 ) | ||||
|                                                         ); | ||||
|                                                     } else { | ||||
|                                                         player.sendMessage(TranslatableCaption.of("backups.backup_load_success")); | ||||
|                                                     } | ||||
|                                                 }) | ||||
|                                 ); | ||||
|                             } | ||||
|                             CmdConfirm.addPending( | ||||
|                                     player, "/plot backup load " + number, | ||||
|                                     () -> backupProfile.restoreBackup(backup, player) | ||||
|                                             .whenComplete((n, error) -> { | ||||
|                                                 if (error != null) { | ||||
|                                                     player.sendMessage( | ||||
|                                                             TranslatableCaption.of("backups.backup_load_failure"), | ||||
|                                                             TagResolver.resolver( | ||||
|                                                                     "reason", | ||||
|                                                                     Tag.inserting(Component.text(error.getMessage())) | ||||
|                                                             ) | ||||
|                                                     ); | ||||
|                                                 } else { | ||||
|                                                     player.sendMessage(TranslatableCaption.of("backups.backup_load_success")); | ||||
|                                                 } | ||||
|                                             }) | ||||
|                             ); | ||||
|                         } | ||||
|                     } | ||||
|                 }); | ||||
|   | ||||
| @@ -27,7 +27,6 @@ import com.sk89q.worldedit.command.util.SuggestionHelper; | ||||
| import com.sk89q.worldedit.world.biome.BiomeType; | ||||
| import com.sk89q.worldedit.world.biome.BiomeTypes; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.TextComponent; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
|  | ||||
|   | ||||
| @@ -20,9 +20,11 @@ package com.plotsquared.core.command; | ||||
|  | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.events.PlotFlagRemoveEvent; | ||||
| import com.plotsquared.core.events.PlayerBuyPlotEvent; | ||||
| import com.plotsquared.core.events.Result; | ||||
| import com.plotsquared.core.player.OfflinePlotPlayer; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| @@ -83,27 +85,36 @@ public class Buy extends Command { | ||||
|         checkTrue(plot.hasOwner(), TranslatableCaption.of("info.plot_unowned")); | ||||
|         checkTrue(!plot.isOwner(player.getUUID()), TranslatableCaption.of("economy.cannot_buy_own")); | ||||
|         Set<Plot> plots = plot.getConnectedPlots(); | ||||
|         int plotCount = Settings.Limit.GLOBAL ? player.getPlotCount() : player.getPlotCount(plot.getWorldName()); | ||||
|         checkTrue( | ||||
|                 player.getPlotCount() + plots.size() <= player.getAllowedPlots(), | ||||
|                 plotCount + plots.size() <= player.getAllowedPlots(), | ||||
|                 TranslatableCaption.of("permission.cant_claim_more_plots"), | ||||
|                 TagResolver.resolver("amount", Tag.inserting(Component.text(player.getAllowedPlots()))) | ||||
|         ); | ||||
|         double price = plot.getFlag(PriceFlag.class); | ||||
|         if (price <= 0) { | ||||
|         double priceFlag = plot.getFlag(PriceFlag.class); | ||||
|         if (priceFlag <= 0) { | ||||
|             throw new CommandException(TranslatableCaption.of("economy.not_for_sale")); | ||||
|         } | ||||
|         checkTrue( | ||||
|                 this.econHandler.isSupported(), | ||||
|                 TranslatableCaption.of("economy.vault_or_consumer_null") | ||||
|         ); | ||||
|         checkTrue( | ||||
|                 this.econHandler.getMoney(player) >= price, | ||||
|                 TranslatableCaption.of("economy.cannot_afford_plot"), | ||||
|                 TagResolver.builder() | ||||
|                         .tag("money", Tag.inserting(Component.text(this.econHandler.format(price)))) | ||||
|                         .tag("balance", Tag.inserting(Component.text(this.econHandler.format(this.econHandler.getMoney(player))))) | ||||
|                         .build() | ||||
|         ); | ||||
|  | ||||
|         PlayerBuyPlotEvent event = this.eventDispatcher.callPlayerBuyPlot(player, plot, priceFlag); | ||||
|         if (event.getEventResult() == Result.DENY) { | ||||
|             throw new CommandException(TranslatableCaption.of("economy.cannot_buy_blocked")); | ||||
|         } | ||||
|  | ||||
|         double price = event.getEventResult() == Result.FORCE ? 0 : event.price(); | ||||
|         if (this.econHandler.getMoney(player) < price) { | ||||
|             throw new CommandException( | ||||
|                     TranslatableCaption.of("economy.cannot_afford_plot"), | ||||
|                     TagResolver.builder() | ||||
|                             .tag("money", Tag.inserting(Component.text(this.econHandler.format(price)))) | ||||
|                             .tag("balance", Tag.inserting(Component.text(this.econHandler.format(this.econHandler.getMoney(player))))) | ||||
|                             .build() | ||||
|             ); | ||||
|         } | ||||
|         this.econHandler.withdrawMoney(player, price); | ||||
|         // Failure | ||||
|         // Success | ||||
| @@ -113,7 +124,8 @@ public class Buy extends Command { | ||||
|                     TagResolver.resolver("money", Tag.inserting(Component.text(this.econHandler.format(price)))) | ||||
|             ); | ||||
|  | ||||
|             this.econHandler.depositMoney(PlotSquared.platform().playerManager().getOfflinePlayer(plot.getOwnerAbs()), price); | ||||
|             OfflinePlotPlayer previousOwner = PlotSquared.platform().playerManager().getOfflinePlayer(plot.getOwnerAbs()); | ||||
|             this.econHandler.depositMoney(previousOwner, price); | ||||
|  | ||||
|             PlotPlayer<?> owner = PlotSquared.platform().playerManager().getPlayerIfExists(plot.getOwnerAbs()); | ||||
|             if (owner != null) { | ||||
| @@ -127,15 +139,17 @@ public class Buy extends Command { | ||||
|                 ); | ||||
|             } | ||||
|             PlotFlag<?, ?> plotFlag = plot.getFlagContainer().getFlag(PriceFlag.class); | ||||
|             PlotFlagRemoveEvent event = this.eventDispatcher.callFlagRemove(plotFlag, plot); | ||||
|             if (event.getEventResult() != Result.DENY) { | ||||
|                 plot.removeFlag(event.getFlag()); | ||||
|             if (this.eventDispatcher.callFlagRemove(plotFlag, plot).getEventResult() != Result.DENY) { | ||||
|                 plot.removeFlag(plotFlag); | ||||
|             } | ||||
|             plot.setOwner(player.getUUID()); | ||||
|             plot.getPlotModificationManager().setSign(player.getName()); | ||||
|             player.sendMessage( | ||||
|                     TranslatableCaption.of("working.claimed"), | ||||
|                     TagResolver.resolver("world", Tag.inserting(Component.text(plot.getArea().getWorldName()))), | ||||
|                     TagResolver.resolver("plot", Tag.inserting(Component.text(plot.getId().toString()))) | ||||
|             ); | ||||
|             this.eventDispatcher.callPostPlayerBuyPlot(player, previousOwner, plot, price); | ||||
|             whenDone.run(Buy.this, CommandResult.SUCCESS); | ||||
|         }, () -> { | ||||
|             this.econHandler.depositMoney(player, price); | ||||
|   | ||||
| @@ -29,7 +29,6 @@ import com.plotsquared.core.plot.flag.implementations.HostileCapFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.MiscCapFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.MobCapFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.VehicleCapFlag; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.ComponentLike; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| @@ -54,8 +53,7 @@ public class Caps extends SubCommand { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||
|             return false; | ||||
|         } | ||||
|         if (!plot.isAdded(player.getUUID()) && !Permissions | ||||
|                 .hasPermission(player, Permission.PERMISSION_ADMIN_CAPS_OTHER)) { | ||||
|         if (!plot.isAdded(player.getUUID()) && !player.hasPermission(Permission.PERMISSION_ADMIN_CAPS_OTHER)) { | ||||
|             player.sendMessage( | ||||
|                     TranslatableCaption.of("permission.no_permission"), | ||||
|                     TagResolver.resolver("node", Tag.inserting(Permission.PERMISSION_ADMIN_CAPS_OTHER)) | ||||
|   | ||||
| @@ -26,7 +26,6 @@ import com.plotsquared.core.events.PlayerClaimPlotEvent; | ||||
| import com.plotsquared.core.events.PlotMergeEvent; | ||||
| import com.plotsquared.core.events.Result; | ||||
| import com.plotsquared.core.location.Direction; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.MetaDataAccess; | ||||
| import com.plotsquared.core.player.PlayerMetaDataKeys; | ||||
| @@ -35,7 +34,6 @@ import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.util.EconHandler; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.PlotExpression; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import net.kyori.adventure.text.Component; | ||||
| @@ -73,8 +71,7 @@ public class Claim extends SubCommand { | ||||
|         if (args.length >= 1) { | ||||
|             schematic = args[0]; | ||||
|         } | ||||
|         Location location = player.getLocation(); | ||||
|         Plot plot = location.getPlotAbs(); | ||||
|         Plot plot = player.getCurrentPlot(); | ||||
|         if (plot == null) { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||
|             return false; | ||||
| @@ -91,7 +88,7 @@ public class Claim extends SubCommand { | ||||
|         boolean force = event.getEventResult() == Result.FORCE; | ||||
|         int currentPlots = Settings.Limit.GLOBAL ? | ||||
|                 player.getPlotCount() : | ||||
|                 player.getPlotCount(location.getWorldName()); | ||||
|                 player.getPlotCount(plot.getWorldName()); | ||||
|  | ||||
|         final PlotArea area = plot.getArea(); | ||||
|  | ||||
| @@ -131,9 +128,8 @@ public class Claim extends SubCommand { | ||||
|                                         .build() | ||||
|                         ); | ||||
|                     } | ||||
|                     if (!Permissions.hasPermission(player, Permission.PERMISSION_CLAIM_SCHEMATIC | ||||
|                             .format(schematic)) && !Permissions.hasPermission( | ||||
|                             player, | ||||
|                     if (!player.hasPermission(Permission.PERMISSION_CLAIM_SCHEMATIC | ||||
|                             .format(schematic)) && !player.hasPermission( | ||||
|                             "plots.admin.command.schematic" | ||||
|                     ) && !force) { | ||||
|                         player.sendMessage( | ||||
| @@ -143,7 +139,7 @@ public class Claim extends SubCommand { | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             if (this.econHandler.isEnabled(area) && !force) { | ||||
|             if (this.econHandler.isEnabled(area) && !force && !player.hasPermission(Permission.PERMISSION_ADMIN_BYPASS_ECON)) { | ||||
|                 PlotExpression costExr = area.getPrices().get("claim"); | ||||
|                 double cost = costExr.evaluate(currentPlots); | ||||
|                 if (cost > 0d) { | ||||
| @@ -188,14 +184,14 @@ public class Claim extends SubCommand { | ||||
|                 player.sendMessage( | ||||
|                         TranslatableCaption.of("economy.removed_granted_plot"), | ||||
|                         TagResolver.builder() | ||||
|                                 .tag("usedGrants", Tag.inserting(Component.text(grants - 1))) | ||||
|                                 .tag("remainingGrants", Tag.inserting(Component.text(grants))) | ||||
|                                 .tag("used_grants", Tag.inserting(Component.text(grants - 1))) | ||||
|                                 .tag("remaining_grants", Tag.inserting(Component.text(grants))) | ||||
|                                 .build() | ||||
|                 ); | ||||
|             } | ||||
|         } | ||||
|         if (!Permissions.hasPermission(player, Permission.PERMISSION_ADMIN_BYPASS_BORDER)) { | ||||
|             int border = area.getBorder(); | ||||
|         if (!player.hasPermission(Permission.PERMISSION_ADMIN_BYPASS_BORDER)) { | ||||
|             int border = area.getBorder(false); | ||||
|             if (border != Integer.MAX_VALUE && plot.getDistanceFromOrigin() > border && !force) { | ||||
|                 player.sendMessage(TranslatableCaption.of("border.denied")); | ||||
|                 return false; | ||||
|   | ||||
| @@ -32,7 +32,6 @@ import com.plotsquared.core.plot.flag.implementations.AnalysisFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.DoneFlag; | ||||
| import com.plotsquared.core.queue.GlobalBlockQueue; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.task.RunnableVal2; | ||||
| import com.plotsquared.core.util.task.RunnableVal3; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| @@ -91,13 +90,14 @@ public class Clear extends Command { | ||||
|         } | ||||
|         boolean force = eventResult == Result.FORCE; | ||||
|         checkTrue( | ||||
|                 force || plot.isOwner(player.getUUID()) || Permissions | ||||
|                         .hasPermission(player, "plots.admin.command.clear"), | ||||
|                 force || plot.isOwner(player.getUUID()) || player.hasPermission("plots.admin.command.clear"), | ||||
|                 TranslatableCaption.of("permission.no_plot_perms") | ||||
|         ); | ||||
|         checkTrue(plot.getRunning() == 0, TranslatableCaption.of("errors.wait_for_timer")); | ||||
|         checkTrue(force || !Settings.Done.RESTRICT_BUILDING || !DoneFlag.isDone(plot) || Permissions | ||||
|                 .hasPermission(player, "plots.continue"), TranslatableCaption.of("done.done_already_done")); | ||||
|         checkTrue( | ||||
|                 force || !Settings.Done.RESTRICT_BUILDING || !DoneFlag.isDone(plot) || player.hasPermission("plots.continue"), | ||||
|                 TranslatableCaption.of("done.done_already_done") | ||||
|         ); | ||||
|         confirm.run(this, () -> { | ||||
|             if (Settings.Teleport.ON_CLEAR) { | ||||
|                 plot.getPlayersInPlot().forEach(playerInPlot -> plot.teleportPlayer(playerInPlot, TeleportCause.COMMAND_CLEAR, | ||||
| @@ -107,38 +107,37 @@ public class Clear extends Command { | ||||
|             } | ||||
|             BackupManager.backup(player, plot, () -> { | ||||
|                 final long start = System.currentTimeMillis(); | ||||
|                 boolean result = plot.getPlotModificationManager().clear(true, false, player, () -> { | ||||
|                     plot.getPlotModificationManager().unlink(); | ||||
|                     TaskManager.runTask(() -> { | ||||
|                         plot.removeRunning(); | ||||
|                         // If the state changes, then mark it as no longer done | ||||
|                         if (DoneFlag.isDone(plot)) { | ||||
|                             PlotFlag<?, ?> plotFlag = | ||||
|                                     plot.getFlagContainer().getFlag(DoneFlag.class); | ||||
|                             PlotFlagRemoveEvent event = this.eventDispatcher | ||||
|                                     .callFlagRemove(plotFlag, plot); | ||||
|                             if (event.getEventResult() != Result.DENY) { | ||||
|                                 plot.removeFlag(event.getFlag()); | ||||
|                             } | ||||
|                 boolean result = plot.getPlotModificationManager().clear(true, false, player, () -> TaskManager.runTask(() -> { | ||||
|                     plot.removeRunning(); | ||||
|                     // If the state changes, then mark it as no longer done | ||||
|                     if (DoneFlag.isDone(plot)) { | ||||
|                         PlotFlag<?, ?> plotFlag = | ||||
|                                 plot.getFlagContainer().getFlag(DoneFlag.class); | ||||
|                         PlotFlagRemoveEvent event = this.eventDispatcher | ||||
|                                 .callFlagRemove(plotFlag, plot); | ||||
|                         if (event.getEventResult() != Result.DENY) { | ||||
|                             plot.removeFlag(event.getFlag()); | ||||
|                         } | ||||
|                         if (!plot.getFlag(AnalysisFlag.class).isEmpty()) { | ||||
|                             PlotFlag<?, ?> plotFlag = | ||||
|                                     plot.getFlagContainer().getFlag(AnalysisFlag.class); | ||||
|                             PlotFlagRemoveEvent event = this.eventDispatcher | ||||
|                                     .callFlagRemove(plotFlag, plot); | ||||
|                             if (event.getEventResult() != Result.DENY) { | ||||
|                                 plot.removeFlag(event.getFlag()); | ||||
|                             } | ||||
|                     } | ||||
|                     if (!plot.getFlag(AnalysisFlag.class).isEmpty()) { | ||||
|                         PlotFlag<?, ?> plotFlag = | ||||
|                                 plot.getFlagContainer().getFlag(AnalysisFlag.class); | ||||
|                         PlotFlagRemoveEvent event = this.eventDispatcher | ||||
|                                 .callFlagRemove(plotFlag, plot); | ||||
|                         if (event.getEventResult() != Result.DENY) { | ||||
|                             plot.removeFlag(event.getFlag()); | ||||
|                         } | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("working.clearing_done"), | ||||
|                                 TagResolver.builder() | ||||
|                                         .tag("amount", Tag.inserting(Component.text(System.currentTimeMillis() - start))) | ||||
|                                         .tag("plot", Tag.inserting(Component.text(plot.getId().toString()))) | ||||
|                                         .build() | ||||
|                         ); | ||||
|                     }); | ||||
|                 }); | ||||
|                     } | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("working.clearing_done"), | ||||
|                             TagResolver.builder() | ||||
|                                     .tag("world", Tag.inserting(Component.text(plot.getArea().getWorldName()))) | ||||
|                                     .tag("amount", Tag.inserting(Component.text(System.currentTimeMillis() - start))) | ||||
|                                     .tag("plot", Tag.inserting(Component.text(plot.getId().toString()))) | ||||
|                                     .build() | ||||
|                     ); | ||||
|                     this.eventDispatcher.callPostPlotClear(player, plot); | ||||
|                 })); | ||||
|                 if (!result) { | ||||
|                     player.sendMessage(TranslatableCaption.of("errors.wait_for_timer")); | ||||
|                 } else { | ||||
|   | ||||
| @@ -33,7 +33,6 @@ import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.PlotCluster; | ||||
| import com.plotsquared.core.plot.PlotId; | ||||
| import com.plotsquared.core.util.ComponentHelper; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.TabCompletions; | ||||
| import com.plotsquared.core.util.query.PlotQuery; | ||||
| import net.kyori.adventure.text.Component; | ||||
| @@ -47,6 +46,7 @@ import java.util.Collections; | ||||
| import java.util.HashSet; | ||||
| import java.util.LinkedList; | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
| import java.util.Set; | ||||
| import java.util.UUID; | ||||
| import java.util.concurrent.TimeoutException; | ||||
| @@ -78,9 +78,8 @@ public class Cluster extends SubCommand { | ||||
|         } | ||||
|         String sub = args[0].toLowerCase(); | ||||
|         switch (sub) { | ||||
|             case "l": | ||||
|             case "list": { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_LIST)) { | ||||
|             case "l", "list" -> { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_CLUSTER_LIST)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -134,9 +133,8 @@ public class Cluster extends SubCommand { | ||||
|                 } | ||||
|                 return true; | ||||
|             } | ||||
|             case "c": | ||||
|             case "create": { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_CREATE)) { | ||||
|             case "c", "create" -> { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_CLUSTER_CREATE)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -213,8 +211,7 @@ public class Cluster extends SubCommand { | ||||
|                 } | ||||
|                 Set<Plot> plots = area.getPlotSelectionOwned(pos1, pos2); | ||||
|                 if (!plots.isEmpty()) { | ||||
|                     if (!Permissions | ||||
|                             .hasPermission(player, Permission.PERMISSION_CLUSTER_CREATE_OTHER)) { | ||||
|                     if (!player.hasPermission(Permission.PERMISSION_CLUSTER_CREATE_OTHER)) { | ||||
|                         UUID uuid = player.getUUID(); | ||||
|                         for (Plot plot : plots) { | ||||
|                             if (!plot.isOwner(uuid)) { | ||||
| @@ -238,10 +235,10 @@ public class Cluster extends SubCommand { | ||||
|                 } else { | ||||
|                     current = player.getPlayerClusterCount(player.getLocation().getWorldName()); | ||||
|                 } | ||||
|                 int allowed = Permissions | ||||
|                         .hasPermissionRange(player, Permission.PERMISSION_CLUSTER_SIZE, | ||||
|                                 Settings.Limit.MAX_PLOTS | ||||
|                         ); | ||||
|                 int allowed = player.hasPermissionRange( | ||||
|                         Permission.PERMISSION_CLUSTER_SIZE, | ||||
|                         Settings.Limit.MAX_PLOTS | ||||
|                 ); | ||||
|                 if (current + cluster.getArea() > allowed) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
| @@ -271,10 +268,8 @@ public class Cluster extends SubCommand { | ||||
|                 ); | ||||
|                 return true; | ||||
|             } | ||||
|             case "disband": | ||||
|             case "del": | ||||
|             case "delete": { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_DELETE)) { | ||||
|             case "disband", "del", "delete" -> { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_CLUSTER_DELETE)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -314,8 +309,7 @@ public class Cluster extends SubCommand { | ||||
|                     } | ||||
|                 } | ||||
|                 if (!cluster.owner.equals(player.getUUID())) { | ||||
|                     if (!Permissions | ||||
|                             .hasPermission(player, Permission.PERMISSION_CLUSTER_DELETE_OTHER)) { | ||||
|                     if (!player.hasPermission(Permission.PERMISSION_CLUSTER_DELETE_OTHER)) { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("permission.no_permission"), | ||||
|                                 TagResolver.resolver( | ||||
| @@ -333,9 +327,8 @@ public class Cluster extends SubCommand { | ||||
|                 )); | ||||
|                 return true; | ||||
|             } | ||||
|             case "res": | ||||
|             case "resize": { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_RESIZE)) { | ||||
|             case "res", "resize" -> { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_CLUSTER_RESIZE)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -378,8 +371,7 @@ public class Cluster extends SubCommand { | ||||
|                     return false; | ||||
|                 } | ||||
|                 if (!cluster.hasHelperRights(player.getUUID())) { | ||||
|                     if (!Permissions | ||||
|                             .hasPermission(player, Permission.PERMISSION_CLUSTER_RESIZE_OTHER)) { | ||||
|                     if (!player.hasPermission(Permission.PERMISSION_CLUSTER_RESIZE_OTHER)) { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("permission.no_permission"), | ||||
|                                 TagResolver.resolver( | ||||
| @@ -406,8 +398,7 @@ public class Cluster extends SubCommand { | ||||
|                 removed.removeAll(newPlots); | ||||
|                 // Check expand / shrink | ||||
|                 if (!removed.isEmpty()) { | ||||
|                     if (!Permissions | ||||
|                             .hasPermission(player, Permission.PERMISSION_CLUSTER_RESIZE_SHRINK)) { | ||||
|                     if (!player.hasPermission(Permission.PERMISSION_CLUSTER_RESIZE_SHRINK)) { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("permission.no_permission"), | ||||
|                                 TagResolver.resolver( | ||||
| @@ -420,8 +411,7 @@ public class Cluster extends SubCommand { | ||||
|                 } | ||||
|                 newPlots.removeAll(existing); | ||||
|                 if (!newPlots.isEmpty()) { | ||||
|                     if (!Permissions | ||||
|                             .hasPermission(player, Permission.PERMISSION_CLUSTER_RESIZE_EXPAND)) { | ||||
|                     if (!player.hasPermission(Permission.PERMISSION_CLUSTER_RESIZE_EXPAND)) { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("permission.no_permission"), | ||||
|                                 TagResolver.resolver( | ||||
| @@ -440,7 +430,8 @@ public class Cluster extends SubCommand { | ||||
|                     current = player.getPlayerClusterCount(player.getLocation().getWorldName()); | ||||
|                 } | ||||
|                 current -= cluster.getArea() + (1 + pos2.getX() - pos1.getX()) * (1 + pos2.getY() - pos1.getY()); | ||||
|                 int allowed = Permissions.hasPermissionRange(player, Permission.PERMISSION_CLUSTER, | ||||
|                 int allowed = player.hasPermissionRange( | ||||
|                         Permission.PERMISSION_CLUSTER, | ||||
|                         Settings.Limit.MAX_PLOTS | ||||
|                 ); | ||||
|                 if (current + cluster.getArea() > allowed) { | ||||
| @@ -457,10 +448,8 @@ public class Cluster extends SubCommand { | ||||
|                 player.sendMessage(TranslatableCaption.of("cluster.cluster_resized")); | ||||
|                 return true; | ||||
|             } | ||||
|             case "add": | ||||
|             case "inv": | ||||
|             case "invite": { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_INVITE)) { | ||||
|             case "add", "inv", "invite" -> { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_CLUSTER_INVITE)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -488,8 +477,7 @@ public class Cluster extends SubCommand { | ||||
|                     return false; | ||||
|                 } | ||||
|                 if (!cluster.hasHelperRights(player.getUUID())) { | ||||
|                     if (!Permissions | ||||
|                             .hasPermission(player, Permission.PERMISSION_CLUSTER_INVITE_OTHER)) { | ||||
|                     if (!player.hasPermission(Permission.PERMISSION_CLUSTER_INVITE_OTHER)) { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("permission.no_permission"), | ||||
|                                 TagResolver.resolver( | ||||
| @@ -529,10 +517,8 @@ public class Cluster extends SubCommand { | ||||
|                         }); | ||||
|                 return true; | ||||
|             } | ||||
|             case "k": | ||||
|             case "remove": | ||||
|             case "kick": { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_KICK)) { | ||||
|             case "k", "remove", "kick" -> { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_CLUSTER_KICK)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -559,8 +545,7 @@ public class Cluster extends SubCommand { | ||||
|                     return false; | ||||
|                 } | ||||
|                 if (!cluster.hasHelperRights(player.getUUID())) { | ||||
|                     if (!Permissions | ||||
|                             .hasPermission(player, Permission.PERMISSION_CLUSTER_KICK_OTHER)) { | ||||
|                     if (!player.hasPermission(Permission.PERMISSION_CLUSTER_KICK_OTHER)) { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("permission.no_permission"), | ||||
|                                 TagResolver.resolver( | ||||
| @@ -612,9 +597,8 @@ public class Cluster extends SubCommand { | ||||
|                         }); | ||||
|                 return true; | ||||
|             } | ||||
|             case "quit": | ||||
|             case "leave": { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_LEAVE)) { | ||||
|             case "quit", "leave" -> { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_CLUSTER_LEAVE)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -674,8 +658,8 @@ public class Cluster extends SubCommand { | ||||
|                 removePlayerPlots(cluster, uuid, player.getLocation().getWorldName()); | ||||
|                 return true; | ||||
|             } | ||||
|             case "members": { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_HELPERS)) { | ||||
|             case "members" -> { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_CLUSTER_HELPERS)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -735,10 +719,8 @@ public class Cluster extends SubCommand { | ||||
|                         }); | ||||
|                 return true; | ||||
|             } | ||||
|             case "spawn": | ||||
|             case "home": | ||||
|             case "tp": { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_TP)) { | ||||
|             case "spawn", "home", "tp" -> { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_CLUSTER_TP)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -770,7 +752,7 @@ public class Cluster extends SubCommand { | ||||
|                 } | ||||
|                 UUID uuid = player.getUUID(); | ||||
|                 if (!cluster.isAdded(uuid)) { | ||||
|                     if (!Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_TP_OTHER)) { | ||||
|                     if (!player.hasPermission(Permission.PERMISSION_CLUSTER_TP_OTHER)) { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("permission.no_permission"), | ||||
|                                 TagResolver.resolver( | ||||
| @@ -785,11 +767,8 @@ public class Cluster extends SubCommand { | ||||
|                 player.sendMessage(TranslatableCaption.of("cluster.cluster_teleporting")); | ||||
|                 return true; | ||||
|             } | ||||
|             case "i": | ||||
|             case "info": | ||||
|             case "show": | ||||
|             case "information": { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_INFO)) { | ||||
|             case "i", "info", "show", "information" -> { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_CLUSTER_INFO)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver( | ||||
| @@ -835,11 +814,7 @@ public class Cluster extends SubCommand { | ||||
|                                 player.sendMessage(TranslatableCaption.of("players.fetching_players_timeout")); | ||||
|                             } else { | ||||
|                                 final String owner; | ||||
|                                 if (username == null) { | ||||
|                                     owner = "unknown"; | ||||
|                                 } else { | ||||
|                                     owner = username; | ||||
|                                 } | ||||
|                                 owner = Objects.requireNonNullElse(username, "unknown"); | ||||
|                                 String name = cluster.getName(); | ||||
|                                 String size = (cluster.getP2().getX() - cluster.getP1().getX() + 1) + "x" + ( | ||||
|                                         cluster.getP2().getY() - cluster.getP1().getY() + 1); | ||||
| @@ -857,10 +832,8 @@ public class Cluster extends SubCommand { | ||||
|                         }); | ||||
|                 return true; | ||||
|             } | ||||
|             case "sh": | ||||
|             case "setspawn": | ||||
|             case "sethome": { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_SETHOME)) { | ||||
|             case "sh", "setspawn", "sethome" -> { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_CLUSTER_SETHOME)) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver("node", Tag.inserting(Permission.PERMISSION_CLUSTER_SETHOME)) | ||||
| @@ -884,8 +857,7 @@ public class Cluster extends SubCommand { | ||||
|                     return false; | ||||
|                 } | ||||
|                 if (!cluster.hasHelperRights(player.getUUID())) { | ||||
|                     if (!Permissions | ||||
|                             .hasPermission(player, Permission.PERMISSION_CLUSTER_SETHOME_OTHER)) { | ||||
|                     if (!player.hasPermission(Permission.PERMISSION_CLUSTER_SETHOME_OTHER)) { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("permission.no_permission"), | ||||
|                                 TagResolver.resolver( | ||||
| @@ -937,37 +909,37 @@ public class Cluster extends SubCommand { | ||||
|     public Collection<Command> tab(final PlotPlayer<?> player, final String[] args, final boolean space) { | ||||
|         if (args.length == 1) { | ||||
|             final List<String> completions = new LinkedList<>(); | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_LIST)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_CLUSTER_LIST)) { | ||||
|                 completions.add("list"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_CREATE)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_CLUSTER_CREATE)) { | ||||
|                 completions.add("create"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_DELETE)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_CLUSTER_DELETE)) { | ||||
|                 completions.add("delete"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_RESIZE)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_CLUSTER_RESIZE)) { | ||||
|                 completions.add("resize"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_INVITE)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_CLUSTER_INVITE)) { | ||||
|                 completions.add("invite"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_KICK)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_CLUSTER_KICK)) { | ||||
|                 completions.add("kick"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_KICK)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_CLUSTER_KICK)) { | ||||
|                 completions.add("leave"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_HELPERS)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_CLUSTER_HELPERS)) { | ||||
|                 completions.add("members"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_INFO)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_CLUSTER_INFO)) { | ||||
|                 completions.add("info"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_TP)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_CLUSTER_TP)) { | ||||
|                 completions.add("tp"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER_SETHOME)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_CLUSTER_SETHOME)) { | ||||
|                 completions.add("sethome"); | ||||
|             } | ||||
|             final List<Command> commands = completions.stream().filter(completion -> completion | ||||
| @@ -982,7 +954,7 @@ public class Cluster extends SubCommand { | ||||
|                             CommandCategory.ADMINISTRATION | ||||
|                     ) { | ||||
|                     }).collect(Collectors.toCollection(LinkedList::new)); | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_CLUSTER) && args[0].length() > 0) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_CLUSTER) && args[0].length() > 0) { | ||||
|                 commands.addAll(TabCompletions.completePlayers(player, args[0], Collections.emptyList())); | ||||
|             } | ||||
|             return commands; | ||||
|   | ||||
| @@ -25,7 +25,6 @@ import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.permissions.PermissionHolder; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.util.MathMan; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.StringComparison; | ||||
| import com.plotsquared.core.util.StringMan; | ||||
| import com.plotsquared.core.util.task.RunnableVal2; | ||||
| @@ -483,7 +482,7 @@ public abstract class Command { | ||||
|             if (message) { | ||||
|                 player.sendMessage(this.required.getErrorMessage()); | ||||
|             } | ||||
|         } else if (!Permissions.hasPermission(player, getPermission())) { | ||||
|         } else if (!player.hasPermission(getPermission())) { | ||||
|             if (message) { | ||||
|                 player.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission"), | ||||
| @@ -560,9 +559,10 @@ public abstract class Command { | ||||
|  | ||||
|     public Collection<Command> tab(PlotPlayer<?> player, String[] args, boolean space) { | ||||
|         switch (args.length) { | ||||
|             case 0: | ||||
|             case 0 -> { | ||||
|                 return this.allCommands; | ||||
|             case 1: | ||||
|             } | ||||
|             case 1 -> { | ||||
|                 String arg = args[0].toLowerCase(); | ||||
|                 if (space) { | ||||
|                     Command cmd = getCommand(arg); | ||||
| @@ -581,13 +581,15 @@ public abstract class Command { | ||||
|                     } | ||||
|                     return commands; | ||||
|                 } | ||||
|             default: | ||||
|             } | ||||
|             default -> { | ||||
|                 Command cmd = getCommand(args[0]); | ||||
|                 if (cmd != null) { | ||||
|                     return cmd.tab(player, Arrays.copyOfRange(args, 1, args.length), space); | ||||
|                 } else { | ||||
|                     return null; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -23,9 +23,10 @@ import com.plotsquared.core.configuration.caption.LocaleHolder; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.ComponentLike; | ||||
| import net.kyori.adventure.text.minimessage.MiniMessage; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
|  | ||||
| /** | ||||
|  * CommandCategory. | ||||
| @@ -83,7 +84,7 @@ public enum CommandCategory implements Caption { | ||||
|     // TODO this method shouldn't be invoked | ||||
|     @Deprecated | ||||
|     @Override | ||||
|     public String toString() { | ||||
|     public @NotNull String toString() { | ||||
|         return this.caption.getComponent(LocaleHolder.console()); | ||||
|     } | ||||
|  | ||||
| @@ -98,6 +99,14 @@ public enum CommandCategory implements Caption { | ||||
|         return MiniMessage.miniMessage().deserialize(getComponent(localeHolder)); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public @NonNull Component toComponent( | ||||
|             @NonNull final LocaleHolder localeHolder, | ||||
|             final @NonNull TagResolver @NonNull ... tagResolvers | ||||
|     ) { | ||||
|         return MiniMessage.miniMessage().deserialize(getComponent(localeHolder)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Checks if a player has access to this command category | ||||
|      * | ||||
| @@ -109,4 +118,5 @@ public enum CommandCategory implements Caption { | ||||
|         return !MainCommand.getInstance().getCommands(this, player).isEmpty(); | ||||
|     } | ||||
|  | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -256,11 +256,11 @@ public class Condense extends SubCommand { | ||||
|                 player.sendMessage(TranslatableCaption.of("condense.default_eval")); | ||||
|                 player.sendMessage( | ||||
|                         TranslatableCaption.of("condense.minimum_radius"), | ||||
|                         TagResolver.resolver("minimumRadius", Tag.inserting(Component.text(minimumRadius))) | ||||
|                         TagResolver.resolver("minimum_radius", Tag.inserting(Component.text(minimumRadius))) | ||||
|                 ); | ||||
|                 player.sendMessage( | ||||
|                         TranslatableCaption.of("condense.maximum_moved"), | ||||
|                         TagResolver.resolver("maxMove", Tag.inserting(Component.text(maxMove))) | ||||
|                         TagResolver.resolver("maximum_moves", Tag.inserting(Component.text(maxMove))) | ||||
|                 ); | ||||
|                 player.sendMessage(TranslatableCaption.of("condense.input_eval")); | ||||
|                 player.sendMessage( | ||||
| @@ -269,7 +269,7 @@ public class Condense extends SubCommand { | ||||
|                 ); | ||||
|                 player.sendMessage( | ||||
|                         TranslatableCaption.of("condense.estimated_moves"), | ||||
|                         TagResolver.resolver("userMove", Tag.inserting(Component.text(userMove))) | ||||
|                         TagResolver.resolver("user_move", Tag.inserting(Component.text(userMove))) | ||||
|                 ); | ||||
|                 player.sendMessage(TranslatableCaption.of("condense.eta")); | ||||
|                 player.sendMessage(TranslatableCaption.of("condense.radius_measured")); | ||||
|   | ||||
| @@ -29,7 +29,6 @@ import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.flag.PlotFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.DoneFlag; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
| @@ -55,8 +54,7 @@ public class Continue extends SubCommand { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||
|             return false; | ||||
|         } | ||||
|         if (!plot.isOwner(player.getUUID()) && !Permissions | ||||
|                 .hasPermission(player, Permission.PERMISSION_ADMIN_COMMAND_CONTINUE)) { | ||||
|         if (!plot.isOwner(player.getUUID()) && !player.hasPermission(Permission.PERMISSION_ADMIN_COMMAND_CONTINUE)) { | ||||
|             player.sendMessage( | ||||
|                     TranslatableCaption.of("permission.no_permission"), | ||||
|                     TagResolver.resolver("node", Tag.inserting( | ||||
| @@ -70,8 +68,8 @@ public class Continue extends SubCommand { | ||||
|             return false; | ||||
|         } | ||||
|         int size = plot.getConnectedPlots().size(); | ||||
|         if (!Settings.Done.COUNTS_TOWARDS_LIMIT && (player.getAllowedPlots() | ||||
|                 < player.getPlotCount() + size)) { | ||||
|         int plotCount = Settings.Limit.GLOBAL ? player.getPlotCount() : player.getPlotCount(plot.getWorldName()); | ||||
|         if (!Settings.Done.COUNTS_TOWARDS_LIMIT && (player.getAllowedPlots() < plotCount + size)) { | ||||
|             player.sendMessage( | ||||
|                     TranslatableCaption.of("permission.cant_claim_more_plots"), | ||||
|                     TagResolver.resolver("amount", Tag.inserting(Component.text(player.getAllowedPlots()))) | ||||
|   | ||||
| @@ -19,11 +19,9 @@ | ||||
| package com.plotsquared.core.command; | ||||
|  | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
| @@ -38,14 +36,12 @@ public class Copy extends SubCommand { | ||||
|  | ||||
|     @Override | ||||
|     public boolean onCommand(final PlotPlayer<?> player, String[] args) { | ||||
|         Location location = player.getLocation(); | ||||
|         Plot plot1 = location.getPlotAbs(); | ||||
|         Plot plot1 = player.getCurrentPlot(); | ||||
|         if (plot1 == null) { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||
|             return false; | ||||
|         } | ||||
|         if (!plot1.isOwner(player.getUUID()) && !Permissions | ||||
|                 .hasPermission(player, Permission.PERMISSION_ADMIN.toString())) { | ||||
|         if (!plot1.isOwner(player.getUUID()) && !player.hasPermission(Permission.PERMISSION_ADMIN.toString())) { | ||||
|             player.sendMessage(TranslatableCaption.of("permission.no_plot_perms")); | ||||
|             return false; | ||||
|         } | ||||
| @@ -71,7 +67,8 @@ public class Copy extends SubCommand { | ||||
|  | ||||
|         plot1.getPlotModificationManager().copy(plot2, player).thenAccept(result -> { | ||||
|             if (result) { | ||||
|                 player.sendMessage(TranslatableCaption.of("move.copy_success"), | ||||
|                 player.sendMessage( | ||||
|                         TranslatableCaption.of("move.copy_success"), | ||||
|                         TagResolver.builder() | ||||
|                                 .tag("origin", Tag.inserting(Component.text(plot1.toString()))) | ||||
|                                 .tag("target", Tag.inserting(Component.text(plot2.toString()))) | ||||
|   | ||||
| @@ -22,7 +22,6 @@ import com.google.inject.Inject; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.generator.HybridPlotWorld; | ||||
| import com.plotsquared.core.generator.HybridUtils; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import net.kyori.adventure.text.Component; | ||||
| @@ -47,8 +46,7 @@ public class CreateRoadSchematic extends SubCommand { | ||||
|  | ||||
|     @Override | ||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||
|         Location location = player.getLocation(); | ||||
|         Plot plot = location.getPlotAbs(); | ||||
|         Plot plot = player.getCurrentPlot(); | ||||
|         if (plot == null) { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||
|             return false; | ||||
| @@ -57,7 +55,7 @@ public class CreateRoadSchematic extends SubCommand { | ||||
|             player.sendMessage(TranslatableCaption.of("schematics.schematic_too_large")); | ||||
|             return false; | ||||
|         } | ||||
|         if (!(location.getPlotArea() instanceof HybridPlotWorld)) { | ||||
|         if (!(plot.getArea() instanceof HybridPlotWorld)) { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); | ||||
|         } | ||||
|         this.hybridUtils.setupRoadSchematic(plot); | ||||
|   | ||||
| @@ -184,7 +184,7 @@ public class DatabaseCommand extends SubCommand { | ||||
|                                                         ); | ||||
|                                                 worldFile.renameTo(newFile); | ||||
|                                             } | ||||
|                                             plot.setId(newId.copy()); | ||||
|                                             plot.setId(newId); | ||||
|                                             plot.setArea(pa); | ||||
|                                             plots.add(plot); | ||||
|                                             continue; | ||||
|   | ||||
| @@ -29,13 +29,10 @@ import com.plotsquared.core.util.WorldUtil; | ||||
| import com.plotsquared.core.util.entity.EntityCategories; | ||||
| import com.plotsquared.core.util.entity.EntityCategory; | ||||
| import com.plotsquared.core.util.query.PlotQuery; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import com.plotsquared.core.uuid.UUIDMapping; | ||||
| import com.sk89q.worldedit.world.entity.EntityType; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.TextComponent; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import net.kyori.adventure.text.format.TextColor; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| @@ -73,7 +70,7 @@ public class Debug extends SubCommand { | ||||
|                     TranslatableCaption.of("commandconfig.command_syntax"), | ||||
|                     TagResolver.resolver( | ||||
|                             "value", | ||||
|                             Tag.inserting(Component.text("/plot debug <loadedchunks | player | debug-players | entitytypes | msg>")) | ||||
|                             Tag.inserting(Component.text("/plot debug <player | debug-players | entitytypes | msg>")) | ||||
|                     ) | ||||
|             ); | ||||
|         } | ||||
| @@ -87,16 +84,6 @@ public class Debug extends SubCommand { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|         if (args.length > 0 && "loadedchunks".equalsIgnoreCase(args[0])) { | ||||
|             final long start = System.currentTimeMillis(); | ||||
|             player.sendMessage(TranslatableCaption.of("debug.fetching_loaded_chunks")); | ||||
|             TaskManager.runTaskAsync(() -> player.sendMessage(StaticCaption | ||||
|                     .of("Loaded chunks: " + this.worldUtil | ||||
|                             .getChunkChunks(player.getLocation().getWorldName()) | ||||
|                             .size() + " (" + (System.currentTimeMillis() | ||||
|                             - start) + "ms) using thread: " + Thread.currentThread().getName()))); | ||||
|             return true; | ||||
|         } | ||||
|         if (args.length > 0 && "uuids".equalsIgnoreCase(args[0])) { | ||||
|             final Collection<UUIDMapping> mappings = PlotSquared.get().getImpromptuUUIDPipeline().getAllImmediately(); | ||||
|             player.sendMessage( | ||||
| @@ -198,7 +185,7 @@ public class Debug extends SubCommand { | ||||
|  | ||||
|     @Override | ||||
|     public Collection<Command> tab(final PlotPlayer<?> player, String[] args, boolean space) { | ||||
|         return Stream.of("loadedchunks", "debug-players", "entitytypes") | ||||
|         return Stream.of("debug-players", "entitytypes") | ||||
|                 .filter(value -> value.startsWith(args[0].toLowerCase(Locale.ENGLISH))) | ||||
|                 .map(value -> new Command(null, false, value, "plots.admin", RequiredType.NONE, null) { | ||||
|                 }).collect(Collectors.toList()); | ||||
|   | ||||
| @@ -19,6 +19,7 @@ | ||||
| package com.plotsquared.core.command; | ||||
|  | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.caption.StaticCaption; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.events.PlotFlagRemoveEvent; | ||||
| @@ -27,7 +28,6 @@ import com.plotsquared.core.generator.HybridUtils; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.expiration.ExpireManager; | ||||
| import com.plotsquared.core.plot.expiration.PlotAnalysis; | ||||
| import com.plotsquared.core.plot.flag.GlobalFlagContainer; | ||||
| import com.plotsquared.core.plot.flag.PlotFlag; | ||||
| @@ -143,10 +143,7 @@ public class DebugExec extends SubCommand { | ||||
|                     return true; | ||||
|                 } | ||||
|                 case "start-expire" -> { | ||||
|                     if (ExpireManager.IMP == null) { | ||||
|                         ExpireManager.IMP = new ExpireManager(this.eventDispatcher); | ||||
|                     } | ||||
|                     if (ExpireManager.IMP.runAutomatedTask()) { | ||||
|                     if (PlotSquared.platform().expireManager().runAutomatedTask()) { | ||||
|                         player.sendMessage(TranslatableCaption.of("debugexec.expiry_started")); | ||||
|                     } else { | ||||
|                         player.sendMessage(TranslatableCaption.of("debugexec.expiry_already_started")); | ||||
| @@ -154,7 +151,7 @@ public class DebugExec extends SubCommand { | ||||
|                     return true; | ||||
|                 } | ||||
|                 case "stop-expire" -> { | ||||
|                     if (ExpireManager.IMP == null || !ExpireManager.IMP.cancelTask()) { | ||||
|                     if (!PlotSquared.platform().expireManager().cancelTask()) { | ||||
|                         player.sendMessage(TranslatableCaption.of("debugexec.task_halted")); | ||||
|                     } else { | ||||
|                         player.sendMessage(TranslatableCaption.of("debugexec.task_cancelled")); | ||||
|   | ||||
| @@ -71,9 +71,9 @@ public class DebugPaste extends SubCommand { | ||||
|                 StringBuilder b = new StringBuilder(); | ||||
|                 b.append( | ||||
|                         """ | ||||
|                          # Welcome to this paste | ||||
|                          # It is meant to provide us at IntellectualSites with better information about your problem | ||||
|                          """ | ||||
|                                 # Welcome to this paste | ||||
|                                 # It is meant to provide us at IntellectualSites with better information about your problem | ||||
|                                 """ | ||||
|                 ); | ||||
|                 b.append("# PlotSquared Information\n"); | ||||
|                 b.append("PlotSquared Version: ").append(PlotSquared.get().getVersion()) | ||||
| @@ -86,7 +86,8 @@ public class DebugPaste extends SubCommand { | ||||
|                 b.append("# WorldEdit implementation:\n"); | ||||
|                 b.append(PlotSquared.platform().worldEditImplementations()).append("\n\n"); | ||||
|                 b.append("# Server Information\n"); | ||||
|                 b.append("Server Version: ").append(PlotSquared.platform().serverImplementation()) | ||||
|                 b.append("Server Version: ").append(PlotSquared.platform().serverBrand()).append(": ") | ||||
|                         .append(PlotSquared.platform().serverImplementation()).append("\n") | ||||
|                         .append("\n"); | ||||
|                 b.append("online_mode: ").append(!Settings.UUID.OFFLINE).append(';') | ||||
|                         .append(!Settings.UUID.OFFLINE).append('\n'); | ||||
|   | ||||
| @@ -22,7 +22,6 @@ import com.google.inject.Inject; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.generator.HybridPlotManager; | ||||
| import com.plotsquared.core.generator.HybridUtils; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| @@ -57,8 +56,7 @@ public class DebugRoadRegen extends SubCommand { | ||||
|  | ||||
|     @Override | ||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||
|         Location location = player.getLocation(); | ||||
|         Plot plot = location.getPlotAbs(); | ||||
|         Plot plot = player.getCurrentPlot(); | ||||
|         if (args.length < 1) { | ||||
|             player.sendMessage( | ||||
|                     TranslatableCaption.of("commandconfig.command_syntax"), | ||||
| @@ -75,24 +73,27 @@ public class DebugRoadRegen extends SubCommand { | ||||
|         } | ||||
|         String kind = args[0].toLowerCase(); | ||||
|         switch (kind) { | ||||
|             case "plot": | ||||
|             case "plot" -> { | ||||
|                 return regenPlot(player); | ||||
|             case "region": | ||||
|             } | ||||
|             case "region" -> { | ||||
|                 return regenRegion(player, Arrays.copyOfRange(args, 1, args.length)); | ||||
|             default: | ||||
|             } | ||||
|             default -> { | ||||
|                 player.sendMessage( | ||||
|                         TranslatableCaption.of("commandconfig.command_syntax"), | ||||
|                         TagResolver.resolver("value", Tag.inserting(Component.text(DebugRoadRegen.USAGE))) | ||||
|                 ); | ||||
|                 return false; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public boolean regenPlot(PlotPlayer<?> player) { | ||||
|         Location location = player.getLocation(); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         PlotArea area = player.getCurrentPlot().getArea(); | ||||
|         if (area == null) { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); | ||||
|             return false; | ||||
|         } | ||||
|         Plot plot = player.getCurrentPlot(); | ||||
|         if (plot == null) { | ||||
| @@ -144,8 +145,7 @@ public class DebugRoadRegen extends SubCommand { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         Location location = player.getLocation(); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         PlotArea area = player.getCurrentPlot().getArea(); | ||||
|         if (area == null) { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); | ||||
|         } | ||||
|   | ||||
| @@ -23,14 +23,12 @@ import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.events.Result; | ||||
| import com.plotsquared.core.events.TeleportCause; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.util.EconHandler; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.PlotExpression; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import net.kyori.adventure.text.Component; | ||||
| @@ -62,8 +60,7 @@ public class Delete extends SubCommand { | ||||
|  | ||||
|     @Override | ||||
|     public boolean onCommand(final PlotPlayer<?> player, String[] args) { | ||||
|         Location location = player.getLocation(); | ||||
|         final Plot plot = location.getPlotAbs(); | ||||
|         final Plot plot = player.getCurrentPlot(); | ||||
|         if (plot == null) { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||
|             return false; | ||||
| @@ -85,8 +82,7 @@ public class Delete extends SubCommand { | ||||
|             return true; | ||||
|         } | ||||
|         boolean force = eventResult == Result.FORCE; | ||||
|         if (!force && !plot.isOwner(player.getUUID()) && !Permissions | ||||
|                 .hasPermission(player, Permission.PERMISSION_ADMIN_COMMAND_DELETE)) { | ||||
|         if (!force && !plot.isOwner(player.getUUID()) && !player.hasPermission(Permission.PERMISSION_ADMIN_COMMAND_DELETE)) { | ||||
|             player.sendMessage(TranslatableCaption.of("permission.no_plot_perms")); | ||||
|             return false; | ||||
|         } | ||||
| @@ -94,7 +90,7 @@ public class Delete extends SubCommand { | ||||
|         final java.util.Set<Plot> plots = plot.getConnectedPlots(); | ||||
|         final int currentPlots = Settings.Limit.GLOBAL ? | ||||
|                 player.getPlotCount() : | ||||
|                 player.getPlotCount(location.getWorldName()); | ||||
|                 player.getPlotCount(plot.getWorldName()); | ||||
|         Runnable run = () -> { | ||||
|             if (plot.getRunning() > 0) { | ||||
|                 player.sendMessage(TranslatableCaption.of("errors.wait_for_timer")); | ||||
| @@ -126,6 +122,7 @@ public class Delete extends SubCommand { | ||||
|                                 "amount", | ||||
|                                 Tag.inserting(Component.text(String.valueOf(System.currentTimeMillis() - start))) | ||||
|                         ), | ||||
|                         TagResolver.resolver("world", Tag.inserting(Component.text(plotArea.getWorldName()))), | ||||
|                         TagResolver.resolver("plot", Tag.inserting(Component.text(plot.getId().toString()))) | ||||
|                 ); | ||||
|                 eventDispatcher.callPostDelete(plot); | ||||
|   | ||||
| @@ -30,7 +30,6 @@ import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.PlayerManager; | ||||
| import com.plotsquared.core.util.TabCompletions; | ||||
| import com.plotsquared.core.util.WorldUtil; | ||||
| @@ -71,8 +70,7 @@ public class Deny extends SubCommand { | ||||
|     @Override | ||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||
|  | ||||
|         Location location = player.getLocation(); | ||||
|         final Plot plot = location.getPlotAbs(); | ||||
|         final Plot plot = player.getCurrentPlot(); | ||||
|         if (plot == null) { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||
|             return false; | ||||
| @@ -81,13 +79,12 @@ public class Deny extends SubCommand { | ||||
|             player.sendMessage(TranslatableCaption.of("info.plot_unowned")); | ||||
|             return false; | ||||
|         } | ||||
|         if (!plot.isOwner(player.getUUID()) && !Permissions | ||||
|                 .hasPermission(player, Permission.PERMISSION_ADMIN_COMMAND_DENY)) { | ||||
|         if (!plot.isOwner(player.getUUID()) && !player.hasPermission(Permission.PERMISSION_ADMIN_COMMAND_DENY)) { | ||||
|             player.sendMessage(TranslatableCaption.of("permission.no_plot_perms")); | ||||
|             return true; | ||||
|         } | ||||
|  | ||||
|         int maxDenySize = Permissions.hasPermissionRange(player, Permission.PERMISSION_DENY, Settings.Limit.MAX_PLOTS); | ||||
|         int maxDenySize = player.hasPermissionRange(Permission.PERMISSION_DENY, Settings.Limit.MAX_PLOTS); | ||||
|         int size = plot.getDenied().size(); | ||||
|         if (size >= maxDenySize) { | ||||
|             player.sendMessage( | ||||
| @@ -108,8 +105,7 @@ public class Deny extends SubCommand { | ||||
|             } else { | ||||
|                 for (UUID uuid : uuids) { | ||||
|                     if (uuid == DBFunc.EVERYONE && !( | ||||
|                             Permissions.hasPermission(player, Permission.PERMISSION_DENY_EVERYONE) || Permissions | ||||
|                                     .hasPermission(player, Permission.PERMISSION_ADMIN_COMMAND_DENY))) { | ||||
|                             player.hasPermission(Permission.PERMISSION_DENY_EVERYONE) || player.hasPermission(Permission.PERMISSION_ADMIN_COMMAND_DENY))) { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("errors.invalid_player"), | ||||
|                                 TagResolver.resolver("value", Tag.inserting(Component.text(args[0]))) | ||||
| @@ -120,10 +116,11 @@ public class Deny extends SubCommand { | ||||
|                     } else if (plot.getDenied().contains(uuid)) { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("member.already_added"), | ||||
|                                 TagResolver.resolver( | ||||
|                                 PlotSquared.platform().playerManager().getUsernameCaption(uuid) | ||||
|                                         .thenApply(caption -> TagResolver.resolver( | ||||
|                                         "player", | ||||
|                                         Tag.inserting(PlayerManager.resolveName(uuid).toComponent(player)) | ||||
|                                 ) | ||||
|                                         Tag.inserting(caption.toComponent(player)) | ||||
|                                 )) | ||||
|                         ); | ||||
|                         return; | ||||
|                     } else { | ||||
|   | ||||
| @@ -21,7 +21,6 @@ package com.plotsquared.core.command; | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.TabCompletions; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| @@ -54,7 +53,7 @@ public class Dislike extends SubCommand { | ||||
|     public Collection<Command> tab(final PlotPlayer<?> player, final String[] args, final boolean space) { | ||||
|         if (args.length == 1) { | ||||
|             final List<String> completions = new LinkedList<>(); | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_ADMIN_COMMAND_PURGE_RATINGS)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_ADMIN_COMMAND_PURGE_RATINGS)) { | ||||
|                 completions.add("purge"); | ||||
|             } | ||||
|             final List<Command> commands = completions.stream().filter(completion -> completion | ||||
| @@ -62,7 +61,7 @@ public class Dislike extends SubCommand { | ||||
|                             .startsWith(args[0].toLowerCase())) | ||||
|                     .map(completion -> new Command(null, true, completion, "", RequiredType.PLAYER, CommandCategory.INFO) { | ||||
|                     }).collect(Collectors.toCollection(LinkedList::new)); | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_RATE) && args[0].length() > 0) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_RATE) && args[0].length() > 0) { | ||||
|                 commands.addAll(TabCompletions.completePlayers(player, args[0], Collections.emptyList())); | ||||
|             } | ||||
|             return commands; | ||||
|   | ||||
| @@ -19,22 +19,20 @@ | ||||
| package com.plotsquared.core.command; | ||||
|  | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.events.PlotDoneEvent; | ||||
| import com.plotsquared.core.events.PlotFlagAddEvent; | ||||
| import com.plotsquared.core.events.Result; | ||||
| import com.plotsquared.core.generator.HybridUtils; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.expiration.ExpireManager; | ||||
| import com.plotsquared.core.plot.expiration.PlotAnalysis; | ||||
| import com.plotsquared.core.plot.flag.PlotFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.DoneFlag; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.task.RunnableVal; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| @@ -62,8 +60,7 @@ public class Done extends SubCommand { | ||||
|  | ||||
|     @Override | ||||
|     public boolean onCommand(final PlotPlayer<?> player, String[] args) { | ||||
|         Location location = player.getLocation(); | ||||
|         final Plot plot = location.getPlotAbs(); | ||||
|         final Plot plot = player.getCurrentPlot(); | ||||
|         if ((plot == null) || !plot.hasOwner()) { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||
|             return false; | ||||
| @@ -77,8 +74,7 @@ public class Done extends SubCommand { | ||||
|             return true; | ||||
|         } | ||||
|         boolean force = event.getEventResult() == Result.FORCE; | ||||
|         if (!force && !plot.isOwner(player.getUUID()) && !Permissions | ||||
|                 .hasPermission(player, Permission.PERMISSION_ADMIN_COMMAND_DONE)) { | ||||
|         if (!force && !plot.isOwner(player.getUUID()) && !player.hasPermission(Permission.PERMISSION_ADMIN_COMMAND_DONE)) { | ||||
|             player.sendMessage(TranslatableCaption.of("permission.no_plot_perms")); | ||||
|             return false; | ||||
|         } | ||||
| @@ -96,7 +92,7 @@ public class Done extends SubCommand { | ||||
|                 TagResolver.resolver("plot", Tag.inserting(Component.text(plot.getId().toString()))) | ||||
|         ); | ||||
|         final Settings.Auto_Clear doneRequirements = Settings.AUTO_CLEAR.get("done"); | ||||
|         if (ExpireManager.IMP == null || doneRequirements == null) { | ||||
|         if (PlotSquared.platform().expireManager() == null || doneRequirements == null || player.hasPermission(Permission.PERMISSION_ADMIN_COMMAND_DONE)) { | ||||
|             finish(plot, player, true); | ||||
|             plot.removeRunning(); | ||||
|         } else { | ||||
| @@ -105,7 +101,7 @@ public class Done extends SubCommand { | ||||
|                 public void run(PlotAnalysis value) { | ||||
|                     plot.removeRunning(); | ||||
|                     boolean result = | ||||
|                             value.getComplexity(doneRequirements) <= doneRequirements.THRESHOLD; | ||||
|                             value.getComplexity(doneRequirements) >= doneRequirements.THRESHOLD; | ||||
|                     finish(plot, player, result); | ||||
|                 } | ||||
|             }); | ||||
|   | ||||
| @@ -20,14 +20,12 @@ package com.plotsquared.core.command; | ||||
|  | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.StaticCaption; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| 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.DoneFlag; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.PlotUploader; | ||||
| import com.plotsquared.core.util.SchematicHandler; | ||||
| import com.plotsquared.core.util.StringMan; | ||||
| @@ -75,7 +73,7 @@ public class Download extends SubCommand { | ||||
|  | ||||
|     @Override | ||||
|     public boolean onCommand(final PlotPlayer<?> player, String[] args) { | ||||
|         String world = player.getLocation().getWorldName(); | ||||
|         String world = player.getCurrentPlot().getWorldName(); | ||||
|         if (!this.plotAreaManager.hasPlotArea(world)) { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); | ||||
|             return false; | ||||
| @@ -89,13 +87,11 @@ public class Download extends SubCommand { | ||||
|             player.sendMessage(TranslatableCaption.of("info.plot_unowned")); | ||||
|             return false; | ||||
|         } | ||||
|         if ((Settings.Done.REQUIRED_FOR_DOWNLOAD && !DoneFlag.isDone(plot)) && !Permissions | ||||
|                 .hasPermission(player, Permission.PERMISSION_ADMIN_COMMAND_DOWNLOAD)) { | ||||
|         if ((Settings.Done.REQUIRED_FOR_DOWNLOAD && !DoneFlag.isDone(plot)) && !player.hasPermission(Permission.PERMISSION_ADMIN_COMMAND_DOWNLOAD)) { | ||||
|             player.sendMessage(TranslatableCaption.of("done.done_not_done")); | ||||
|             return false; | ||||
|         } | ||||
|         if (!plot.isOwner(player.getUUID()) && !Permissions | ||||
|                 .hasPermission(player, Permission.PERMISSION_ADMIN.toString())) { | ||||
|         if (!plot.isOwner(player.getUUID()) && !player.hasPermission(Permission.PERMISSION_ADMIN.toString())) { | ||||
|             player.sendMessage(TranslatableCaption.of("permission.no_plot_perms")); | ||||
|             return false; | ||||
|         } | ||||
| @@ -113,7 +109,7 @@ public class Download extends SubCommand { | ||||
|             upload(player, plot); | ||||
|         } else if (args.length == 1 && StringMan | ||||
|                 .isEqualIgnoreCaseToAny(args[0], "mcr", "world", "mca")) { | ||||
|             if (!Permissions.hasPermission(player, Permission.PERMISSION_DOWNLOAD_WORLD)) { | ||||
|             if (!player.hasPermission(Permission.PERMISSION_DOWNLOAD_WORLD)) { | ||||
|                 player.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission"), | ||||
|                         TagResolver.resolver( | ||||
| @@ -139,7 +135,9 @@ public class Download extends SubCommand { | ||||
|                     } | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("web.generation_link_success_legacy_world"), | ||||
|                             TagResolver.resolver("url", Tag.inserting(Component.text(url.toString()))) | ||||
|                             TagResolver.builder() | ||||
|                                     .tag("url", Tag.preProcessParsed(url.toString())) | ||||
|                                     .build() | ||||
|                     ); | ||||
|                 } | ||||
|             }); | ||||
| @@ -158,10 +156,10 @@ public class Download extends SubCommand { | ||||
|     public Collection<Command> tab(final PlotPlayer<?> player, final String[] args, final boolean space) { | ||||
|         if (args.length == 1) { | ||||
|             final List<String> completions = new LinkedList<>(); | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_DOWNLOAD)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_DOWNLOAD)) { | ||||
|                 completions.add("schem"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_DOWNLOAD_WORLD)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_DOWNLOAD_WORLD)) { | ||||
|                 completions.add("world"); | ||||
|             } | ||||
|             final List<Command> commands = completions.stream().filter(completion -> completion | ||||
| @@ -176,7 +174,7 @@ public class Download extends SubCommand { | ||||
|                             CommandCategory.ADMINISTRATION | ||||
|                     ) { | ||||
|                     }).collect(Collectors.toCollection(LinkedList::new)); | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_DOWNLOAD) && args[0].length() > 0) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_DOWNLOAD) && args[0].length() > 0) { | ||||
|                 commands.addAll(TabCompletions.completePlayers(player, args[0], Collections.emptyList())); | ||||
|             } | ||||
|             return commands; | ||||
| @@ -188,21 +186,24 @@ public class Download extends SubCommand { | ||||
|         if (Settings.Web.LEGACY_WEBINTERFACE) { | ||||
|             schematicHandler | ||||
|                     .getCompoundTag(plot) | ||||
|                     .whenComplete((compoundTag, throwable) -> { | ||||
|                         schematicHandler.upload(compoundTag, null, null, new RunnableVal<>() { | ||||
|                             @Override | ||||
|                             public void run(URL value) { | ||||
|                                 player.sendMessage( | ||||
|                                         TranslatableCaption.of("web.generation_link_success"), | ||||
|                                         TagResolver.builder() | ||||
|                                                 .tag("download", Tag.preProcessParsed(value.toString())) | ||||
|                                                 .tag("delete", Tag.preProcessParsed("Not available")) | ||||
|                                                 .build() | ||||
|                                 ); | ||||
|                                 player.sendMessage(StaticCaption.of(value.toString())); | ||||
|                     .whenComplete((compoundTag, throwable) -> schematicHandler.upload( | ||||
|                             compoundTag, | ||||
|                             null, | ||||
|                             null, | ||||
|                             new RunnableVal<>() { | ||||
|                                 @Override | ||||
|                                 public void run(URL value) { | ||||
|                                     plot.removeRunning(); | ||||
|                                     player.sendMessage( | ||||
|                                             TranslatableCaption.of("web.generation_link_success"), | ||||
|                                             TagResolver.builder() | ||||
|                                                     .tag("download", Tag.preProcessParsed(value.toString())) | ||||
|                                                     .tag("delete", Tag.preProcessParsed("Not available")) | ||||
|                                                     .build() | ||||
|                                     ); | ||||
|                                 } | ||||
|                             } | ||||
|                         }); | ||||
|                     }); | ||||
|                     )); | ||||
|             return; | ||||
|         } | ||||
|         // TODO legacy support | ||||
|   | ||||
| @@ -27,7 +27,6 @@ import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.events.PlotFlagAddEvent; | ||||
| import com.plotsquared.core.events.PlotFlagRemoveEvent; | ||||
| import com.plotsquared.core.events.Result; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| @@ -39,7 +38,6 @@ import com.plotsquared.core.plot.flag.types.IntegerFlag; | ||||
| import com.plotsquared.core.plot.flag.types.ListFlag; | ||||
| import com.plotsquared.core.util.EventDispatcher; | ||||
| import com.plotsquared.core.util.MathMan; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.StringComparison; | ||||
| import com.plotsquared.core.util.StringMan; | ||||
| import com.plotsquared.core.util.helpmenu.HelpMenu; | ||||
| @@ -104,9 +102,10 @@ public final class FlagCommand extends Command { | ||||
|         if (flag instanceof IntegerFlag && MathMan.isInteger(value)) { | ||||
|             try { | ||||
|                 int numeric = Integer.parseInt(value); | ||||
|                 // Getting full permission without ".<amount>" at the end | ||||
|                 perm = perm.substring(0, perm.length() - value.length() - 1); | ||||
|                 boolean result = false; | ||||
|                 if (numeric > 0) { | ||||
|                 if (numeric >= 0) { | ||||
|                     int checkRange = PlotSquared.get().getPlatform().equalsIgnoreCase("bukkit") ? | ||||
|                             numeric : | ||||
|                             Settings.Limit.MAX_PLOTS; | ||||
| @@ -132,7 +131,7 @@ public final class FlagCommand extends Command { | ||||
|                             key.toLowerCase(), | ||||
|                             entry.toString().toLowerCase() | ||||
|                     ); | ||||
|                     final boolean result = Permissions.hasPermission(player, permission); | ||||
|                     final boolean result = player.hasPermission(permission); | ||||
|                     if (!result) { | ||||
|                         player.sendMessage( | ||||
|                                 TranslatableCaption.of("permission.no_permission"), | ||||
| @@ -159,9 +158,9 @@ public final class FlagCommand extends Command { | ||||
|         boolean result; | ||||
|         String basePerm = Permission.PERMISSION_SET_FLAG_KEY.format(key.toLowerCase()); | ||||
|         if (flag.isValuedPermission()) { | ||||
|             result = Permissions.hasKeyedPermission(player, basePerm, value); | ||||
|             result = player.hasKeyedPermission(basePerm, value); | ||||
|         } else { | ||||
|             result = Permissions.hasPermission(player, basePerm); | ||||
|             result = player.hasPermission(basePerm); | ||||
|             perm = basePerm; | ||||
|         } | ||||
|         if (!result) { | ||||
| @@ -179,8 +178,7 @@ public final class FlagCommand extends Command { | ||||
|      * @return {@code true} if the player is allowed to modify the flags at their current location | ||||
|      */ | ||||
|     private static boolean checkRequirements(final @NonNull PlotPlayer<?> player) { | ||||
|         final Location location = player.getLocation(); | ||||
|         final Plot plot = location.getPlotAbs(); | ||||
|         final Plot plot = player.getCurrentPlot(); | ||||
|         if (plot == null) { | ||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||
|             return false; | ||||
| @@ -189,8 +187,7 @@ public final class FlagCommand extends Command { | ||||
|             player.sendMessage(TranslatableCaption.of("working.plot_not_claimed")); | ||||
|             return false; | ||||
|         } | ||||
|         if (!plot.isOwner(player.getUUID()) && !Permissions | ||||
|                 .hasPermission(player, Permission.PERMISSION_SET_FLAG_OTHER)) { | ||||
|         if (!plot.isOwner(player.getUUID()) && !player.hasPermission(Permission.PERMISSION_SET_FLAG_OTHER)) { | ||||
|             player.sendMessage( | ||||
|                     TranslatableCaption.of("permission.no_permission"), | ||||
|                     TagResolver.resolver("node", Tag.inserting(Permission.PERMISSION_SET_FLAG_OTHER)) | ||||
| @@ -345,7 +342,7 @@ public final class FlagCommand extends Command { | ||||
|         if (plotFlag == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = player.getLocation().getPlotAbs(); | ||||
|         Plot plot = player.getCurrentPlot(); | ||||
|         PlotFlagAddEvent event = eventDispatcher.callFlagAdd(plotFlag, plot); | ||||
|         if (event.getEventResult() == Result.DENY) { | ||||
|             player.sendMessage( | ||||
| @@ -410,7 +407,7 @@ public final class FlagCommand extends Command { | ||||
|         if (plotFlag == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = player.getLocation().getPlotAbs(); | ||||
|         Plot plot = player.getCurrentPlot(); | ||||
|         PlotFlagAddEvent event = eventDispatcher.callFlagAdd(plotFlag, plot); | ||||
|         if (event.getEventResult() == Result.DENY) { | ||||
|             player.sendMessage( | ||||
| @@ -420,7 +417,7 @@ public final class FlagCommand extends Command { | ||||
|             return; | ||||
|         } | ||||
|         boolean force = event.getEventResult() == Result.FORCE; | ||||
|         final PlotFlag localFlag = player.getLocation().getPlotAbs().getFlagContainer() | ||||
|         final PlotFlag localFlag = player.getCurrentPlot().getFlagContainer() | ||||
|                 .getFlag(event.getFlag().getClass()); | ||||
|         if (!force) { | ||||
|             for (String entry : args[1].split(",")) { | ||||
| @@ -445,7 +442,7 @@ public final class FlagCommand extends Command { | ||||
|             return; | ||||
|         } | ||||
|         boolean result = | ||||
|                 player.getLocation().getPlotAbs().setFlag(localFlag.merge(parsed.getValue())); | ||||
|                 player.getCurrentPlot().setFlag(localFlag.merge(parsed.getValue())); | ||||
|         if (!result) { | ||||
|             player.sendMessage(TranslatableCaption.of("flag.flag_not_added")); | ||||
|             return; | ||||
| @@ -485,7 +482,8 @@ public final class FlagCommand extends Command { | ||||
|         if (flag == null) { | ||||
|             return; | ||||
|         } | ||||
|         final Plot plot = player.getLocation().getPlotAbs(); | ||||
|         final Plot plot = player.getCurrentPlot(); | ||||
|         final PlotFlag<?, ?> flagWithOldValue = plot.getFlagContainer().getFlag(flag.getClass()); | ||||
|         PlotFlagRemoveEvent event = eventDispatcher.callFlagRemove(flag, plot); | ||||
|         if (event.getEventResult() == Result.DENY) { | ||||
|             player.sendMessage( | ||||
| @@ -496,7 +494,7 @@ public final class FlagCommand extends Command { | ||||
|         } | ||||
|         boolean force = event.getEventResult() == Result.FORCE; | ||||
|         flag = event.getFlag(); | ||||
|         if (!force && !Permissions.hasPermission(player, Permission.PERMISSION_SET_FLAG_KEY.format(args[0].toLowerCase()))) { | ||||
|         if (!force && !player.hasPermission(Permission.PERMISSION_SET_FLAG_KEY.format(args[0].toLowerCase()))) { | ||||
|             if (args.length != 2) { | ||||
|                 player.sendMessage( | ||||
|                         TranslatableCaption.of("permission.no_permission"), | ||||
| @@ -682,12 +680,12 @@ public final class FlagCommand extends Command { | ||||
|                     TranslatableCaption.of("flag.flag_info_example"), | ||||
|                     TagResolver.builder() | ||||
|                             .tag("command", Tag.preProcessParsed("/plot flag set")) | ||||
|                             .tag("flag", Tag.inserting(Component.text(plotFlag.getName()))) | ||||
|                             .tag("value", Tag.inserting(Component.text(plotFlag.getExample()))) | ||||
|                             .tag("flag", Tag.preProcessParsed(plotFlag.getName())) | ||||
|                             .tag("value", Tag.preProcessParsed(plotFlag.getExample())) | ||||
|                             .build() | ||||
|             ); | ||||
|             // Default value | ||||
|             final String defaultValue = player.getLocation().getPlotArea().getFlagContainer() | ||||
|             final String defaultValue = player.getCurrentPlot().getArea().getFlagContainer() | ||||
|                     .getFlagErased(plotFlag.getClass()).toString(); | ||||
|             player.sendMessage( | ||||
|                     TranslatableCaption.of("flag.flag_info_default_value"), | ||||
|   | ||||
| @@ -26,13 +26,11 @@ import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.MetaDataAccess; | ||||
| import com.plotsquared.core.player.PlayerMetaDataKeys; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.PlayerManager; | ||||
| import com.plotsquared.core.util.TabCompletions; | ||||
| import com.plotsquared.core.util.task.RunnableVal; | ||||
| import com.plotsquared.core.util.task.RunnableVal2; | ||||
| import com.plotsquared.core.util.task.RunnableVal3; | ||||
| import com.plotsquared.core.uuid.UUIDMapping; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
| @@ -42,6 +40,7 @@ import java.util.Collections; | ||||
| import java.util.LinkedList; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.UUID; | ||||
| import java.util.concurrent.CompletableFuture; | ||||
| import java.util.concurrent.TimeoutException; | ||||
| import java.util.stream.Collectors; | ||||
| @@ -71,7 +70,7 @@ public class Grant extends Command { | ||||
|         final String arg0 = args[0].toLowerCase(); | ||||
|         switch (arg0) { | ||||
|             case "add", "check" -> { | ||||
|                 if (!Permissions.hasPermission(player, Permission.PERMISSION_GRANT.format(arg0))) { | ||||
|                 if (!player.hasPermission(Permission.PERMISSION_GRANT.format(arg0))) { | ||||
|                     player.sendMessage( | ||||
|                             TranslatableCaption.of("permission.no_permission"), | ||||
|                             TagResolver.resolver("node", Tag.inserting(Component.text(Permission.PERMISSION_GRANT.format(arg0)))) | ||||
| @@ -90,8 +89,8 @@ public class Grant extends Command { | ||||
|                                 TagResolver.resolver("value", Tag.inserting(Component.text(String.valueOf(uuids)))) | ||||
|                         ); | ||||
|                     } else { | ||||
|                         final UUIDMapping uuid = uuids.toArray(new UUIDMapping[0])[0]; | ||||
|                         PlotPlayer<?> pp = PlotSquared.platform().playerManager().getPlayerIfExists(uuid.getUuid()); | ||||
|                         final UUID uuid = uuids.iterator().next(); | ||||
|                         PlotPlayer<?> pp = PlotSquared.platform().playerManager().getPlayerIfExists(uuid); | ||||
|                         if (pp != null) { | ||||
|                             try (final MetaDataAccess<Integer> access = pp.accessPersistentMetaData( | ||||
|                                     PlayerMetaDataKeys.PERSISTENT_GRANTED_PLOTS)) { | ||||
| @@ -102,10 +101,14 @@ public class Grant extends Command { | ||||
|                                     ); | ||||
|                                 } else { | ||||
|                                     access.set(access.get().orElse(0) + 1); | ||||
|                                     player.sendMessage( | ||||
|                                             TranslatableCaption.of("grants.added"), | ||||
|                                             TagResolver.resolver("grants", Tag.inserting(Component.text(access.get().orElse(0)))) | ||||
|                                     ); | ||||
|                                 } | ||||
|                             } | ||||
|                         } else { | ||||
|                             DBFunc.getPersistentMeta(uuid.getUuid(), new RunnableVal<>() { | ||||
|                             DBFunc.getPersistentMeta(uuid, new RunnableVal<>() { | ||||
|                                 @Override | ||||
|                                 public void run(Map<String, byte[]> value) { | ||||
|                                     final byte[] array = value.get("grantedPlots"); | ||||
| @@ -130,7 +133,7 @@ public class Grant extends Command { | ||||
|                                         boolean replace = array != null; | ||||
|                                         String key = "grantedPlots"; | ||||
|                                         byte[] rawData = Ints.toByteArray(amount); | ||||
|                                         DBFunc.addPersistentMeta(uuid.getUuid(), key, rawData, replace); | ||||
|                                         DBFunc.addPersistentMeta(uuid, key, rawData, replace); | ||||
|                                         player.sendMessage( | ||||
|                                                 TranslatableCaption.of("grants.added"), | ||||
|                                                 TagResolver.resolver("grants", Tag.inserting(Component.text(amount))) | ||||
| @@ -152,10 +155,10 @@ public class Grant extends Command { | ||||
|     public Collection<Command> tab(final PlotPlayer<?> player, final String[] args, final boolean space) { | ||||
|         if (args.length == 1) { | ||||
|             final List<String> completions = new LinkedList<>(); | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_GRANT_ADD)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_GRANT_ADD)) { | ||||
|                 completions.add("add"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_GRANT_CHECK)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_GRANT_CHECK)) { | ||||
|                 completions.add("check"); | ||||
|             } | ||||
|             final List<Command> commands = completions.stream().filter(completion -> completion | ||||
| @@ -170,12 +173,18 @@ public class Grant extends Command { | ||||
|                             CommandCategory.ADMINISTRATION | ||||
|                     ) { | ||||
|                     }).collect(Collectors.toCollection(LinkedList::new)); | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_GRANT_SINGLE) && args[0].length() > 0) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_GRANT_SINGLE) && args[0].length() > 0) { | ||||
|                 commands.addAll(TabCompletions.completePlayers(player, args[0], Collections.emptyList())); | ||||
|             } | ||||
|             return commands; | ||||
|         } else if (args.length == 2) { | ||||
|             final String subcommand = args[0].toLowerCase(); | ||||
|             if ((subcommand.equals("add") && player.hasPermission(Permission.PERMISSION_GRANT_ADD)) || | ||||
|                 (subcommand.equals("check") && player.hasPermission(Permission.PERMISSION_GRANT_CHECK))) { | ||||
|                 return TabCompletions.completePlayers(player, args[1], Collections.emptyList()); | ||||
|             } | ||||
|         } | ||||
|         return TabCompletions.completePlayers(player, String.join(",", args).trim(), Collections.emptyList()); | ||||
|         return Collections.emptyList(); | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -60,9 +60,10 @@ public class Help extends Command { | ||||
|             RunnableVal2<Command, CommandResult> whenDone | ||||
|     ) { | ||||
|         switch (args.length) { | ||||
|             case 0: | ||||
|             case 0 -> { | ||||
|                 return displayHelp(player, null, 0); | ||||
|             case 1: | ||||
|             } | ||||
|             case 1 -> { | ||||
|                 if (MathMan.isInteger(args[0])) { | ||||
|                     try { | ||||
|                         return displayHelp(player, null, Integer.parseInt(args[0])); | ||||
| @@ -72,7 +73,8 @@ public class Help extends Command { | ||||
|                 } else { | ||||
|                     return displayHelp(player, args[0], 1); | ||||
|                 } | ||||
|             case 2: | ||||
|             } | ||||
|             case 2 -> { | ||||
|                 if (MathMan.isInteger(args[1])) { | ||||
|                     try { | ||||
|                         return displayHelp(player, args[0], Integer.parseInt(args[1])); | ||||
| @@ -81,8 +83,8 @@ public class Help extends Command { | ||||
|                     } | ||||
|                 } | ||||
|                 return CompletableFuture.completedFuture(false); | ||||
|             default: | ||||
|                 sendUsage(player); | ||||
|             } | ||||
|             default -> sendUsage(player); | ||||
|         } | ||||
|         return CompletableFuture.completedFuture(true); | ||||
|     } | ||||
| @@ -111,33 +113,34 @@ public class Help extends Command { | ||||
|             } | ||||
|             if (cat == null && page == 0) { | ||||
|                 TextComponent.Builder builder = Component.text(); | ||||
|                 builder.append(MINI_MESSAGE.deserialize(TranslatableCaption.of("help.help_header").getComponent(player))); | ||||
|                 builder.append(TranslatableCaption.of("help.help_header").toComponent(player)); | ||||
|                 for (CommandCategory c : CommandCategory.values()) { | ||||
|                     if (!c.canAccess(player)) { | ||||
|                         continue; | ||||
|                     } | ||||
|                     builder.append(Component.newline()).append(MINI_MESSAGE | ||||
|                             .deserialize( | ||||
|                                     TranslatableCaption.of("help.help_info_item").getComponent(player), | ||||
|                                     TagResolver.builder() | ||||
|                                             .tag("command",  Tag.inserting(Component.text("/plot help"))) | ||||
|                                             .tag("category",  Tag.inserting(Component.text(c.name().toLowerCase()))) | ||||
|                                             .tag("category_desc",  Tag.inserting(c.toComponent(player))) | ||||
|                                             .build() | ||||
|                             )); | ||||
|                     builder.append(Component.newline()); | ||||
|                     builder.append(TranslatableCaption.of("help.help_info_item").toComponent( | ||||
|                             player, TagResolver.builder() | ||||
|                                     .tag("command", Tag.inserting(Component.text("/plot help"))) | ||||
|                                     .tag("category", Tag.inserting(Component.text(c.name().toLowerCase()))) | ||||
|                                     .tag("category_desc", Tag.inserting(c.toComponent(player))) | ||||
|                                     .build() | ||||
|                     )); | ||||
|                 } | ||||
|                 builder.append(Component.newline()).append(MINI_MESSAGE | ||||
|                         .deserialize( | ||||
|                                 TranslatableCaption.of("help.help_info_item").getComponent(player), | ||||
|                                 TagResolver.builder() | ||||
|                                         .tag("command",  Tag.inserting(Component.text("/plot help"))) | ||||
|                                         .tag("category",  Tag.inserting(Component.text("all"))) | ||||
|                                         .tag("category_desc",  Tag.inserting(Component.text("Display all commands"))) | ||||
|                                         .build() | ||||
|                         )); | ||||
|                 builder.append(Component.newline()).append(MINI_MESSAGE.deserialize(TranslatableCaption | ||||
|                         .of("help.help_footer") | ||||
|                         .getComponent(player))); | ||||
|                 builder.append(Component.newline()); | ||||
|                 builder.append(TranslatableCaption.of("help.help_info_item").toComponent( | ||||
|                         player, TagResolver.builder() | ||||
|                                 .tag("command", Tag.inserting(Component.text("/plot help"))) | ||||
|                                 .tag("category", Tag.inserting(Component.text("all"))) | ||||
|                                 .tag( | ||||
|                                         "category_desc", Tag.inserting(TranslatableCaption | ||||
|                                                 .of("help.help_display_all_commands") | ||||
|                                                 .toComponent(player)) | ||||
|                                 ) | ||||
|                                 .build() | ||||
|                 )); | ||||
|                 builder.append(Component.newline()); | ||||
|                 builder.append(TranslatableCaption.of("help.help_footer").toComponent(player)); | ||||
|                 player.sendMessage(StaticCaption.of(MINI_MESSAGE.serialize(builder.asComponent()))); | ||||
|                 return true; | ||||
|             } | ||||
|   | ||||
| @@ -29,7 +29,6 @@ import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.PlotId; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.MathMan; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.TabCompletions; | ||||
| import com.plotsquared.core.util.query.PlotQuery; | ||||
| import com.plotsquared.core.util.query.SortingStrategy; | ||||
| @@ -108,8 +107,7 @@ public class HomeCommand extends Command { | ||||
|         // /plot home <[area;]x;y> | ||||
|         // /plot home <area> <x;y> | ||||
|         // /plot home <area> <page> | ||||
|         if (!Permissions.hasPermission(player, Permission.PERMISSION_VISIT_OWNED) && !Permissions | ||||
|                 .hasPermission(player, Permission.PERMISSION_HOME)) { | ||||
|         if (!player.hasPermission(Permission.PERMISSION_VISIT_OWNED) && !player.hasPermission(Permission.PERMISSION_HOME)) { | ||||
|             player.sendMessage( | ||||
|                     TranslatableCaption.of("permission.no_permission"), | ||||
|                     TagResolver.resolver("node", Tag.inserting(Component.text(Permission.PERMISSION_VISIT_OWNED.toString()))) | ||||
|   | ||||
| @@ -29,7 +29,6 @@ import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.comment.CommentInbox; | ||||
| import com.plotsquared.core.plot.comment.CommentManager; | ||||
| import com.plotsquared.core.plot.comment.PlotComment; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.StringMan; | ||||
| import com.plotsquared.core.util.TabCompletions; | ||||
| import com.plotsquared.core.util.task.RunnableVal; | ||||
| @@ -86,24 +85,24 @@ public class Inbox extends SubCommand { | ||||
|         for (int x = page * 12; x < max; x++) { | ||||
|             PlotComment comment = comments[x]; | ||||
|             Component commentColored; | ||||
|             if (player.getName().equals(comment.senderName)) { | ||||
|             if (player.getName().equals(comment.senderName())) { | ||||
|                 commentColored = MINI_MESSAGE | ||||
|                         .deserialize( | ||||
|                                 TranslatableCaption.of("list.comment_list_by_lister").getComponent(player), | ||||
|                                 TagResolver.resolver("comment", Tag.inserting(Component.text(comment.comment))) | ||||
|                                 TagResolver.resolver("comment", Tag.inserting(Component.text(comment.comment()))) | ||||
|                         ); | ||||
|             } else { | ||||
|                 commentColored = MINI_MESSAGE | ||||
|                         .deserialize( | ||||
|                                 TranslatableCaption.of("list.comment_list_by_other").getComponent(player), | ||||
|                                 TagResolver.resolver("comment", Tag.inserting(Component.text(comment.comment))) | ||||
|                                 TagResolver.resolver("comment", Tag.inserting(Component.text(comment.comment()))) | ||||
|                         ); | ||||
|             } | ||||
|             TagResolver resolver = TagResolver.builder() | ||||
|                     .tag("number", Tag.inserting(Component.text(x))) | ||||
|                     .tag("world", Tag.inserting(Component.text(comment.world))) | ||||
|                     .tag("plot_id", Tag.inserting(Component.text(comment.id.getX() + ";" + comment.id.getY()))) | ||||
|                     .tag("commenter", Tag.inserting(Component.text(comment.senderName))) | ||||
|                     .tag("world", Tag.inserting(Component.text(comment.world()))) | ||||
|                     .tag("plot_id", Tag.inserting(Component.text(comment.id().getX() + ";" + comment.id().getY()))) | ||||
|                     .tag("commenter", Tag.inserting(Component.text(comment.senderName()))) | ||||
|                     .tag("comment", Tag.inserting(commentColored)) | ||||
|                     .build(); | ||||
|             builder.append(MINI_MESSAGE | ||||
| @@ -138,7 +137,7 @@ public class Inbox extends SubCommand { | ||||
|                                 int unread = 0; | ||||
|                                 for (PlotComment comment : value) { | ||||
|                                     total++; | ||||
|                                     if (comment.timestamp > CommentManager | ||||
|                                     if (comment.timestamp() > CommentManager | ||||
|                                             .getTimestamp(player, inbox.toString())) { | ||||
|                                         unread++; | ||||
|                                     } | ||||
| @@ -191,7 +190,7 @@ public class Inbox extends SubCommand { | ||||
|         final int page; | ||||
|         if (args.length > 1) { | ||||
|             switch (args[1].toLowerCase()) { | ||||
|                 case "delete": | ||||
|                 case "delete" -> { | ||||
|                     if (!inbox.canModify(plot, player)) { | ||||
|                         player.sendMessage(TranslatableCaption.of("comment.no_perm_inbox_modify")); | ||||
|                         return false; | ||||
| @@ -226,7 +225,6 @@ public class Inbox extends SubCommand { | ||||
|                         ); | ||||
|                         return false; | ||||
|                     } | ||||
|  | ||||
|                     if (!inbox.getComments(plot, new RunnableVal<>() { | ||||
|                         @Override | ||||
|                         public void run(List<PlotComment> value) { | ||||
| @@ -243,7 +241,7 @@ public class Inbox extends SubCommand { | ||||
|                             if (success) { | ||||
|                                 player.sendMessage( | ||||
|                                         TranslatableCaption.of("comment.comment_removed_success"), | ||||
|                                         TagResolver.resolver("value", Tag.inserting(Component.text(comment.comment))) | ||||
|                                         TagResolver.resolver("value", Tag.inserting(Component.text(comment.comment()))) | ||||
|                                 ); | ||||
|                             } else { | ||||
|                                 player.sendMessage( | ||||
| @@ -255,7 +253,8 @@ public class Inbox extends SubCommand { | ||||
|                         return false; | ||||
|                     } | ||||
|                     return true; | ||||
|                 case "clear": | ||||
|                 } | ||||
|                 case "clear" -> { | ||||
|                     if (!inbox.canModify(plot, player)) { | ||||
|                         player.sendMessage(TranslatableCaption.of("comment.no_perm_inbox_modify")); | ||||
|                     } | ||||
| @@ -269,13 +268,15 @@ public class Inbox extends SubCommand { | ||||
|                         plot.getPlotCommentContainer().removeComments(comments); | ||||
|                     } | ||||
|                     return true; | ||||
|                 default: | ||||
|                 } | ||||
|                 default -> { | ||||
|                     try { | ||||
|                         page = Integer.parseInt(args[1]); | ||||
|                     } catch (NumberFormatException ignored) { | ||||
|                         sendUsage(player); | ||||
|                         return false; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } else { | ||||
|             page = 1; | ||||
| @@ -300,13 +301,13 @@ public class Inbox extends SubCommand { | ||||
|     public Collection<Command> tab(final PlotPlayer<?> player, final String[] args, final boolean space) { | ||||
|         if (args.length == 1) { | ||||
|             final List<String> completions = new LinkedList<>(); | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_INBOX_READ_OWNER)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_INBOX_READ_OWNER)) { | ||||
|                 completions.add("owner"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_INBOX_READ_PUBLIC)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_INBOX_READ_PUBLIC)) { | ||||
|                 completions.add("public"); | ||||
|             } | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_INBOX_READ_REPORT)) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_INBOX_READ_REPORT)) { | ||||
|                 completions.add("report"); | ||||
|             } | ||||
|             final List<Command> commands = completions.stream().filter(completion -> completion | ||||
| @@ -314,7 +315,7 @@ public class Inbox extends SubCommand { | ||||
|                             .startsWith(args[0].toLowerCase())) | ||||
|                     .map(completion -> new Command(null, true, completion, "", RequiredType.PLAYER, CommandCategory.CHAT) { | ||||
|                     }).collect(Collectors.toCollection(LinkedList::new)); | ||||
|             if (Permissions.hasPermission(player, Permission.PERMISSION_INBOX) && args[0].length() > 0) { | ||||
|             if (player.hasPermission(Permission.PERMISSION_INBOX) && args[0].length() > 0) { | ||||
|                 commands.addAll(TabCompletions.completePlayers(player, args[0], Collections.emptyList())); | ||||
|             } | ||||
|             return commands; | ||||
|   | ||||
| @@ -20,14 +20,12 @@ package com.plotsquared.core.command; | ||||
|  | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.Caption; | ||||
| import com.plotsquared.core.configuration.caption.StaticCaption; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.database.DBFunc; | ||||
| 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.HideInfoFlag; | ||||
| import com.plotsquared.core.util.Permissions; | ||||
| import com.plotsquared.core.util.TabCompletions; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| @@ -53,8 +51,9 @@ public class Info extends SubCommand { | ||||
|             arg = args[0]; | ||||
|             switch (arg) { | ||||
|                 // TODO: (re?)implement /plot info inv. (it was never properly implemented) | ||||
|                 case "trusted", "alias", "biome", "denied", "flags", "id", "size", "members", "creationdate", "seen", "owner", "rating", "likes" -> plot = Plot | ||||
|                         .getPlotFromString(player, null, false); | ||||
|                 case "trusted", "alias", "biome", "denied", "flags", "id", "size", "members", "creationdate", "seen", "owner", "rating", "likes" -> | ||||
|                         plot = Plot | ||||
|                                 .getPlotFromString(player, null, false); | ||||
|                 default -> { | ||||
|                     plot = Plot.getPlotFromString(player, arg, false); | ||||
|                     if (args.length == 2) { | ||||
| @@ -131,13 +130,9 @@ public class Info extends SubCommand { | ||||
|             info = getCaption(arg); | ||||
|             if (info == null) { | ||||
|                 if (Settings.Ratings.USE_LIKES) { | ||||
|                     player.sendMessage(StaticCaption.of( | ||||
|                             "&6Categories&7: &amembers&7, &aalias&7, &abiome&7, &aseen&7, &adenied&7, &aflags&7, &aid&7, &asize&7, &atrusted&7, " | ||||
|                                     + "&aowner&7, " + " &alikes")); | ||||
|                     player.sendMessage(TranslatableCaption.of("info.plot_info_categories.use_likes")); | ||||
|                 } else { | ||||
|                     player.sendMessage(StaticCaption.of( | ||||
|                             "&6Categories&7: &amembers&7, &aalias&7, &abiome&7, &aseen&7, &adenied&7, &aflags&7, &aid&7, &asize&7, &atrusted&7, " | ||||
|                                     + "&aowner&7, " + " &arating")); | ||||
|                     player.sendMessage(TranslatableCaption.of("info.plot_info_categories.use_rating")); | ||||
|                 } | ||||
|                 return false; | ||||
|             } | ||||
| @@ -152,7 +147,7 @@ public class Info extends SubCommand { | ||||
|     @Override | ||||
|     public Collection<Command> tab(PlotPlayer<?> player, String[] args, boolean space) { | ||||
|         final List<String> completions = new LinkedList<>(); | ||||
|         if (Permissions.hasPermission(player, Permission.PERMISSION_AREA_INFO_FORCE)) { | ||||
|         if (player.hasPermission(Permission.PERMISSION_AREA_INFO_FORCE)) { | ||||
|             completions.add("-f"); | ||||
|         } | ||||
|  | ||||
| @@ -162,7 +157,7 @@ public class Info extends SubCommand { | ||||
|                 .map(completion -> new Command(null, true, completion, "", RequiredType.PLAYER, CommandCategory.INFO) { | ||||
|                 }).collect(Collectors.toCollection(LinkedList::new)); | ||||
|  | ||||
|         if (Permissions.hasPermission(player, Permission.PERMISSION_AREA_INFO_FORCE) && args[0].length() > 0) { | ||||
|         if (player.hasPermission(Permission.PERMISSION_AREA_INFO_FORCE) && args[0].length() > 0) { | ||||
|             commands.addAll(TabCompletions.completePlayers(player, args[0], Collections.emptyList())); | ||||
|         } | ||||
|  | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user