mirror of
				https://github.com/IntellectualSites/PlotSquared.git
				synced 2025-10-25 07:33:44 +02:00 
			
		
		
		
	Compare commits
	
		
			135 Commits
		
	
	
		
			feat/chore
			...
			fix/genera
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | d1a6c021fc | ||
|   | a388629d62 | ||
|   | 6991bc79d6 | ||
|   | a191000142 | ||
|   | 2cf57dda47 | ||
|   | e60a016164 | ||
|   | e2f6b0a6fe | ||
|   | 8002d170f5 | ||
|   | 52e8ba6ddd | ||
|   | 2f1fe633af | ||
|   | dc30408ed0 | ||
|   | 313ea8b266 | ||
|   | 1ad0e700a5 | ||
|   | 590f442a0f | ||
|   | 308527e822 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1f32909ffd | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1785693fe4 | ||
|   | df738b279d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b7b25252c6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7123f51bb5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 897afa894e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2238609551 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 255959232b | ||
|   | 622c9f1d13 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e7aff3982e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | db7ea780f9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | db188150d7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cfd8401515 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c887cbe28c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 59183c1412 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 439fb3a8ea | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 78d6ca1deb | ||
|   | 03aa1be5a3 | ||
|   | 2e3832f1bd | ||
|   | 05af41f832 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b613318a29 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c2f8356042 | ||
|   | 109e6059f8 | ||
|   | f01c287f89 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fd8cf3c475 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8c9957edeb | ||
|   | aaba2b5b1a | ||
|   | 02a65c8855 | ||
|   | 8a5fa26796 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 13cbb7e083 | ||
|   | 4d8d5b3a9f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cd6a32cf44 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 273c0ad989 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 774da7183b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e083015ab2 | ||
|   | 3f577d039b | ||
|   | 4d2e4a3d1a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e5d36579b1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f0cde251bd | ||
|   | 58016bb1c8 | ||
|   | e5943ba627 | ||
|   | 07dfdeef2c | ||
|   | 025b08e716 | ||
|   | 921435689e | ||
|   | aae154b23a | ||
|   | 0f33465d76 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 438f1d9656 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cb38aeef93 | ||
|   | 7be0655b86 | ||
|   | 774298bef5 | ||
|   | 6fc25bc034 | ||
|   | 1054018e1e | ||
|   | 0508a7f6b6 | ||
|   | da0a57a48c | ||
|   | 87859b002b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4f4ba07bd2 | ||
|   | 53771a3ece | ||
|   | f020a6c6da | ||
|   | d7e158747e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4d64ea83ec | ||
|   | 7ab334562c | ||
|   | fe1ef36f7e | ||
|   | 70bc02985f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3ec7e992a3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9acaa9c554 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e132c01331 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 24e4e51884 | ||
|   | 84ec090df1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6c6ea1c1b4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 97989face1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f471c02330 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 94322d5982 | ||
|   | 6cbb894249 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6f0fa19601 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0b692459e6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0d410ed869 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 711fba0b2a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 671a27fa6f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 03de685dc4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f616885206 | ||
|   | 7da4eb1ab5 | ||
|   | 629646ab06 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 74a1a1f954 | ||
|   | aa44078018 | ||
|   | bfbf406418 | ||
|   | 2accedf264 | ||
|   | 6ef0d58480 | ||
|   | fbf4a638b4 | ||
|   | 9abfa21078 | ||
|   | b84599b4b3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 058983cdd5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6ba3694121 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8e8e31b80e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | da2e66c1f8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d5d6fcb859 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 96f73331f9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 020947d90c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7dbd0bcff8 | ||
|   | 9626302f04 | ||
|   | 1b4a347e8b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 19e6ed4b9b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6b1b0f2d6a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e499bc02ec | ||
|   | 62084fffdd | ||
|   | d012f79349 | ||
|   | 139d6efc70 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6f5cb917f2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9bf66f1b7c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 92875ebe7f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 25a4545f14 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3fdeed019b | ||
|   | f3400df811 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | beb7cb40f4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2c0bb03e5c | ||
|   | e8c170686c | ||
|   | ff8676cde3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 817effb735 | ||
|   | a0a3d8828a | ||
|   | 8741bfcf88 | ||
|   | 6a6c113e5b | 
							
								
								
									
										3
									
								
								.github/renovate.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.github/renovate.json
									
									
									
									
										vendored
									
									
								
							| @@ -2,7 +2,8 @@ | |||||||
|   "$schema": "https://docs.renovatebot.com/renovate-schema.json", |   "$schema": "https://docs.renovatebot.com/renovate-schema.json", | ||||||
|   "extends": [ |   "extends": [ | ||||||
|     "config:recommended", |     "config:recommended", | ||||||
|     ":semanticCommitsDisabled" |     ":semanticCommitsDisabled", | ||||||
|  |     "schedule:earlyMondays" | ||||||
|   ], |   ], | ||||||
|   "automerge": true, |   "automerge": true, | ||||||
|   "labels": [ |   "labels": [ | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ jobs: | |||||||
|           DISCORD_WEBHOOK: ${{ secrets.DISCORD_WEBHOOK }} |           DISCORD_WEBHOOK: ${{ secrets.DISCORD_WEBHOOK }} | ||||||
|           DISCORD_USERNAME: PlotSquared Release |           DISCORD_USERNAME: PlotSquared Release | ||||||
|           DISCORD_AVATAR: https://raw.githubusercontent.com/IntellectualSites/Assets/main/plugins/PlotSquared/PlotSquared.png |           DISCORD_AVATAR: https://raw.githubusercontent.com/IntellectualSites/Assets/main/plugins/PlotSquared/PlotSquared.png | ||||||
|         uses: Ilshidur/action-discord@0.3.2 |         uses: Ilshidur/action-discord@0.4.0 | ||||||
|         with: |         with: | ||||||
|           args: | |           args: | | ||||||
|             "<@&525015541815967744> <@&679322738552471574> <@&699293353862496266>" |             "<@&525015541815967744> <@&679322738552471574> <@&699293353862496266>" | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								.github/workflows/build-pr.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/build-pr.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,11 +9,11 @@ jobs: | |||||||
|         os: [ ubuntu-latest, windows-latest, macos-latest ] |         os: [ ubuntu-latest, windows-latest, macos-latest ] | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout Repository |       - name: Checkout Repository | ||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v5 | ||||||
|       - name: Validate Gradle Wrapper |       - name: Validate Gradle Wrapper | ||||||
|         uses: gradle/actions/wrapper-validation@v4 |         uses: gradle/actions/wrapper-validation@v5 | ||||||
|       - name: Setup Java |       - name: Setup Java | ||||||
|         uses: actions/setup-java@v4 |         uses: actions/setup-java@v5 | ||||||
|         with: |         with: | ||||||
|           distribution: temurin |           distribution: temurin | ||||||
|           java-version: 21 |           java-version: 21 | ||||||
|   | |||||||
							
								
								
									
										18
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										18
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,11 +9,11 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout Repository |       - name: Checkout Repository | ||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v5 | ||||||
|       - name: Validate Gradle Wrapper |       - name: Validate Gradle Wrapper | ||||||
|         uses: gradle/actions/wrapper-validation@v4 |         uses: gradle/actions/wrapper-validation@v5 | ||||||
|       - name: Setup Java |       - name: Setup Java | ||||||
|         uses: actions/setup-java@v4 |         uses: actions/setup-java@v5 | ||||||
|         with: |         with: | ||||||
|           distribution: temurin |           distribution: temurin | ||||||
|           java-version: 21 |           java-version: 21 | ||||||
| @@ -29,18 +29,18 @@ jobs: | |||||||
|           fi |           fi | ||||||
|       - name: Publish Release |       - name: Publish Release | ||||||
|         if: ${{ runner.os == 'Linux' && env.STATUS == 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main'}} |         if: ${{ runner.os == 'Linux' && env.STATUS == 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main'}} | ||||||
|         run: ./gradlew publishToSonatype closeAndReleaseSonatypeStagingRepository |         run: ./gradlew publishAndReleaseToMavenCentral --no-configuration-cache | ||||||
|         env: |         env: | ||||||
|           ORG_GRADLE_PROJECT_sonatypeUsername: ${{ secrets.SONATYPE_USERNAME }} |           ORG_GRADLE_PROJECT_mavenCentralUsername: ${{ secrets.CENTRAL_USERNAME }} | ||||||
|           ORG_GRADLE_PROJECT_sonatypePassword: ${{ secrets.SONATYPE_PASSWORD }} |           ORG_GRADLE_PROJECT_mavenCentralPassword: ${{ secrets.CENTRAL_PASSWORD }} | ||||||
|           ORG_GRADLE_PROJECT_signingKey: ${{ secrets.SIGNING_KEY }} |           ORG_GRADLE_PROJECT_signingKey: ${{ secrets.SIGNING_KEY }} | ||||||
|           ORG_GRADLE_PROJECT_signingPassword: ${{ secrets.SIGNING_PASSWORD }} |           ORG_GRADLE_PROJECT_signingPassword: ${{ secrets.SIGNING_PASSWORD }} | ||||||
|       - name: Publish Snapshot |       - name: Publish Snapshot | ||||||
|         if: ${{ runner.os == 'Linux' && env.STATUS != 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main' }} |         if: ${{ runner.os == 'Linux' && env.STATUS != 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main' }} | ||||||
|         run: ./gradlew publishToSonatype |         run: ./gradlew publishAllPublicationsToMavenCentralRepository | ||||||
|         env: |         env: | ||||||
|           ORG_GRADLE_PROJECT_sonatypeUsername: ${{ secrets.SONATYPE_USERNAME }} |           ORG_GRADLE_PROJECT_mavenCentralUsername: ${{ secrets.CENTRAL_USERNAME }} | ||||||
|           ORG_GRADLE_PROJECT_sonatypePassword: ${{ secrets.SONATYPE_PASSWORD }} |           ORG_GRADLE_PROJECT_mavenCentralPassword: ${{ secrets.CENTRAL_PASSWORD }} | ||||||
|       - name: Publish core javadoc |       - name: Publish core javadoc | ||||||
|         if: ${{ runner.os == 'Linux' && env.STATUS == 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main'}} |         if: ${{ runner.os == 'Linux' && env.STATUS == 'release' && github.event_name == 'push' && github.ref == 'refs/heads/main'}} | ||||||
|         uses: cpina/github-action-push-to-another-repository@main |         uses: cpina/github-action-push-to-another-repository@main | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
								
							| @@ -20,17 +20,17 @@ jobs: | |||||||
|         language: [ 'java' ] |         language: [ 'java' ] | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout repository |       - name: Checkout repository | ||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v5 | ||||||
|       - name: Setup Java |       - name: Setup Java | ||||||
|         uses: actions/setup-java@v4 |         uses: actions/setup-java@v5 | ||||||
|         with: |         with: | ||||||
|           distribution: temurin |           distribution: temurin | ||||||
|           java-version: 21 |           java-version: 21 | ||||||
|       - name: Initialize CodeQL |       - name: Initialize CodeQL | ||||||
|         uses: github/codeql-action/init@v3 |         uses: github/codeql-action/init@v4 | ||||||
|         with: |         with: | ||||||
|           languages: ${{ matrix.language }} |           languages: ${{ matrix.language }} | ||||||
|       - name: Autobuild |       - name: Autobuild | ||||||
|         uses: github/codeql-action/autobuild@v3 |         uses: github/codeql-action/autobuild@v4 | ||||||
|       - name: Perform CodeQL Analysis |       - name: Perform CodeQL Analysis | ||||||
|         uses: github/codeql-action/analyze@v3 |         uses: github/codeql-action/analyze@v4 | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ import com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar | |||||||
| repositories { | repositories { | ||||||
|     maven { |     maven { | ||||||
|         name = "PlaceholderAPI" |         name = "PlaceholderAPI" | ||||||
|         url = uri("https://repo.extendedclip.com/content/repositories/placeholderapi/") |         url = uri("https://repo.extendedclip.com/releases/") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     maven { |     maven { | ||||||
| @@ -17,6 +17,19 @@ 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 { | dependencies { | ||||||
|     api(projects.plotsquaredCore) |     api(projects.plotsquaredCore) | ||||||
|  |  | ||||||
| @@ -28,20 +41,13 @@ dependencies { | |||||||
|     implementation(libs.paperlib) |     implementation(libs.paperlib) | ||||||
|  |  | ||||||
|     // Plugins |     // Plugins | ||||||
|     compileOnly(libs.worldeditBukkit) { |     compileOnly(libs.worldeditBukkit) | ||||||
|         exclude(group = "org.bukkit") |  | ||||||
|         exclude(group = "org.spigotmc") |  | ||||||
|     } |  | ||||||
|     compileOnly(libs.faweBukkit) { isTransitive = false } |     compileOnly(libs.faweBukkit) { isTransitive = false } | ||||||
|     testImplementation(libs.faweBukkit) { isTransitive = false } |     testImplementation(libs.faweBukkit) { isTransitive = false } | ||||||
|     compileOnly(libs.vault) { |     compileOnly(libs.vault) | ||||||
|         exclude(group = "org.bukkit") |  | ||||||
|     } |  | ||||||
|     compileOnly(libs.placeholderapi) |     compileOnly(libs.placeholderapi) | ||||||
|     compileOnly(libs.luckperms) |     compileOnly(libs.luckperms) | ||||||
|     compileOnly(libs.essentialsx) { |     compileOnly(libs.essentialsx) | ||||||
|         exclude(group = "org.spigotmc") |  | ||||||
|     } |  | ||||||
|     compileOnly(libs.mvdwapi) { isTransitive = false } |     compileOnly(libs.mvdwapi) { isTransitive = false } | ||||||
|  |  | ||||||
|     // Other libraries |     // Other libraries | ||||||
| @@ -74,7 +80,7 @@ tasks.named<ShadowJar>("shadowJar") { | |||||||
|     relocate("net.kyori.examination", "com.plotsquared.core.configuration.examination") |     relocate("net.kyori.examination", "com.plotsquared.core.configuration.examination") | ||||||
|     relocate("io.papermc.lib", "com.plotsquared.bukkit.paperlib") |     relocate("io.papermc.lib", "com.plotsquared.bukkit.paperlib") | ||||||
|     relocate("org.bstats", "com.plotsquared.metrics") |     relocate("org.bstats", "com.plotsquared.metrics") | ||||||
|     relocate("org.enginehub", "com.plotsquared.squirrelid") |     relocate("org.enginehub.squirrelid", "com.plotsquared.squirrelid") | ||||||
|     relocate("org.khelekore.prtree", "com.plotsquared.prtree") |     relocate("org.khelekore.prtree", "com.plotsquared.prtree") | ||||||
|     relocate("com.google.inject", "com.plotsquared.google") |     relocate("com.google.inject", "com.plotsquared.google") | ||||||
|     relocate("org.aopalliance", "com.plotsquared.core.aopalliance") |     relocate("org.aopalliance", "com.plotsquared.core.aopalliance") | ||||||
|   | |||||||
| @@ -46,6 +46,7 @@ import com.plotsquared.bukkit.listener.WorldEvents; | |||||||
| import com.plotsquared.bukkit.placeholder.PAPIPlaceholders; | import com.plotsquared.bukkit.placeholder.PAPIPlaceholders; | ||||||
| import com.plotsquared.bukkit.placeholder.PlaceholderFormatter; | import com.plotsquared.bukkit.placeholder.PlaceholderFormatter; | ||||||
| import com.plotsquared.bukkit.player.BukkitPlayerManager; | import com.plotsquared.bukkit.player.BukkitPlayerManager; | ||||||
|  | import com.plotsquared.bukkit.schematic.StateWrapper; | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
| import com.plotsquared.bukkit.util.BukkitWorld; | import com.plotsquared.bukkit.util.BukkitWorld; | ||||||
| import com.plotsquared.bukkit.util.SetGenCB; | import com.plotsquared.bukkit.util.SetGenCB; | ||||||
| @@ -252,6 +253,11 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|         return Bukkit.getVersion(); |         return Bukkit.getVersion(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NonNull String serverBrand() { | ||||||
|  |         return Bukkit.getName(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     @SuppressWarnings("deprecation") // Paper deprecation |     @SuppressWarnings("deprecation") // Paper deprecation | ||||||
|     public void onEnable() { |     public void onEnable() { | ||||||
| @@ -284,6 +290,18 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         // Validate compatibility of StateWrapper with the current running server version | ||||||
|  |         // Do this always, even if it's not required, to prevent running servers which fail to restore plot backups or | ||||||
|  |         // inserting broken plot / road templates. | ||||||
|  |         try { | ||||||
|  |             var instance = StateWrapper.INSTANCE; | ||||||
|  |         } catch (Exception e) { | ||||||
|  |             LOGGER.error("Failed to initialize required classes for restoring tile entities. " + | ||||||
|  |                     "PlotSquared will disable itself to prevent possible damages.", e); | ||||||
|  |             getServer().getPluginManager().disablePlugin(this); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         // We create the injector after PlotSquared has been initialized, so that we have access |         // We create the injector after PlotSquared has been initialized, so that we have access | ||||||
|         // to generated instances and settings |         // to generated instances and settings | ||||||
|         this.injector = Guice |         this.injector = Guice | ||||||
| @@ -832,7 +850,8 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|                         case "HOPPER_MINECART": |                         case "HOPPER_MINECART": | ||||||
|                         case "MINECART_MOB_SPAWNER": |                         case "MINECART_MOB_SPAWNER": | ||||||
|                         case "SPAWNER_MINECART": |                         case "SPAWNER_MINECART": | ||||||
|                         case "ENDER_CRYSTAL": |                         case "END_CRYSTAL": | ||||||
|  |                         case "ENDER_CRYSTAL": // Backwards compatibility for 1.20.4 | ||||||
|                         case "MINECART_TNT": |                         case "MINECART_TNT": | ||||||
|                         case "TNT_MINECART": |                         case "TNT_MINECART": | ||||||
|                         case "CHEST_BOAT": |                         case "CHEST_BOAT": | ||||||
| @@ -950,6 +969,8 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | |||||||
|                         case "ENDERMITE": |                         case "ENDERMITE": | ||||||
|                         case "ENDER_DRAGON": |                         case "ENDER_DRAGON": | ||||||
|                         case "GHAST": |                         case "GHAST": | ||||||
|  |                         case "HAPPY_GHAST": // 1.21.6+ | ||||||
|  |                         case "GHASTLING": // 1.21.6+ | ||||||
|                         case "GIANT": |                         case "GIANT": | ||||||
|                         case "GUARDIAN": |                         case "GUARDIAN": | ||||||
|                         case "HORSE": |                         case "HORSE": | ||||||
|   | |||||||
| @@ -115,7 +115,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                 this.dataByte = getOrdinal(Boat.Type.values(), boat.getBoatType()); |                 this.dataByte = getOrdinal(Boat.Type.values(), boat.getBoatType()); | ||||||
|                 storeInventory(boat); |                 storeInventory(boat); | ||||||
|             } |             } | ||||||
|             case "ARROW", "EGG", "ENDER_CRYSTAL", "ENDER_PEARL", "ENDER_SIGNAL", "EXPERIENCE_ORB", "FALLING_BLOCK", "FIREBALL", |             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", |                     "FIREWORK", "FISHING_HOOK", "LEASH_HITCH", "LIGHTNING", "MINECART", "MINECART_COMMAND", "MINECART_MOB_SPAWNER", | ||||||
|                     "MINECART_TNT", "PLAYER", "PRIMED_TNT", "SLIME", "SMALL_FIREBALL", "SNOWBALL", "MINECART_FURNACE", "SPLASH_POTION", |                     "MINECART_TNT", "PLAYER", "PRIMED_TNT", "SLIME", "SMALL_FIREBALL", "SNOWBALL", "MINECART_FURNACE", "SPLASH_POTION", | ||||||
|                     "THROWN_EXP_BOTTLE", "WITHER_SKULL", "UNKNOWN", "SPECTRAL_ARROW", "SHULKER_BULLET", "DRAGON_FIREBALL", "AREA_EFFECT_CLOUD", |                     "THROWN_EXP_BOTTLE", "WITHER_SKULL", "UNKNOWN", "SPECTRAL_ARROW", "SHULKER_BULLET", "DRAGON_FIREBALL", "AREA_EFFECT_CLOUD", | ||||||
| @@ -272,7 +272,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                 this.dataByte = (byte) entity1.getPhase().ordinal(); |                 this.dataByte = (byte) entity1.getPhase().ordinal(); | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             case "SKELETON", "WITHER_SKELETON", "GUARDIAN", "ELDER_GUARDIAN", "GHAST", "MAGMA_CUBE", "SQUID", "PIG_ZOMBIE", "HOGLIN", |             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", |                     "ZOMBIFIED_PIGLIN", "PIGLIN", "PIGLIN_BRUTE", "ZOMBIE", "WITHER", "WITCH", "SPIDER", "CAVE_SPIDER", "SILVERFISH", | ||||||
|                     "GIANT", "ENDERMAN", "CREEPER", "BLAZE", "SHULKER", "SNOWMAN", "SNOW_GOLEM" -> { |                     "GIANT", "ENDERMAN", "CREEPER", "BLAZE", "SHULKER", "SNOWMAN", "SNOW_GOLEM" -> { | ||||||
|                 storeLiving((LivingEntity) entity); |                 storeLiving((LivingEntity) entity); | ||||||
| @@ -511,7 +511,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                 ((Slime) entity).setSize(this.dataByte); |                 ((Slime) entity).setSize(this.dataByte); | ||||||
|                 return entity; |                 return entity; | ||||||
|             } */ |             } */ | ||||||
|             case "ARROW", "EGG", "ENDER_CRYSTAL", "ENDER_PEARL", "ENDER_SIGNAL", "DROPPED_ITEM", "EXPERIENCE_ORB", "FALLING_BLOCK", |             case "ARROW", "EGG", "END_CRYSTAL", "ENDER_CRYSTAL", "ENDER_PEARL", "ENDER_SIGNAL", "DROPPED_ITEM", "EXPERIENCE_ORB", "FALLING_BLOCK", | ||||||
|                     "FIREBALL", "FIREWORK", "FISHING_HOOK", "LEASH_HITCH", "LIGHTNING", "MINECART", "MINECART_COMMAND", |                     "FIREBALL", "FIREWORK", "FISHING_HOOK", "LEASH_HITCH", "LIGHTNING", "MINECART", "MINECART_COMMAND", | ||||||
|                     "MINECART_MOB_SPAWNER", "MINECART_TNT", "PLAYER", "PRIMED_TNT", "SMALL_FIREBALL", "SNOWBALL", |                     "MINECART_MOB_SPAWNER", "MINECART_TNT", "PLAYER", "PRIMED_TNT", "SMALL_FIREBALL", "SNOWBALL", | ||||||
|                     "SPLASH_POTION", "THROWN_EXP_BOTTLE", "SPECTRAL_ARROW", "SHULKER_BULLET", "AREA_EFFECT_CLOUD", |                     "SPLASH_POTION", "THROWN_EXP_BOTTLE", "SPECTRAL_ARROW", "SHULKER_BULLET", "AREA_EFFECT_CLOUD", | ||||||
| @@ -676,7 +676,7 @@ public final class ReplicatingEntityWrapper extends EntityWrapper { | |||||||
|                 restoreLiving((LivingEntity) entity); |                 restoreLiving((LivingEntity) entity); | ||||||
|                 return entity; |                 return entity; | ||||||
|             } |             } | ||||||
|             case "ENDERMITE", "GHAST", "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" -> { |             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); |                 restoreLiving((LivingEntity) entity); | ||||||
|                 return entity; |                 return entity; | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -90,12 +90,7 @@ public class BukkitPlotGenerator extends ChunkGenerator implements GeneratorWrap | |||||||
|         this.plotGenerator = generator; |         this.plotGenerator = generator; | ||||||
|         this.platformGenerator = this; |         this.platformGenerator = this; | ||||||
|         this.populators = new ArrayList<>(); |         this.populators = new ArrayList<>(); | ||||||
|         int minecraftMinorVersion = PlotSquared.platform().serverVersion()[1]; |  | ||||||
|         if (minecraftMinorVersion >= 17) { |  | ||||||
|         this.populators.add(new BlockStatePopulator(this.plotGenerator)); |         this.populators.add(new BlockStatePopulator(this.plotGenerator)); | ||||||
|         } else { |  | ||||||
|             this.populators.add(new LegacyBlockStatePopulator(this.plotGenerator)); |  | ||||||
|         } |  | ||||||
|         this.full = true; |         this.full = true; | ||||||
|         this.useNewGenerationMethods = PlotSquared.platform().serverVersion()[1] >= 19; |         this.useNewGenerationMethods = PlotSquared.platform().serverVersion()[1] >= 19; | ||||||
|         this.biomeProvider = new BukkitPlotBiomeProvider(); |         this.biomeProvider = new BukkitPlotBiomeProvider(); | ||||||
|   | |||||||
| @@ -36,6 +36,7 @@ import org.checkerframework.checker.nullness.qual.NonNull; | |||||||
|  |  | ||||||
| import java.util.Random; | import java.util.Random; | ||||||
|  |  | ||||||
|  | @Deprecated(since = "TODO") | ||||||
| final class LegacyBlockStatePopulator extends BlockPopulator { | final class LegacyBlockStatePopulator extends BlockPopulator { | ||||||
|  |  | ||||||
|     private final IndependentPlotGenerator plotGenerator; |     private final IndependentPlotGenerator plotGenerator; | ||||||
|   | |||||||
| @@ -28,7 +28,6 @@ import com.plotsquared.core.plot.flag.implementations.CopperOxideFlag; | |||||||
| import com.plotsquared.core.plot.flag.implementations.MiscInteractFlag; | import com.plotsquared.core.plot.flag.implementations.MiscInteractFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.SculkSensorInteractFlag; | import com.plotsquared.core.plot.flag.implementations.SculkSensorInteractFlag; | ||||||
| import com.plotsquared.core.util.PlotFlagUtil; | import com.plotsquared.core.util.PlotFlagUtil; | ||||||
| import org.bukkit.Material; |  | ||||||
| import org.bukkit.block.Block; | import org.bukkit.block.Block; | ||||||
| import org.bukkit.entity.Entity; | import org.bukkit.entity.Entity; | ||||||
| import org.bukkit.entity.Item; | import org.bukkit.entity.Item; | ||||||
| @@ -42,31 +41,11 @@ import org.bukkit.event.block.BlockReceiveGameEvent; | |||||||
|  |  | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Objects; | import java.util.Objects; | ||||||
| import java.util.Set; |  | ||||||
| import java.util.UUID; | import java.util.UUID; | ||||||
|  |  | ||||||
| @SuppressWarnings("unused") | @SuppressWarnings("unused") | ||||||
| public class BlockEventListener117 implements Listener { | public class BlockEventListener117 implements Listener { | ||||||
|  |  | ||||||
|     private static final Set<Material> COPPER_OXIDIZING = Set.of( |  | ||||||
|             Material.COPPER_BLOCK, |  | ||||||
|             Material.EXPOSED_COPPER, |  | ||||||
|             Material.WEATHERED_COPPER, |  | ||||||
|             Material.OXIDIZED_COPPER, |  | ||||||
|             Material.CUT_COPPER, |  | ||||||
|             Material.EXPOSED_CUT_COPPER, |  | ||||||
|             Material.WEATHERED_CUT_COPPER, |  | ||||||
|             Material.OXIDIZED_CUT_COPPER, |  | ||||||
|             Material.CUT_COPPER_STAIRS, |  | ||||||
|             Material.EXPOSED_CUT_COPPER_STAIRS, |  | ||||||
|             Material.WEATHERED_CUT_COPPER_STAIRS, |  | ||||||
|             Material.OXIDIZED_CUT_COPPER_STAIRS, |  | ||||||
|             Material.CUT_COPPER_SLAB, |  | ||||||
|             Material.EXPOSED_CUT_COPPER_SLAB, |  | ||||||
|             Material.WEATHERED_CUT_COPPER_SLAB, |  | ||||||
|             Material.OXIDIZED_CUT_COPPER_SLAB |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     @Inject |     @Inject | ||||||
|     public BlockEventListener117() { |     public BlockEventListener117() { | ||||||
|     } |     } | ||||||
| @@ -134,7 +113,7 @@ public class BlockEventListener117 implements Listener { | |||||||
|     public void onBlockFertilize(BlockFertilizeEvent event) { |     public void onBlockFertilize(BlockFertilizeEvent event) { | ||||||
|         Block block = event.getBlock(); |         Block block = event.getBlock(); | ||||||
|         List<org.bukkit.block.BlockState> blocks = event.getBlocks(); |         List<org.bukkit.block.BlockState> blocks = event.getBlocks(); | ||||||
|         Location location = BukkitUtil.adapt(blocks.get(0).getLocation()); |         Location location = BukkitUtil.adapt(block.getLocation()); | ||||||
|  |  | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
| @@ -184,7 +163,7 @@ public class BlockEventListener117 implements Listener { | |||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         if (COPPER_OXIDIZING.contains(event.getNewState().getType())) { |         if (event.getNewState().getType().name().contains("COPPER")) { | ||||||
|             if (!plot.getFlag(CopperOxideFlag.class)) { |             if (!plot.getFlag(CopperOxideFlag.class)) { | ||||||
|                 plot.debug("Copper could not oxide because copper-oxide = false"); |                 plot.debug("Copper could not oxide because copper-oxide = false"); | ||||||
|                 event.setCancelled(true); |                 event.setCancelled(true); | ||||||
|   | |||||||
| @@ -160,7 +160,7 @@ public class EntityEventListener implements Listener { | |||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             case "REINFORCEMENTS", "NATURAL", "MOUNT", "PATROL", "RAID", "SHEARED", "SILVERFISH_BLOCK", "ENDER_PEARL", |             case "REINFORCEMENTS", "NATURAL", "MOUNT", "PATROL", "RAID", "SILVERFISH_BLOCK", "ENDER_PEARL", | ||||||
|                  "TRAP", "VILLAGE_DEFENSE", "VILLAGE_INVASION", "BEEHIVE", "CHUNK_GEN", "NETHER_PORTAL", |                  "TRAP", "VILLAGE_DEFENSE", "VILLAGE_INVASION", "BEEHIVE", "CHUNK_GEN", "NETHER_PORTAL", | ||||||
|                  "FROZEN", "SPELL", "DEFAULT" -> { |                  "FROZEN", "SPELL", "DEFAULT" -> { | ||||||
|                 if (!area.isMobSpawning()) { |                 if (!area.isMobSpawning()) { | ||||||
|   | |||||||
| @@ -29,6 +29,7 @@ import com.destroystokyo.paper.event.player.PlayerLaunchProjectileEvent; | |||||||
| import com.destroystokyo.paper.event.server.AsyncTabCompleteEvent; | import com.destroystokyo.paper.event.server.AsyncTabCompleteEvent; | ||||||
| import com.google.inject.Inject; | import com.google.inject.Inject; | ||||||
| import com.plotsquared.bukkit.util.BukkitUtil; | import com.plotsquared.bukkit.util.BukkitUtil; | ||||||
|  | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.command.Command; | import com.plotsquared.core.command.Command; | ||||||
| import com.plotsquared.core.command.MainCommand; | import com.plotsquared.core.command.MainCommand; | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
| @@ -38,6 +39,7 @@ import com.plotsquared.core.permissions.Permission; | |||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| import com.plotsquared.core.plot.PlotArea; | import com.plotsquared.core.plot.PlotArea; | ||||||
|  | import com.plotsquared.core.plot.PlotAreaType; | ||||||
| import com.plotsquared.core.plot.flag.FlagContainer; | import com.plotsquared.core.plot.flag.FlagContainer; | ||||||
| import com.plotsquared.core.plot.flag.implementations.BeaconEffectsFlag; | import com.plotsquared.core.plot.flag.implementations.BeaconEffectsFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.DoneFlag; | import com.plotsquared.core.plot.flag.implementations.DoneFlag; | ||||||
| @@ -48,6 +50,7 @@ import com.plotsquared.core.plot.flag.types.BooleanFlag; | |||||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | import com.plotsquared.core.plot.world.PlotAreaManager; | ||||||
| import com.plotsquared.core.util.PlotFlagUtil; | import com.plotsquared.core.util.PlotFlagUtil; | ||||||
| import io.papermc.paper.event.entity.EntityMoveEvent; | import io.papermc.paper.event.entity.EntityMoveEvent; | ||||||
|  | import io.papermc.paper.event.world.StructuresLocateEvent; | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | import net.kyori.adventure.text.minimessage.tag.Tag; | ||||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| @@ -199,7 +202,7 @@ public class PaperListener implements Listener { | |||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             case "REINFORCEMENTS", "NATURAL", "MOUNT", "PATROL", "RAID", "SHEARED", "SILVERFISH_BLOCK", "ENDER_PEARL", "TRAP", "VILLAGE_DEFENSE", "VILLAGE_INVASION", "BEEHIVE", "CHUNK_GEN" -> { |             case "REINFORCEMENTS", "NATURAL", "MOUNT", "PATROL", "RAID", "SILVERFISH_BLOCK", "ENDER_PEARL", "TRAP", "VILLAGE_DEFENSE", "VILLAGE_INVASION", "BEEHIVE", "CHUNK_GEN" -> { | ||||||
|                 if (!area.isMobSpawning()) { |                 if (!area.isMobSpawning()) { | ||||||
|                     event.setShouldAbortSpawn(true); |                     event.setShouldAbortSpawn(true); | ||||||
|                     event.setCancelled(true); |                     event.setCancelled(true); | ||||||
| @@ -458,6 +461,21 @@ public class PaperListener implements Listener { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Don't let the server die when populating cartographers (villager offering maps) in classic plot worlds | ||||||
|  |      * (as those don't generate POIs) | ||||||
|  |      */ | ||||||
|  |     @EventHandler(ignoreCancelled = true, priority = EventPriority.HIGHEST) | ||||||
|  |     public void onStructuresLocate(StructuresLocateEvent event) { | ||||||
|  |         if (!PlotSquared.get().getPlotAreaManager().hasPlotArea(event.getWorld().getName())) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         final PlotArea area = PlotSquared.get().getPlotAreaManager().getPlotAreaByString(event.getWorld().getName()); | ||||||
|  |         if (area != null && area.getType() == PlotAreaType.NORMAL) { | ||||||
|  |             event.setCancelled(true); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     private boolean getBooleanFlagValue( |     private boolean getBooleanFlagValue( | ||||||
|             @NonNull FlagContainer container, |             @NonNull FlagContainer container, | ||||||
|             @NonNull Class<? extends BooleanFlag<?>> flagClass, |             @NonNull Class<? extends BooleanFlag<?>> flagClass, | ||||||
|   | |||||||
| @@ -64,9 +64,11 @@ import com.plotsquared.core.plot.flag.implementations.PreventCreativeCopyFlag; | |||||||
| import com.plotsquared.core.plot.flag.implementations.TamedInteractFlag; | import com.plotsquared.core.plot.flag.implementations.TamedInteractFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.TileDropFlag; | import com.plotsquared.core.plot.flag.implementations.TileDropFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.UntrustedVisitFlag; | import com.plotsquared.core.plot.flag.implementations.UntrustedVisitFlag; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.UseFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.VehicleBreakFlag; | import com.plotsquared.core.plot.flag.implementations.VehicleBreakFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.VehicleUseFlag; | import com.plotsquared.core.plot.flag.implementations.VehicleUseFlag; | ||||||
| import com.plotsquared.core.plot.flag.implementations.VillagerInteractFlag; | import com.plotsquared.core.plot.flag.implementations.VillagerInteractFlag; | ||||||
|  | import com.plotsquared.core.plot.flag.types.BlockTypeWrapper; | ||||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | import com.plotsquared.core.plot.world.PlotAreaManager; | ||||||
| import com.plotsquared.core.util.EventDispatcher; | import com.plotsquared.core.util.EventDispatcher; | ||||||
| import com.plotsquared.core.util.MathMan; | import com.plotsquared.core.util.MathMan; | ||||||
| @@ -77,7 +79,9 @@ import com.plotsquared.core.util.task.TaskManager; | |||||||
| import com.plotsquared.core.util.task.TaskTime; | import com.plotsquared.core.util.task.TaskTime; | ||||||
| import com.sk89q.worldedit.WorldEdit; | import com.sk89q.worldedit.WorldEdit; | ||||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||||
|  | import com.sk89q.worldedit.util.Enums; | ||||||
| import com.sk89q.worldedit.world.block.BlockType; | import com.sk89q.worldedit.world.block.BlockType; | ||||||
|  | import com.sk89q.worldedit.world.block.BlockTypes; | ||||||
| import io.papermc.lib.PaperLib; | import io.papermc.lib.PaperLib; | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| import net.kyori.adventure.text.minimessage.MiniMessage; | import net.kyori.adventure.text.minimessage.MiniMessage; | ||||||
| @@ -88,10 +92,8 @@ import org.bukkit.Bukkit; | |||||||
| import org.bukkit.FluidCollisionMode; | import org.bukkit.FluidCollisionMode; | ||||||
| import org.bukkit.Material; | import org.bukkit.Material; | ||||||
| import org.bukkit.block.Block; | import org.bukkit.block.Block; | ||||||
| import org.bukkit.block.BlockFace; |  | ||||||
| import org.bukkit.block.BlockState; | import org.bukkit.block.BlockState; | ||||||
| import org.bukkit.block.Sign; | import org.bukkit.block.Sign; | ||||||
| import org.bukkit.block.data.Waterlogged; |  | ||||||
| import org.bukkit.command.PluginCommand; | import org.bukkit.command.PluginCommand; | ||||||
| import org.bukkit.entity.ArmorStand; | import org.bukkit.entity.ArmorStand; | ||||||
| import org.bukkit.entity.Boat; | import org.bukkit.entity.Boat; | ||||||
| @@ -154,9 +156,12 @@ import org.bukkit.util.Vector; | |||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.lang.reflect.Field; | import java.lang.reflect.Field; | ||||||
|  | import java.util.Collections; | ||||||
| import java.util.HashSet; | import java.util.HashSet; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Locale; | import java.util.Locale; | ||||||
|  | import java.util.Objects; | ||||||
|  | import java.util.Optional; | ||||||
| import java.util.Set; | import java.util.Set; | ||||||
| import java.util.UUID; | import java.util.UUID; | ||||||
|  |  | ||||||
| @@ -180,7 +185,17 @@ public class PlayerEventListener implements Listener { | |||||||
|             Material.WRITABLE_BOOK, |             Material.WRITABLE_BOOK, | ||||||
|             Material.WRITTEN_BOOK |             Material.WRITTEN_BOOK | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * The correct EntityType for End Crystal, determined once at class loading time. | ||||||
|  |      * Tries END_CRYSTAL first (1.21+), falls back to ENDER_CRYSTAL (1.20.4 and older). | ||||||
|  |      */ | ||||||
|  |     private static final EntityType END_CRYSTAL_ENTITY_TYPE = Objects.requireNonNull( | ||||||
|  |             Enums.findByValue(EntityType.class, "END_CRYSTAL", "ENDER_CRYSTAL") | ||||||
|  |     ); | ||||||
|  |  | ||||||
|     private static final Set<String> DYES; |     private static final Set<String> DYES; | ||||||
|  |  | ||||||
|     static { |     static { | ||||||
|         Set<String> mutableDyes = new HashSet<>(Set.of( |         Set<String> mutableDyes = new HashSet<>(Set.of( | ||||||
|                 "WHITE_DYE", |                 "WHITE_DYE", | ||||||
| @@ -533,12 +548,14 @@ public class PlayerEventListener implements Listener { | |||||||
|         // Delayed |         // Delayed | ||||||
|  |  | ||||||
|         // Async |         // Async | ||||||
|         TaskManager.runTaskLaterAsync(() -> { |         TaskManager.runTaskLaterAsync( | ||||||
|  |                 () -> { | ||||||
|                     if (!player.hasPlayedBefore() && player.isOnline()) { |                     if (!player.hasPlayedBefore() && player.isOnline()) { | ||||||
|                         player.saveData(); |                         player.saveData(); | ||||||
|                     } |                     } | ||||||
|                     this.eventDispatcher.doJoinTask(pp); |                     this.eventDispatcher.doJoinTask(pp); | ||||||
|         }, TaskTime.seconds(1L)); |                 }, TaskTime.seconds(1L) | ||||||
|  |         ); | ||||||
|  |  | ||||||
|         if (pp.hasPermission(Permission.PERMISSION_ADMIN_UPDATE_NOTIFICATION.toString()) && Settings.Enabled_Components.UPDATE_NOTIFICATIONS |         if (pp.hasPermission(Permission.PERMISSION_ADMIN_UPDATE_NOTIFICATION.toString()) && Settings.Enabled_Components.UPDATE_NOTIFICATIONS | ||||||
|                 && PremiumVerification.isPremium() && UpdateUtility.hasUpdate) { |                 && PremiumVerification.isPremium() && UpdateUtility.hasUpdate) { | ||||||
| @@ -593,14 +610,20 @@ public class PlayerEventListener implements Listener { | |||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|                 Plot plot = area.getPlot(location); |                 Plot plot = area.getPlot(location); | ||||||
|                 if (plot != null) { |                 if (plot != null && !plot.equals(lastPlot)) { | ||||||
|                     final boolean result = DenyTeleportFlag.allowsTeleport(pp, plot); |                     final boolean result = DenyTeleportFlag.allowsTeleport(pp, plot); | ||||||
|                     // there is one possibility to still allow teleportation: |                     // there is one possibility to still allow teleportation: | ||||||
|                     // to is identical to the plot's home location, and untrusted-visit is true |                     // to is identical to the plot's home location, and untrusted-visit is true | ||||||
|                     // i.e. untrusted-visit can override deny-teleport |                     // i.e. untrusted-visit can override deny-teleport | ||||||
|                     // this is acceptable, because otherwise it wouldn't make sense to have both flags set |                     // this is acceptable, because otherwise it wouldn't make sense to have both flags set | ||||||
|                     if (result || (plot.getFlag(UntrustedVisitFlag.class) && plot.getHomeSynchronous().equals(BukkitUtil.adaptComplete(to)))) { |                     if (result || (plot.getFlag(UntrustedVisitFlag.class) && plot | ||||||
|                         plotListener.plotEntry(pp, plot); |                             .getHomeSynchronous() | ||||||
|  |                             .equals(BukkitUtil.adaptComplete(to)))) { | ||||||
|  |                         // returns false if the player is not allowed to enter the plot (if they are denied, for example) | ||||||
|  |                         // don't let the move event cancel the entry after teleport, but rather catch and cancel early (#4647) | ||||||
|  |                         if (!plotListener.plotEntry(pp, plot)) { | ||||||
|  |                             event.setCancelled(true); | ||||||
|  |                         } | ||||||
|                     } else { |                     } else { | ||||||
|                         pp.sendMessage( |                         pp.sendMessage( | ||||||
|                                 TranslatableCaption.of("deny.no_enter"), |                                 TranslatableCaption.of("deny.no_enter"), | ||||||
| @@ -939,12 +962,15 @@ public class PlayerEventListener implements Listener { | |||||||
|         builder.tag("plot_id", Tag.inserting(Component.text(id.toString()))); |         builder.tag("plot_id", Tag.inserting(Component.text(id.toString()))); | ||||||
|         builder.tag("sender", Tag.inserting(Component.text(sender))); |         builder.tag("sender", Tag.inserting(Component.text(sender))); | ||||||
|         if (plotPlayer.hasPermission("plots.chat.color")) { |         if (plotPlayer.hasPermission("plots.chat.color")) { | ||||||
|             builder.tag("msg", Tag.inserting(MiniMessage.miniMessage().deserialize( |             builder.tag( | ||||||
|  |                     "msg", Tag.inserting(MiniMessage.miniMessage().deserialize( | ||||||
|                             message, |                             message, | ||||||
|                     TagResolver.resolver(StandardTags.color(), StandardTags.gradient(), |                             TagResolver.resolver( | ||||||
|  |                                     StandardTags.color(), StandardTags.gradient(), | ||||||
|                                     StandardTags.rainbow(), StandardTags.decorations() |                                     StandardTags.rainbow(), StandardTags.decorations() | ||||||
|                             ) |                             ) | ||||||
|             ))); |                     )) | ||||||
|  |             ); | ||||||
|         } else { |         } else { | ||||||
|             builder.tag("msg", Tag.inserting(Component.text(message))); |             builder.tag("msg", Tag.inserting(Component.text(message))); | ||||||
|         } |         } | ||||||
| @@ -1253,7 +1279,9 @@ public class PlayerEventListener implements Listener { | |||||||
|                 eventType = PlayerBlockEventType.INTERACT_BLOCK; |                 eventType = PlayerBlockEventType.INTERACT_BLOCK; | ||||||
|                 blocktype1 = BukkitAdapter.asBlockType(block.getType()); |                 blocktype1 = BukkitAdapter.asBlockType(block.getType()); | ||||||
|  |  | ||||||
|                 if (INTERACTABLE_MATERIALS != null ? INTERACTABLE_MATERIALS.contains(blockType.name()) : blockType.isInteractable()) { |                 if (INTERACTABLE_MATERIALS != null | ||||||
|  |                         ? INTERACTABLE_MATERIALS.contains(blockType.name()) | ||||||
|  |                         : blockType.isInteractable()) { | ||||||
|                     if (!player.isSneaking()) { |                     if (!player.isSneaking()) { | ||||||
|                         break; |                         break; | ||||||
|                     } |                     } | ||||||
| @@ -1296,6 +1324,17 @@ public class PlayerEventListener implements Listener { | |||||||
|                     //Allow all players to eat while also allowing the block place event to be fired |                     //Allow all players to eat while also allowing the block place event to be fired | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|  |                 // Process creature spawning of armor stands & end crystals here if spawned by the player in order to be able to | ||||||
|  |                 // reset the player's hand item if spawning needs to be cancelled. | ||||||
|  |                 if (type == Material.ARMOR_STAND || type == Material.END_CRYSTAL) { | ||||||
|  |                     Plot plot = location.getOwnedPlotAbs(); | ||||||
|  |                     EntityType entityType = type == Material.ARMOR_STAND ? EntityType.ARMOR_STAND : END_CRYSTAL_ENTITY_TYPE; | ||||||
|  |                     if (BukkitEntityUtil.checkEntity(entityType, plot)) { | ||||||
|  |                         event.setCancelled(true); | ||||||
|  |                         break; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 // Continue with normal place event checks | ||||||
|                 if (type == Material.ARMOR_STAND) { |                 if (type == Material.ARMOR_STAND) { | ||||||
|                     location = BukkitUtil.adapt(block.getRelative(event.getBlockFace()).getLocation()); |                     location = BukkitUtil.adapt(block.getRelative(event.getBlockFace()).getLocation()); | ||||||
|                     eventType = PlayerBlockEventType.PLACE_MISC; |                     eventType = PlayerBlockEventType.PLACE_MISC; | ||||||
| @@ -1370,22 +1409,7 @@ public class PlayerEventListener implements Listener { | |||||||
|  |  | ||||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) |     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||||
|     public void onBucketEmpty(PlayerBucketEmptyEvent event) { |     public void onBucketEmpty(PlayerBucketEmptyEvent event) { | ||||||
|         BlockFace bf = event.getBlockFace(); |         final Block block = event.getBlock(); | ||||||
|         // Note: a month after Bukkit 1.14.4 released, they added the API method |  | ||||||
|         // PlayerBucketEmptyEvent#getBlock(), which returns the block the |  | ||||||
|         // bucket contents is going to be placed at. Currently we determine this |  | ||||||
|         // block ourselves to retain compatibility with 1.13. |  | ||||||
|         final Block block; |  | ||||||
|         // if the block can be waterlogged, the event might waterlog the block |  | ||||||
|         // sometimes |  | ||||||
|         if (event.getBlockClicked().getBlockData() instanceof Waterlogged waterlogged |  | ||||||
|                 && !waterlogged.isWaterlogged() && event.getBucket() != Material.LAVA_BUCKET) { |  | ||||||
|             block = event.getBlockClicked(); |  | ||||||
|         } else { |  | ||||||
|             block = event.getBlockClicked().getLocation() |  | ||||||
|                     .add(bf.getModX(), bf.getModY(), bf.getModZ()) |  | ||||||
|                     .getBlock(); |  | ||||||
|         } |  | ||||||
|         Location location = BukkitUtil.adapt(block.getLocation()); |         Location location = BukkitUtil.adapt(block.getLocation()); | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
| @@ -1393,6 +1417,16 @@ public class PlayerEventListener implements Listener { | |||||||
|         } |         } | ||||||
|         BukkitPlayer pp = BukkitUtil.adapt(event.getPlayer()); |         BukkitPlayer pp = BukkitUtil.adapt(event.getPlayer()); | ||||||
|         Plot plot = area.getPlot(location); |         Plot plot = area.getPlot(location); | ||||||
|  |         final List<BlockTypeWrapper> use = | ||||||
|  |                 Optional.ofNullable(plot).map(p -> p.getFlag(UseFlag.class)).orElse(area.isRoadFlags() ? | ||||||
|  |                         area.getFlag(UseFlag.class) : Collections.emptyList()); | ||||||
|  |         BlockType type = BukkitAdapter.asBlockType(block.getType()); | ||||||
|  |         for (final BlockTypeWrapper blockTypeWrapper : use) { | ||||||
|  |             if (blockTypeWrapper.accepts(BlockTypes.AIR) || blockTypeWrapper | ||||||
|  |                     .accepts(type)) { | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             if (pp.hasPermission(Permission.PERMISSION_ADMIN_BUILD_ROAD)) { |             if (pp.hasPermission(Permission.PERMISSION_ADMIN_BUILD_ROAD)) { | ||||||
|                 return; |                 return; | ||||||
| @@ -1464,6 +1498,16 @@ public class PlayerEventListener implements Listener { | |||||||
|         Player player = event.getPlayer(); |         Player player = event.getPlayer(); | ||||||
|         BukkitPlayer plotPlayer = BukkitUtil.adapt(player); |         BukkitPlayer plotPlayer = BukkitUtil.adapt(player); | ||||||
|         Plot plot = area.getPlot(location); |         Plot plot = area.getPlot(location); | ||||||
|  |         final List<BlockTypeWrapper> use = | ||||||
|  |                 Optional.ofNullable(plot).map(p -> p.getFlag(UseFlag.class)).orElse(area.isRoadFlags() ? | ||||||
|  |                         area.getFlag(UseFlag.class) : Collections.emptyList()); | ||||||
|  |         BlockType type = BukkitAdapter.asBlockType(blockClicked.getType()); | ||||||
|  |         for (final BlockTypeWrapper blockTypeWrapper : use) { | ||||||
|  |             if (blockTypeWrapper.accepts(BlockTypes.AIR) || blockTypeWrapper | ||||||
|  |                     .accepts(type)) { | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             if (plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_BUILD_ROAD)) { |             if (plotPlayer.hasPermission(Permission.PERMISSION_ADMIN_BUILD_ROAD)) { | ||||||
|                 return; |                 return; | ||||||
|   | |||||||
| @@ -47,6 +47,7 @@ import org.bukkit.event.entity.LingeringPotionSplashEvent; | |||||||
| import org.bukkit.event.entity.PotionSplashEvent; | import org.bukkit.event.entity.PotionSplashEvent; | ||||||
| import org.bukkit.event.entity.ProjectileHitEvent; | import org.bukkit.event.entity.ProjectileHitEvent; | ||||||
| import org.bukkit.event.entity.ProjectileLaunchEvent; | import org.bukkit.event.entity.ProjectileLaunchEvent; | ||||||
|  | import org.bukkit.event.player.PlayerEggThrowEvent; | ||||||
| import org.bukkit.projectiles.BlockProjectileSource; | import org.bukkit.projectiles.BlockProjectileSource; | ||||||
| import org.bukkit.projectiles.ProjectileSource; | import org.bukkit.projectiles.ProjectileSource; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
| @@ -157,14 +158,26 @@ public class ProjectileEventListener implements Listener { | |||||||
|  |  | ||||||
|     @EventHandler |     @EventHandler | ||||||
|     public void onProjectileHit(ProjectileHitEvent event) { |     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()); |         Location location = BukkitUtil.adapt(entity.getLocation()); | ||||||
|         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { |         if (!this.plotAreaManager.hasPlotArea(location.getWorldName())) { | ||||||
|             return; |             return false; | ||||||
|         } |         } | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
|             return; |             return false; | ||||||
|         } |         } | ||||||
|         Plot plot = area.getPlot(location); |         Plot plot = area.getPlot(location); | ||||||
|         ProjectileSource shooter = entity.getShooter(); |         ProjectileSource shooter = entity.getShooter(); | ||||||
| @@ -172,15 +185,14 @@ public class ProjectileEventListener implements Listener { | |||||||
|             if (!((Player) shooter).isOnline()) { |             if (!((Player) shooter).isOnline()) { | ||||||
|                 if (plot != null) { |                 if (plot != null) { | ||||||
|                     if (plot.isAdded(((Player) shooter).getUniqueId()) || plot.getFlag(ProjectilesFlag.class)) { |                     if (plot.isAdded(((Player) shooter).getUniqueId()) || plot.getFlag(ProjectilesFlag.class)) { | ||||||
|                         return; |                         return false; | ||||||
|                     } |                     } | ||||||
|                 } else if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, ProjectilesFlag.class, true)) { |                 } else if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, ProjectilesFlag.class, true)) { | ||||||
|                     return; |                     return false; | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 entity.remove(); |                 entity.remove(); | ||||||
|                 event.setCancelled(true); |                 return true; | ||||||
|                 return; |  | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             PlotPlayer<?> pp = BukkitUtil.adapt((Player) shooter); |             PlotPlayer<?> pp = BukkitUtil.adapt((Player) shooter); | ||||||
| @@ -189,38 +201,36 @@ public class ProjectileEventListener implements Listener { | |||||||
|                         Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED |                         Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED | ||||||
|                 )) { |                 )) { | ||||||
|                     entity.remove(); |                     entity.remove(); | ||||||
|                     event.setCancelled(true); |                     return true; | ||||||
|                 } |                 } | ||||||
|                 return; |                 return false; | ||||||
|             } |             } | ||||||
|             if (plot.isAdded(pp.getUUID()) || pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER) || plot.getFlag( |             if (plot.isAdded(pp.getUUID()) || pp.hasPermission(Permission.PERMISSION_ADMIN_PROJECTILE_OTHER) || plot.getFlag( | ||||||
|                     ProjectilesFlag.class) || (entity instanceof FishHook && plot.getFlag( |                     ProjectilesFlag.class) || (entity instanceof FishHook && plot.getFlag( | ||||||
|                     FishingFlag.class))) { |                     FishingFlag.class))) { | ||||||
|                 return; |                 return false; | ||||||
|             } |             } | ||||||
|             entity.remove(); |             entity.remove(); | ||||||
|             event.setCancelled(true); |             return true; | ||||||
|             return; |  | ||||||
|         } |         } | ||||||
|         if (!(shooter instanceof Entity) && shooter != null) { |         if (!(shooter instanceof Entity) && shooter != null) { | ||||||
|             if (plot == null) { |             if (plot == null) { | ||||||
|                 entity.remove(); |                 entity.remove(); | ||||||
|                 event.setCancelled(true); |                 return true; | ||||||
|                 return; |  | ||||||
|             } |             } | ||||||
|             Location sLoc = |             Location sLoc = | ||||||
|                     BukkitUtil.adapt(((BlockProjectileSource) shooter).getBlock().getLocation()); |                     BukkitUtil.adapt(((BlockProjectileSource) shooter).getBlock().getLocation()); | ||||||
|             if (!area.contains(sLoc.getX(), sLoc.getZ())) { |             if (!area.contains(sLoc.getX(), sLoc.getZ())) { | ||||||
|                 entity.remove(); |                 entity.remove(); | ||||||
|                 event.setCancelled(true); |                 return true; | ||||||
|                 return; |  | ||||||
|             } |             } | ||||||
|             Plot sPlot = area.getOwnedPlotAbs(sLoc); |             Plot sPlot = area.getOwnedPlotAbs(sLoc); | ||||||
|             if (sPlot == null || !PlotHandler.sameOwners(plot, sPlot)) { |             if (sPlot == null || !PlotHandler.sameOwners(plot, sPlot)) { | ||||||
|                 entity.remove(); |                 entity.remove(); | ||||||
|                 event.setCancelled(true); |                 return true; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -44,6 +44,7 @@ import java.util.List; | |||||||
| import java.util.Queue; | import java.util.Queue; | ||||||
| import java.util.concurrent.LinkedBlockingQueue; | import java.util.concurrent.LinkedBlockingQueue; | ||||||
| import java.util.concurrent.TimeUnit; | import java.util.concurrent.TimeUnit; | ||||||
|  | import java.util.concurrent.TimeoutException; | ||||||
| import java.util.concurrent.atomic.AtomicInteger; | import java.util.concurrent.atomic.AtomicInteger; | ||||||
| import java.util.function.Consumer; | import java.util.function.Consumer; | ||||||
|  |  | ||||||
| @@ -226,16 +227,22 @@ public final class BukkitChunkCoordinator extends ChunkCoordinator { | |||||||
|             loadingChunks.incrementAndGet(); |             loadingChunks.incrementAndGet(); | ||||||
|             PaperLib |             PaperLib | ||||||
|                     .getChunkAtAsync(this.bukkitWorld, chunk.getX(), chunk.getZ(), shouldGen, true) |                     .getChunkAtAsync(this.bukkitWorld, chunk.getX(), chunk.getZ(), shouldGen, true) | ||||||
|                     .completeOnTimeout(null, 10L, TimeUnit.SECONDS) |                     .orTimeout(10L, TimeUnit.SECONDS) | ||||||
|                     .whenComplete((chunkObject, throwable) -> { |                     .whenComplete((chunkObject, throwable) -> { | ||||||
|                         loadingChunks.decrementAndGet(); |                         loadingChunks.decrementAndGet(); | ||||||
|                         if (throwable != null) { |                         if (throwable != null) { | ||||||
|  |                             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); |                                 LOGGER.error("Failed to load chunk {}", chunk, throwable); | ||||||
|                                 // We want one less because this couldn't be processed |                                 // We want one less because this couldn't be processed | ||||||
|                                 this.expectedSize.decrementAndGet(); |                                 this.expectedSize.decrementAndGet(); | ||||||
|  |                             } | ||||||
|                         } else if (chunkObject == null) { |                         } else if (chunkObject == null) { | ||||||
|                             LOGGER.warn("Timed out awaiting chunk load {}", chunk); |                             if (shouldGen) { | ||||||
|                             this.requestedChunks.offer(chunk); |                                 LOGGER.error("Null chunk returned for chunk at {}", chunk); | ||||||
|  |                             } | ||||||
|                         } else if (PlotSquared.get().isMainThread(Thread.currentThread())) { |                         } else if (PlotSquared.get().isMainThread(Thread.currentThread())) { | ||||||
|                             this.processChunk(chunkObject); |                             this.processChunk(chunkObject); | ||||||
|                         } else { |                         } else { | ||||||
|   | |||||||
| @@ -52,6 +52,7 @@ import org.checkerframework.checker.nullness.qual.NonNull; | |||||||
|  |  | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
|  | import java.util.Objects; | ||||||
| import java.util.function.Consumer; | import java.util.function.Consumer; | ||||||
|  |  | ||||||
| public class BukkitQueueCoordinator extends BasicQueueCoordinator { | public class BukkitQueueCoordinator extends BasicQueueCoordinator { | ||||||
| @@ -210,8 +211,13 @@ public class BukkitQueueCoordinator extends BasicQueueCoordinator { | |||||||
|                             BaseBlock block = getWorld().getBlock(blockVector3).toBaseBlock(tag); |                             BaseBlock block = getWorld().getBlock(blockVector3).toBaseBlock(tag); | ||||||
|                             getWorld().setBlock(blockVector3, block, getSideEffectSet(SideEffectState.NONE)); |                             getWorld().setBlock(blockVector3, block, getSideEffectSet(SideEffectState.NONE)); | ||||||
|                         } catch (WorldEditException ignored) { |                         } catch (WorldEditException ignored) { | ||||||
|                             StateWrapper sw = new StateWrapper(tag); |                             StateWrapper.INSTANCE.restore( | ||||||
|                             sw.restoreTag(getWorld().getName(), blockVector3.getX(), blockVector3.getY(), blockVector3.getZ()); |                                     getWorld().getName(), | ||||||
|  |                                     blockVector3.getX(), | ||||||
|  |                                     blockVector3.getY(), | ||||||
|  |                                     blockVector3.getZ(), | ||||||
|  |                                     tag | ||||||
|  |                             ); | ||||||
|                         } |                         } | ||||||
|                     }); |                     }); | ||||||
|                 } |                 } | ||||||
| @@ -295,9 +301,7 @@ public class BukkitQueueCoordinator extends BasicQueueCoordinator { | |||||||
|             existing.setBlockData(blockData, false); |             existing.setBlockData(blockData, false); | ||||||
|             if (block.hasNbtData()) { |             if (block.hasNbtData()) { | ||||||
|                 CompoundTag tag = block.getNbtData(); |                 CompoundTag tag = block.getNbtData(); | ||||||
|                 StateWrapper sw = new StateWrapper(tag); |                 StateWrapper.INSTANCE.restore(existing, Objects.requireNonNull(tag)); | ||||||
|  |  | ||||||
|                 sw.restoreTag(existing); |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -34,6 +34,8 @@ import org.bukkit.entity.EntityType; | |||||||
| import org.bukkit.generator.LimitedRegion; | import org.bukkit.generator.LimitedRegion; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
|  | import java.util.Objects; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Wraps a {@link LimitedRegion} inside a {@link com.plotsquared.core.queue.QueueCoordinator} so it can be written to. |  * Wraps a {@link LimitedRegion} inside a {@link com.plotsquared.core.queue.QueueCoordinator} so it can be written to. | ||||||
|  * |  * | ||||||
| @@ -44,7 +46,6 @@ public class LimitedRegionWrapperQueue extends DelegateQueueCoordinator { | |||||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + LimitedRegionWrapperQueue.class.getSimpleName()); |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + LimitedRegionWrapperQueue.class.getSimpleName()); | ||||||
|  |  | ||||||
|     private final LimitedRegion limitedRegion; |     private final LimitedRegion limitedRegion; | ||||||
|     private boolean useOtherRestoreTagMethod = false; |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * @since 6.9.0 |      * @since 6.9.0 | ||||||
| @@ -64,20 +65,11 @@ public class LimitedRegionWrapperQueue extends DelegateQueueCoordinator { | |||||||
|         boolean result = setBlock(x, y, z, id.toImmutableState()); |         boolean result = setBlock(x, y, z, id.toImmutableState()); | ||||||
|         if (result && id.hasNbtData()) { |         if (result && id.hasNbtData()) { | ||||||
|             CompoundTag tag = id.getNbtData(); |             CompoundTag tag = id.getNbtData(); | ||||||
|             StateWrapper sw = new StateWrapper(tag); |  | ||||||
|             try { |             try { | ||||||
|                 if (useOtherRestoreTagMethod && getWorld() != null) { |                 StateWrapper.INSTANCE.restore(limitedRegion.getBlockState(x, y, z).getBlock(), Objects.requireNonNull(tag)); | ||||||
|                     sw.restoreTag(getWorld().getName(), x, y, z); |  | ||||||
|                 } else { |  | ||||||
|                     sw.restoreTag(limitedRegion.getBlockState(x, y, z).getBlock()); |  | ||||||
|                 } |  | ||||||
|             } catch (IllegalArgumentException e) { |             } catch (IllegalArgumentException e) { | ||||||
|                 LOGGER.error("Error attempting to populate tile entity into the world at location {},{},{}", x, y, z, e); |                 LOGGER.error("Error attempting to populate tile entity into the world at location {},{},{}", x, y, z, e); | ||||||
|                 return false; |                 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; |         return result; | ||||||
| @@ -113,9 +105,8 @@ public class LimitedRegionWrapperQueue extends DelegateQueueCoordinator { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean setTile(final int x, final int y, final int z, @NonNull final CompoundTag tag) { |     public boolean setTile(final int x, final int y, final int z, @NonNull final CompoundTag tag) { | ||||||
|         StateWrapper sw = new StateWrapper(tag); |  | ||||||
|         try { |         try { | ||||||
|             return sw.restoreTag(limitedRegion.getBlockState(x, y, z).getBlock()); |             return StateWrapper.INSTANCE.restore(limitedRegion.getBlockState(x, y, z).getBlock(), tag); | ||||||
|         } catch (IllegalArgumentException e) { |         } catch (IllegalArgumentException e) { | ||||||
|             LOGGER.error("Error attempting to populate tile entity into the world at location {},{},{}", x, y, z, e); |             LOGGER.error("Error attempting to populate tile entity into the world at location {},{},{}", x, y, z, e); | ||||||
|             return false; |             return false; | ||||||
|   | |||||||
| @@ -27,6 +27,8 @@ import com.plotsquared.core.util.WorldUtil; | |||||||
| import com.sk89q.jnbt.CompoundTag; | import com.sk89q.jnbt.CompoundTag; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
|  | import java.util.Objects; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Schematic Handler. |  * Schematic Handler. | ||||||
|  */ |  */ | ||||||
| @@ -39,8 +41,8 @@ public class BukkitSchematicHandler extends SchematicHandler { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean restoreTile(QueueCoordinator queue, CompoundTag ct, int x, int y, int z) { |     public boolean restoreTile(QueueCoordinator queue, CompoundTag tag, int x, int y, int z) { | ||||||
|         return new StateWrapper(ct).restoreTag(queue.getWorld().getName(), x, y, z); |         return StateWrapper.INSTANCE.restore(Objects.requireNonNull(queue.getWorld()).getName(), x, y, z, tag); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -18,332 +18,35 @@ | |||||||
|  */ |  */ | ||||||
| package com.plotsquared.bukkit.schematic; | 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.plotsquared.bukkit.util.BukkitUtil; | ||||||
| import com.sk89q.jnbt.ByteTag; |  | ||||||
| import com.sk89q.jnbt.CompoundTag; | import com.sk89q.jnbt.CompoundTag; | ||||||
| import com.sk89q.jnbt.ListTag; |  | ||||||
| import com.sk89q.jnbt.ShortTag; |  | ||||||
| import com.sk89q.jnbt.StringTag; |  | ||||||
| import com.sk89q.jnbt.Tag; |  | ||||||
| import com.sk89q.worldedit.blocks.BaseItemStack; |  | ||||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; |  | ||||||
| import com.sk89q.worldedit.world.item.ItemType; |  | ||||||
| import 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.World; | ||||||
| import org.bukkit.block.Banner; |  | ||||||
| import org.bukkit.block.Block; | 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.ItemStack; |  | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  | import org.jetbrains.annotations.ApiStatus; | ||||||
|  |  | ||||||
| import java.util.ArrayList; | @ApiStatus.Internal | ||||||
| import java.util.HashMap; | public sealed interface StateWrapper permits StateWrapperSpigot { | ||||||
| import java.util.List; |  | ||||||
| import java.util.Map; |  | ||||||
| import java.util.Map.Entry; |  | ||||||
| import java.util.Objects; |  | ||||||
| import java.util.UUID; |  | ||||||
|  |  | ||||||
| public class StateWrapper { |     StateWrapper INSTANCE = Factory.createStateWrapper(); | ||||||
|  |  | ||||||
|     public CompoundTag tag; |     boolean restore(final @NonNull Block block, final @NonNull CompoundTag data); | ||||||
|  |  | ||||||
|     private boolean paperErrorTextureSent = false; |     default boolean restore(final String worldName, final int x, final int y, final int z, final CompoundTag data) { | ||||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + StateWrapper.class.getSimpleName()); |         final World world = BukkitUtil.getWorld(worldName); | ||||||
|  |  | ||||||
|     public StateWrapper(CompoundTag tag) { |  | ||||||
|         this.tag = tag; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static String jsonToColourCode(String str) { |  | ||||||
|         str = str.replace("{\"extra\":", "").replace("],\"text\":\"\"}", "]") |  | ||||||
|                 .replace("[{\"color\":\"black\",\"text\":\"", "&0") |  | ||||||
|                 .replace("[{\"color\":\"dark_blue\",\"text\":\"", "&1") |  | ||||||
|                 .replace("[{\"color\":\"dark_green\",\"text\":\"", "&2") |  | ||||||
|                 .replace("[{\"color\":\"dark_aqua\",\"text\":\"", "&3") |  | ||||||
|                 .replace("[{\"color\":\"dark_red\",\"text\":\"", "&4") |  | ||||||
|                 .replace("[{\"color\":\"dark_purple\",\"text\":\"", "&5") |  | ||||||
|                 .replace("[{\"color\":\"gold\",\"text\":\"", "&6") |  | ||||||
|                 .replace("[{\"color\":\"gray\",\"text\":\"", "&7") |  | ||||||
|                 .replace("[{\"color\":\"dark_gray\",\"text\":\"", "&8") |  | ||||||
|                 .replace("[{\"color\":\"blue\",\"text\":\"", "&9") |  | ||||||
|                 .replace("[{\"color\":\"green\",\"text\":\"", "&a") |  | ||||||
|                 .replace("[{\"color\":\"aqua\",\"text\":\"", "&b") |  | ||||||
|                 .replace("[{\"color\":\"red\",\"text\":\"", "&c") |  | ||||||
|                 .replace("[{\"color\":\"light_purple\",\"text\":\"", "&d") |  | ||||||
|                 .replace("[{\"color\":\"yellow\",\"text\":\"", "&e") |  | ||||||
|                 .replace("[{\"color\":\"white\",\"text\":\"", "&f") |  | ||||||
|                 .replace("[{\"obfuscated\":true,\"text\":\"", "&k") |  | ||||||
|                 .replace("[{\"bold\":true,\"text\":\"", "&l") |  | ||||||
|                 .replace("[{\"strikethrough\":true,\"text\":\"", "&m") |  | ||||||
|                 .replace("[{\"underlined\":true,\"text\":\"", "&n") |  | ||||||
|                 .replace("[{\"italic\":true,\"text\":\"", "&o").replace("[{\"color\":\"black\",", "&0") |  | ||||||
|                 .replace("[{\"color\":\"dark_blue\",", "&1") |  | ||||||
|                 .replace("[{\"color\":\"dark_green\",", "&2") |  | ||||||
|                 .replace("[{\"color\":\"dark_aqua\",", "&3").replace("[{\"color\":\"dark_red\",", "&4") |  | ||||||
|                 .replace("[{\"color\":\"dark_purple\",", "&5").replace("[{\"color\":\"gold\",", "&6") |  | ||||||
|                 .replace("[{\"color\":\"gray\",", "&7").replace("[{\"color\":\"dark_gray\",", "&8") |  | ||||||
|                 .replace("[{\"color\":\"blue\",", "&9").replace("[{\"color\":\"green\",", "&a") |  | ||||||
|                 .replace("[{\"color\":\"aqua\",", "&b").replace("[{\"color\":\"red\",", "&c") |  | ||||||
|                 .replace("[{\"color\":\"light_purple\",", "&d").replace("[{\"color\":\"yellow\",", "&e") |  | ||||||
|                 .replace("[{\"color\":\"white\",", "&f").replace("[{\"obfuscated\":true,", "&k") |  | ||||||
|                 .replace("[{\"bold\":true,", "&l").replace("[{\"strikethrough\":true,", "&m") |  | ||||||
|                 .replace("[{\"underlined\":true,", "&n").replace("[{\"italic\":true,", "&o") |  | ||||||
|                 .replace("{\"color\":\"black\",\"text\":\"", "&0") |  | ||||||
|                 .replace("{\"color\":\"dark_blue\",\"text\":\"", "&1") |  | ||||||
|                 .replace("{\"color\":\"dark_green\",\"text\":\"", "&2") |  | ||||||
|                 .replace("{\"color\":\"dark_aqua\",\"text\":\"", "&3") |  | ||||||
|                 .replace("{\"color\":\"dark_red\",\"text\":\"", "&4") |  | ||||||
|                 .replace("{\"color\":\"dark_purple\",\"text\":\"", "&5") |  | ||||||
|                 .replace("{\"color\":\"gold\",\"text\":\"", "&6") |  | ||||||
|                 .replace("{\"color\":\"gray\",\"text\":\"", "&7") |  | ||||||
|                 .replace("{\"color\":\"dark_gray\",\"text\":\"", "&8") |  | ||||||
|                 .replace("{\"color\":\"blue\",\"text\":\"", "&9") |  | ||||||
|                 .replace("{\"color\":\"green\",\"text\":\"", "&a") |  | ||||||
|                 .replace("{\"color\":\"aqua\",\"text\":\"", "&b") |  | ||||||
|                 .replace("{\"color\":\"red\",\"text\":\"", "&c") |  | ||||||
|                 .replace("{\"color\":\"light_purple\",\"text\":\"", "&d") |  | ||||||
|                 .replace("{\"color\":\"yellow\",\"text\":\"", "&e") |  | ||||||
|                 .replace("{\"color\":\"white\",\"text\":\"", "&f") |  | ||||||
|                 .replace("{\"obfuscated\":true,\"text\":\"", "&k") |  | ||||||
|                 .replace("{\"bold\":true,\"text\":\"", "&l") |  | ||||||
|                 .replace("{\"strikethrough\":true,\"text\":\"", "&m") |  | ||||||
|                 .replace("{\"underlined\":true,\"text\":\"", "&n") |  | ||||||
|                 .replace("{\"italic\":true,\"text\":\"", "&o").replace("{\"color\":\"black\",", "&0") |  | ||||||
|                 .replace("{\"color\":\"dark_blue\",", "&1").replace("{\"color\":\"dark_green\",", "&2") |  | ||||||
|                 .replace("{\"color\":\"dark_aqua\",", "&3").replace("{\"color\":\"dark_red\",", "&4") |  | ||||||
|                 .replace("{\"color\":\"dark_purple\",", "&5").replace("{\"color\":\"gold\",", "&6") |  | ||||||
|                 .replace("{\"color\":\"gray\",", "&7").replace("{\"color\":\"dark_gray\",", "&8") |  | ||||||
|                 .replace("{\"color\":\"blue\",", "&9").replace("{\"color\":\"green\",", "&a") |  | ||||||
|                 .replace("{\"color\":\"aqua\",", "&b").replace("{\"color\":\"red\",", "&c") |  | ||||||
|                 .replace("{\"color\":\"light_purple\",", "&d").replace("{\"color\":\"yellow\",", "&e") |  | ||||||
|                 .replace("{\"color\":\"white\",", "&f").replace("{\"obfuscated\":true,", "&k") |  | ||||||
|                 .replace("{\"bold\":true,", "&l").replace("{\"strikethrough\":true,", "&m") |  | ||||||
|                 .replace("{\"underlined\":true,", "&n").replace("{\"italic\":true,", "&o") |  | ||||||
|                 .replace("\"color\":\"black\",\"text\":\"", "&0") |  | ||||||
|                 .replace("\"color\":\"dark_blue\",\"text\":\"", "&1") |  | ||||||
|                 .replace("\"color\":\"dark_green\",\"text\":\"", "&2") |  | ||||||
|                 .replace("\"color\":\"dark_aqua\",\"text\":\"", "&3") |  | ||||||
|                 .replace("\"color\":\"dark_red\",\"text\":\"", "&4") |  | ||||||
|                 .replace("\"color\":\"dark_purple\",\"text\":\"", "&5") |  | ||||||
|                 .replace("\"color\":\"gold\",\"text\":\"", "&6") |  | ||||||
|                 .replace("\"color\":\"gray\",\"text\":\"", "&7") |  | ||||||
|                 .replace("\"color\":\"dark_gray\",\"text\":\"", "&8") |  | ||||||
|                 .replace("\"color\":\"blue\",\"text\":\"", "&9") |  | ||||||
|                 .replace("\"color\":\"green\",\"text\":\"", "&a") |  | ||||||
|                 .replace("\"color\":\"aqua\",\"text\":\"", "&b") |  | ||||||
|                 .replace("\"color\":\"red\",\"text\":\"", "&c") |  | ||||||
|                 .replace("\"color\":\"light_purple\",\"text\":\"", "&d") |  | ||||||
|                 .replace("\"color\":\"yellow\",\"text\":\"", "&e") |  | ||||||
|                 .replace("\"color\":\"white\",\"text\":\"", "&f") |  | ||||||
|                 .replace("\"obfuscated\":true,\"text\":\"", "&k") |  | ||||||
|                 .replace("\"bold\":true,\"text\":\"", "&l") |  | ||||||
|                 .replace("\"strikethrough\":true,\"text\":\"", "&m") |  | ||||||
|                 .replace("\"underlined\":true,\"text\":\"", "&n") |  | ||||||
|                 .replace("\"italic\":true,\"text\":\"", "&o").replace("\"color\":\"black\",", "&0") |  | ||||||
|                 .replace("\"color\":\"dark_blue\",", "&1").replace("\"color\":\"dark_green\",", "&2") |  | ||||||
|                 .replace("\"color\":\"dark_aqua\",", "&3").replace("\"color\":\"dark_red\",", "&4") |  | ||||||
|                 .replace("\"color\":\"dark_purple\",", "&5").replace("\"color\":\"gold\",", "&6") |  | ||||||
|                 .replace("\"color\":\"gray\",", "&7").replace("\"color\":\"dark_gray\",", "&8") |  | ||||||
|                 .replace("\"color\":\"blue\",", "&9").replace("\"color\":\"green\",", "&a") |  | ||||||
|                 .replace("\"color\":\"aqua\",", "&b").replace("\"color\":\"red\",", "&c") |  | ||||||
|                 .replace("\"color\":\"light_purple\",", "&d").replace("\"color\":\"yellow\",", "&e") |  | ||||||
|                 .replace("\"color\":\"white\",", "&f").replace("\"obfuscated\":true,", "&k") |  | ||||||
|                 .replace("\"bold\":true,", "&l").replace("\"strikethrough\":true,", "&m") |  | ||||||
|                 .replace("\"underlined\":true,", "&n").replace("\"italic\":true,", "&o") |  | ||||||
|                 .replace("[{\"text\":\"", "&0").replace("{\"text\":\"", "&0").replace("\"},", "") |  | ||||||
|                 .replace("\"}]", "").replace("\"}", ""); |  | ||||||
|         str = ChatColor.translateAlternateColorCodes('&', str); |  | ||||||
|         return str; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Restore the TileEntity data to the given world at the given coordinates. |  | ||||||
|      * |  | ||||||
|      * @param worldName World name |  | ||||||
|      * @param x         x position |  | ||||||
|      * @param y         y position |  | ||||||
|      * @param z         z position |  | ||||||
|      * @return true if successful |  | ||||||
|      */ |  | ||||||
|     public boolean restoreTag(String worldName, int x, int y, int z) { |  | ||||||
|         World world = BukkitUtil.getWorld(worldName); |  | ||||||
|         if (world == null) { |         if (world == null) { | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|         return restoreTag(world.getBlockAt(x, y, z)); |         return this.restore(world.getBlockAt(x, y, z), data); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     @ApiStatus.Internal | ||||||
|      * Restore the TileEntity data to the given block |     final class Factory { | ||||||
|      * |  | ||||||
|      * @param block Block to restore to |         private static StateWrapper createStateWrapper() { | ||||||
|      * @return true if successful |             return new StateWrapperSpigot(); | ||||||
|      */ |         } | ||||||
|     @SuppressWarnings("deprecation") // #setLine is needed for Spigot compatibility |  | ||||||
|     public boolean restoreTag(@NonNull Block block) { |  | ||||||
|         if (this.tag == null) { |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
|         org.bukkit.block.BlockState state = block.getState(); |  | ||||||
|         switch (getId()) { |  | ||||||
|             case "chest", "beacon", "brewingstand", "dispenser", "dropper", "furnace", "hopper", "shulkerbox" -> { |  | ||||||
|                 if (!(state instanceof Container container)) { |  | ||||||
|                     return false; |  | ||||||
|                 } |  | ||||||
|                 List<Tag> itemsTag = this.tag.getListTag("Items").getValue(); |  | ||||||
|                 Inventory inv = container.getSnapshotInventory(); |  | ||||||
|                 for (Tag itemTag : itemsTag) { |  | ||||||
|                     CompoundTag itemComp = (CompoundTag) itemTag; |  | ||||||
|                     ItemType type = ItemType.REGISTRY.get(itemComp.getString("id").toLowerCase()); |  | ||||||
|                     if (type == null) { |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
|                     int count = itemComp.getByte("Count"); |  | ||||||
|                     int slot = itemComp.getByte("Slot"); |  | ||||||
|                     CompoundTag tag = (CompoundTag) itemComp.getValue().get("tag"); |  | ||||||
|                     BaseItemStack baseItemStack = new BaseItemStack(type, tag, count); |  | ||||||
|                     ItemStack itemStack = BukkitAdapter.adapt(baseItemStack); |  | ||||||
|                     inv.setItem(slot, itemStack); |  | ||||||
|                 } |  | ||||||
|                 container.update(true, false); |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|             case "sign" -> { |  | ||||||
|                 if (state instanceof Sign sign) { |  | ||||||
|                     sign.setLine(0, jsonToColourCode(tag.getString("Text1"))); |  | ||||||
|                     sign.setLine(1, jsonToColourCode(tag.getString("Text2"))); |  | ||||||
|                     sign.setLine(2, jsonToColourCode(tag.getString("Text3"))); |  | ||||||
|                     sign.setLine(3, jsonToColourCode(tag.getString("Text4"))); |  | ||||||
|                     state.update(true); |  | ||||||
|                     return true; |  | ||||||
|                 } |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
|             case "skull" -> { |  | ||||||
|                 if (state instanceof Skull skull) { |  | ||||||
|                     CompoundTag skullOwner = ((CompoundTag) this.tag.getValue().get("SkullOwner")); |  | ||||||
|                     if (skullOwner == null) { |  | ||||||
|                         return true; |  | ||||||
|                     } |  | ||||||
|                     String player = skullOwner.getString("Name"); |  | ||||||
|  |  | ||||||
|                     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; |  | ||||||
|                     } |  | ||||||
|                     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; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getId() { |  | ||||||
|         String tileid = this.tag.getString("id").toLowerCase(); |  | ||||||
|         if (tileid.startsWith("minecraft:")) { |  | ||||||
|             tileid = tileid.replace("minecraft:", ""); |  | ||||||
|         } |  | ||||||
|         return tileid; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public List<CompoundTag> serializeInventory(ItemStack[] items) { |  | ||||||
|         List<CompoundTag> tags = new ArrayList<>(); |  | ||||||
|         for (int i = 0; i < items.length; ++i) { |  | ||||||
|             if (items[i] != null) { |  | ||||||
|                 Map<String, Tag> tagData = serializeItem(items[i]); |  | ||||||
|                 tagData.put("Slot", new ByteTag((byte) i)); |  | ||||||
|                 tags.add(new CompoundTag(tagData)); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return tags; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Map<String, Tag> serializeItem(ItemStack item) { |  | ||||||
|         Map<String, Tag> data = new HashMap<>(); |  | ||||||
|         data.put("id", new StringTag(item.getType().name())); |  | ||||||
|         data.put("Damage", new ShortTag(item.getDurability())); |  | ||||||
|         data.put("Count", new ByteTag((byte) item.getAmount())); |  | ||||||
|         if (!item.getEnchantments().isEmpty()) { |  | ||||||
|             List<CompoundTag> enchantmentList = new ArrayList<>(); |  | ||||||
|             for (Entry<Enchantment, Integer> entry : item.getEnchantments().entrySet()) { |  | ||||||
|                 Map<String, Tag> enchantment = new HashMap<>(); |  | ||||||
|                 enchantment.put("id", new StringTag(entry.getKey().toString())); |  | ||||||
|                 enchantment.put("lvl", new ShortTag(entry.getValue().shortValue())); |  | ||||||
|                 enchantmentList.add(new CompoundTag(enchantment)); |  | ||||||
|             } |  | ||||||
|             Map<String, Tag> auxData = new HashMap<>(); |  | ||||||
|             auxData.put("ench", new ListTag(CompoundTag.class, enchantmentList)); |  | ||||||
|             data.put("tag", new CompoundTag(auxData)); |  | ||||||
|         } |  | ||||||
|         return data; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,263 @@ | |||||||
|  | /* | ||||||
|  |  * PlotSquared, a land and world management plugin for Minecraft. | ||||||
|  |  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||||
|  |  * Copyright (C) IntellectualSites team and contributors | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 3 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  | package com.plotsquared.bukkit.schematic; | ||||||
|  |  | ||||||
|  | import com.plotsquared.core.util.ReflectionHelper; | ||||||
|  | import com.plotsquared.core.util.ReflectionUtils; | ||||||
|  | import com.sk89q.jnbt.CompoundTag; | ||||||
|  | import com.sk89q.worldedit.bukkit.WorldEditPlugin; | ||||||
|  | import com.sk89q.worldedit.bukkit.adapter.BukkitImplAdapter; | ||||||
|  | import com.sk89q.worldedit.bukkit.adapter.Refraction; | ||||||
|  | import com.sk89q.worldedit.extension.platform.NoCapablePlatformException; | ||||||
|  | import org.apache.logging.log4j.LogManager; | ||||||
|  | import org.apache.logging.log4j.Logger; | ||||||
|  | import org.bukkit.Bukkit; | ||||||
|  | import org.bukkit.block.Block; | ||||||
|  | import org.bukkit.block.BlockState; | ||||||
|  | import org.bukkit.block.Sign; | ||||||
|  | import org.bukkit.block.sign.Side; | ||||||
|  | import org.bukkit.block.sign.SignSide; | ||||||
|  | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
|  | import java.lang.invoke.MethodHandle; | ||||||
|  | import java.lang.invoke.MethodHandles; | ||||||
|  | import java.lang.invoke.MethodType; | ||||||
|  | import java.lang.reflect.Field; | ||||||
|  | import java.lang.reflect.Method; | ||||||
|  | import java.lang.reflect.Modifier; | ||||||
|  | import java.util.Arrays; | ||||||
|  | import java.util.Optional; | ||||||
|  |  | ||||||
|  | final class StateWrapperSpigot implements StateWrapper { | ||||||
|  |  | ||||||
|  |     private static final boolean FORCE_UPDATE_STATE = true; | ||||||
|  |     private static final boolean UPDATE_TRIGGER_PHYSICS = false; | ||||||
|  |     private static final String CRAFTBUKKIT_PACKAGE = Bukkit.getServer().getClass().getPackageName(); | ||||||
|  |  | ||||||
|  |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + StateWrapperSpigot.class.getSimpleName()); | ||||||
|  |     private static final MethodHandles.Lookup LOOKUP = MethodHandles.lookup(); | ||||||
|  |  | ||||||
|  |     private static BukkitImplAdapter ADAPTER = null; | ||||||
|  |     private static Class<?> LIN_TAG_CLASS = null; | ||||||
|  |     private static Class<?> CRAFT_BLOCK_ENTITY_STATE_CLASS = null; | ||||||
|  |     private static Field CRAFT_SIGN_SIDE_SIGN_TEXT = null; | ||||||
|  |     private static Field CRAFT_SIGN_SIDE_LINES = null; | ||||||
|  |     private static MethodHandle PAPERWEIGHT_ADAPTER_FROM_NATIVE = null; | ||||||
|  |     private static MethodHandle CRAFT_BLOCK_ENTITY_STATE_LOAD_DATA = null; | ||||||
|  |     private static MethodHandle CRAFT_BLOCK_ENTITY_STATE_UPDATE = null; | ||||||
|  |     private static MethodHandle CRAFT_BLOCK_ENTITY_STATE_GET_SNAPSHOT = null; | ||||||
|  |     private static MethodHandle SIGN_BLOCK_ENTITY_SET_TEXT = null; | ||||||
|  |     private static MethodHandle DECODER_PARSE = null; | ||||||
|  |     private static MethodHandle DATA_RESULT_RESULT = null; | ||||||
|  |     private static MethodHandle TO_LIN_TAG = null; | ||||||
|  |  | ||||||
|  |     private static Object SIGN_TEXT_DIRECT_CODEC = null; | ||||||
|  |     private static Object NBT_OPS_INSTANCE = null; | ||||||
|  |  | ||||||
|  |     public StateWrapperSpigot() { | ||||||
|  |         try { | ||||||
|  |             ReflectionUtils.RefClass worldEditPluginRefClass = ReflectionUtils.getRefClass(WorldEditPlugin.class); | ||||||
|  |             WorldEditPlugin worldEditPlugin = (WorldEditPlugin) worldEditPluginRefClass | ||||||
|  |                     .getMethod("getInstance") | ||||||
|  |                     .of(null) | ||||||
|  |                     .call(); | ||||||
|  |             ADAPTER = (BukkitImplAdapter) worldEditPluginRefClass | ||||||
|  |                     .getMethod("getBukkitImplAdapter") | ||||||
|  |                     .of(worldEditPlugin) | ||||||
|  |                     .call(); | ||||||
|  |             LIN_TAG_CLASS = Class.forName("org.enginehub.linbus.tree.LinTag"); // provided WE / FAWE version is too old | ||||||
|  |             PAPERWEIGHT_ADAPTER_FROM_NATIVE = findPaperweightAdapterFromNativeMethodHandle(ADAPTER.getClass()); | ||||||
|  |             TO_LIN_TAG = findToLinTagMethodHandle(); | ||||||
|  |         } catch (NoSuchMethodException | ClassNotFoundException | IllegalAccessException | NoCapablePlatformException e) { | ||||||
|  |             throw new RuntimeException("Failed to access required WorldEdit classes or methods", e); | ||||||
|  |         } | ||||||
|  |         try { | ||||||
|  |             final Class<?> SIGN_TEXT_CLASS = Class.forName("net.minecraft.world.level.block.entity.SignText"); | ||||||
|  |             final Class<?> CRAFT_SIGN_SIDE_CLASS = Class.forName(CRAFTBUKKIT_PACKAGE + ".block.sign.CraftSignSide"); | ||||||
|  |             CRAFT_SIGN_SIDE_SIGN_TEXT = CRAFT_SIGN_SIDE_CLASS.getDeclaredField("signText"); | ||||||
|  |             CRAFT_SIGN_SIDE_SIGN_TEXT.setAccessible(true); | ||||||
|  |             CRAFT_SIGN_SIDE_LINES = CRAFT_SIGN_SIDE_CLASS.getDeclaredField("lines"); | ||||||
|  |             CRAFT_SIGN_SIDE_LINES.setAccessible(true); | ||||||
|  |             CRAFT_BLOCK_ENTITY_STATE_CLASS = Class.forName(CRAFTBUKKIT_PACKAGE + ".block.CraftBlockEntityState"); | ||||||
|  |             CRAFT_BLOCK_ENTITY_STATE_LOAD_DATA = findCraftBlockEntityStateLoadDataMethodHandle(CRAFT_BLOCK_ENTITY_STATE_CLASS); | ||||||
|  |             CRAFT_BLOCK_ENTITY_STATE_UPDATE = findCraftBlockEntityStateUpdateMethodHandle(CRAFT_BLOCK_ENTITY_STATE_CLASS); | ||||||
|  |             CRAFT_BLOCK_ENTITY_STATE_GET_SNAPSHOT = findCraftBlockEntityStateSnapshotMethodHandle(CRAFT_BLOCK_ENTITY_STATE_CLASS); | ||||||
|  |             SIGN_BLOCK_ENTITY_SET_TEXT = findSignBlockEntitySetTextMethodHandle( | ||||||
|  |                     Class.forName(Refraction.pickName( | ||||||
|  |                             "net.minecraft.world.level.block.entity.SignBlockEntity", | ||||||
|  |                             "net.minecraft.world.level.block.entity.TileEntitySign" | ||||||
|  |                     )), | ||||||
|  |                     SIGN_TEXT_CLASS | ||||||
|  |             ); | ||||||
|  |             final Class<?> CODEC_CLASS = Class.forName("com.mojang.serialization.Codec"); | ||||||
|  |             final Class<?> DECODER_CLASS = Class.forName("com.mojang.serialization.Decoder"); | ||||||
|  |             final Class<?> DATA_RESULT_CLASS = Class.forName("com.mojang.serialization.DataResult"); | ||||||
|  |             final Class<?> DYNAMIC_OPS_CLASS = Class.forName("com.mojang.serialization.DynamicOps"); | ||||||
|  |             final Class<?> NBT_OPS_CLASS = Class.forName(Refraction.pickName( | ||||||
|  |                     "net.minecraft.nbt.NbtOps", | ||||||
|  |                     "net.minecraft.nbt.DynamicOpsNBT" | ||||||
|  |             )); | ||||||
|  |             SIGN_TEXT_DIRECT_CODEC = Arrays.stream(SIGN_TEXT_CLASS.getFields()) | ||||||
|  |                     .filter(field -> Modifier.isStatic(field.getModifiers()) && Modifier.isPublic(field.getModifiers())) | ||||||
|  |                     .filter(field -> field.getType() == CODEC_CLASS) | ||||||
|  |                     .findFirst().orElseThrow().get(null); | ||||||
|  |             DECODER_PARSE = LOOKUP.findVirtual( | ||||||
|  |                     DECODER_CLASS, "parse", MethodType.methodType( | ||||||
|  |                             DATA_RESULT_CLASS, DYNAMIC_OPS_CLASS, Object.class | ||||||
|  |                     ) | ||||||
|  |             ); | ||||||
|  |             NBT_OPS_INSTANCE = Arrays.stream(NBT_OPS_CLASS.getFields()) | ||||||
|  |                     .filter(field -> Modifier.isStatic(field.getModifiers()) && Modifier.isPublic(field.getModifiers())) | ||||||
|  |                     .filter(field -> field.getType() == NBT_OPS_CLASS) | ||||||
|  |                     .findFirst().orElseThrow().get(null); | ||||||
|  |             DATA_RESULT_RESULT = LOOKUP.findVirtual( | ||||||
|  |                     DATA_RESULT_CLASS, "result", | ||||||
|  |                     MethodType.methodType(Optional.class) | ||||||
|  |             ); | ||||||
|  |         } catch (Throwable e) { | ||||||
|  |             throw new RuntimeException("Failed to initialize required native method accessors", e); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean restore(final @NonNull Block block, final @NonNull CompoundTag data) { | ||||||
|  |         try { | ||||||
|  |             final BlockState blockState = block.getState(); | ||||||
|  |             if (!CRAFT_BLOCK_ENTITY_STATE_CLASS.isAssignableFrom(blockState.getClass())) { | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |             // get native tag | ||||||
|  |             Object nativeTag = PAPERWEIGHT_ADAPTER_FROM_NATIVE.invoke(ADAPTER, TO_LIN_TAG.invoke(data)); | ||||||
|  |             // load block entity data | ||||||
|  |             CRAFT_BLOCK_ENTITY_STATE_LOAD_DATA.invoke(blockState, nativeTag); | ||||||
|  |  | ||||||
|  |             // signs need to be handled explicitly (at least during worldgen) | ||||||
|  |             if (blockState instanceof Sign sign) { | ||||||
|  |                 if (data.getValue().get("front_text") instanceof CompoundTag textTag) { | ||||||
|  |                     setSignContents(true, sign.getSide(Side.FRONT), blockState, textTag); | ||||||
|  |                 } | ||||||
|  |                 if (data.getValue().get("back_text") instanceof CompoundTag textTag) { | ||||||
|  |                     setSignContents(false, sign.getSide(Side.BACK), blockState, textTag); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             CRAFT_BLOCK_ENTITY_STATE_UPDATE.invoke(blockState, FORCE_UPDATE_STATE, UPDATE_TRIGGER_PHYSICS); | ||||||
|  |         } catch (Throwable e) { | ||||||
|  |             LOGGER.error("Failed to update tile entity", e); | ||||||
|  |         } | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static void setSignContents(boolean front, SignSide side, BlockState blockState, CompoundTag data) throws Throwable { | ||||||
|  |         Object nativeTag = PAPERWEIGHT_ADAPTER_FROM_NATIVE.invoke(ADAPTER, TO_LIN_TAG.invoke(data)); | ||||||
|  |         Object dataResult = DECODER_PARSE.invoke(SIGN_TEXT_DIRECT_CODEC, NBT_OPS_INSTANCE, nativeTag); | ||||||
|  |         //noinspection rawtypes | ||||||
|  |         Object signText = ((Optional) DATA_RESULT_RESULT.invoke(dataResult)).orElseThrow(); | ||||||
|  |  | ||||||
|  |         // set the SignText on the underlying tile entity snapshot (SignBlockEntity) | ||||||
|  |         SIGN_BLOCK_ENTITY_SET_TEXT.invoke(CRAFT_BLOCK_ENTITY_STATE_GET_SNAPSHOT.invoke(blockState), signText, front); | ||||||
|  |         // and update the SignText field on the CraftSignSide - changes are otherwise not reflected | ||||||
|  |         CRAFT_SIGN_SIDE_SIGN_TEXT.set(side, signText); | ||||||
|  |  | ||||||
|  |         // reset cached lines to null, so it can be re-retrieved from SignText (for API access etc.) | ||||||
|  |         CRAFT_SIGN_SIDE_LINES.set(side, null); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Finds the {@code toLinTag} method on the {@code ToLinTag} interface, if lin-bus is available in the classpath. | ||||||
|  |      * <br /> | ||||||
|  |      * Required to access the underlying lin tag of the used JNBT tag by PlotSquared, so it can be converted into the platforms | ||||||
|  |      * native tag later. | ||||||
|  |      * | ||||||
|  |      * @return the MethodHandle for {@code toLinTag}, or {@code null} if lin-bus is not available in the classpath. | ||||||
|  |      * @throws ClassNotFoundException if the {@code ToLinTag} class could not be found. | ||||||
|  |      * @throws NoSuchMethodException  if no {@code toLinTag} method exists. | ||||||
|  |      * @throws IllegalAccessException shouldn't happen. | ||||||
|  |      */ | ||||||
|  |     private static MethodHandle findToLinTagMethodHandle() throws ClassNotFoundException, | ||||||
|  |             NoSuchMethodException, IllegalAccessException { | ||||||
|  |         return LOOKUP.findVirtual( | ||||||
|  |                 Class.forName("org.enginehub.linbus.tree.ToLinTag"), | ||||||
|  |                 "toLinTag", | ||||||
|  |                 MethodType.methodType(LIN_TAG_CLASS) | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Find the method (handle) to convert from native (= WE/FAWE) NBT tags to minecraft NBT tags. | ||||||
|  |      * <br /> | ||||||
|  |      * Depending on the used version of WE/FAWE, this differs: | ||||||
|  |      * <ul> | ||||||
|  |      *     <li>On WE versions post LinBus introduction: {@code fromNative(org.enginehub.linbus.tree.LinTag)}</li> | ||||||
|  |      *     <li>On FAWE versions post LinBus introduction: {@code fromNativeLin(org.enginehub.linbus.tree.LinTag)}</li> | ||||||
|  |      * </ul> | ||||||
|  |      * | ||||||
|  |      * @param adapterClass The bukkit adapter implementation class | ||||||
|  |      * @return the method. | ||||||
|  |      * @throws IllegalAccessException shouldn't happen as private lookup is used. | ||||||
|  |      * @throws NoSuchMethodException  if the method couldn't be found. | ||||||
|  |      */ | ||||||
|  |     private static MethodHandle findPaperweightAdapterFromNativeMethodHandle(Class<?> adapterClass) throws | ||||||
|  |             IllegalAccessException, NoSuchMethodException { | ||||||
|  |         final MethodHandles.Lookup lookup = MethodHandles.privateLookupIn(adapterClass, LOOKUP); | ||||||
|  |         try { | ||||||
|  |             // FAWE | ||||||
|  |             return lookup.findVirtual(adapterClass, "fromNativeLin", MethodType.methodType(Object.class, LIN_TAG_CLASS)); | ||||||
|  |         } catch (NoSuchMethodException e) { | ||||||
|  |             // WE | ||||||
|  |             return lookup.findVirtual(adapterClass, "fromNative", MethodType.methodType(Object.class, LIN_TAG_CLASS)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static MethodHandle findCraftBlockEntityStateLoadDataMethodHandle(Class<?> craftBlockEntityStateClass) throws | ||||||
|  |             NoSuchMethodException, IllegalAccessException { | ||||||
|  |         for (final Method method : craftBlockEntityStateClass.getMethods()) { | ||||||
|  |             if (method.getName().equals("loadData") && method.getParameterCount() == 1) { | ||||||
|  |                 return LOOKUP.unreflect(method); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         throw new NoSuchMethodException("Couldn't find #loadData(CompoundTag) in " + craftBlockEntityStateClass.getName()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static MethodHandle findCraftBlockEntityStateUpdateMethodHandle(Class<?> craftBlockEntityStateClass) throws | ||||||
|  |             NoSuchMethodException, IllegalAccessException { | ||||||
|  |         return LOOKUP.unreflect(ReflectionHelper.findMethod( | ||||||
|  |                 craftBlockEntityStateClass, | ||||||
|  |                 MethodType.methodType(Boolean.TYPE, Boolean.TYPE, Boolean.TYPE), | ||||||
|  |                 Modifier.PUBLIC | ||||||
|  |         ).orElseThrow(() -> new NoSuchMethodException("Couldn't lookup CraftBlockEntityState#update(boolean, boolean) boolean"))); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static MethodHandle findCraftBlockEntityStateSnapshotMethodHandle(Class<?> craftBlockEntityStateClass) throws | ||||||
|  |             IllegalAccessException, NoSuchMethodException { | ||||||
|  |         // doesn't seem to be obfuscated, but protected | ||||||
|  |         final MethodHandles.Lookup lookup = MethodHandles.privateLookupIn(craftBlockEntityStateClass, LOOKUP); | ||||||
|  |         return lookup.unreflect(craftBlockEntityStateClass.getDeclaredMethod("getSnapshot")); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static MethodHandle findSignBlockEntitySetTextMethodHandle(Class<?> signBlockEntity, Class<?> signText) throws | ||||||
|  |             NoSuchMethodException, IllegalAccessException { | ||||||
|  |         return LOOKUP.unreflect(ReflectionHelper.findMethod( | ||||||
|  |                 signBlockEntity, | ||||||
|  |                 MethodType.methodType(Boolean.TYPE, signText, Boolean.TYPE), | ||||||
|  |                 Modifier.PUBLIC | ||||||
|  |         ).orElseThrow(() -> new NoSuchMethodException("Couldn't lookup SignBlockEntity#setText(SignText, boolean) boolean"))); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -354,13 +354,17 @@ public class BukkitEntityUtil { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static boolean checkEntity(Entity entity, Plot plot) { |     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() |         if (plot == null || !plot.hasOwner() || plot.getFlags().isEmpty() && plot.getArea() | ||||||
|                 .getFlagContainer().getFlagMap().isEmpty()) { |                 .getFlagContainer().getFlagMap().isEmpty()) { | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         final com.sk89q.worldedit.world.entity.EntityType entityType = |         final com.sk89q.worldedit.world.entity.EntityType entityType = | ||||||
|                 BukkitAdapter.adapt(entity.getType()); |                 BukkitAdapter.adapt(type); | ||||||
|  |  | ||||||
|         if (EntityCategories.PLAYER.contains(entityType)) { |         if (EntityCategories.PLAYER.contains(entityType)) { | ||||||
|             return false; |             return false; | ||||||
|   | |||||||
| @@ -120,6 +120,14 @@ public interface PlotPlatform<P> extends LocaleHolder { | |||||||
|      */ |      */ | ||||||
|     @NonNull String serverImplementation(); |     @NonNull String serverImplementation(); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Gets the server brand name | ||||||
|  |      * | ||||||
|  |      * @return server brand | ||||||
|  |      * @since 7.5.3 | ||||||
|  |      */ | ||||||
|  |     @NonNull String serverBrand(); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Gets the native server code package prefix. |      * Gets the native server code package prefix. | ||||||
|      * |      * | ||||||
|   | |||||||
| @@ -18,6 +18,8 @@ | |||||||
|  */ |  */ | ||||||
| package com.plotsquared.core.backup; | 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 org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  |  | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
| @@ -30,12 +32,14 @@ import java.nio.file.Path; | |||||||
|  */ |  */ | ||||||
| public class Backup { | public class Backup { | ||||||
|  |  | ||||||
|  |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + Backup.class.getSimpleName()); | ||||||
|  |  | ||||||
|     private final BackupProfile owner; |     private final BackupProfile owner; | ||||||
|     private final long creationTime; |     private final long creationTime; | ||||||
|     @Nullable |     @Nullable | ||||||
|     private final Path file; |     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.owner = owner; | ||||||
|         this.creationTime = creationTime; |         this.creationTime = creationTime; | ||||||
|         this.file = file; |         this.file = file; | ||||||
| @@ -49,7 +53,7 @@ public class Backup { | |||||||
|             try { |             try { | ||||||
|                 Files.deleteIfExists(file); |                 Files.deleteIfExists(file); | ||||||
|             } catch (final IOException e) { |             } 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.Inject; | ||||||
| import com.google.inject.assistedinject.Assisted; | import com.google.inject.assistedinject.Assisted; | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | 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.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| import com.plotsquared.core.plot.schematic.Schematic; | import com.plotsquared.core.plot.schematic.Schematic; | ||||||
| import com.plotsquared.core.util.SchematicHandler; | import com.plotsquared.core.util.SchematicHandler; | ||||||
| import com.plotsquared.core.util.task.RunnableVal; | import com.plotsquared.core.util.task.RunnableVal; | ||||||
| import com.plotsquared.core.util.task.TaskManager; | 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.NonNull; | ||||||
| import org.checkerframework.checker.nullness.qual.Nullable; | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  |  | ||||||
| @@ -51,7 +52,7 @@ import java.util.concurrent.CompletableFuture; | |||||||
|  */ |  */ | ||||||
| public class PlayerBackupProfile implements BackupProfile { | 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 UUID owner; | ||||||
|     private final Plot plot; |     private final Plot plot; | ||||||
| @@ -87,7 +88,7 @@ public class PlayerBackupProfile implements BackupProfile { | |||||||
|                 Files.createDirectory(path); |                 Files.createDirectory(path); | ||||||
|             } |             } | ||||||
|         } catch (final Exception e) { |         } catch (final Exception e) { | ||||||
|             e.printStackTrace(); |             LOGGER.error("Error resolving {} from {}", child, parent, e); | ||||||
|         } |         } | ||||||
|         return path; |         return path; | ||||||
|     } |     } | ||||||
| @@ -104,7 +105,7 @@ public class PlayerBackupProfile implements BackupProfile { | |||||||
|                     try { |                     try { | ||||||
|                         Files.createDirectories(path); |                         Files.createDirectories(path); | ||||||
|                     } catch (IOException e) { |                     } catch (IOException e) { | ||||||
|                         e.printStackTrace(); |                         LOGGER.error("Error creating directory {}", path, e); | ||||||
|                         return Collections.emptyList(); |                         return Collections.emptyList(); | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
| @@ -117,11 +118,11 @@ public class PlayerBackupProfile implements BackupProfile { | |||||||
|                             backups.add( |                             backups.add( | ||||||
|                                     new Backup(this, basicFileAttributes.creationTime().toMillis(), file)); |                                     new Backup(this, basicFileAttributes.creationTime().toMillis(), file)); | ||||||
|                         } catch (IOException e) { |                         } catch (IOException e) { | ||||||
|                             e.printStackTrace(); |                             LOGGER.error("Error getting attributes for file {} to create backup", file, e); | ||||||
|                         } |                         } | ||||||
|                     }); |                     }); | ||||||
|                 } catch (IOException e) { |                 } catch (IOException e) { | ||||||
|                     e.printStackTrace(); |                     LOGGER.error("Error walking files from {}", path, e); | ||||||
|                 } |                 } | ||||||
|                 backups.sort(Comparator.comparingLong(Backup::getCreationTime).reversed()); |                 backups.sort(Comparator.comparingLong(Backup::getCreationTime).reversed()); | ||||||
|                 return (this.backupCache = backups); |                 return (this.backupCache = backups); | ||||||
| @@ -133,7 +134,7 @@ public class PlayerBackupProfile implements BackupProfile { | |||||||
|     public void destroy() { |     public void destroy() { | ||||||
|         this.listBackups().whenCompleteAsync((backups, error) -> { |         this.listBackups().whenCompleteAsync((backups, error) -> { | ||||||
|             if (error != null) { |             if (error != null) { | ||||||
|                 error.printStackTrace(); |                 LOGGER.error("Error while listing backups", error); | ||||||
|             } |             } | ||||||
|             backups.forEach(Backup::delete); |             backups.forEach(Backup::delete); | ||||||
|             this.backupCache = null; |             this.backupCache = null; | ||||||
| @@ -141,10 +142,12 @@ public class PlayerBackupProfile implements BackupProfile { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     public @NonNull Path getBackupDirectory() { |     public @NonNull Path getBackupDirectory() { | ||||||
|         return resolve(resolve( |         return resolve( | ||||||
|  |                 resolve( | ||||||
|                         resolve(backupManager.getBackupPath(), Objects.requireNonNull(plot.getArea().toString(), "plot area id")), |                         resolve(backupManager.getBackupPath(), Objects.requireNonNull(plot.getArea().toString(), "plot area id")), | ||||||
|                         Objects.requireNonNull(plot.getId().toDashSeparatedString(), "plot id") |                         Objects.requireNonNull(plot.getId().toDashSeparatedString(), "plot id") | ||||||
|         ), Objects.requireNonNull(owner.toString(), "owner")); |                 ), Objects.requireNonNull(owner.toString(), "owner") | ||||||
|  |         ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -156,7 +159,8 @@ public class PlayerBackupProfile implements BackupProfile { | |||||||
|                     backups.get(backups.size() - 1).delete(); |                     backups.get(backups.size() - 1).delete(); | ||||||
|                 } |                 } | ||||||
|                 final List<Plot> plots = Collections.singletonList(plot); |                 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(), () -> |                         "%world%-%id%-" + System.currentTimeMillis(), () -> | ||||||
|                                 future.complete(new Backup(this, System.currentTimeMillis(), null)) |                                 future.complete(new Backup(this, System.currentTimeMillis(), null)) | ||||||
|                 ); |                 ); | ||||||
| @@ -180,7 +184,7 @@ public class PlayerBackupProfile implements BackupProfile { | |||||||
|                 try { |                 try { | ||||||
|                     schematic = this.schematicHandler.getSchematic(backup.getFile().toFile()); |                     schematic = this.schematicHandler.getSchematic(backup.getFile().toFile()); | ||||||
|                 } catch (SchematicHandler.UnsupportedFormatException e) { |                 } catch (SchematicHandler.UnsupportedFormatException e) { | ||||||
|                     e.printStackTrace(); |                     LOGGER.error("Unsupported format for backup {}", backup.getFile(), e); | ||||||
|                 } |                 } | ||||||
|                 if (schematic == null) { |                 if (schematic == null) { | ||||||
|                     future.completeExceptionally(new IllegalArgumentException( |                     future.completeExceptionally(new IllegalArgumentException( | ||||||
| @@ -200,10 +204,9 @@ public class PlayerBackupProfile implements BackupProfile { | |||||||
|                                     if (value) { |                                     if (value) { | ||||||
|                                         future.complete(null); |                                         future.complete(null); | ||||||
|                                     } else { |                                     } else { | ||||||
|                                         future.completeExceptionally(new RuntimeException(MINI_MESSAGE.escapeTags( |                                         future.completeExceptionally(new PlotSquaredException( | ||||||
|                                                 TranslatableCaption |                                                 TranslatableCaption | ||||||
|                                                         .of("schematics.schematic_paste_failed") |                                                         .of("schematics.schematic_paste_failed"))); | ||||||
|                                                         .getComponent(ConsolePlayer.getConsole())))); |  | ||||||
|                                     } |                                     } | ||||||
|                                 } |                                 } | ||||||
|                             } |                             } | ||||||
|   | |||||||
| @@ -32,6 +32,8 @@ import com.plotsquared.core.util.task.TaskManager; | |||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | import net.kyori.adventure.text.minimessage.tag.Tag; | ||||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
|  | import org.apache.logging.log4j.LogManager; | ||||||
|  | import org.apache.logging.log4j.Logger; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
| import org.checkerframework.checker.nullness.qual.Nullable; | import org.checkerframework.checker.nullness.qual.Nullable; | ||||||
|  |  | ||||||
| @@ -47,6 +49,7 @@ import java.util.concurrent.TimeUnit; | |||||||
| @Singleton | @Singleton | ||||||
| public class SimpleBackupManager implements BackupManager { | public class SimpleBackupManager implements BackupManager { | ||||||
|  |  | ||||||
|  |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + SimpleBackupManager.class.getSimpleName()); | ||||||
|     private final Path backupPath; |     private final Path backupPath; | ||||||
|     private final boolean automaticBackup; |     private final boolean automaticBackup; | ||||||
|     private final int backupLimit; |     private final int backupLimit; | ||||||
| @@ -112,7 +115,12 @@ public class SimpleBackupManager implements BackupManager { | |||||||
|                                 TagResolver.resolver("reason", Tag.inserting(Component.text(throwable.getMessage()))) |                                 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 { |                 } else { | ||||||
|                     if (player != null) { |                     if (player != null) { | ||||||
|                         player.sendMessage(TranslatableCaption.of("backups.backup_automatic_finished")); |                         player.sendMessage(TranslatableCaption.of("backups.backup_automatic_finished")); | ||||||
| @@ -128,6 +136,7 @@ public class SimpleBackupManager implements BackupManager { | |||||||
|         return this.automaticBackup; |         return this.automaticBackup; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @NonNull | ||||||
|     public Path getBackupPath() { |     public Path getBackupPath() { | ||||||
|         return this.backupPath; |         return this.backupPath; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -21,7 +21,6 @@ package com.plotsquared.core.command; | |||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| @@ -58,8 +57,7 @@ public class Alias extends SubCommand { | |||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Location location = player.getLocation(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         Plot plot = location.getPlotAbs(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
|   | |||||||
| @@ -24,6 +24,7 @@ import com.plotsquared.core.backup.BackupProfile; | |||||||
| import com.plotsquared.core.backup.NullBackupProfile; | import com.plotsquared.core.backup.NullBackupProfile; | ||||||
| import com.plotsquared.core.backup.PlayerBackupProfile; | import com.plotsquared.core.backup.PlayerBackupProfile; | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
|  | import com.plotsquared.core.exception.PlotSquaredException; | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| @@ -32,6 +33,8 @@ import com.plotsquared.core.util.task.RunnableVal3; | |||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | import net.kyori.adventure.text.minimessage.tag.Tag; | ||||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
|  | import org.apache.logging.log4j.LogManager; | ||||||
|  | import org.apache.logging.log4j.Logger; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.nio.file.Files; | import java.nio.file.Files; | ||||||
| @@ -57,6 +60,8 @@ import java.util.stream.Stream; | |||||||
|         permission = "plots.backup") |         permission = "plots.backup") | ||||||
| public final class Backup extends Command { | public final class Backup extends Command { | ||||||
|  |  | ||||||
|  |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + Backup.class.getSimpleName()); | ||||||
|  |  | ||||||
|     private final BackupManager backupManager; |     private final BackupManager backupManager; | ||||||
|  |  | ||||||
|     @Inject |     @Inject | ||||||
| @@ -326,19 +331,28 @@ public final class Backup extends Command { | |||||||
|             if (backupProfile instanceof NullBackupProfile) { |             if (backupProfile instanceof NullBackupProfile) { | ||||||
|                 player.sendMessage( |                 player.sendMessage( | ||||||
|                         TranslatableCaption.of("backups.backup_impossible"), |                         TranslatableCaption.of("backups.backup_impossible"), | ||||||
|                         TagResolver.resolver("plot", Tag.inserting( |                         TagResolver.resolver( | ||||||
|  |                                 "plot", Tag.inserting( | ||||||
|                                         TranslatableCaption.of("generic.generic_other").toComponent(player) |                                         TranslatableCaption.of("generic.generic_other").toComponent(player) | ||||||
|                         )) |                                 ) | ||||||
|  |                         ) | ||||||
|                 ); |                 ); | ||||||
|             } else { |             } else { | ||||||
|                 backupProfile.listBackups().whenComplete((backups, throwable) -> { |                 backupProfile.listBackups().whenComplete((backups, throwable) -> { | ||||||
|                     if (throwable != null) { |                     if (throwable != null) { | ||||||
|  |                         Component reason; | ||||||
|  |                         if (throwable instanceof PlotSquaredException pe) { | ||||||
|  |                             reason = pe.getCaption().toComponent(player); | ||||||
|  |                         } else { | ||||||
|  |                             reason = Component.text(throwable.getMessage()); | ||||||
|  |                         } | ||||||
|                         player.sendMessage( |                         player.sendMessage( | ||||||
|                                 TranslatableCaption.of("backups.backup_load_failure"), |                                 TranslatableCaption.of("backups.backup_load_failure"), | ||||||
|                                 TagResolver.resolver("reason", Tag.inserting(Component.text(throwable.getMessage()))) |                                 TagResolver.resolver("reason", Tag.inserting(reason)) | ||||||
|                         ); |                         ); | ||||||
|                         throwable.printStackTrace(); |                         LOGGER.error("Error loading player ({}) backup", player.getName(), throwable); | ||||||
|                     } else { |                         return; | ||||||
|  |                     } | ||||||
|                     if (number < 1 || number > backups.size()) { |                     if (number < 1 || number > backups.size()) { | ||||||
|                         player.sendMessage( |                         player.sendMessage( | ||||||
|                                 TranslatableCaption.of("backups.backup_impossible"), |                                 TranslatableCaption.of("backups.backup_impossible"), | ||||||
| @@ -364,7 +378,8 @@ public final class Backup extends Command { | |||||||
|                                     ) |                                     ) | ||||||
|                             ); |                             ); | ||||||
|                         } else { |                         } else { | ||||||
|                                 CmdConfirm.addPending(player, "/plot backup load " + number, |                             CmdConfirm.addPending( | ||||||
|  |                                     player, "/plot backup load " + number, | ||||||
|                                     () -> backupProfile.restoreBackup(backup, player) |                                     () -> backupProfile.restoreBackup(backup, player) | ||||||
|                                             .whenComplete((n, error) -> { |                                             .whenComplete((n, error) -> { | ||||||
|                                                 if (error != null) { |                                                 if (error != null) { | ||||||
| @@ -382,7 +397,6 @@ public final class Backup extends Command { | |||||||
|                             ); |                             ); | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                     } |  | ||||||
|                 }); |                 }); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -20,6 +20,7 @@ package com.plotsquared.core.command; | |||||||
|  |  | ||||||
| import com.google.inject.Inject; | import com.google.inject.Inject; | ||||||
| import com.plotsquared.core.PlotSquared; | import com.plotsquared.core.PlotSquared; | ||||||
|  | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.events.PlayerBuyPlotEvent; | import com.plotsquared.core.events.PlayerBuyPlotEvent; | ||||||
| import com.plotsquared.core.events.Result; | import com.plotsquared.core.events.Result; | ||||||
| @@ -84,8 +85,9 @@ public class Buy extends Command { | |||||||
|         checkTrue(plot.hasOwner(), TranslatableCaption.of("info.plot_unowned")); |         checkTrue(plot.hasOwner(), TranslatableCaption.of("info.plot_unowned")); | ||||||
|         checkTrue(!plot.isOwner(player.getUUID()), TranslatableCaption.of("economy.cannot_buy_own")); |         checkTrue(!plot.isOwner(player.getUUID()), TranslatableCaption.of("economy.cannot_buy_own")); | ||||||
|         Set<Plot> plots = plot.getConnectedPlots(); |         Set<Plot> plots = plot.getConnectedPlots(); | ||||||
|  |         int plotCount = Settings.Limit.GLOBAL ? player.getPlotCount() : player.getPlotCount(plot.getWorldName()); | ||||||
|         checkTrue( |         checkTrue( | ||||||
|                 player.getPlotCount() + plots.size() <= player.getAllowedPlots(), |                 plotCount + plots.size() <= player.getAllowedPlots(), | ||||||
|                 TranslatableCaption.of("permission.cant_claim_more_plots"), |                 TranslatableCaption.of("permission.cant_claim_more_plots"), | ||||||
|                 TagResolver.resolver("amount", Tag.inserting(Component.text(player.getAllowedPlots()))) |                 TagResolver.resolver("amount", Tag.inserting(Component.text(player.getAllowedPlots()))) | ||||||
|         ); |         ); | ||||||
| @@ -144,6 +146,7 @@ public class Buy extends Command { | |||||||
|             plot.getPlotModificationManager().setSign(player.getName()); |             plot.getPlotModificationManager().setSign(player.getName()); | ||||||
|             player.sendMessage( |             player.sendMessage( | ||||||
|                     TranslatableCaption.of("working.claimed"), |                     TranslatableCaption.of("working.claimed"), | ||||||
|  |                     TagResolver.resolver("world", Tag.inserting(Component.text(plot.getArea().getWorldName()))), | ||||||
|                     TagResolver.resolver("plot", Tag.inserting(Component.text(plot.getId().toString()))) |                     TagResolver.resolver("plot", Tag.inserting(Component.text(plot.getId().toString()))) | ||||||
|             ); |             ); | ||||||
|             this.eventDispatcher.callPostPlayerBuyPlot(player, previousOwner, plot, price); |             this.eventDispatcher.callPostPlayerBuyPlot(player, previousOwner, plot, price); | ||||||
|   | |||||||
| @@ -26,7 +26,6 @@ import com.plotsquared.core.events.PlayerClaimPlotEvent; | |||||||
| import com.plotsquared.core.events.PlotMergeEvent; | import com.plotsquared.core.events.PlotMergeEvent; | ||||||
| import com.plotsquared.core.events.Result; | import com.plotsquared.core.events.Result; | ||||||
| import com.plotsquared.core.location.Direction; | import com.plotsquared.core.location.Direction; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.MetaDataAccess; | import com.plotsquared.core.player.MetaDataAccess; | ||||||
| import com.plotsquared.core.player.PlayerMetaDataKeys; | import com.plotsquared.core.player.PlayerMetaDataKeys; | ||||||
| @@ -72,8 +71,7 @@ public class Claim extends SubCommand { | |||||||
|         if (args.length >= 1) { |         if (args.length >= 1) { | ||||||
|             schematic = args[0]; |             schematic = args[0]; | ||||||
|         } |         } | ||||||
|         Location location = player.getLocation(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         Plot plot = location.getPlotAbs(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
| @@ -90,7 +88,7 @@ public class Claim extends SubCommand { | |||||||
|         boolean force = event.getEventResult() == Result.FORCE; |         boolean force = event.getEventResult() == Result.FORCE; | ||||||
|         int currentPlots = Settings.Limit.GLOBAL ? |         int currentPlots = Settings.Limit.GLOBAL ? | ||||||
|                 player.getPlotCount() : |                 player.getPlotCount() : | ||||||
|                 player.getPlotCount(location.getWorldName()); |                 player.getPlotCount(plot.getWorldName()); | ||||||
|  |  | ||||||
|         final PlotArea area = plot.getArea(); |         final PlotArea area = plot.getArea(); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -131,6 +131,7 @@ public class Clear extends Command { | |||||||
|                     player.sendMessage( |                     player.sendMessage( | ||||||
|                             TranslatableCaption.of("working.clearing_done"), |                             TranslatableCaption.of("working.clearing_done"), | ||||||
|                             TagResolver.builder() |                             TagResolver.builder() | ||||||
|  |                                     .tag("world", Tag.inserting(Component.text(plot.getArea().getWorldName()))) | ||||||
|                                     .tag("amount", Tag.inserting(Component.text(System.currentTimeMillis() - start))) |                                     .tag("amount", Tag.inserting(Component.text(System.currentTimeMillis() - start))) | ||||||
|                                     .tag("plot", Tag.inserting(Component.text(plot.getId().toString()))) |                                     .tag("plot", Tag.inserting(Component.text(plot.getId().toString()))) | ||||||
|                                     .build() |                                     .build() | ||||||
|   | |||||||
| @@ -24,6 +24,7 @@ import com.plotsquared.core.configuration.caption.TranslatableCaption; | |||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| import net.kyori.adventure.text.minimessage.MiniMessage; | import net.kyori.adventure.text.minimessage.MiniMessage; | ||||||
|  | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.jetbrains.annotations.NotNull; | ||||||
|  |  | ||||||
| @@ -98,6 +99,14 @@ public enum CommandCategory implements Caption { | |||||||
|         return MiniMessage.miniMessage().deserialize(getComponent(localeHolder)); |         return MiniMessage.miniMessage().deserialize(getComponent(localeHolder)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NonNull Component toComponent( | ||||||
|  |             @NonNull final LocaleHolder localeHolder, | ||||||
|  |             final @NonNull TagResolver @NonNull ... tagResolvers | ||||||
|  |     ) { | ||||||
|  |         return MiniMessage.miniMessage().deserialize(getComponent(localeHolder)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Checks if a player has access to this command category |      * Checks if a player has access to this command category | ||||||
|      * |      * | ||||||
|   | |||||||
| @@ -68,8 +68,8 @@ public class Continue extends SubCommand { | |||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|         int size = plot.getConnectedPlots().size(); |         int size = plot.getConnectedPlots().size(); | ||||||
|         if (!Settings.Done.COUNTS_TOWARDS_LIMIT && (player.getAllowedPlots() |         int plotCount = Settings.Limit.GLOBAL ? player.getPlotCount() : player.getPlotCount(plot.getWorldName()); | ||||||
|                 < player.getPlotCount() + size)) { |         if (!Settings.Done.COUNTS_TOWARDS_LIMIT && (player.getAllowedPlots() < plotCount + size)) { | ||||||
|             player.sendMessage( |             player.sendMessage( | ||||||
|                     TranslatableCaption.of("permission.cant_claim_more_plots"), |                     TranslatableCaption.of("permission.cant_claim_more_plots"), | ||||||
|                     TagResolver.resolver("amount", Tag.inserting(Component.text(player.getAllowedPlots()))) |                     TagResolver.resolver("amount", Tag.inserting(Component.text(player.getAllowedPlots()))) | ||||||
|   | |||||||
| @@ -19,7 +19,6 @@ | |||||||
| package com.plotsquared.core.command; | package com.plotsquared.core.command; | ||||||
|  |  | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| @@ -37,8 +36,7 @@ public class Copy extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(final PlotPlayer<?> player, String[] args) { |     public boolean onCommand(final PlotPlayer<?> player, String[] args) { | ||||||
|         Location location = player.getLocation(); |         Plot plot1 = player.getCurrentPlot(); | ||||||
|         Plot plot1 = location.getPlotAbs(); |  | ||||||
|         if (plot1 == null) { |         if (plot1 == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
|   | |||||||
| @@ -22,7 +22,6 @@ import com.google.inject.Inject; | |||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.generator.HybridPlotWorld; | import com.plotsquared.core.generator.HybridPlotWorld; | ||||||
| import com.plotsquared.core.generator.HybridUtils; | import com.plotsquared.core.generator.HybridUtils; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| @@ -47,8 +46,7 @@ public class CreateRoadSchematic extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|         Location location = player.getLocation(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         Plot plot = location.getPlotAbs(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
| @@ -57,7 +55,7 @@ public class CreateRoadSchematic extends SubCommand { | |||||||
|             player.sendMessage(TranslatableCaption.of("schematics.schematic_too_large")); |             player.sendMessage(TranslatableCaption.of("schematics.schematic_too_large")); | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|         if (!(location.getPlotArea() instanceof HybridPlotWorld)) { |         if (!(plot.getArea() instanceof HybridPlotWorld)) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); | ||||||
|         } |         } | ||||||
|         this.hybridUtils.setupRoadSchematic(plot); |         this.hybridUtils.setupRoadSchematic(plot); | ||||||
|   | |||||||
| @@ -86,7 +86,8 @@ public class DebugPaste extends SubCommand { | |||||||
|                 b.append("# WorldEdit implementation:\n"); |                 b.append("# WorldEdit implementation:\n"); | ||||||
|                 b.append(PlotSquared.platform().worldEditImplementations()).append("\n\n"); |                 b.append(PlotSquared.platform().worldEditImplementations()).append("\n\n"); | ||||||
|                 b.append("# Server Information\n"); |                 b.append("# Server Information\n"); | ||||||
|                 b.append("Server Version: ").append(PlotSquared.platform().serverImplementation()) |                 b.append("Server Version: ").append(PlotSquared.platform().serverBrand()).append(": ") | ||||||
|  |                         .append(PlotSquared.platform().serverImplementation()).append("\n") | ||||||
|                         .append("\n"); |                         .append("\n"); | ||||||
|                 b.append("online_mode: ").append(!Settings.UUID.OFFLINE).append(';') |                 b.append("online_mode: ").append(!Settings.UUID.OFFLINE).append(';') | ||||||
|                         .append(!Settings.UUID.OFFLINE).append('\n'); |                         .append(!Settings.UUID.OFFLINE).append('\n'); | ||||||
|   | |||||||
| @@ -22,7 +22,6 @@ import com.google.inject.Inject; | |||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.generator.HybridPlotManager; | import com.plotsquared.core.generator.HybridPlotManager; | ||||||
| import com.plotsquared.core.generator.HybridUtils; | import com.plotsquared.core.generator.HybridUtils; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| import com.plotsquared.core.plot.PlotArea; | import com.plotsquared.core.plot.PlotArea; | ||||||
| @@ -57,8 +56,7 @@ public class DebugRoadRegen extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|         Location location = player.getLocation(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         Plot plot = location.getPlotAbs(); |  | ||||||
|         if (args.length < 1) { |         if (args.length < 1) { | ||||||
|             player.sendMessage( |             player.sendMessage( | ||||||
|                     TranslatableCaption.of("commandconfig.command_syntax"), |                     TranslatableCaption.of("commandconfig.command_syntax"), | ||||||
| @@ -92,8 +90,7 @@ public class DebugRoadRegen extends SubCommand { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     public boolean regenPlot(PlotPlayer<?> player) { |     public boolean regenPlot(PlotPlayer<?> player) { | ||||||
|         Location location = player.getLocation(); |         PlotArea area = player.getContextualPlotArea(); | ||||||
|         PlotArea area = location.getPlotArea(); |  | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); | ||||||
|             return false; |             return false; | ||||||
| @@ -148,10 +145,10 @@ public class DebugRoadRegen extends SubCommand { | |||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Location location = player.getLocation(); |         PlotArea area = player.getContextualPlotArea(); | ||||||
|         PlotArea area = location.getPlotArea(); |  | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); | ||||||
|  |             return false; | ||||||
|         } |         } | ||||||
|         Plot plot = player.getCurrentPlot(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         PlotManager manager = area.getPlotManager(); |         PlotManager manager = area.getPlotManager(); | ||||||
|   | |||||||
| @@ -23,7 +23,6 @@ import com.plotsquared.core.configuration.Settings; | |||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.events.Result; | import com.plotsquared.core.events.Result; | ||||||
| import com.plotsquared.core.events.TeleportCause; | import com.plotsquared.core.events.TeleportCause; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| @@ -61,8 +60,7 @@ public class Delete extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(final PlotPlayer<?> player, String[] args) { |     public boolean onCommand(final PlotPlayer<?> player, String[] args) { | ||||||
|         Location location = player.getLocation(); |         final Plot plot = player.getCurrentPlot(); | ||||||
|         final Plot plot = location.getPlotAbs(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
| @@ -92,7 +90,7 @@ public class Delete extends SubCommand { | |||||||
|         final java.util.Set<Plot> plots = plot.getConnectedPlots(); |         final java.util.Set<Plot> plots = plot.getConnectedPlots(); | ||||||
|         final int currentPlots = Settings.Limit.GLOBAL ? |         final int currentPlots = Settings.Limit.GLOBAL ? | ||||||
|                 player.getPlotCount() : |                 player.getPlotCount() : | ||||||
|                 player.getPlotCount(location.getWorldName()); |                 player.getPlotCount(plot.getWorldName()); | ||||||
|         Runnable run = () -> { |         Runnable run = () -> { | ||||||
|             if (plot.getRunning() > 0) { |             if (plot.getRunning() > 0) { | ||||||
|                 player.sendMessage(TranslatableCaption.of("errors.wait_for_timer")); |                 player.sendMessage(TranslatableCaption.of("errors.wait_for_timer")); | ||||||
| @@ -124,6 +122,7 @@ public class Delete extends SubCommand { | |||||||
|                                 "amount", |                                 "amount", | ||||||
|                                 Tag.inserting(Component.text(String.valueOf(System.currentTimeMillis() - start))) |                                 Tag.inserting(Component.text(String.valueOf(System.currentTimeMillis() - start))) | ||||||
|                         ), |                         ), | ||||||
|  |                         TagResolver.resolver("world", Tag.inserting(Component.text(plotArea.getWorldName()))), | ||||||
|                         TagResolver.resolver("plot", Tag.inserting(Component.text(plot.getId().toString()))) |                         TagResolver.resolver("plot", Tag.inserting(Component.text(plot.getId().toString()))) | ||||||
|                 ); |                 ); | ||||||
|                 eventDispatcher.callPostDelete(plot); |                 eventDispatcher.callPostDelete(plot); | ||||||
|   | |||||||
| @@ -70,8 +70,7 @@ public class Deny extends SubCommand { | |||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|  |  | ||||||
|         Location location = player.getLocation(); |         final Plot plot = player.getCurrentPlot(); | ||||||
|         final Plot plot = location.getPlotAbs(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
|   | |||||||
| @@ -26,7 +26,6 @@ import com.plotsquared.core.events.PlotDoneEvent; | |||||||
| import com.plotsquared.core.events.PlotFlagAddEvent; | import com.plotsquared.core.events.PlotFlagAddEvent; | ||||||
| import com.plotsquared.core.events.Result; | import com.plotsquared.core.events.Result; | ||||||
| import com.plotsquared.core.generator.HybridUtils; | import com.plotsquared.core.generator.HybridUtils; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| @@ -61,8 +60,7 @@ public class Done extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(final PlotPlayer<?> player, String[] args) { |     public boolean onCommand(final PlotPlayer<?> player, String[] args) { | ||||||
|         Location location = player.getLocation(); |         final Plot plot = player.getCurrentPlot(); | ||||||
|         final Plot plot = location.getPlotAbs(); |  | ||||||
|         if ((plot == null) || !plot.hasOwner()) { |         if ((plot == null) || !plot.hasOwner()) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
|   | |||||||
| @@ -73,7 +73,7 @@ public class Download extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(final PlotPlayer<?> player, String[] args) { |     public boolean onCommand(final PlotPlayer<?> player, String[] args) { | ||||||
|         String world = player.getLocation().getWorldName(); |         String world = player.getCurrentPlot().getWorldName(); | ||||||
|         if (!this.plotAreaManager.hasPlotArea(world)) { |         if (!this.plotAreaManager.hasPlotArea(world)) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); | ||||||
|             return false; |             return false; | ||||||
|   | |||||||
| @@ -27,7 +27,6 @@ import com.plotsquared.core.configuration.caption.TranslatableCaption; | |||||||
| import com.plotsquared.core.events.PlotFlagAddEvent; | import com.plotsquared.core.events.PlotFlagAddEvent; | ||||||
| import com.plotsquared.core.events.PlotFlagRemoveEvent; | import com.plotsquared.core.events.PlotFlagRemoveEvent; | ||||||
| import com.plotsquared.core.events.Result; | import com.plotsquared.core.events.Result; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| @@ -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 |      * @return {@code true} if the player is allowed to modify the flags at their current location | ||||||
|      */ |      */ | ||||||
|     private static boolean checkRequirements(final @NonNull PlotPlayer<?> player) { |     private static boolean checkRequirements(final @NonNull PlotPlayer<?> player) { | ||||||
|         final Location location = player.getLocation(); |         final Plot plot = player.getCurrentPlot(); | ||||||
|         final Plot plot = location.getPlotAbs(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
| @@ -344,7 +342,7 @@ public final class FlagCommand extends Command { | |||||||
|         if (plotFlag == null) { |         if (plotFlag == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Plot plot = player.getLocation().getPlotAbs(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         PlotFlagAddEvent event = eventDispatcher.callFlagAdd(plotFlag, plot); |         PlotFlagAddEvent event = eventDispatcher.callFlagAdd(plotFlag, plot); | ||||||
|         if (event.getEventResult() == Result.DENY) { |         if (event.getEventResult() == Result.DENY) { | ||||||
|             player.sendMessage( |             player.sendMessage( | ||||||
| @@ -409,7 +407,7 @@ public final class FlagCommand extends Command { | |||||||
|         if (plotFlag == null) { |         if (plotFlag == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Plot plot = player.getLocation().getPlotAbs(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         PlotFlagAddEvent event = eventDispatcher.callFlagAdd(plotFlag, plot); |         PlotFlagAddEvent event = eventDispatcher.callFlagAdd(plotFlag, plot); | ||||||
|         if (event.getEventResult() == Result.DENY) { |         if (event.getEventResult() == Result.DENY) { | ||||||
|             player.sendMessage( |             player.sendMessage( | ||||||
| @@ -419,7 +417,7 @@ public final class FlagCommand extends Command { | |||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         boolean force = event.getEventResult() == Result.FORCE; |         boolean force = event.getEventResult() == Result.FORCE; | ||||||
|         final PlotFlag localFlag = player.getLocation().getPlotAbs().getFlagContainer() |         final PlotFlag localFlag = player.getCurrentPlot().getFlagContainer() | ||||||
|                 .getFlag(event.getFlag().getClass()); |                 .getFlag(event.getFlag().getClass()); | ||||||
|         if (!force) { |         if (!force) { | ||||||
|             for (String entry : args[1].split(",")) { |             for (String entry : args[1].split(",")) { | ||||||
| @@ -444,7 +442,7 @@ public final class FlagCommand extends Command { | |||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         boolean result = |         boolean result = | ||||||
|                 player.getLocation().getPlotAbs().setFlag(localFlag.merge(parsed.getValue())); |                 player.getCurrentPlot().setFlag(localFlag.merge(parsed.getValue())); | ||||||
|         if (!result) { |         if (!result) { | ||||||
|             player.sendMessage(TranslatableCaption.of("flag.flag_not_added")); |             player.sendMessage(TranslatableCaption.of("flag.flag_not_added")); | ||||||
|             return; |             return; | ||||||
| @@ -484,7 +482,7 @@ public final class FlagCommand extends Command { | |||||||
|         if (flag == null) { |         if (flag == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         final Plot plot = player.getLocation().getPlotAbs(); |         final Plot plot = player.getCurrentPlot(); | ||||||
|         final PlotFlag<?, ?> flagWithOldValue = plot.getFlagContainer().getFlag(flag.getClass()); |         final PlotFlag<?, ?> flagWithOldValue = plot.getFlagContainer().getFlag(flag.getClass()); | ||||||
|         PlotFlagRemoveEvent event = eventDispatcher.callFlagRemove(flag, plot); |         PlotFlagRemoveEvent event = eventDispatcher.callFlagRemove(flag, plot); | ||||||
|         if (event.getEventResult() == Result.DENY) { |         if (event.getEventResult() == Result.DENY) { | ||||||
| @@ -687,7 +685,7 @@ public final class FlagCommand extends Command { | |||||||
|                             .build() |                             .build() | ||||||
|             ); |             ); | ||||||
|             // Default value |             // Default value | ||||||
|             final String defaultValue = player.getLocation().getPlotArea().getFlagContainer() |             final String defaultValue = player.getCurrentPlot().getArea().getFlagContainer() | ||||||
|                     .getFlagErased(plotFlag.getClass()).toString(); |                     .getFlagErased(plotFlag.getClass()).toString(); | ||||||
|             player.sendMessage( |             player.sendMessage( | ||||||
|                     TranslatableCaption.of("flag.flag_info_default_value"), |                     TranslatableCaption.of("flag.flag_info_default_value"), | ||||||
|   | |||||||
| @@ -101,6 +101,10 @@ public class Grant extends Command { | |||||||
|                                     ); |                                     ); | ||||||
|                                 } else { |                                 } else { | ||||||
|                                     access.set(access.get().orElse(0) + 1); |                                     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 { |                         } else { | ||||||
| @@ -173,8 +177,14 @@ public class Grant extends Command { | |||||||
|                 commands.addAll(TabCompletions.completePlayers(player, args[0], Collections.emptyList())); |                 commands.addAll(TabCompletions.completePlayers(player, args[0], Collections.emptyList())); | ||||||
|             } |             } | ||||||
|             return commands; |             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(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -113,38 +113,34 @@ public class Help extends Command { | |||||||
|             } |             } | ||||||
|             if (cat == null && page == 0) { |             if (cat == null && page == 0) { | ||||||
|                 TextComponent.Builder builder = Component.text(); |                 TextComponent.Builder builder = Component.text(); | ||||||
|                 builder.append(MINI_MESSAGE.deserialize(TranslatableCaption.of("help.help_header").getComponent(player))); |                 builder.append(TranslatableCaption.of("help.help_header").toComponent(player)); | ||||||
|                 for (CommandCategory c : CommandCategory.values()) { |                 for (CommandCategory c : CommandCategory.values()) { | ||||||
|                     if (!c.canAccess(player)) { |                     if (!c.canAccess(player)) { | ||||||
|                         continue; |                         continue; | ||||||
|                     } |                     } | ||||||
|                     builder.append(Component.newline()).append(MINI_MESSAGE |                     builder.append(Component.newline()); | ||||||
|                             .deserialize( |                     builder.append(TranslatableCaption.of("help.help_info_item").toComponent( | ||||||
|                                     TranslatableCaption.of("help.help_info_item").getComponent(player), |                             player, TagResolver.builder() | ||||||
|                                     TagResolver.builder() |  | ||||||
|                                     .tag("command", Tag.inserting(Component.text("/plot help"))) |                                     .tag("command", Tag.inserting(Component.text("/plot help"))) | ||||||
|                                     .tag("category", Tag.inserting(Component.text(c.name().toLowerCase()))) |                                     .tag("category", Tag.inserting(Component.text(c.name().toLowerCase()))) | ||||||
|                                     .tag("category_desc", Tag.inserting(c.toComponent(player))) |                                     .tag("category_desc", Tag.inserting(c.toComponent(player))) | ||||||
|                                     .build() |                                     .build() | ||||||
|                     )); |                     )); | ||||||
|                 } |                 } | ||||||
|                 builder.append(Component.newline()).append(MINI_MESSAGE |                 builder.append(Component.newline()); | ||||||
|                         .deserialize( |                 builder.append(TranslatableCaption.of("help.help_info_item").toComponent( | ||||||
|                                 TranslatableCaption.of("help.help_info_item").getComponent(player), |                         player, TagResolver.builder() | ||||||
|                                 TagResolver.builder() |  | ||||||
|                                 .tag("command", Tag.inserting(Component.text("/plot help"))) |                                 .tag("command", Tag.inserting(Component.text("/plot help"))) | ||||||
|                                 .tag("category", Tag.inserting(Component.text("all"))) |                                 .tag("category", Tag.inserting(Component.text("all"))) | ||||||
|                                 .tag( |                                 .tag( | ||||||
|                                                 "category_desc", |                                         "category_desc", Tag.inserting(TranslatableCaption | ||||||
|                                                 Tag.inserting(TranslatableCaption |  | ||||||
|                                                 .of("help.help_display_all_commands") |                                                 .of("help.help_display_all_commands") | ||||||
|                                                 .toComponent(player)) |                                                 .toComponent(player)) | ||||||
|                                 ) |                                 ) | ||||||
|                                 .build() |                                 .build() | ||||||
|                 )); |                 )); | ||||||
|                 builder.append(Component.newline()).append(MINI_MESSAGE.deserialize(TranslatableCaption |                 builder.append(Component.newline()); | ||||||
|                         .of("help.help_footer") |                 builder.append(TranslatableCaption.of("help.help_footer").toComponent(player)); | ||||||
|                         .getComponent(player))); |  | ||||||
|                 player.sendMessage(StaticCaption.of(MINI_MESSAGE.serialize(builder.asComponent()))); |                 player.sendMessage(StaticCaption.of(MINI_MESSAGE.serialize(builder.asComponent()))); | ||||||
|                 return true; |                 return true; | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -20,7 +20,6 @@ package com.plotsquared.core.command; | |||||||
|  |  | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.configuration.caption.Caption; | 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.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.database.DBFunc; | import com.plotsquared.core.database.DBFunc; | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| @@ -131,13 +130,9 @@ public class Info extends SubCommand { | |||||||
|             info = getCaption(arg); |             info = getCaption(arg); | ||||||
|             if (info == null) { |             if (info == null) { | ||||||
|                 if (Settings.Ratings.USE_LIKES) { |                 if (Settings.Ratings.USE_LIKES) { | ||||||
|                     player.sendMessage(StaticCaption.of( |                     player.sendMessage(TranslatableCaption.of("info.plot_info_categories.use_likes")); | ||||||
|                             "&6Categories&7: &amembers&7, &aalias&7, &abiome&7, &aseen&7, &adenied&7, &aflags&7, &aid&7, &asize&7, &atrusted&7, " |  | ||||||
|                                     + "&aowner&7, " + " &alikes")); |  | ||||||
|                 } else { |                 } else { | ||||||
|                     player.sendMessage(StaticCaption.of( |                     player.sendMessage(TranslatableCaption.of("info.plot_info_categories.use_rating")); | ||||||
|                             "&6Categories&7: &amembers&7, &aalias&7, &abiome&7, &aseen&7, &adenied&7, &aflags&7, &aid&7, &asize&7, &atrusted&7, " |  | ||||||
|                                     + "&aowner&7, " + " &arating")); |  | ||||||
|                 } |                 } | ||||||
|                 return false; |                 return false; | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -65,8 +65,7 @@ public class Kick extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|         Location location = player.getLocation(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         Plot plot = location.getPlot(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
| @@ -124,7 +123,7 @@ public class Kick extends SubCommand { | |||||||
|                         ); |                         ); | ||||||
|                         return; |                         return; | ||||||
|                     } |                     } | ||||||
|                     Location spawn = this.worldUtil.getSpawn(location.getWorldName()); |                     Location spawn = this.worldUtil.getSpawn(plot.getWorldName()); | ||||||
|                     player2.sendMessage(TranslatableCaption.of("kick.you_got_kicked")); |                     player2.sendMessage(TranslatableCaption.of("kick.you_got_kicked")); | ||||||
|                     if (plot.equals(spawn.getPlot())) { |                     if (plot.equals(spawn.getPlot())) { | ||||||
|                         Location newSpawn = this.worldUtil.getSpawn(this.plotAreaManager.getAllWorlds()[0]); |                         Location newSpawn = this.worldUtil.getSpawn(this.plotAreaManager.getAllWorlds()[0]); | ||||||
| @@ -148,8 +147,7 @@ public class Kick extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public Collection<Command> tab(final PlotPlayer<?> player, final String[] args, final boolean space) { |     public Collection<Command> tab(final PlotPlayer<?> player, final String[] args, final boolean space) { | ||||||
|         Location location = player.getLocation(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         Plot plot = location.getPlotAbs(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             return Collections.emptyList(); |             return Collections.emptyList(); | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -150,8 +150,8 @@ public class ListCmd extends SubCommand { | |||||||
|             page = 0; |             page = 0; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         String world = player.getLocation().getWorldName(); |         PlotArea area = player.getContextualPlotArea(); | ||||||
|         PlotArea area = player.getApplicablePlotArea(); |         String world = area != null ? area.getWorldName() : ""; | ||||||
|         String arg = args[0].toLowerCase(); |         String arg = args[0].toLowerCase(); | ||||||
|         final boolean[] sort = new boolean[]{true}; |         final boolean[] sort = new boolean[]{true}; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -68,11 +68,6 @@ public class Load extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(final PlotPlayer<?> player, final String[] args) { |     public boolean onCommand(final PlotPlayer<?> player, final String[] args) { | ||||||
|         final String world = player.getLocation().getWorldName(); |  | ||||||
|         if (!this.plotAreaManager.hasPlotArea(world)) { |  | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot_world")); |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
|         final Plot plot = player.getCurrentPlot(); |         final Plot plot = player.getCurrentPlot(); | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|   | |||||||
| @@ -44,8 +44,13 @@ import org.apache.logging.log4j.Logger; | |||||||
| import java.util.Arrays; | import java.util.Arrays; | ||||||
| import java.util.LinkedList; | import java.util.LinkedList; | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  | import java.util.Objects; | ||||||
|  | import java.util.Optional; | ||||||
| import java.util.concurrent.CompletableFuture; | import java.util.concurrent.CompletableFuture; | ||||||
|  |  | ||||||
|  | import javax.annotation.Nonnull; | ||||||
|  | import javax.annotation.Nullable; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * PlotSquared command class. |  * PlotSquared command class. | ||||||
|  */ |  */ | ||||||
| @@ -147,8 +152,7 @@ public class MainCommand extends Command { | |||||||
|                 try { |                 try { | ||||||
|                     injector.getInstance(command); |                     injector.getInstance(command); | ||||||
|                 } catch (final Exception e) { |                 } catch (final Exception e) { | ||||||
|                     LOGGER.error("Failed to register command {}", command.getCanonicalName()); |                     LOGGER.error("Failed to register command {}", command.getCanonicalName(), e); | ||||||
|                     e.printStackTrace(); |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
| @@ -236,110 +240,179 @@ public class MainCommand extends Command { | |||||||
|             RunnableVal3<Command, Runnable, Runnable> confirm, |             RunnableVal3<Command, Runnable, Runnable> confirm, | ||||||
|             RunnableVal2<Command, CommandResult> whenDone |             RunnableVal2<Command, CommandResult> whenDone | ||||||
|     ) { |     ) { | ||||||
|         // Optional command scope // |         prepareArguments(new CommandExecutionData(player, args, confirm, whenDone, null)) | ||||||
|         Location location = null; |                 .thenCompose(executionData -> { | ||||||
|         Plot plot = null; |                     if (executionData.isEmpty()) { | ||||||
|         boolean tp = false; |  | ||||||
|         if (args.length >= 2) { |  | ||||||
|             PlotArea area = player.getApplicablePlotArea(); |  | ||||||
|             Plot newPlot = Plot.fromString(area, args[0]); |  | ||||||
|             if (newPlot != null && (player instanceof ConsolePlayer || newPlot.getArea() |  | ||||||
|                     .equals(area) || player.hasPermission(Permission.PERMISSION_ADMIN) |  | ||||||
|                     || player.hasPermission(Permission.PERMISSION_ADMIN_AREA_SUDO)) |  | ||||||
|                     && !newPlot.isDenied(player.getUUID())) { |  | ||||||
|                 final Location newLoc; |  | ||||||
|                 if (newPlot.getArea() instanceof SinglePlotArea) { |  | ||||||
|                     newLoc = newPlot.isLoaded() ? newPlot.getCenterSynchronous() : Location.at("", 0, 0, 0); |  | ||||||
|                 } else { |  | ||||||
|                     newLoc = newPlot.getCenterSynchronous(); |  | ||||||
|                 } |  | ||||||
|                 if (player.canTeleport(newLoc)) { |  | ||||||
|                     // Save meta |  | ||||||
|                     try (final MetaDataAccess<Location> locationMetaDataAccess |  | ||||||
|                                  = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LOCATION)) { |  | ||||||
|                         location = locationMetaDataAccess.get().orElse(null); |  | ||||||
|                         locationMetaDataAccess.set(newLoc); |  | ||||||
|                     } |  | ||||||
|                     try (final MetaDataAccess<Plot> plotMetaDataAccess |  | ||||||
|                                  = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { |  | ||||||
|                         plot = plotMetaDataAccess.get().orElse(null); |  | ||||||
|                         plotMetaDataAccess.set(newPlot); |  | ||||||
|                     } |  | ||||||
|                     tp = true; |  | ||||||
|                 } else { |  | ||||||
|                     player.sendMessage(TranslatableCaption.of("border.denied")); |  | ||||||
|                         return CompletableFuture.completedFuture(false); |                         return CompletableFuture.completedFuture(false); | ||||||
|                     } |                     } | ||||||
|                 // Trim command |                     var data = executionData.get(); | ||||||
|                 args = Arrays.copyOfRange(args, 1, args.length); |                     try { | ||||||
|  |                         return super.execute(data.player(), data.args(), data.confirm(), data.whenDone()); | ||||||
|  |                     } catch (CommandException e) { | ||||||
|  |                         throw e; | ||||||
|  |                     } catch (Throwable e) { | ||||||
|  |                         LOGGER.error("A error occurred while executing plot command", e); | ||||||
|  |                         String message = e.getMessage(); | ||||||
|  |                         if (message != null) { | ||||||
|  |                             data.player().sendMessage( | ||||||
|  |                                     TranslatableCaption.of("errors.error"), | ||||||
|  |                                     TagResolver.resolver("value", Tag.inserting(Component.text(message))) | ||||||
|  |                             ); | ||||||
|  |                         } else { | ||||||
|  |                             data.player().sendMessage( | ||||||
|  |                                     TranslatableCaption.of("errors.error_console")); | ||||||
|                         } |                         } | ||||||
|             if (args.length >= 2 && !args[0].isEmpty() && args[0].charAt(0) == '-') { |                     } finally { | ||||||
|                 if ("f".equals(args[0].substring(1))) { |                         if (data.postCommandData() != null) { | ||||||
|                     confirm = new RunnableVal3<>() { |                             resetCommandScope(data.player(), data.postCommandData()); | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                     return CompletableFuture.completedFuture(true); | ||||||
|  |                 }); | ||||||
|  |         return CompletableFuture.completedFuture(true); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private CompletableFuture<Optional<CommandExecutionData>> prepareArguments(CommandExecutionData data) { | ||||||
|  |         if (data.args().length >= 2) { | ||||||
|  |             PlotArea area = data.player().getApplicablePlotArea(); | ||||||
|  |             Plot newPlot = Plot.fromString(area, data.args()[0], data.player()); | ||||||
|  |             return preparePlotArgument(newPlot, data, area) | ||||||
|  |                     .thenApply(d -> d.flatMap(x -> prepareFlagArgument(x, area))); | ||||||
|  |         } else { | ||||||
|  |             return CompletableFuture.completedFuture(Optional.of(data)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private CompletableFuture<Optional<CommandExecutionData>> preparePlotArgument( | ||||||
|  |             @Nullable Plot newPlot, | ||||||
|  |             @Nonnull CommandExecutionData data, | ||||||
|  |             @Nullable PlotArea area | ||||||
|  |     ) { | ||||||
|  |         if (newPlot == null) { | ||||||
|  |             return CompletableFuture.completedFuture(Optional.of(data)); | ||||||
|  |         } | ||||||
|  |         final PlotPlayer<?> player = data.player(); | ||||||
|  |         final boolean isAdmin = player instanceof ConsolePlayer || player.hasPermission(Permission.PERMISSION_ADMIN); | ||||||
|  |         final boolean isDenied = newPlot.isDenied(player.getUUID()); | ||||||
|  |         if (!isAdmin) { | ||||||
|  |             if (isDenied) { | ||||||
|  |                 throw new CommandException(TranslatableCaption.of("deny.cannot_interact")); | ||||||
|  |             } | ||||||
|  |             if (area != null && area.equals(newPlot.getArea()) && !player.hasPermission(Permission.PERMISSION_ADMIN_AREA_SUDO)) { | ||||||
|  |                 return CompletableFuture.completedFuture(Optional.of(data)); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return fetchPlotCenterLocation(newPlot) | ||||||
|  |                 .thenApply(newLoc -> { | ||||||
|  |                     if (!player.canTeleport(newLoc)) { | ||||||
|  |                         player.sendMessage(TranslatableCaption.of("border.denied")); | ||||||
|  |                         return Optional.empty(); | ||||||
|  |                     } | ||||||
|  |                     // Save meta | ||||||
|  |                     var originalCommandMeta = setCommandScope(player, new TemporaryCommandMeta(newLoc, newPlot)); | ||||||
|  |                     return Optional.of(new CommandExecutionData( | ||||||
|  |                             player, | ||||||
|  |                             Arrays.copyOfRange(data.args(), 1, data.args().length), // Trimmed command | ||||||
|  |                             data.confirm(), | ||||||
|  |                             data.whenDone(), | ||||||
|  |                             originalCommandMeta | ||||||
|  |                     )); | ||||||
|  |                 }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private Optional<CommandExecutionData> prepareFlagArgument(@Nonnull CommandExecutionData data, @Nonnull PlotArea area) { | ||||||
|  |         if (data.args().length >= 2 && !data.args()[0].isEmpty() && data.args()[0].charAt(0) == '-') { | ||||||
|  |             if ("f".equals(data.args()[0].substring(1))) { | ||||||
|  |                 return Optional.of(new CommandExecutionData( | ||||||
|  |                         data.player(), | ||||||
|  |                         Arrays.copyOfRange(data.args(), 1, data.args().length), // Trimmed command | ||||||
|  |                         createForcedConfirmation(data.player(), area), | ||||||
|  |                         data.whenDone(), | ||||||
|  |                         data.postCommandData() | ||||||
|  |                 )); | ||||||
|  |             } else { | ||||||
|  |                 data.player().sendMessage(TranslatableCaption.of("errors.invalid_command_flag")); | ||||||
|  |                 return Optional.empty(); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return Optional.of(data); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private CompletableFuture<Location> fetchPlotCenterLocation(Plot plot) { | ||||||
|  |         if (plot.getArea() instanceof SinglePlotArea && !plot.isLoaded()) { | ||||||
|  |             return CompletableFuture.completedFuture(Location.at("", 0, 0, 0)); | ||||||
|  |         } | ||||||
|  |         CompletableFuture<Location> future = new CompletableFuture<>(); | ||||||
|  |         plot.getCenter(future::complete); | ||||||
|  |         return future; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private @Nonnull RunnableVal3<Command, Runnable, Runnable> createForcedConfirmation(@Nonnull PlotPlayer<?> player, | ||||||
|  |                                                                                         @Nullable PlotArea area) { | ||||||
|  |         return new RunnableVal3<>() { | ||||||
|             @Override |             @Override | ||||||
|             public void run(Command cmd, Runnable success, Runnable failure) { |             public void run(Command cmd, Runnable success, Runnable failure) { | ||||||
|                             if (area != null && PlotSquared.platform().econHandler().isEnabled(area)) { |                 if (area != null && PlotSquared.platform().econHandler().isEnabled(area) | ||||||
|                                 PlotExpression priceEval = |                         && Optional.of(area.getPrices().get(cmd.getFullId())) | ||||||
|                                         area.getPrices().get(cmd.getFullId()); |                         .map(priceEval -> priceEval.evaluate(0d)) | ||||||
|                                 double price = priceEval != null ? priceEval.evaluate(0d) : 0d; |                         .filter(price -> price != 0d) | ||||||
|                                 if (price != 0d |                         .filter(price -> PlotSquared.platform().econHandler().getMoney(player) < price) | ||||||
|                                         && PlotSquared.platform().econHandler().getMoney(player) < price) { |                         .isPresent()) { | ||||||
|                     if (failure != null) { |                     if (failure != null) { | ||||||
|                         failure.run(); |                         failure.run(); | ||||||
|                     } |                     } | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|                             } |  | ||||||
|                 if (success != null) { |                 if (success != null) { | ||||||
|                     success.run(); |                     success.run(); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }; |         }; | ||||||
|                     args = Arrays.copyOfRange(args, 1, args.length); |  | ||||||
|                 } else { |  | ||||||
|                     player.sendMessage(TranslatableCaption.of("errors.invalid_command_flag")); |  | ||||||
|                     return CompletableFuture.completedFuture(false); |  | ||||||
|     } |     } | ||||||
|             } |  | ||||||
|         } |     private @Nonnull TemporaryCommandMeta setCommandScope(@Nonnull PlotPlayer<?> player, @Nonnull TemporaryCommandMeta commandMeta) { | ||||||
|         try { |         Objects.requireNonNull(commandMeta.location()); | ||||||
|             super.execute(player, args, confirm, whenDone); |         Objects.requireNonNull(commandMeta.plot()); | ||||||
|         } catch (CommandException e) { |         Location location; | ||||||
|             throw e; |         Plot plot; | ||||||
|         } catch (Throwable e) { |  | ||||||
|             e.printStackTrace(); |  | ||||||
|             String message = e.getMessage(); |  | ||||||
|             if (message != null) { |  | ||||||
|                 player.sendMessage( |  | ||||||
|                         TranslatableCaption.of("errors.error"), |  | ||||||
|                         TagResolver.resolver("value", Tag.inserting(Component.text(message))) |  | ||||||
|                 ); |  | ||||||
|             } else { |  | ||||||
|                 player.sendMessage( |  | ||||||
|                         TranslatableCaption.of("errors.error_console")); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         // Reset command scope // |  | ||||||
|         if (tp && !(player instanceof ConsolePlayer)) { |  | ||||||
|         try (final MetaDataAccess<Location> locationMetaDataAccess |         try (final MetaDataAccess<Location> locationMetaDataAccess | ||||||
|                      = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LOCATION)) { |                      = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LOCATION)) { | ||||||
|                 if (location == null) { |             location = locationMetaDataAccess.get().orElse(null); | ||||||
|  |             locationMetaDataAccess.set(commandMeta.location()); | ||||||
|  |         } | ||||||
|  |         try (final MetaDataAccess<Plot> plotMetaDataAccess | ||||||
|  |                      = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { | ||||||
|  |             plot = plotMetaDataAccess.get().orElse(null); | ||||||
|  |             plotMetaDataAccess.set(commandMeta.plot()); | ||||||
|  |         } | ||||||
|  |         return new TemporaryCommandMeta(location, plot); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void resetCommandScope(@Nonnull PlotPlayer<?> player, @Nonnull TemporaryCommandMeta commandMeta) { | ||||||
|  |         try (final MetaDataAccess<Location> locationMetaDataAccess | ||||||
|  |                      = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LOCATION)) { | ||||||
|  |             if (commandMeta.location() == null) { | ||||||
|                 locationMetaDataAccess.remove(); |                 locationMetaDataAccess.remove(); | ||||||
|             } else { |             } else { | ||||||
|                     locationMetaDataAccess.set(location); |                 locationMetaDataAccess.set(commandMeta.location()); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         try (final MetaDataAccess<Plot> plotMetaDataAccess |         try (final MetaDataAccess<Plot> plotMetaDataAccess | ||||||
|                      = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { |                      = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { | ||||||
|                 if (plot == null) { |             if (commandMeta.plot() == null) { | ||||||
|                 plotMetaDataAccess.remove(); |                 plotMetaDataAccess.remove(); | ||||||
|             } else { |             } else { | ||||||
|                     plotMetaDataAccess.set(plot); |                 plotMetaDataAccess.set(commandMeta.plot()); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|         return CompletableFuture.completedFuture(true); |  | ||||||
|     } |     private record CommandExecutionData(@Nonnull PlotPlayer<?> player, @Nonnull String[] args, | ||||||
|  |                     @Nonnull RunnableVal3<Command, Runnable, Runnable> confirm, | ||||||
|  |                     @Nonnull RunnableVal2<Command, CommandResult> whenDone, | ||||||
|  |                     @Nullable TemporaryCommandMeta postCommandData) {} | ||||||
|  |  | ||||||
|  |     private record TemporaryCommandMeta(@Nullable Location location, @Nullable Plot plot) {} | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean canExecute(PlotPlayer<?> player, boolean message) { |     public boolean canExecute(PlotPlayer<?> player, boolean message) { | ||||||
|   | |||||||
| @@ -20,7 +20,6 @@ package com.plotsquared.core.command; | |||||||
|  |  | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.events.TeleportCause; | import com.plotsquared.core.events.TeleportCause; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
|  |  | ||||||
| @@ -36,8 +35,7 @@ public class Middle extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] arguments) { |     public boolean onCommand(PlotPlayer<?> player, String[] arguments) { | ||||||
|         Location location = player.getLocation(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         Plot plot = location.getPlot(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
|   | |||||||
| @@ -20,7 +20,6 @@ package com.plotsquared.core.command; | |||||||
|  |  | ||||||
| import com.google.inject.Inject; | import com.google.inject.Inject; | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| @@ -55,8 +54,7 @@ public class Move extends SubCommand { | |||||||
|             RunnableVal3<Command, Runnable, Runnable> confirm, |             RunnableVal3<Command, Runnable, Runnable> confirm, | ||||||
|             RunnableVal2<Command, CommandResult> whenDone |             RunnableVal2<Command, CommandResult> whenDone | ||||||
|     ) { |     ) { | ||||||
|         Location location = player.getLocation(); |         Plot plot1 = player.getCurrentPlot(); | ||||||
|         Plot plot1 = location.getPlotAbs(); |  | ||||||
|         if (plot1 == null) { |         if (plot1 == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return CompletableFuture.completedFuture(false); |             return CompletableFuture.completedFuture(false); | ||||||
|   | |||||||
| @@ -23,7 +23,6 @@ import com.plotsquared.core.configuration.caption.TranslatableCaption; | |||||||
| import com.plotsquared.core.events.PlotFlagAddEvent; | import com.plotsquared.core.events.PlotFlagAddEvent; | ||||||
| import com.plotsquared.core.events.PlotFlagRemoveEvent; | import com.plotsquared.core.events.PlotFlagRemoveEvent; | ||||||
| import com.plotsquared.core.events.Result; | import com.plotsquared.core.events.Result; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| @@ -57,7 +56,7 @@ public class Music extends SubCommand { | |||||||
|                     "music_disc_far", "music_disc_mall", "music_disc_mellohi", "music_disc_stal", |                     "music_disc_far", "music_disc_mall", "music_disc_mellohi", "music_disc_stal", | ||||||
|                     "music_disc_strad", "music_disc_ward", "music_disc_11", "music_disc_wait", "music_disc_otherside", |                     "music_disc_strad", "music_disc_ward", "music_disc_11", "music_disc_wait", "music_disc_otherside", | ||||||
|                     "music_disc_pigstep", "music_disc_5", "music_disc_relic", "music_disc_creator", |                     "music_disc_pigstep", "music_disc_5", "music_disc_relic", "music_disc_creator", | ||||||
|                     "music_disc_creator_music_box", "music_disc_precipice" |                     "music_disc_creator_music_box", "music_disc_precipice", "music_disc_tears", "music_disc_lava_chicken" | ||||||
|             ); |             ); | ||||||
|  |  | ||||||
|     // make sure all discs and the bedrock ("cancel") fit into the inventory |     // make sure all discs and the bedrock ("cancel") fit into the inventory | ||||||
| @@ -74,8 +73,7 @@ public class Music extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|         Location location = player.getLocation(); |         final Plot plot = player.getCurrentPlot(); | ||||||
|         final Plot plot = location.getPlotAbs(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
|   | |||||||
| @@ -21,7 +21,6 @@ package com.plotsquared.core.command; | |||||||
| import com.google.inject.Inject; | import com.google.inject.Inject; | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.database.DBFunc; | import com.plotsquared.core.database.DBFunc; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| @@ -56,8 +55,7 @@ public class Remove extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|         Location location = player.getLocation(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         Plot plot = location.getPlotAbs(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
| @@ -132,8 +130,7 @@ public class Remove extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public Collection<Command> tab(final PlotPlayer<?> player, final String[] args, final boolean space) { |     public Collection<Command> tab(final PlotPlayer<?> player, final String[] args, final boolean space) { | ||||||
|         Location location = player.getLocation(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         Plot plot = location.getPlotAbs(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             return Collections.emptyList(); |             return Collections.emptyList(); | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -22,7 +22,6 @@ import com.google.common.collect.Lists; | |||||||
| import com.google.inject.Inject; | import com.google.inject.Inject; | ||||||
| import com.plotsquared.core.configuration.Settings; | import com.plotsquared.core.configuration.Settings; | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.ConsolePlayer; | import com.plotsquared.core.player.ConsolePlayer; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| @@ -102,8 +101,7 @@ public class SchematicCmd extends SubCommand { | |||||||
|                     ); |                     ); | ||||||
|                     break; |                     break; | ||||||
|                 } |                 } | ||||||
|                 Location loc = player.getLocation(); |                 final Plot plot = player.getCurrentPlot(); | ||||||
|                 final Plot plot = loc.getPlotAbs(); |  | ||||||
|                 if (plot == null) { |                 if (plot == null) { | ||||||
|                     player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |                     player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|                     return false; |                     return false; | ||||||
| @@ -247,8 +245,7 @@ public class SchematicCmd extends SubCommand { | |||||||
|                     player.sendMessage(TranslatableCaption.of("error.task_in_process")); |                     player.sendMessage(TranslatableCaption.of("error.task_in_process")); | ||||||
|                     return false; |                     return false; | ||||||
|                 } |                 } | ||||||
|                 Location location = player.getLocation(); |                 Plot plot = player.getCurrentPlot(); | ||||||
|                 Plot plot = location.getPlotAbs(); |  | ||||||
|                 if (plot == null) { |                 if (plot == null) { | ||||||
|                     player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |                     player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|                     return false; |                     return false; | ||||||
|   | |||||||
| @@ -78,7 +78,7 @@ public class Set extends SubCommand { | |||||||
|  |  | ||||||
|             @Override |             @Override | ||||||
|             public boolean set(PlotPlayer<?> player, final Plot plot, String value) { |             public boolean set(PlotPlayer<?> player, final Plot plot, String value) { | ||||||
|                 final PlotArea plotArea = player.getLocation().getPlotArea(); |                 final PlotArea plotArea = player.getContextualPlotArea(); | ||||||
|                 if (plotArea == null) { |                 if (plotArea == null) { | ||||||
|                     return false; |                     return false; | ||||||
|                 } |                 } | ||||||
|   | |||||||
| @@ -19,7 +19,6 @@ | |||||||
| package com.plotsquared.core.command; | package com.plotsquared.core.command; | ||||||
|  |  | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| @@ -32,8 +31,7 @@ public abstract class SetCommand extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|         Location location = player.getLocation(); |         Plot plot = player.getCurrentPlot(); | ||||||
|         Plot plot = location.getPlotAbs(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
|   | |||||||
| @@ -19,7 +19,6 @@ | |||||||
| package com.plotsquared.core.command; | package com.plotsquared.core.command; | ||||||
|  |  | ||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| @@ -44,8 +43,7 @@ public class Swap extends SubCommand { | |||||||
|             RunnableVal3<Command, Runnable, Runnable> confirm, |             RunnableVal3<Command, Runnable, Runnable> confirm, | ||||||
|             RunnableVal2<Command, CommandResult> whenDone |             RunnableVal2<Command, CommandResult> whenDone | ||||||
|     ) { |     ) { | ||||||
|         Location location = player.getLocation(); |         Plot plot1 = player.getCurrentPlot(); | ||||||
|         Plot plot1 = location.getPlotAbs(); |  | ||||||
|         if (plot1 == null) { |         if (plot1 == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return CompletableFuture.completedFuture(false); |             return CompletableFuture.completedFuture(false); | ||||||
| @@ -79,8 +77,10 @@ public class Swap extends SubCommand { | |||||||
|         String p1 = plot1.toString(); |         String p1 = plot1.toString(); | ||||||
|         String p2 = plot2.toString(); |         String p2 = plot2.toString(); | ||||||
|  |  | ||||||
|         return plot1.getPlotModificationManager().move(plot2, player, () -> { |         return plot1.getPlotModificationManager().move( | ||||||
|         }, true).thenApply(result -> { |                 plot2, player, () -> { | ||||||
|  |                 }, true | ||||||
|  |         ).thenApply(result -> { | ||||||
|             if (result) { |             if (result) { | ||||||
|                 player.sendMessage( |                 player.sendMessage( | ||||||
|                         TranslatableCaption.of("swap.swap_success"), |                         TranslatableCaption.of("swap.swap_success"), | ||||||
|   | |||||||
| @@ -22,7 +22,6 @@ import com.google.inject.Inject; | |||||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||||
| import com.plotsquared.core.events.PlotUnlinkEvent; | import com.plotsquared.core.events.PlotUnlinkEvent; | ||||||
| import com.plotsquared.core.events.Result; | import com.plotsquared.core.events.Result; | ||||||
| import com.plotsquared.core.location.Location; |  | ||||||
| import com.plotsquared.core.permissions.Permission; | import com.plotsquared.core.permissions.Permission; | ||||||
| import com.plotsquared.core.player.PlotPlayer; | import com.plotsquared.core.player.PlotPlayer; | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
| @@ -51,8 +50,7 @@ public class Unlink extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(final PlotPlayer<?> player, String[] args) { |     public boolean onCommand(final PlotPlayer<?> player, String[] args) { | ||||||
|         Location location = player.getLocation(); |         final Plot plot = player.getCurrentPlot(); | ||||||
|         final Plot plot = location.getPlotAbs(); |  | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); |             player.sendMessage(TranslatableCaption.of("errors.not_in_plot")); | ||||||
|             return false; |             return false; | ||||||
|   | |||||||
| @@ -20,6 +20,7 @@ package com.plotsquared.core.configuration.caption; | |||||||
|  |  | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| import net.kyori.adventure.text.ComponentLike; | import net.kyori.adventure.text.ComponentLike; | ||||||
|  | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -44,6 +45,16 @@ public interface Caption { | |||||||
|      */ |      */ | ||||||
|     @NonNull Component toComponent(@NonNull LocaleHolder localeHolder); |     @NonNull Component toComponent(@NonNull LocaleHolder localeHolder); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get the Adventure {@link ComponentLike} for this caption while applying custom {@link TagResolver} | ||||||
|  |      * (apart from the default {@code core.prefix}) | ||||||
|  |      * @param localeHolder Local holder | ||||||
|  |      * @param tagResolvers custom tag resolvers to replace placeholders / parameters | ||||||
|  |      * @return {@link ComponentLike} | ||||||
|  |      * @since 7.5.4 | ||||||
|  |      */ | ||||||
|  |     @NonNull Component toComponent(@NonNull LocaleHolder localeHolder, @NonNull TagResolver @NonNull... tagResolvers); | ||||||
|  |  | ||||||
|     @NonNull String toString(); |     @NonNull String toString(); | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -21,6 +21,7 @@ package com.plotsquared.core.configuration.caption; | |||||||
| import com.google.common.base.Preconditions; | import com.google.common.base.Preconditions; | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| import net.kyori.adventure.text.minimessage.MiniMessage; | import net.kyori.adventure.text.minimessage.MiniMessage; | ||||||
|  | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| public final class StaticCaption implements Caption { | public final class StaticCaption implements Caption { | ||||||
| @@ -51,6 +52,14 @@ public final class StaticCaption implements Caption { | |||||||
|         return MiniMessage.miniMessage().deserialize(this.value); |         return MiniMessage.miniMessage().deserialize(this.value); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NonNull Component toComponent( | ||||||
|  |             @NonNull final LocaleHolder localeHolder, | ||||||
|  |             final @NonNull TagResolver @NonNull ... tagResolvers | ||||||
|  |     ) { | ||||||
|  |         return MiniMessage.miniMessage().deserialize(this.value, tagResolvers); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public @NonNull String toString() { |     public @NonNull String toString() { | ||||||
|         return "StaticCaption(" + value + ")"; |         return "StaticCaption(" + value + ")"; | ||||||
|   | |||||||
| @@ -27,6 +27,7 @@ import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | |||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.jetbrains.annotations.NotNull; | ||||||
|  |  | ||||||
|  | import java.util.Arrays; | ||||||
| import java.util.Locale; | import java.util.Locale; | ||||||
| import java.util.regex.Pattern; | import java.util.regex.Pattern; | ||||||
|  |  | ||||||
| @@ -96,13 +97,23 @@ public final class TranslatableCaption implements NamespacedCaption { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public @NonNull Component toComponent(@NonNull final LocaleHolder localeHolder) { |     public @NonNull Component toComponent(@NonNull final LocaleHolder localeHolder) { | ||||||
|  |         return this.toComponent(localeHolder, new TagResolver[0]); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public @NonNull Component toComponent( | ||||||
|  |             @NonNull final LocaleHolder localeHolder, | ||||||
|  |             final @NonNull TagResolver @NonNull ... tagResolvers | ||||||
|  |     ) { | ||||||
|         if (getKey().equals("core.prefix")) { |         if (getKey().equals("core.prefix")) { | ||||||
|             return MiniMessage.miniMessage().deserialize(getComponent(localeHolder)); |             return MiniMessage.miniMessage().deserialize(getComponent(localeHolder)); | ||||||
|         } |         } | ||||||
|         return MiniMessage.miniMessage().deserialize(getComponent(localeHolder), TagResolver.resolver( |         TagResolver[] finalResolvers = Arrays.copyOf(tagResolvers, tagResolvers.length + 1); | ||||||
|  |         finalResolvers[finalResolvers.length - 1] = TagResolver.resolver( | ||||||
|                 "prefix", |                 "prefix", | ||||||
|                 Tag.inserting(TranslatableCaption.of("core.prefix").toComponent(localeHolder)) |                 Tag.inserting(TranslatableCaption.of("core.prefix").toComponent(localeHolder)) | ||||||
|         )); |         ); | ||||||
|  |         return MiniMessage.miniMessage().deserialize(getComponent(localeHolder), finalResolvers); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|   | |||||||
| @@ -0,0 +1,62 @@ | |||||||
|  | /* | ||||||
|  |  * PlotSquared, a land and world management plugin for Minecraft. | ||||||
|  |  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||||
|  |  * Copyright (C) IntellectualSites team and contributors | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 3 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  | package com.plotsquared.core.exception; | ||||||
|  |  | ||||||
|  | import com.plotsquared.core.configuration.caption.Caption; | ||||||
|  | import com.plotsquared.core.configuration.caption.LocaleHolder; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Internal use only. Used to allow adventure captions to be used in an exception | ||||||
|  |  * | ||||||
|  |  * @since 7.5.7 | ||||||
|  |  */ | ||||||
|  | public final class PlotSquaredException extends RuntimeException { | ||||||
|  |  | ||||||
|  |     private final Caption caption; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Create a new instance with the given caption | ||||||
|  |      * | ||||||
|  |      * @param caption caption | ||||||
|  |      */ | ||||||
|  |     public PlotSquaredException(Caption caption) { | ||||||
|  |         this.caption = caption; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Create a new instance with the given caption and cause | ||||||
|  |      * | ||||||
|  |      * @param caption caption | ||||||
|  |      * @param cause   cause | ||||||
|  |      */ | ||||||
|  |     public PlotSquaredException(Caption caption, Exception cause) { | ||||||
|  |         super(cause); | ||||||
|  |         this.caption = caption; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public String getMessage() { | ||||||
|  |         return caption.getComponent(LocaleHolder.console()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public Caption getCaption() { | ||||||
|  |         return caption; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -290,7 +290,7 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|      * |      * | ||||||
|      * @return the plot the player is standing on or null if standing on a road or not in a {@link PlotArea} |      * @return the plot the player is standing on or null if standing on a road or not in a {@link PlotArea} | ||||||
|      */ |      */ | ||||||
|     public Plot getCurrentPlot() { |     public @Nullable Plot getCurrentPlot() { | ||||||
|         try (final MetaDataAccess<Plot> lastPlotAccess = |         try (final MetaDataAccess<Plot> lastPlotAccess = | ||||||
|                      this.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { |                      this.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { | ||||||
|             if (lastPlotAccess.get().orElse(null) == null && !Settings.Enabled_Components.EVENTS) { |             if (lastPlotAccess.get().orElse(null) == null && !Settings.Enabled_Components.EVENTS) { | ||||||
| @@ -319,7 +319,7 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|      */ |      */ | ||||||
|     public int getPlotCount() { |     public int getPlotCount() { | ||||||
|         if (!Settings.Limit.GLOBAL) { |         if (!Settings.Limit.GLOBAL) { | ||||||
|             return getPlotCount(getLocation().getWorldName()); |             return getPlotCount(getContextualWorldName()); | ||||||
|         } |         } | ||||||
|         final AtomicInteger count = new AtomicInteger(0); |         final AtomicInteger count = new AtomicInteger(0); | ||||||
|         final UUID uuid = getUUID(); |         final UUID uuid = getUUID(); | ||||||
| @@ -339,7 +339,7 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|  |  | ||||||
|     public int getClusterCount() { |     public int getClusterCount() { | ||||||
|         if (!Settings.Limit.GLOBAL) { |         if (!Settings.Limit.GLOBAL) { | ||||||
|             return getClusterCount(getLocation().getWorldName()); |             return getClusterCount(getContextualWorldName()); | ||||||
|         } |         } | ||||||
|         final AtomicInteger count = new AtomicInteger(0); |         final AtomicInteger count = new AtomicInteger(0); | ||||||
|         this.plotAreaManager.forEachPlotArea(value -> { |         this.plotAreaManager.forEachPlotArea(value -> { | ||||||
| @@ -352,6 +352,34 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|         return count.get(); |         return count.get(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * {@return the world name at the player's contextual position} | ||||||
|  |      * The contextual position can be affected when using a command with | ||||||
|  |      * an explicit plot override, e.g., `/plot <id> info`. | ||||||
|  |      */ | ||||||
|  |     private @NonNull String getContextualWorldName() { | ||||||
|  |         Plot current = getCurrentPlot(); | ||||||
|  |         if (current != null) { | ||||||
|  |             return current.getWorldName(); | ||||||
|  |         } | ||||||
|  |         return getLocation().getWorldName(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * {@return the plot area at the player's contextual position} | ||||||
|  |      * The contextual position can be affected when using a command with | ||||||
|  |      * an explicit plot override, e.g., `/plot <id> info`. | ||||||
|  |      * | ||||||
|  |      * @since TODO | ||||||
|  |      */ | ||||||
|  |     public @Nullable PlotArea getContextualPlotArea() { | ||||||
|  |         Plot current = getCurrentPlot(); | ||||||
|  |         if (current != null) { | ||||||
|  |             return current.getArea(); | ||||||
|  |         } | ||||||
|  |         return getLocation().getPlotArea(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the number of plots this player owns in the world. |      * Get the number of plots this player owns in the world. | ||||||
|      * |      * | ||||||
| @@ -408,8 +436,12 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     public PlotArea getApplicablePlotArea() { |     public PlotArea getApplicablePlotArea() { | ||||||
|  |         Plot plot = getCurrentPlot(); | ||||||
|  |         if (plot == null) { | ||||||
|             return this.plotAreaManager.getApplicablePlotArea(getLocation()); |             return this.plotAreaManager.getApplicablePlotArea(getLocation()); | ||||||
|         } |         } | ||||||
|  |         return plot.getArea(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public @NonNull RequiredType getSuperCaller() { |     public @NonNull RequiredType getSuperCaller() { | ||||||
| @@ -444,7 +476,7 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get this player's UUID. |      * Get this player's UUID. | ||||||
|      * === !IMPORTANT ===<br> |      * <p>=== !IMPORTANT ===</p> | ||||||
|      * The UUID is dependent on the mode chosen in the settings.yml and may not be the same as Bukkit has |      * The UUID is dependent on the mode chosen in the settings.yml and may not be the same as Bukkit has | ||||||
|      * (especially if using an old version of Bukkit that does not support UUIDs) |      * (especially if using an old version of Bukkit that does not support UUIDs) | ||||||
|      * |      * | ||||||
| @@ -614,16 +646,16 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|             PlotId id = plot.getId(); |             PlotId id = plot.getId(); | ||||||
|             int x = id.getX(); |             int x = id.getX(); | ||||||
|             int z = id.getY(); |             int z = id.getY(); | ||||||
|             ByteBuffer buffer = ByteBuffer.allocate(13); |             ByteBuffer buffer = ByteBuffer.allocate(14); | ||||||
|             buffer.putShort((short) x); |             buffer.putShort((short) x); | ||||||
|             buffer.putShort((short) z); |             buffer.putShort((short) z); | ||||||
|             Location location = getLocation(); |             Location location = getLocation(); | ||||||
|             buffer.putInt(location.getX()); |             buffer.putInt(location.getX()); | ||||||
|             buffer.put((byte) location.getY()); |             buffer.putShort((short) location.getY()); | ||||||
|             buffer.putInt(location.getZ()); |             buffer.putInt(location.getZ()); | ||||||
|             setPersistentMeta("quitLoc", buffer.array()); |             setPersistentMeta("quitLocV2", buffer.array()); | ||||||
|         } else if (hasPersistentMeta("quitLoc")) { |         } else if (hasPersistentMeta("quitLocV2")) { | ||||||
|             removePersistentMeta("quitLoc"); |             removePersistentMeta("quitLocV2"); | ||||||
|         } |         } | ||||||
|         if (plot != null) { |         if (plot != null) { | ||||||
|             this.eventDispatcher.callLeave(this, plot); |             this.eventDispatcher.callLeave(this, plot); | ||||||
| @@ -678,7 +710,8 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|  |  | ||||||
|     public void populatePersistentMetaMap() { |     public void populatePersistentMetaMap() { | ||||||
|         if (Settings.Enabled_Components.PERSISTENT_META) { |         if (Settings.Enabled_Components.PERSISTENT_META) { | ||||||
|             DBFunc.getPersistentMeta(getUUID(), new RunnableVal<>() { |             DBFunc.getPersistentMeta( | ||||||
|  |                     getUUID(), new RunnableVal<>() { | ||||||
|                         @Override |                         @Override | ||||||
|                         public void run(Map<String, byte[]> value) { |                         public void run(Map<String, byte[]> value) { | ||||||
|                             try { |                             try { | ||||||
| @@ -700,11 +733,18 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|                                     return; |                                     return; | ||||||
|                                 } |                                 } | ||||||
|                                 PlotArea area = ((SinglePlotAreaManager) manager).getArea(); |                                 PlotArea area = ((SinglePlotAreaManager) manager).getArea(); | ||||||
|  |                                 boolean V2 = false; | ||||||
|                                 byte[] arr = PlotPlayer.this.getPersistentMeta("quitLoc"); |                                 byte[] arr = PlotPlayer.this.getPersistentMeta("quitLoc"); | ||||||
|  |                                 if (arr == null) { | ||||||
|  |                                     arr = PlotPlayer.this.getPersistentMeta("quitLocV2"); | ||||||
|                                     if (arr == null) { |                                     if (arr == null) { | ||||||
|                                         return; |                                         return; | ||||||
|                                     } |                                     } | ||||||
|  |                                     V2 = true; | ||||||
|  |                                     removePersistentMeta("quitLocV2"); | ||||||
|  |                                 } else { | ||||||
|                                     removePersistentMeta("quitLoc"); |                                     removePersistentMeta("quitLoc"); | ||||||
|  |                                 } | ||||||
|  |  | ||||||
|                                 if (!getMeta("teleportOnLogin", true)) { |                                 if (!getMeta("teleportOnLogin", true)) { | ||||||
|                                     return; |                                     return; | ||||||
| @@ -714,7 +754,7 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|                                 final int plotZ = quitWorld.getShort(); |                                 final int plotZ = quitWorld.getShort(); | ||||||
|                                 PlotId id = PlotId.of(plotX, plotZ); |                                 PlotId id = PlotId.of(plotX, plotZ); | ||||||
|                                 int x = quitWorld.getInt(); |                                 int x = quitWorld.getInt(); | ||||||
|                         int y = quitWorld.get() & 0xFF; |                                 int y = V2 ? quitWorld.getShort() : (quitWorld.get() & 0xFF); | ||||||
|                                 int z = quitWorld.getInt(); |                                 int z = quitWorld.getInt(); | ||||||
|                                 Plot plot = area.getOwnedPlot(id); |                                 Plot plot = area.getOwnedPlot(id); | ||||||
|  |  | ||||||
| @@ -748,10 +788,11 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|                                     } |                                     } | ||||||
|                                 } |                                 } | ||||||
|                             } catch (Throwable e) { |                             } catch (Throwable e) { | ||||||
|                         e.printStackTrace(); |                                 LOGGER.error("Error populating persistent meta for player {}", PlotPlayer.this.getName(), e); | ||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|             }); |                     } | ||||||
|  |             ); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -819,7 +860,8 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @SuppressWarnings("unchecked") |     @SuppressWarnings("unchecked") | ||||||
|     @Nullable <T> T getPersistentMeta(final @NonNull MetaDataKey<T> key) { |     @Nullable | ||||||
|  |     <T> T getPersistentMeta(final @NonNull MetaDataKey<T> key) { | ||||||
|         final byte[] value = this.getPersistentMeta(key.toString()); |         final byte[] value = this.getPersistentMeta(key.toString()); | ||||||
|         if (value == null) { |         if (value == null) { | ||||||
|             return null; |             return null; | ||||||
| @@ -989,9 +1031,11 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|             if (throwable != null) { |             if (throwable != null) { | ||||||
|                 sendMessage( |                 sendMessage( | ||||||
|                         TranslatableCaption.of("errors.error"), |                         TranslatableCaption.of("errors.error"), | ||||||
|                         TagResolver.resolver("value", Tag.inserting( |                         TagResolver.resolver( | ||||||
|  |                                 "value", Tag.inserting( | ||||||
|                                         Component.text("Failed to resolve asynchronous caption replacements") |                                         Component.text("Failed to resolve asynchronous caption replacements") | ||||||
|                         )) |                                 ) | ||||||
|  |                         ) | ||||||
|                 ); |                 ); | ||||||
|                 LOGGER.error("Failed to resolve asynchronous tagresolver(s) for " + caption, throwable); |                 LOGGER.error("Failed to resolve asynchronous tagresolver(s) for " + caption, throwable); | ||||||
|             } else { |             } else { | ||||||
|   | |||||||
| @@ -321,7 +321,8 @@ public class Plot { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the plot from a string. |      * Get the plot from a string. Performs a check to ensure Plot#getBottomAbs is not outside world bounds | ||||||
|  |      * (x/z +/- 30,000,000) to prevent crashes | ||||||
|      * |      * | ||||||
|      * @param player  Provides a context for what world to search in. Prefixing the term with 'world_name;' will override this context. |      * @param player  Provides a context for what world to search in. Prefixing the term with 'world_name;' will override this context. | ||||||
|      * @param arg     The search term |      * @param arg     The search term | ||||||
| @@ -332,6 +333,31 @@ public class Plot { | |||||||
|             final @Nullable PlotPlayer<?> player, |             final @Nullable PlotPlayer<?> player, | ||||||
|             final @Nullable String arg, |             final @Nullable String arg, | ||||||
|             final boolean message |             final boolean message | ||||||
|  |     ) { | ||||||
|  |         Plot plot = getPlotFromStringUnchecked(player, arg, message); | ||||||
|  |         if (plot != null && !WorldUtil.isValidLocation(plot.getBottomAbs())) { | ||||||
|  |             if (message) { | ||||||
|  |                 (player == null ? ConsolePlayer.getConsole() : player).sendMessage(TranslatableCaption.of( | ||||||
|  |                         "invalid.world_location_plot")); | ||||||
|  |             } | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         return plot; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get the plot from a string. Does not perform a check on world bounds. | ||||||
|  |      * | ||||||
|  |      * @param player  Provides a context for what world to search in. Prefixing the term with 'world_name;' will override this context. | ||||||
|  |      * @param arg     The search term | ||||||
|  |      * @param message If a message should be sent to the player if a plot cannot be found | ||||||
|  |      * @return The plot if only 1 result is found, or null | ||||||
|  |      * @since 7.5.5 | ||||||
|  |      */ | ||||||
|  |     public static @Nullable Plot getPlotFromStringUnchecked( | ||||||
|  |             final @Nullable PlotPlayer<?> player, | ||||||
|  |             final @Nullable String arg, | ||||||
|  |             final boolean message | ||||||
|     ) { |     ) { | ||||||
|         if (arg == null) { |         if (arg == null) { | ||||||
|             if (player == null) { |             if (player == null) { | ||||||
| @@ -389,13 +415,51 @@ public class Plot { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Gets a plot from a string e.g. [area];[id] |      * Gets a plot from a string e.g. [area];[id]. Performs a check to ensure Plot#getBottomAbs is not outside world bounds | ||||||
|  |      * (x/z +/- 30,000,000) to prevent crashes | ||||||
|      * |      * | ||||||
|      * @param defaultArea if no area is specified |      * @param defaultArea if no area is specified | ||||||
|      * @param string      plot id/area + id |      * @param string      plot id/area + id | ||||||
|      * @return New or existing plot object |      * @return New or existing plot object | ||||||
|      */ |      */ | ||||||
|     public static @Nullable Plot fromString(final @Nullable PlotArea defaultArea, final @NonNull String string) { |     public static @Nullable Plot fromString(final @Nullable PlotArea defaultArea, final @NonNull String string) { | ||||||
|  |         return fromString(defaultArea, string, null); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Gets a plot from a string e.g. [area];[id]. Performs a check to ensure Plot#getBottomAbs is not outside world bounds | ||||||
|  |      * (x/z +/- 30,000,000) to prevent crashes | ||||||
|  |      * | ||||||
|  |      * @param defaultArea if no area is specified | ||||||
|  |      * @param string      plot id/area + id | ||||||
|  |      * @param player      {@link PlotPlayer} player to notify if plot is invalid (outside bounds) | ||||||
|  |      * @return New or existing plot object | ||||||
|  |      * @since 7.5.5 | ||||||
|  |      */ | ||||||
|  |     public static @Nullable Plot fromString( | ||||||
|  |             final @Nullable PlotArea defaultArea, | ||||||
|  |             final @NonNull String string, | ||||||
|  |             final @Nullable PlotPlayer<?> player | ||||||
|  |     ) { | ||||||
|  |         Plot plot = fromStringUnchecked(defaultArea, string); | ||||||
|  |         if (plot != null && !WorldUtil.isValidLocation(plot.getBottomAbs())) { | ||||||
|  |             if (player != null) { | ||||||
|  |                 player.sendMessage(TranslatableCaption.of("invalid.world_location_plot")); | ||||||
|  |             } | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         return plot; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Gets a plot from a string e.g. [area];[id]. Does not perform a check on world bounds. | ||||||
|  |      * | ||||||
|  |      * @param defaultArea if no area is specified | ||||||
|  |      * @param string      plot id/area + id | ||||||
|  |      * @return New or existing plot object | ||||||
|  |      * @since 7.5.5 | ||||||
|  |      */ | ||||||
|  |     public static @Nullable Plot fromStringUnchecked(final @Nullable PlotArea defaultArea, final @NonNull String string) { | ||||||
|         final String[] split = string.split("[;,]"); |         final String[] split = string.split("[;,]"); | ||||||
|         if (split.length == 2) { |         if (split.length == 2) { | ||||||
|             if (defaultArea != null) { |             if (defaultArea != null) { | ||||||
| @@ -419,7 +483,8 @@ public class Plot { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Return a new/cached plot object at a given location. |      * Return a new/cached plot object at a given location. Does not check world bounds for potential crashes, these should be | ||||||
|  |      * performed before (or after) this method is used. | ||||||
|      * |      * | ||||||
|      * <p> |      * <p> | ||||||
|      * Use {@link PlotPlayer#getCurrentPlot()} if a player is expected here. |      * Use {@link PlotPlayer#getCurrentPlot()} if a player is expected here. | ||||||
| @@ -476,7 +541,7 @@ public class Plot { | |||||||
|      * |      * | ||||||
|      * @return World name |      * @return World name | ||||||
|      */ |      */ | ||||||
|     public @Nullable String getWorldName() { |     public @NonNull String getWorldName() { | ||||||
|         return area.getWorldName(); |         return area.getWorldName(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -1717,6 +1782,7 @@ public class Plot { | |||||||
|         } |         } | ||||||
|         player.sendMessage( |         player.sendMessage( | ||||||
|                 TranslatableCaption.of("working.claimed"), |                 TranslatableCaption.of("working.claimed"), | ||||||
|  |                 TagResolver.resolver("world", Tag.inserting(Component.text(this.getWorldName()))), | ||||||
|                 TagResolver.resolver("plot", Tag.inserting(Component.text(this.getId().toString()))) |                 TagResolver.resolver("plot", Tag.inserting(Component.text(this.getId().toString()))) | ||||||
|         ); |         ); | ||||||
|         if (teleport) { |         if (teleport) { | ||||||
|   | |||||||
| @@ -371,8 +371,7 @@ public final class PlotModificationManager { | |||||||
|                             manager.createRoadSouthEast(current, queue); |                             manager.createRoadSouthEast(current, queue); | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                 } |                 } else if (current.isMerged(Direction.SOUTH)) { | ||||||
|                 if (current.isMerged(Direction.SOUTH)) { |  | ||||||
|                     manager.createRoadSouth(current, queue); |                     manager.createRoadSouth(current, queue); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -63,4 +63,18 @@ public class PlotTitle { | |||||||
|         return subtitle; |         return subtitle; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Provides a string representation of this plot title value (used in placeholders). | ||||||
|  |      * | ||||||
|  |      * @return the plot title representation in the format {@code "<title>" "<subtitle>"} | ||||||
|  |      * @since 7.5.5 | ||||||
|  |      */ | ||||||
|  |     @Override | ||||||
|  |     public String toString() { | ||||||
|  |         return "\"%s\" \"%s\"".formatted( | ||||||
|  |                 this.title != null ? this.title : "", | ||||||
|  |                 this.subtitle != null ? this.subtitle : "" | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -58,7 +58,7 @@ public class SinglePlot extends Plot { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public String getWorldName() { |     public @NonNull String getWorldName() { | ||||||
|         return getId().toUnderscoreSeparatedString(); |         return getId().toUnderscoreSeparatedString(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -0,0 +1,76 @@ | |||||||
|  | /* | ||||||
|  |  * PlotSquared, a land and world management plugin for Minecraft. | ||||||
|  |  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||||
|  |  * Copyright (C) IntellectualSites team and contributors | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 3 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  | package com.plotsquared.core.util; | ||||||
|  |  | ||||||
|  | import org.jetbrains.annotations.ApiStatus; | ||||||
|  |  | ||||||
|  | import java.lang.invoke.MethodHandles; | ||||||
|  | import java.lang.invoke.MethodType; | ||||||
|  | import java.lang.reflect.Method; | ||||||
|  | import java.util.Optional; | ||||||
|  |  | ||||||
|  | @ApiStatus.Internal | ||||||
|  | public final class ReflectionHelper { | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Find a (declared) method with an unknown or potentially obfuscated name by its signature and optional modifiers. | ||||||
|  |      * <br> | ||||||
|  |      * The method - if private - is not made accessible. Either call {@link Method#setAccessible(boolean)} or | ||||||
|  |      * use a {@link java.lang.invoke.MethodHandles.Lookup#privateLookupIn(Class, MethodHandles.Lookup) private lookup}. | ||||||
|  |      * | ||||||
|  |      * @param holder    The class providing the method. | ||||||
|  |      * @param signature The signature of the method, identified by parameter types and the return type. | ||||||
|  |      * @param modifiers All possible modifiers of the method that should be validated. | ||||||
|  |      * @return The method, if one has been found. Otherwise, an empty Optional. | ||||||
|  |      * @throws RuntimeException if multiple matching methods have been found. | ||||||
|  |      * @see java.lang.reflect.Modifier | ||||||
|  |      */ | ||||||
|  |     public static Optional<Method> findMethod(Class<?> holder, MethodType signature, int... modifiers) { | ||||||
|  |         Method found = null; | ||||||
|  |         outer: | ||||||
|  |         for (final Method method : holder.getDeclaredMethods()) { | ||||||
|  |             if (method.getParameterCount() != signature.parameterCount()) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |             if (!signature.returnType().isAssignableFrom(method.getReturnType())) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             for (final int modifier : modifiers) { | ||||||
|  |                 if ((method.getModifiers() & modifier) == 0) { | ||||||
|  |                     continue outer; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             Class<?>[] parameterTypes = signature.parameterArray(); | ||||||
|  |             for (int i = 0; i < parameterTypes.length; i++) { | ||||||
|  |                 // validate expected parameter is either the same type or subtype of actual parameter | ||||||
|  |                 if (!parameterTypes[i].isAssignableFrom(method.getParameterTypes()[i])) { | ||||||
|  |                     continue outer; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             if (found != null) { | ||||||
|  |                 throw new RuntimeException("Found ambiguous method by selector: " + method + " vs " + found); | ||||||
|  |             } | ||||||
|  |             found = method; | ||||||
|  |         } | ||||||
|  |         return Optional.ofNullable(found); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -135,6 +135,7 @@ public abstract class SchematicHandler { | |||||||
|         } |         } | ||||||
|         final String filename; |         final String filename; | ||||||
|         final String website; |         final String website; | ||||||
|  |         final @Nullable UUID finalUuid = uuid; | ||||||
|         if (uuid == null) { |         if (uuid == null) { | ||||||
|             uuid = UUID.randomUUID(); |             uuid = UUID.randomUUID(); | ||||||
|             website = Settings.Web.URL + "upload.php?" + uuid; |             website = Settings.Web.URL + "upload.php?" + uuid; | ||||||
| @@ -144,10 +145,11 @@ public abstract class SchematicHandler { | |||||||
|             filename = file + '.' + extension; |             filename = file + '.' + extension; | ||||||
|         } |         } | ||||||
|         final URL url; |         final URL url; | ||||||
|  |         String uri = Settings.Web.URL + "?key=" + uuid + "&type=" + extension; | ||||||
|         try { |         try { | ||||||
|             url = URI.create(Settings.Web.URL + "?key=" + uuid + "&type=" + extension).toURL(); |             url = URI.create(uri).toURL(); | ||||||
|         } catch (MalformedURLException e) { |         } catch (MalformedURLException e) { | ||||||
|             e.printStackTrace(); |             LOGGER.error("Malformed URI `{}`", uri, e); | ||||||
|             whenDone.run(); |             whenDone.run(); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| @@ -193,7 +195,7 @@ public abstract class SchematicHandler { | |||||||
|                 } |                 } | ||||||
|                 TaskManager.runTask(whenDone); |                 TaskManager.runTask(whenDone); | ||||||
|             } catch (IOException e) { |             } catch (IOException e) { | ||||||
|                 e.printStackTrace(); |                 LOGGER.error("Error while uploading schematic for UUID {}", finalUuid, e); | ||||||
|                 TaskManager.runTask(whenDone); |                 TaskManager.runTask(whenDone); | ||||||
|             } |             } | ||||||
|         }); |         }); | ||||||
| @@ -388,8 +390,14 @@ public abstract class SchematicHandler { | |||||||
|             } |             } | ||||||
|             queue.enqueue(); |             queue.enqueue(); | ||||||
|         } catch (Exception e) { |         } catch (Exception e) { | ||||||
|             e.printStackTrace(); |  | ||||||
|             TaskManager.runTask(whenDone); |             TaskManager.runTask(whenDone); | ||||||
|  |             LOGGER.error( | ||||||
|  |                     "Error pasting schematic to plot {};{} for player {}", | ||||||
|  |                     plot.getArea(), | ||||||
|  |                     plot.getId(), | ||||||
|  |                     actor == null ? "null" : actor.getName(), | ||||||
|  |                     e | ||||||
|  |             ); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -456,7 +464,7 @@ public abstract class SchematicHandler { | |||||||
|                 Clipboard clip = reader.read(); |                 Clipboard clip = reader.read(); | ||||||
|                 return new Schematic(clip); |                 return new Schematic(clip); | ||||||
|             } catch (IOException e) { |             } catch (IOException e) { | ||||||
|                 e.printStackTrace(); |                 LOGGER.error("Error reading schematic from file {}", file.getAbsolutePath(), e); | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             throw new UnsupportedFormatException("This schematic format is not recognised or supported."); |             throw new UnsupportedFormatException("This schematic format is not recognised or supported."); | ||||||
| @@ -470,7 +478,7 @@ public abstract class SchematicHandler { | |||||||
|             InputStream inputStream = Channels.newInputStream(readableByteChannel); |             InputStream inputStream = Channels.newInputStream(readableByteChannel); | ||||||
|             return getSchematic(inputStream); |             return getSchematic(inputStream); | ||||||
|         } catch (IOException e) { |         } catch (IOException e) { | ||||||
|             e.printStackTrace(); |             LOGGER.error("Error reading schematic from {}", url, e); | ||||||
|         } |         } | ||||||
|         return null; |         return null; | ||||||
|     } |     } | ||||||
| @@ -486,7 +494,7 @@ public abstract class SchematicHandler { | |||||||
|                 Clipboard clip = schematicReader.read(); |                 Clipboard clip = schematicReader.read(); | ||||||
|                 return new Schematic(clip); |                 return new Schematic(clip); | ||||||
|             } catch (IOException e) { |             } catch (IOException e) { | ||||||
|                 e.printStackTrace(); |                 LOGGER.error("Error reading schematic", e); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         return null; |         return null; | ||||||
| @@ -515,7 +523,7 @@ public abstract class SchematicHandler { | |||||||
|             } |             } | ||||||
|             return schematics; |             return schematics; | ||||||
|         } catch (JsonParseException | IOException e) { |         } catch (JsonParseException | IOException e) { | ||||||
|             e.printStackTrace(); |             LOGGER.error("Error retrieving saves for UUID {}", uuid, e); | ||||||
|         } |         } | ||||||
|         return null; |         return null; | ||||||
|     } |     } | ||||||
| @@ -532,7 +540,7 @@ public abstract class SchematicHandler { | |||||||
|                 try (NBTOutputStream nos = new NBTOutputStream(new GZIPOutputStream(output, true))) { |                 try (NBTOutputStream nos = new NBTOutputStream(new GZIPOutputStream(output, true))) { | ||||||
|                     nos.writeNamedTag("Schematic", tag); |                     nos.writeNamedTag("Schematic", tag); | ||||||
|                 } catch (IOException e1) { |                 } catch (IOException e1) { | ||||||
|                     e1.printStackTrace(); |                     LOGGER.error("Error uploading schematic for UUID {}", uuid, e1); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }, whenDone); |         }, whenDone); | ||||||
| @@ -556,9 +564,9 @@ public abstract class SchematicHandler { | |||||||
|                 nbtStream.writeNamedTag("Schematic", tag); |                 nbtStream.writeNamedTag("Schematic", tag); | ||||||
|             } |             } | ||||||
|         } catch (FileNotFoundException e) { |         } catch (FileNotFoundException e) { | ||||||
|             e.printStackTrace(); |             LOGGER.error("Error saving schematic at {}", path, e); | ||||||
|         } catch (IOException e) { |         } catch (IOException e) { | ||||||
|             e.printStackTrace(); |             LOGGER.error("Error saving schematic at {}", path, e); | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|         return true; |         return true; | ||||||
| @@ -581,7 +589,7 @@ public abstract class SchematicHandler { | |||||||
|         schematic.put("BlockData", new ByteArrayTag(buffer.toByteArray())); |         schematic.put("BlockData", new ByteArrayTag(buffer.toByteArray())); | ||||||
|         schematic.put("BlockEntities", new ListTag(CompoundTag.class, tileEntities)); |         schematic.put("BlockEntities", new ListTag(CompoundTag.class, tileEntities)); | ||||||
|  |  | ||||||
|         if (biomeBuffer.size() == 0 || biomePalette.size() == 0) { |         if (biomeBuffer.size() == 0 || biomePalette.isEmpty()) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -733,10 +741,7 @@ public abstract class SchematicHandler { | |||||||
|                                     } |                                     } | ||||||
|                                     BaseBlock block = aabb.getWorld().getFullBlock(point); |                                     BaseBlock block = aabb.getWorld().getFullBlock(point); | ||||||
|                                     if (block.getNbtData() != null) { |                                     if (block.getNbtData() != null) { | ||||||
|                                         Map<String, Tag> values = new HashMap<>(); |                                         Map<String, Tag> values = new HashMap<>(block.getNbtData().getValue()); | ||||||
|                                         for (Map.Entry<String, Tag> entry : block.getNbtData().getValue().entrySet()) { |  | ||||||
|                                             values.put(entry.getKey(), entry.getValue()); |  | ||||||
|                                         } |  | ||||||
|  |  | ||||||
|                                         // Positions are kept in NBT, we don't want that. |                                         // Positions are kept in NBT, we don't want that. | ||||||
|                                         values.remove("x"); |                                         values.remove("x"); | ||||||
|   | |||||||
| @@ -25,21 +25,16 @@ import com.plotsquared.core.player.PlotPlayer; | |||||||
| import com.plotsquared.core.util.StringMan; | import com.plotsquared.core.util.StringMan; | ||||||
| import net.kyori.adventure.text.Component; | import net.kyori.adventure.text.Component; | ||||||
| import net.kyori.adventure.text.ComponentLike; | import net.kyori.adventure.text.ComponentLike; | ||||||
| import net.kyori.adventure.text.minimessage.MiniMessage; |  | ||||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | import net.kyori.adventure.text.minimessage.tag.Tag; | ||||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||||
| import org.jetbrains.annotations.NotNull; | import org.jetbrains.annotations.NotNull; | ||||||
|  |  | ||||||
| public class HelpObject implements ComponentLike { | public class HelpObject implements ComponentLike { | ||||||
|  |  | ||||||
|     static final MiniMessage MINI_MESSAGE = MiniMessage.miniMessage(); |  | ||||||
|  |  | ||||||
|     private final Component rendered; |     private final Component rendered; | ||||||
|  |  | ||||||
|     public HelpObject(final Command command, final String label, final PlotPlayer<?> audience) { |     public HelpObject(final Command command, final String label, final PlotPlayer<?> audience) { | ||||||
|         rendered = MINI_MESSAGE.deserialize( |         this.rendered = TranslatableCaption.of("help.help_item").toComponent(audience, TagResolver.builder() | ||||||
|                 TranslatableCaption.of("help.help_item").getComponent(audience), |  | ||||||
|                 TagResolver.builder() |  | ||||||
|                 .tag("usage", Tag.inserting(Component.text(command.getUsage().replace("{label}", label)))) |                 .tag("usage", Tag.inserting(Component.text(command.getUsage().replace("{label}", label)))) | ||||||
|                 .tag("alias", Tag.inserting(Component.text( |                 .tag("alias", Tag.inserting(Component.text( | ||||||
|                         command.getAliases().isEmpty() ? "" : StringMan.join(command.getAliases(), " | ") |                         command.getAliases().isEmpty() ? "" : StringMan.join(command.getAliases(), " | ") | ||||||
| @@ -47,8 +42,7 @@ public class HelpObject implements ComponentLike { | |||||||
|                 .tag("desc", Tag.inserting(command.getDescription().toComponent(audience))) |                 .tag("desc", Tag.inserting(command.getDescription().toComponent(audience))) | ||||||
|                 .tag("arguments", Tag.inserting(Component.text(buildArgumentList(command.getRequiredArguments())))) |                 .tag("arguments", Tag.inserting(Component.text(buildArgumentList(command.getRequiredArguments())))) | ||||||
|                 .tag("label", Tag.inserting(Component.text(label))) |                 .tag("label", Tag.inserting(Component.text(label))) | ||||||
|                         .build() |                 .build()); | ||||||
|         ); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private String buildArgumentList(final Argument<?>[] arguments) { |     private String buildArgumentList(final Argument<?>[] arguments) { | ||||||
|   | |||||||
| @@ -49,10 +49,12 @@ import java.util.Collection; | |||||||
| import java.util.Collections; | import java.util.Collections; | ||||||
| import java.util.Locale; | import java.util.Locale; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
|  | import java.util.Set; | ||||||
| import java.util.TimeZone; | import java.util.TimeZone; | ||||||
| import java.util.UUID; | import java.util.UUID; | ||||||
| import java.util.concurrent.TimeUnit; | import java.util.concurrent.TimeUnit; | ||||||
| import java.util.function.BiFunction; | import java.util.function.BiFunction; | ||||||
|  | import java.util.stream.Collectors; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Registry that contains {@link Placeholder placeholders} |  * Registry that contains {@link Placeholder placeholders} | ||||||
| @@ -127,6 +129,22 @@ public final class PlaceholderRegistry { | |||||||
|             } |             } | ||||||
|             return legacyComponent(TranslatableCaption.of("info.unknown"), player); |             return legacyComponent(TranslatableCaption.of("info.unknown"), player); | ||||||
|         }); |         }); | ||||||
|  |         this.createPlaceholder("currentplot_owners", (player, plot) -> { | ||||||
|  |             if (plot.getFlag(ServerPlotFlag.class)) { | ||||||
|  |                 return legacyComponent(TranslatableCaption.of("info.server"), player); | ||||||
|  |             } | ||||||
|  |             final Set<UUID> plotOwners = plot.getOwners(); | ||||||
|  |             if (plotOwners.isEmpty()) { | ||||||
|  |                 return legacyComponent(TranslatableCaption.of("generic.generic_unowned"), player); | ||||||
|  |             } | ||||||
|  |             return plotOwners.stream().map(PlotSquared.platform().playerManager()::getUsernameCaption).map(f -> { | ||||||
|  |                 try { | ||||||
|  |                     return f.get(Settings.UUID.BLOCKING_TIMEOUT, TimeUnit.MILLISECONDS).getComponent(player); | ||||||
|  |                 } catch (final Exception ignored) { | ||||||
|  |                     return legacyComponent(TranslatableCaption.of("info.unknown"), player); | ||||||
|  |                 } | ||||||
|  |             }).collect(Collectors.joining(", ")); | ||||||
|  |         }); | ||||||
|         this.createPlaceholder("currentplot_members", (player, plot) -> { |         this.createPlaceholder("currentplot_members", (player, plot) -> { | ||||||
|             if (plot.getMembers().isEmpty() && plot.getTrusted().isEmpty()) { |             if (plot.getMembers().isEmpty() && plot.getTrusted().isEmpty()) { | ||||||
|                 return legacyComponent(TranslatableCaption.of("info.none"), player); |                 return legacyComponent(TranslatableCaption.of("info.none"), player); | ||||||
|   | |||||||
| @@ -234,6 +234,7 @@ | |||||||
|   "invalid.not_valid_number": "<prefix><red>That's not a valid number within the range: </red><gray><value></gray>", |   "invalid.not_valid_number": "<prefix><red>That's not a valid number within the range: </red><gray><value></gray>", | ||||||
|   "invalid.not_valid_plot_id": "<prefix><red>That's not a valid plot ID.</red>", |   "invalid.not_valid_plot_id": "<prefix><red>That's not a valid plot ID.</red>", | ||||||
|   "invalid.origin_cant_be_target": "<prefix><red>The origin and target location cannot be the same.</red>", |   "invalid.origin_cant_be_target": "<prefix><red>The origin and target location cannot be the same.</red>", | ||||||
|  |   "invalid.world_location_plot": "<prefix><red>The target plot is invalid.</red>", | ||||||
|   "invalid.found_no_plots": "<prefix><red>Found no plots with your search query.</red>", |   "invalid.found_no_plots": "<prefix><red>Found no plots with your search query.</red>", | ||||||
|   "invalid.number_not_in_range": "<prefix><red>That's not a valid number within the range: <gray>(<min>, <max>)</gray></red>", |   "invalid.number_not_in_range": "<prefix><red>That's not a valid number within the range: <gray>(<min>, <max>)</gray></red>", | ||||||
|   "invalid.number_not_positive": "<red>That's not a positive number: <gray><value></gray></red>", |   "invalid.number_not_positive": "<red>That's not a positive number: <gray><value></gray></red>", | ||||||
| @@ -353,6 +354,8 @@ | |||||||
|   "info.infinite": "<gray>Infinite</gray>", |   "info.infinite": "<gray>Infinite</gray>", | ||||||
|   "info.plot_unowned": "<prefix><red>The current plot must have an owner to perform this action.</red>", |   "info.plot_unowned": "<prefix><red>The current plot must have an owner to perform this action.</red>", | ||||||
|   "info.plot_info_unclaimed": "<prefix><gray>Plot <gold><plot></gold> is not yet claimed.</gray>", |   "info.plot_info_unclaimed": "<prefix><gray>Plot <gold><plot></gold> is not yet claimed.</gray>", | ||||||
|  |   "info.plot_info_categories.use_rating": "<prefix><gold>Categories: </gold> <gray>members, alias, biome, seen, denied, flags, id, size, trusted, owner, rating</gray>", | ||||||
|  |   "info.plot_info_categories.use_likes": "<prefix><gold>Categories: </gold> <gray>members, alias, biome, seen, denied, flags, id, size, trusted, owner, likes</gray>", | ||||||
|   "info.plot_info_header": "<dark_gray><strikethrough>--------- <reset><gold>INFO </gold><dark_gray><strikethrough>---------</dark_gray><reset>", |   "info.plot_info_header": "<dark_gray><strikethrough>--------- <reset><gold>INFO </gold><dark_gray><strikethrough>---------</dark_gray><reset>", | ||||||
|   "info.plot_info_hidden": "<prefix><red>You cannot view the information about this plot.</red>", |   "info.plot_info_hidden": "<prefix><red>You cannot view the information about this plot.</red>", | ||||||
|   "info.plot_info_format": "<header>\n<gold>ID: <gray><id></gray>\nCreation: <gray><creationdate></gray>\nArea: <gray><area></gray>\nAlias: <gray><alias></gray>\nOwner: <gray><owner></gray>\nBiome: <gray><biome></gray>\nCan Build: <gray><build></gray>\nRating: <gray><rating></gray>\nSeen: <gray><seen></gray>\nTrusted: <gray><trusted></gray>\nMembers: <gray><members></gray>\nDenied: <gray><denied></gray>\nFlags: <gray><flags></gray>\nDescription: <gray><desc></gray></gold>\n<footer>", |   "info.plot_info_format": "<header>\n<gold>ID: <gray><id></gray>\nCreation: <gray><creationdate></gray>\nArea: <gray><area></gray>\nAlias: <gray><alias></gray>\nOwner: <gray><owner></gray>\nBiome: <gray><biome></gray>\nCan Build: <gray><build></gray>\nRating: <gray><rating></gray>\nSeen: <gray><seen></gray>\nTrusted: <gray><trusted></gray>\nMembers: <gray><members></gray>\nDenied: <gray><denied></gray>\nFlags: <gray><flags></gray>\nDescription: <gray><desc></gray></gold>\n<footer>", | ||||||
| @@ -390,11 +393,11 @@ | |||||||
|   "info.area_list_tooltip": "<gold>Claimed=</gold><gray><claimed></gray>\n<gold>Usage=</gold><gray><usage></gray>\n<gold>Clusters=</gold><gray><clusters></gray>\n<gold>Region=</gold><gray><region></gray>\n<gold>Generator=</gold><gray><generator></gray>", |   "info.area_list_tooltip": "<gold>Claimed=</gold><gray><claimed></gray>\n<gold>Usage=</gold><gray><usage></gray>\n<gold>Clusters=</gold><gray><clusters></gray>\n<gold>Region=</gold><gray><region></gray>\n<gold>Generator=</gold><gray><generator></gray>", | ||||||
|   "info.area_list_item": "<click:run_command:'<command_tp>'><hover:show_text:'<command_tp>'><dark_gray>[</dark_gray><gold><number></gold><dark_gray>]</dark_gray></hover></click> <click:run_command:'<command_info>'><hover:show_text:'<hover_info>'><gold><area_name></gold></hover></click><gray> - </gray><gray><area_type>:<area_terrain></gray>", |   "info.area_list_item": "<click:run_command:'<command_tp>'><hover:show_text:'<command_tp>'><dark_gray>[</dark_gray><gold><number></gold><dark_gray>]</dark_gray></hover></click> <click:run_command:'<command_info>'><hover:show_text:'<hover_info>'><gold><area_name></gold></hover></click><gray> - </gray><gray><area_type>:<area_terrain></gray>", | ||||||
|   "working.generating_component": "<prefix><gold>Started generating component from your settings.</gold>", |   "working.generating_component": "<prefix><gold>Started generating component from your settings.</gold>", | ||||||
|   "working.clearing_done": "<prefix><dark_aqua>Plot </dark_aqua><gold><plot></gold><dark_aqua> has been cleared! Took </dark_aqua><gold><amount>ms</gold><dark_aqua>.</dark_aqua>", |   "working.clearing_done": "<prefix><dark_aqua>Plot </dark_aqua><gold><world>;<plot></gold><dark_aqua> has been cleared! Took </dark_aqua><gold><amount>ms</gold><dark_aqua>.</dark_aqua>", | ||||||
|   "working.deleting_done": "<prefix><dark_aqua>Plot </dark_aqua><gold><plot></gold><dark_aqua> has been deleted! Took </dark_aqua><gold><amount>ms</gold><dark_aqua>.</dark_aqua>", |   "working.deleting_done": "<prefix><dark_aqua>Plot </dark_aqua><gold><world>;<plot></gold><dark_aqua> has been deleted! Took </dark_aqua><gold><amount>ms</gold><dark_aqua>.</dark_aqua>", | ||||||
|   "working.plot_not_claimed": "<prefix><gray>Plot not claimed.</gray>", |   "working.plot_not_claimed": "<prefix><gray>Plot not claimed.</gray>", | ||||||
|   "working.plot_is_claimed": "<prefix><gray>This plot is already claimed.</gray>", |   "working.plot_is_claimed": "<prefix><gray>This plot is already claimed.</gray>", | ||||||
|   "working.claimed": "<prefix><dark_aqua>You successfully claimed the plot </dark_aqua><gold><plot></gold><dark_aqua>.</dark_aqua>", |   "working.claimed": "<prefix><dark_aqua>You successfully claimed the plot </dark_aqua><gold><world>;<plot></gold><dark_aqua>.</dark_aqua>", | ||||||
|   "working.progress": "<prefix><gray>Current progress: </gray><gold><progress></gold><gray>%</gray>", |   "working.progress": "<prefix><gray>Current progress: </gray><gold><progress></gold><gray>%</gray>", | ||||||
|   "working.component_complete": "<prefix><gold>Component generation has finished for plot <plot>.</gold>", |   "working.component_complete": "<prefix><gold>Component generation has finished for plot <plot>.</gold>", | ||||||
|   "list.comment_list_header_paged": "<gray>(Page </gray><gold><cur></gold><gray>/</gray><gold><max></gold><gray>) </gray><gold>List of <amount> comment(s):</gold>", |   "list.comment_list_header_paged": "<gray>(Page </gray><gold><cur></gold><gray>/</gray><gold><max></gold><gray>) </gray><gold>List of <amount> comment(s):</gold>", | ||||||
| @@ -412,6 +415,7 @@ | |||||||
|   "deny.denied_added": "<prefix><dark_aqua>You successfully denied the player from this plot.</dark_aqua>", |   "deny.denied_added": "<prefix><dark_aqua>You successfully denied the player from this plot.</dark_aqua>", | ||||||
|   "deny.no_enter": "<prefix><red>You are denied from the plot <red><gold><plot></gold><red> and therefore not allowed to enter.</red>", |   "deny.no_enter": "<prefix><red>You are denied from the plot <red><gold><plot></gold><red> and therefore not allowed to enter.</red>", | ||||||
|   "deny.you_got_denied": "<prefix><red>You are denied from the plot you were previously on, and got teleported to spawn.</red>", |   "deny.you_got_denied": "<prefix><red>You are denied from the plot you were previously on, and got teleported to spawn.</red>", | ||||||
|  |   "deny.cannot_interact": "<prefix><red>You are denied from the plot <red><gold><plot></gold><red> and therefore cannot interact with it.</red>", | ||||||
|   "deny.cant_remove_owner": "<prefix><red>You can't remove the plot owner.</red>", |   "deny.cant_remove_owner": "<prefix><red>You can't remove the plot owner.</red>", | ||||||
|   "kick.player_not_in_plot": "<prefix><red>The player <gray><player></gray> is not on this plot.</red>", |   "kick.player_not_in_plot": "<prefix><red>The player <gray><player></gray> is not on this plot.</red>", | ||||||
|   "kick.cannot_kick_player": "<prefix><red>You cannot kick the player <gray><player></gray>.</red>", |   "kick.cannot_kick_player": "<prefix><red>You cannot kick the player <gray><player></gray>.</red>", | ||||||
| @@ -454,7 +458,7 @@ | |||||||
|   "category.command_category_debug": "<gray>Debug</gray>", |   "category.command_category_debug": "<gray>Debug</gray>", | ||||||
|   "category.command_category_administration": "<gray>Admin</gray>", |   "category.command_category_administration": "<gray>Admin</gray>", | ||||||
|   "grants.granted_plots": "<prefix><gold>Result: <gray><amount> </gray>grants left.</gold>", |   "grants.granted_plots": "<prefix><gold>Result: <gray><amount> </gray>grants left.</gold>", | ||||||
|   "grants.added": "<prefix><gold><grants></gold> <gray>grant(s) have been added.</gray>", |   "grants.added": "<prefix><gold>1</gold> <gray>grant has been added. (<grants> total grants)</gray>", | ||||||
|   "events.event_denied": "<prefix><gold><value> </gold><gray>Cancelled by external plugin.</gray>", |   "events.event_denied": "<prefix><gold><value> </gold><gray>Cancelled by external plugin.</gray>", | ||||||
|   "backups.backup_impossible": "<prefix><red>Backups are not enabled for this plot: <plot>.</red>", |   "backups.backup_impossible": "<prefix><red>Backups are not enabled for this plot: <plot>.</red>", | ||||||
|   "backups.backup_save_success": "<prefix><gold>The backup was created successfully.</gold>", |   "backups.backup_save_success": "<prefix><gold>The backup was created successfully.</gold>", | ||||||
|   | |||||||
| @@ -37,12 +37,7 @@ class ClickStripTransformTest { | |||||||
|     void removeClickEvent() { |     void removeClickEvent() { | ||||||
|         var commonAction = ClickEvent.Action.OPEN_FILE; |         var commonAction = ClickEvent.Action.OPEN_FILE; | ||||||
|         var transform = new ClickStripTransform(EnumSet.of(commonAction)); |         var transform = new ClickStripTransform(EnumSet.of(commonAction)); | ||||||
|         var component = Component.text("Hello") |         var component = Component.text("Hello").clickEvent(ClickEvent.openFile("World")); | ||||||
|                 .clickEvent(ClickEvent.clickEvent( |  | ||||||
|                                 commonAction, |  | ||||||
|                                 "World" |  | ||||||
|                         ) |  | ||||||
|                 ); |  | ||||||
|         var transformedComponent = transform.transform(component); |         var transformedComponent = transform.transform(component); | ||||||
|         Assertions.assertNull(transformedComponent.clickEvent()); |         Assertions.assertNull(transformedComponent.clickEvent()); | ||||||
|     } |     } | ||||||
| @@ -52,10 +47,7 @@ class ClickStripTransformTest { | |||||||
|     void ignoreClickEvent() { |     void ignoreClickEvent() { | ||||||
|         var actionToRemove = ClickEvent.Action.SUGGEST_COMMAND; |         var actionToRemove = ClickEvent.Action.SUGGEST_COMMAND; | ||||||
|         var transform = new ClickStripTransform(EnumSet.of(actionToRemove)); |         var transform = new ClickStripTransform(EnumSet.of(actionToRemove)); | ||||||
|         var originalClickEvent = ClickEvent.clickEvent( |         var originalClickEvent = ClickEvent.changePage(1337); | ||||||
|                 ClickEvent.Action.CHANGE_PAGE, |  | ||||||
|                 "World" |  | ||||||
|         ); |  | ||||||
|         var component = Component.text("Hello") |         var component = Component.text("Hello") | ||||||
|                 .clickEvent(originalClickEvent); |                 .clickEvent(originalClickEvent); | ||||||
|         var transformedComponent = transform.transform(component); |         var transformedComponent = transform.transform(component); | ||||||
| @@ -76,12 +68,12 @@ class ClickStripTransformTest { | |||||||
|                 .insertion("DEF"); |                 .insertion("DEF"); | ||||||
|         var component = Component.text("Hello ") |         var component = Component.text("Hello ") | ||||||
|                 .append( |                 .append( | ||||||
|                         inner.clickEvent(ClickEvent.clickEvent(ClickEvent.Action.OPEN_URL, "https://example.org")) |                         inner.clickEvent(ClickEvent.openUrl("https://example.org")) | ||||||
|                 ); |                 ); | ||||||
|         var transformedComponent = transform.transform(component); |         var transformedComponent = transform.transform(component); | ||||||
|         Assertions.assertFalse(transformedComponent.children().isEmpty()); // child still exists |         Assertions.assertFalse(transformedComponent.children().isEmpty()); // child still exists | ||||||
|         Assertions.assertEquals(inner, transformedComponent.children().get(0)); // only the click event has changed |         Assertions.assertEquals(inner, transformedComponent.children().getFirst()); // only the click event has changed | ||||||
|         Assertions.assertNull(transformedComponent.children().get(0).clickEvent()); |         Assertions.assertNull(transformedComponent.children().getFirst().clickEvent()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,79 @@ | |||||||
|  | /* | ||||||
|  |  * PlotSquared, a land and world management plugin for Minecraft. | ||||||
|  |  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||||
|  |  * Copyright (C) IntellectualSites team and contributors | ||||||
|  |  * | ||||||
|  |  * This program is free software: you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License as published by | ||||||
|  |  * the Free Software Foundation, either version 3 of the License, or | ||||||
|  |  * (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||||
|  |  */ | ||||||
|  | package com.plotsquared.core.util; | ||||||
|  |  | ||||||
|  | import org.junit.jupiter.api.Test; | ||||||
|  |  | ||||||
|  | import java.lang.invoke.MethodType; | ||||||
|  | import java.lang.reflect.Modifier; | ||||||
|  | import java.util.Collection; | ||||||
|  |  | ||||||
|  | import static org.junit.jupiter.api.Assertions.assertEquals; | ||||||
|  | import static org.junit.jupiter.api.Assertions.assertThrows; | ||||||
|  |  | ||||||
|  | class ReflectionHelperTest { | ||||||
|  |  | ||||||
|  |     @Test | ||||||
|  |     void findMethod() throws NoSuchMethodException { | ||||||
|  |         assertThrows( | ||||||
|  |                 RuntimeException.class, () -> | ||||||
|  |                         ReflectionHelper.findMethod(MethodTesterClass.class, MethodType.methodType(String.class)) | ||||||
|  |         ); | ||||||
|  |         assertEquals( | ||||||
|  |                 MethodTesterClass.class.getMethod("methodThree"), | ||||||
|  |                 ReflectionHelper.findMethod(MethodTesterClass.class, MethodType.methodType(String.class), Modifier.PUBLIC) | ||||||
|  |                         .orElse(null) | ||||||
|  |         ); | ||||||
|  |         assertEquals( | ||||||
|  |                 MethodTesterClass.class.getDeclaredMethod("methodFour", String.class, Collection.class), | ||||||
|  |                 ReflectionHelper.findMethod(MethodTesterClass.class, MethodType.methodType( | ||||||
|  |                         String.class, String.class, Collection.class | ||||||
|  |                 )).orElse(null) | ||||||
|  |         ); | ||||||
|  |         // check that helper allows super classes of parameters when searching | ||||||
|  |         assertEquals( | ||||||
|  |                 MethodTesterClass.class.getDeclaredMethod("methodFour", String.class, Collection.class), | ||||||
|  |                 ReflectionHelper.findMethod(MethodTesterClass.class, MethodType.methodType( | ||||||
|  |                         String.class, String.class, Object.class | ||||||
|  |                 )).orElse(null) | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @SuppressWarnings("unused") | ||||||
|  |     private static class MethodTesterClass { | ||||||
|  |  | ||||||
|  |         private static String methodOne() { | ||||||
|  |             return ""; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private static String methodTwo() { | ||||||
|  |             return ""; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public static String methodThree() { | ||||||
|  |             return ""; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         protected static String methodFour(String param, Collection<String> paramList) { | ||||||
|  |             return ""; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -2,18 +2,16 @@ import com.diffplug.gradle.spotless.SpotlessPlugin | |||||||
| import com.github.jengelman.gradle.plugins.shadow.ShadowPlugin | import com.github.jengelman.gradle.plugins.shadow.ShadowPlugin | ||||||
| import groovy.json.JsonSlurper | import groovy.json.JsonSlurper | ||||||
| import xyz.jpenilla.runpaper.task.RunServer | import xyz.jpenilla.runpaper.task.RunServer | ||||||
| import java.net.URI |  | ||||||
|  |  | ||||||
| plugins { | plugins { | ||||||
|     java |     java | ||||||
|     `java-library` |     `java-library` | ||||||
|     `maven-publish` |  | ||||||
|     signing |     signing | ||||||
|  |  | ||||||
|     alias(libs.plugins.shadow) |     alias(libs.plugins.shadow) | ||||||
|     alias(libs.plugins.spotless) |     alias(libs.plugins.spotless) | ||||||
|     alias(libs.plugins.grgit) |     alias(libs.plugins.grgit) | ||||||
|     alias(libs.plugins.nexus) |     alias(libs.plugins.publish) | ||||||
|  |  | ||||||
|     eclipse |     eclipse | ||||||
|     idea |     idea | ||||||
| @@ -22,7 +20,7 @@ plugins { | |||||||
| } | } | ||||||
|  |  | ||||||
| group = "com.intellectualsites.plotsquared" | group = "com.intellectualsites.plotsquared" | ||||||
| version = "7.5.1-SNAPSHOT" | version = "7.5.9-SNAPSHOT" | ||||||
|  |  | ||||||
| if (!File("$rootDir/.git").exists()) { | if (!File("$rootDir/.git").exists()) { | ||||||
|     logger.lifecycle(""" |     logger.lifecycle(""" | ||||||
| @@ -41,16 +39,6 @@ subprojects { | |||||||
|     repositories { |     repositories { | ||||||
|         mavenCentral() |         mavenCentral() | ||||||
|  |  | ||||||
|         maven { |  | ||||||
|             name = "Sonatype OSS" |  | ||||||
|             url = uri("https://oss.sonatype.org/content/repositories/snapshots/") |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         maven { |  | ||||||
|             name = "Sonatype OSS (S01)" |  | ||||||
|             url = uri("https://s01.oss.sonatype.org/content/repositories/snapshots/") |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         maven { |         maven { | ||||||
|             name = "Jitpack" |             name = "Jitpack" | ||||||
|             url = uri("https://jitpack.io") |             url = uri("https://jitpack.io") | ||||||
| @@ -68,7 +56,7 @@ subprojects { | |||||||
|     apply { |     apply { | ||||||
|         plugin<JavaPlugin>() |         plugin<JavaPlugin>() | ||||||
|         plugin<JavaLibraryPlugin>() |         plugin<JavaLibraryPlugin>() | ||||||
|         plugin<MavenPublishPlugin>() |         plugin<com.vanniktech.maven.publish.MavenPublishPlugin>() | ||||||
|         plugin<ShadowPlugin>() |         plugin<ShadowPlugin>() | ||||||
|         plugin<SpotlessPlugin>() |         plugin<SpotlessPlugin>() | ||||||
|         plugin<SigningPlugin>() |         plugin<SigningPlugin>() | ||||||
| @@ -77,10 +65,16 @@ subprojects { | |||||||
|         plugin<IdeaPlugin>() |         plugin<IdeaPlugin>() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     configurations.matching { it.name == "signatures" }.configureEach { | ||||||
|  |         attributes { | ||||||
|  |             attribute(Attribute.of("signatures-unique", String::class.java), "true") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     dependencies { |     dependencies { | ||||||
|         // Tests |         // Tests | ||||||
|         testImplementation("org.junit.jupiter:junit-jupiter:5.12.0") |         testImplementation("org.junit.jupiter:junit-jupiter:6.0.0") | ||||||
|         testRuntimeOnly("org.junit.platform:junit-platform-launcher:1.12.0") |         testRuntimeOnly("org.junit.platform:junit-platform-launcher:6.0.0") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     plugins.withId("java") { |     plugins.withId("java") { | ||||||
| @@ -107,15 +101,16 @@ subprojects { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     java { |     afterEvaluate { | ||||||
|         withSourcesJar() |  | ||||||
|         withJavadocJar() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|         val javaComponent = components["java"] as AdhocComponentWithVariants |         val javaComponent = components["java"] as AdhocComponentWithVariants | ||||||
|     javaComponent.withVariantsFromConfiguration(configurations["shadowRuntimeElements"]) { |         configurations.findByName("shadowRuntimeElements")?.let { shadowRuntimeElements -> | ||||||
|  |             javaComponent.withVariantsFromConfiguration(shadowRuntimeElements) { | ||||||
|                 skip() |                 skip() | ||||||
|             } |             } | ||||||
|  |         } ?: run { | ||||||
|  |             logger.warn("Configuration 'shadowRuntimeElements' does not exist.") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     signing { |     signing { | ||||||
|         if (!project.hasProperty("skip.signing") && !version.toString().endsWith("-SNAPSHOT")) { |         if (!project.hasProperty("skip.signing") && !version.toString().endsWith("-SNAPSHOT")) { | ||||||
| @@ -127,14 +122,15 @@ subprojects { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     publishing { |     mavenPublishing { | ||||||
|         publications { |         coordinates( | ||||||
|             create<MavenPublication>("maven") { |             groupId = "$group", | ||||||
|                 from(components["java"]) |             artifactId = project.name, | ||||||
|  |             version = "${project.version}", | ||||||
|  |         ) | ||||||
|  |  | ||||||
|         pom { |         pom { | ||||||
|  |             name.set(project.name) | ||||||
|                     name.set(project.name + " " + project.version) |  | ||||||
|             description.set("PlotSquared, a land and world management plugin for Minecraft.") |             description.set("PlotSquared, a land and world management plugin for Minecraft.") | ||||||
|             url.set("https://github.com/IntellectualSites/PlotSquared") |             url.set("https://github.com/IntellectualSites/PlotSquared") | ||||||
|  |  | ||||||
| @@ -185,8 +181,8 @@ subprojects { | |||||||
|                 system.set("GitHub") |                 system.set("GitHub") | ||||||
|                 url.set("https://github.com/IntellectualSites/PlotSquared/issues") |                 url.set("https://github.com/IntellectualSites/PlotSquared/issues") | ||||||
|             } |             } | ||||||
|                 } |  | ||||||
|             } |             publishToMavenCentral() | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -194,7 +190,6 @@ subprojects { | |||||||
|  |  | ||||||
|         compileJava { |         compileJava { | ||||||
|             options.compilerArgs.add("-parameters") |             options.compilerArgs.add("-parameters") | ||||||
|             options.isDeprecation = true |  | ||||||
|             options.encoding = "UTF-8" |             options.encoding = "UTF-8" | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -217,27 +212,18 @@ subprojects { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| nexusPublishing { |  | ||||||
|     this.repositories { |  | ||||||
|         sonatype { |  | ||||||
|             nexusUrl.set(URI.create("https://s01.oss.sonatype.org/service/local/")) |  | ||||||
|             snapshotRepositoryUrl.set(URI.create("https://s01.oss.sonatype.org/content/repositories/snapshots/")) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| tasks.getByName<Jar>("jar") { | tasks.getByName<Jar>("jar") { | ||||||
|     enabled = false |     enabled = false | ||||||
| } | } | ||||||
|  |  | ||||||
| val supportedVersions = listOf("1.19.4", "1.20.6", "1.21.1", "1.21.3", "1.21.4") | val supportedVersions = listOf("1.19.4", "1.20.6", "1.21.1", "1.21.3", "1.21.4", "1.21.5", "1.21.6", "1.21.7", "1.21.8") | ||||||
| tasks { | tasks { | ||||||
|     register("cacheLatestFaweArtifact") { |     register("cacheLatestFaweArtifact") { | ||||||
|         val lastSuccessfulBuildUrl = uri("https://ci.athion.net/job/FastAsyncWorldEdit/lastSuccessfulBuild/api/json").toURL() |         val lastSuccessfulBuildUrl = uri("https://ci.athion.net/job/FastAsyncWorldEdit/lastSuccessfulBuild/api/json").toURL() | ||||||
|         val artifact = ((JsonSlurper().parse(lastSuccessfulBuildUrl) as Map<*, *>)["artifacts"] as List<*>) |         val artifact = ((JsonSlurper().parse(lastSuccessfulBuildUrl) as Map<*, *>)["artifacts"] as List<*>) | ||||||
|                 .map { it as Map<*, *> } |                 .map { it as Map<*, *> } | ||||||
|                 .map { it["fileName"] as String } |                 .map { it["fileName"] as String } | ||||||
|                 .first { it -> it.contains("Bukkit") } |                 .first { it -> it.contains("Paper") } | ||||||
|         project.ext["faweArtifact"] = artifact |         project.ext["faweArtifact"] = artifact | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,42 +2,42 @@ | |||||||
| # Platform expectations | # Platform expectations | ||||||
| paper = "1.20.4-R0.1-SNAPSHOT" | paper = "1.20.4-R0.1-SNAPSHOT" | ||||||
| guice = "7.0.0" | guice = "7.0.0" | ||||||
| spotbugs = "4.9.1" | spotbugs = "4.9.8" | ||||||
| checkerqual = "3.49.0" | checkerqual = "3.51.1" | ||||||
| gson = "2.10" | gson = "2.10" | ||||||
| guava = "31.1-jre" | guava = "31.1-jre" | ||||||
| snakeyaml = "2.0" | snakeyaml = "2.0" | ||||||
| adventure = "4.19.0" | adventure = "4.25.0" | ||||||
| adventure-bukkit = "4.3.4" | adventure-bukkit = "4.4.1" | ||||||
| log4j = "2.19.0" | log4j = "2.19.0" | ||||||
|  |  | ||||||
| # Plugins | # Plugins | ||||||
| worldedit = "7.2.20" | worldedit = "7.2.20" | ||||||
| fawe = "2.13.0" | fawe = "2.14.0" | ||||||
| placeholderapi = "2.11.6" | placeholderapi = "2.11.6" | ||||||
| luckperms = "5.4" | luckperms = "5.5" | ||||||
| essentialsx = "2.20.1" | essentialsx = "2.21.2" | ||||||
| mvdwapi = "3.1.1" | mvdwapi = "3.1.1" | ||||||
|  |  | ||||||
| # Third party | # Third party | ||||||
| prtree = "2.0.1" | prtree = "2.0.1" | ||||||
| aopalliance = "1.0" | aopalliance = "1.0" | ||||||
| cloud-services = "1.8.4" | cloud-services = "1.8.4" | ||||||
| arkitektonika = "2.1.3" | arkitektonika = "2.1.4" | ||||||
| squirrelid = "0.3.2" | squirrelid = "0.3.2" | ||||||
| paster = "1.1.6" | paster = "1.1.7" | ||||||
| bstats = "3.1.0" | bstats = "3.1.0" | ||||||
| paperlib = "1.0.8" | paperlib = "1.0.8" | ||||||
| informative-annotations = "1.5" | informative-annotations = "1.6" | ||||||
| vault = "1.7.1" | vault = "1.7.1" | ||||||
| serverlib = "2.3.6" | serverlib = "2.3.7" | ||||||
|  |  | ||||||
| # Gradle plugins | # Gradle plugins | ||||||
| shadow = "8.3.6" | shadow = "9.2.2" | ||||||
| grgit = "4.1.1" | grgit = "4.1.1" | ||||||
| spotless = "7.0.2" | spotless = "8.0.0" | ||||||
| nexus = "2.0.0" | publish = "0.34.0" | ||||||
| runPaper = "2.3.1" | runPaper = "3.0.2" | ||||||
|  |  | ||||||
| [libraries] | [libraries] | ||||||
| # Platform expectations | # Platform expectations | ||||||
| @@ -81,5 +81,5 @@ serverlib = { group = "dev.notmyfault.serverlib", name = "ServerLib", version.re | |||||||
| shadow = { id = "com.gradleup.shadow", version.ref = "shadow" } | shadow = { id = "com.gradleup.shadow", version.ref = "shadow" } | ||||||
| grgit = { id = "org.ajoberstar.grgit", version.ref = "grgit" } | grgit = { id = "org.ajoberstar.grgit", version.ref = "grgit" } | ||||||
| spotless = { id = "com.diffplug.spotless", version.ref = "spotless" } | spotless = { id = "com.diffplug.spotless", version.ref = "spotless" } | ||||||
| nexus = { id = "io.github.gradle-nexus.publish-plugin", version.ref = "nexus" } | publish = { id = "com.vanniktech.maven.publish", version.ref = "publish" } | ||||||
| runPaper = { id = "xyz.jpenilla.run-paper", version.ref = "runPaper" } | runPaper = { id = "xyz.jpenilla.run-paper", version.ref = "runPaper" } | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								gradle/wrapper/gradle-wrapper.jar
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gradle/wrapper/gradle-wrapper.jar
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										2
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| distributionBase=GRADLE_USER_HOME | distributionBase=GRADLE_USER_HOME | ||||||
| distributionPath=wrapper/dists | distributionPath=wrapper/dists | ||||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.12.1-bin.zip | distributionUrl=https\://services.gradle.org/distributions/gradle-9.1.0-bin.zip | ||||||
| networkTimeout=10000 | networkTimeout=10000 | ||||||
| validateDistributionUrl=true | validateDistributionUrl=true | ||||||
| zipStoreBase=GRADLE_USER_HOME | zipStoreBase=GRADLE_USER_HOME | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								gradlew
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								gradlew
									
									
									
									
										vendored
									
									
								
							| @@ -1,7 +1,7 @@ | |||||||
| #!/bin/sh | #!/bin/sh | ||||||
|  |  | ||||||
| # | # | ||||||
| # Copyright © 2015-2021 the original authors. | # Copyright © 2015 the original authors. | ||||||
| # | # | ||||||
| # Licensed under the Apache License, Version 2.0 (the "License"); | # Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
| # you may not use this file except in compliance with the License. | # you may not use this file except in compliance with the License. | ||||||
| @@ -114,7 +114,6 @@ case "$( uname )" in                #( | |||||||
|   NONSTOP* )        nonstop=true ;; |   NONSTOP* )        nonstop=true ;; | ||||||
| esac | esac | ||||||
|  |  | ||||||
| CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # Determine the Java command to use to start the JVM. | # Determine the Java command to use to start the JVM. | ||||||
| @@ -172,7 +171,6 @@ fi | |||||||
| # For Cygwin or MSYS, switch paths to Windows format before running java | # For Cygwin or MSYS, switch paths to Windows format before running java | ||||||
| if "$cygwin" || "$msys" ; then | if "$cygwin" || "$msys" ; then | ||||||
|     APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) |     APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) | ||||||
|     CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" ) |  | ||||||
|  |  | ||||||
|     JAVACMD=$( cygpath --unix "$JAVACMD" ) |     JAVACMD=$( cygpath --unix "$JAVACMD" ) | ||||||
|  |  | ||||||
| @@ -205,15 +203,14 @@ fi | |||||||
| DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' | DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' | ||||||
|  |  | ||||||
| # Collect all arguments for the java command: | # Collect all arguments for the java command: | ||||||
| #   * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments, | #   * DEFAULT_JVM_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments, | ||||||
| #     and any embedded shellness will be escaped. | #     and any embedded shellness will be escaped. | ||||||
| #   * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be | #   * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be | ||||||
| #     treated as '${Hostname}' itself on the command line. | #     treated as '${Hostname}' itself on the command line. | ||||||
|  |  | ||||||
| set -- \ | set -- \ | ||||||
|         "-Dorg.gradle.appname=$APP_BASE_NAME" \ |         "-Dorg.gradle.appname=$APP_BASE_NAME" \ | ||||||
|         -classpath "$CLASSPATH" \ |         -jar "$APP_HOME/gradle/wrapper/gradle-wrapper.jar" \ | ||||||
|         org.gradle.wrapper.GradleWrapperMain \ |  | ||||||
|         "$@" |         "$@" | ||||||
|  |  | ||||||
| # Stop when "xargs" is not available. | # Stop when "xargs" is not available. | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								gradlew.bat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								gradlew.bat
									
									
									
									
										vendored
									
									
								
							| @@ -70,11 +70,10 @@ goto fail | |||||||
| :execute | :execute | ||||||
| @rem Setup the command line | @rem Setup the command line | ||||||
|  |  | ||||||
| set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @rem Execute Gradle | @rem Execute Gradle | ||||||
| "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -jar "%APP_HOME%\gradle\wrapper\gradle-wrapper.jar" %* | ||||||
|  |  | ||||||
| :end | :end | ||||||
| @rem End local scope for the variables with windows NT shell | @rem End local scope for the variables with windows NT shell | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user