mirror of
				https://github.com/IntellectualSites/PlotSquared.git
				synced 2025-10-22 22:23:45 +02:00 
			
		
		
		
	Compare commits
	
		
			9 Commits
		
	
	
		
			fix/genera
			...
			feat/chore
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | f2924f8a73 | ||
|   | 670c241d76 | ||
|   | 9c64c2e5fa | ||
|   | fb2165e4e0 | ||
|   | 30867354cc | ||
|   | 36de196d3d | ||
|   | 60f81c4732 | ||
|   | d9a46635fd | ||
|   | 424806c5f4 | 
							
								
								
									
										3
									
								
								.github/renovate.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.github/renovate.json
									
									
									
									
										vendored
									
									
								
							| @@ -2,8 +2,7 @@ | |||||||
|   "$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.4.0 |         uses: Ilshidur/action-discord@0.3.2 | ||||||
|         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@v5 |         uses: actions/checkout@v4 | ||||||
|       - name: Validate Gradle Wrapper |       - name: Validate Gradle Wrapper | ||||||
|         uses: gradle/actions/wrapper-validation@v5 |         uses: gradle/actions/wrapper-validation@v4 | ||||||
|       - name: Setup Java |       - name: Setup Java | ||||||
|         uses: actions/setup-java@v5 |         uses: actions/setup-java@v4 | ||||||
|         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@v5 |         uses: actions/checkout@v4 | ||||||
|       - name: Validate Gradle Wrapper |       - name: Validate Gradle Wrapper | ||||||
|         uses: gradle/actions/wrapper-validation@v5 |         uses: gradle/actions/wrapper-validation@v4 | ||||||
|       - name: Setup Java |       - name: Setup Java | ||||||
|         uses: actions/setup-java@v5 |         uses: actions/setup-java@v4 | ||||||
|         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 publishAndReleaseToMavenCentral --no-configuration-cache |         run: ./gradlew publishToSonatype closeAndReleaseSonatypeStagingRepository | ||||||
|         env: |         env: | ||||||
|           ORG_GRADLE_PROJECT_mavenCentralUsername: ${{ secrets.CENTRAL_USERNAME }} |           ORG_GRADLE_PROJECT_sonatypeUsername: ${{ secrets.SONATYPE_USERNAME }} | ||||||
|           ORG_GRADLE_PROJECT_mavenCentralPassword: ${{ secrets.CENTRAL_PASSWORD }} |           ORG_GRADLE_PROJECT_sonatypePassword: ${{ secrets.SONATYPE_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 publishAllPublicationsToMavenCentralRepository |         run: ./gradlew publishToSonatype | ||||||
|         env: |         env: | ||||||
|           ORG_GRADLE_PROJECT_mavenCentralUsername: ${{ secrets.CENTRAL_USERNAME }} |           ORG_GRADLE_PROJECT_sonatypeUsername: ${{ secrets.SONATYPE_USERNAME }} | ||||||
|           ORG_GRADLE_PROJECT_mavenCentralPassword: ${{ secrets.CENTRAL_PASSWORD }} |           ORG_GRADLE_PROJECT_sonatypePassword: ${{ secrets.SONATYPE_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@v5 |         uses: actions/checkout@v4 | ||||||
|       - name: Setup Java |       - name: Setup Java | ||||||
|         uses: actions/setup-java@v5 |         uses: actions/setup-java@v4 | ||||||
|         with: |         with: | ||||||
|           distribution: temurin |           distribution: temurin | ||||||
|           java-version: 21 |           java-version: 21 | ||||||
|       - name: Initialize CodeQL |       - name: Initialize CodeQL | ||||||
|         uses: github/codeql-action/init@v4 |         uses: github/codeql-action/init@v3 | ||||||
|         with: |         with: | ||||||
|           languages: ${{ matrix.language }} |           languages: ${{ matrix.language }} | ||||||
|       - name: Autobuild |       - name: Autobuild | ||||||
|         uses: github/codeql-action/autobuild@v4 |         uses: github/codeql-action/autobuild@v3 | ||||||
|       - name: Perform CodeQL Analysis |       - name: Perform CodeQL Analysis | ||||||
|         uses: github/codeql-action/analyze@v4 |         uses: github/codeql-action/analyze@v3 | ||||||
|   | |||||||
| @@ -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/releases/") |         url = uri("https://repo.extendedclip.com/content/repositories/placeholderapi/") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     maven { |     maven { | ||||||
| @@ -17,19 +17,6 @@ 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) | ||||||
|  |  | ||||||
| @@ -41,13 +28,20 @@ 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 | ||||||
| @@ -80,7 +74,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.squirrelid", "com.plotsquared.squirrelid") |     relocate("org.enginehub", "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") | ||||||
|   | |||||||
| @@ -252,11 +252,6 @@ 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() { | ||||||
| @@ -837,8 +832,7 @@ 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 "END_CRYSTAL": |                         case "ENDER_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": | ||||||
| @@ -956,8 +950,6 @@ 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", "END_CRYSTAL", "ENDER_CRYSTAL", "ENDER_PEARL", "ENDER_SIGNAL", "EXPERIENCE_ORB", "FALLING_BLOCK", "FIREBALL", |             case "ARROW", "EGG", "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", "HAPPY_GHAST", "GHASTLING", "MAGMA_CUBE", "SQUID", "PIG_ZOMBIE", "HOGLIN", |             case "SKELETON", "WITHER_SKELETON", "GUARDIAN", "ELDER_GUARDIAN", "GHAST", "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", "END_CRYSTAL", "ENDER_CRYSTAL", "ENDER_PEARL", "ENDER_SIGNAL", "DROPPED_ITEM", "EXPERIENCE_ORB", "FALLING_BLOCK", |             case "ARROW", "EGG", "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", "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" -> { |             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" -> { | ||||||
|                 restoreLiving((LivingEntity) entity); |                 restoreLiving((LivingEntity) entity); | ||||||
|                 return entity; |                 return entity; | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -90,7 +90,12 @@ 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,7 +36,6 @@ 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,6 +28,7 @@ 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; | ||||||
| @@ -41,11 +42,31 @@ 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() { | ||||||
|     } |     } | ||||||
| @@ -113,7 +134,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(block.getLocation()); |         Location location = BukkitUtil.adapt(blocks.get(0).getLocation()); | ||||||
|  |  | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
| @@ -163,7 +184,7 @@ public class BlockEventListener117 implements Listener { | |||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         if (event.getNewState().getType().name().contains("COPPER")) { |         if (COPPER_OXIDIZING.contains(event.getNewState().getType())) { | ||||||
|             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", "SILVERFISH_BLOCK", "ENDER_PEARL", |             case "REINFORCEMENTS", "NATURAL", "MOUNT", "PATROL", "RAID", "SHEARED", "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,7 +29,6 @@ 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; | ||||||
| @@ -39,7 +38,6 @@ 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; | ||||||
| @@ -50,7 +48,6 @@ 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; | ||||||
| @@ -202,7 +199,7 @@ public class PaperListener implements Listener { | |||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             case "REINFORCEMENTS", "NATURAL", "MOUNT", "PATROL", "RAID", "SILVERFISH_BLOCK", "ENDER_PEARL", "TRAP", "VILLAGE_DEFENSE", "VILLAGE_INVASION", "BEEHIVE", "CHUNK_GEN" -> { |             case "REINFORCEMENTS", "NATURAL", "MOUNT", "PATROL", "RAID", "SHEARED", "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); | ||||||
| @@ -461,21 +458,6 @@ 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,11 +64,9 @@ 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; | ||||||
| @@ -79,9 +77,7 @@ 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; | ||||||
| @@ -92,8 +88,10 @@ 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; | ||||||
| @@ -156,12 +154,9 @@ 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; | ||||||
|  |  | ||||||
| @@ -185,17 +180,7 @@ 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", | ||||||
| @@ -548,14 +533,12 @@ 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) { | ||||||
| @@ -610,20 +593,14 @@ public class PlayerEventListener implements Listener { | |||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|                 Plot plot = area.getPlot(location); |                 Plot plot = area.getPlot(location); | ||||||
|                 if (plot != null && !plot.equals(lastPlot)) { |                 if (plot != null) { | ||||||
|                     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 |                     if (result || (plot.getFlag(UntrustedVisitFlag.class) && plot.getHomeSynchronous().equals(BukkitUtil.adaptComplete(to)))) { | ||||||
|                             .getHomeSynchronous() |                         plotListener.plotEntry(pp, plot); | ||||||
|                             .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"), | ||||||
| @@ -962,15 +939,12 @@ 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( |             builder.tag("msg", Tag.inserting(MiniMessage.miniMessage().deserialize( | ||||||
|                     "msg", Tag.inserting(MiniMessage.miniMessage().deserialize( |  | ||||||
|                     message, |                     message, | ||||||
|                             TagResolver.resolver( |                     TagResolver.resolver(StandardTags.color(), StandardTags.gradient(), | ||||||
|                                     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))); | ||||||
|         } |         } | ||||||
| @@ -1279,9 +1253,7 @@ 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 |                 if (INTERACTABLE_MATERIALS != null ? INTERACTABLE_MATERIALS.contains(blockType.name()) : blockType.isInteractable()) { | ||||||
|                         ? INTERACTABLE_MATERIALS.contains(blockType.name()) |  | ||||||
|                         : blockType.isInteractable()) { |  | ||||||
|                     if (!player.isSneaking()) { |                     if (!player.isSneaking()) { | ||||||
|                         break; |                         break; | ||||||
|                     } |                     } | ||||||
| @@ -1324,17 +1296,6 @@ 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; | ||||||
| @@ -1409,7 +1370,22 @@ 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) { | ||||||
|         final Block block = event.getBlock(); |         BlockFace bf = event.getBlockFace(); | ||||||
|  |         // 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) { | ||||||
| @@ -1417,16 +1393,6 @@ 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; | ||||||
| @@ -1498,16 +1464,6 @@ 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,7 +47,6 @@ 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; | ||||||
| @@ -158,26 +157,14 @@ public class ProjectileEventListener implements Listener { | |||||||
|  |  | ||||||
|     @EventHandler |     @EventHandler | ||||||
|     public void onProjectileHit(ProjectileHitEvent event) { |     public void onProjectileHit(ProjectileHitEvent event) { | ||||||
|         if (cancelProjectileHit(event.getEntity())) { |         Projectile entity = 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 false; |             return; | ||||||
|         } |         } | ||||||
|         PlotArea area = location.getPlotArea(); |         PlotArea area = location.getPlotArea(); | ||||||
|         if (area == null) { |         if (area == null) { | ||||||
|             return false; |             return; | ||||||
|         } |         } | ||||||
|         Plot plot = area.getPlot(location); |         Plot plot = area.getPlot(location); | ||||||
|         ProjectileSource shooter = entity.getShooter(); |         ProjectileSource shooter = entity.getShooter(); | ||||||
| @@ -185,14 +172,15 @@ 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 false; |                         return; | ||||||
|                     } |                     } | ||||||
|                 } else if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, ProjectilesFlag.class, true)) { |                 } else if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, ProjectilesFlag.class, true)) { | ||||||
|                     return false; |                     return; | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 entity.remove(); |                 entity.remove(); | ||||||
|                 return true; |                 event.setCancelled(true); | ||||||
|  |                 return; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             PlotPlayer<?> pp = BukkitUtil.adapt((Player) shooter); |             PlotPlayer<?> pp = BukkitUtil.adapt((Player) shooter); | ||||||
| @@ -201,36 +189,38 @@ public class ProjectileEventListener implements Listener { | |||||||
|                         Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED |                         Permission.PERMISSION_ADMIN_PROJECTILE_UNOWNED | ||||||
|                 )) { |                 )) { | ||||||
|                     entity.remove(); |                     entity.remove(); | ||||||
|                     return true; |                     event.setCancelled(true); | ||||||
|                 } |                 } | ||||||
|                 return false; |                 return; | ||||||
|             } |             } | ||||||
|             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 false; |                 return; | ||||||
|             } |             } | ||||||
|             entity.remove(); |             entity.remove(); | ||||||
|             return true; |             event.setCancelled(true); | ||||||
|  |             return; | ||||||
|         } |         } | ||||||
|         if (!(shooter instanceof Entity) && shooter != null) { |         if (!(shooter instanceof Entity) && shooter != null) { | ||||||
|             if (plot == null) { |             if (plot == null) { | ||||||
|                 entity.remove(); |                 entity.remove(); | ||||||
|                 return true; |                 event.setCancelled(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(); | ||||||
|                 return true; |                 event.setCancelled(true); | ||||||
|  |                 return; | ||||||
|             } |             } | ||||||
|             Plot sPlot = area.getOwnedPlotAbs(sLoc); |             Plot sPlot = area.getOwnedPlotAbs(sLoc); | ||||||
|             if (sPlot == null || !PlotHandler.sameOwners(plot, sPlot)) { |             if (sPlot == null || !PlotHandler.sameOwners(plot, sPlot)) { | ||||||
|                 entity.remove(); |                 entity.remove(); | ||||||
|                 return true; |                 event.setCancelled(true); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         return false; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -44,7 +44,6 @@ 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; | ||||||
|  |  | ||||||
| @@ -227,22 +226,16 @@ 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) | ||||||
|                     .orTimeout(10L, TimeUnit.SECONDS) |                     .completeOnTimeout(null, 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) { | ||||||
|                             if (shouldGen) { |                             LOGGER.warn("Timed out awaiting chunk load {}", chunk); | ||||||
|                                 LOGGER.error("Null chunk returned for chunk at {}", chunk); |                             this.requestedChunks.offer(chunk); | ||||||
|                             } |  | ||||||
|                         } else if (PlotSquared.get().isMainThread(Thread.currentThread())) { |                         } else if (PlotSquared.get().isMainThread(Thread.currentThread())) { | ||||||
|                             this.processChunk(chunkObject); |                             this.processChunk(chunkObject); | ||||||
|                         } else { |                         } else { | ||||||
|   | |||||||
| @@ -52,7 +52,6 @@ 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 { | ||||||
| @@ -211,13 +210,8 @@ 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.INSTANCE.restore( |                             StateWrapper sw = new StateWrapper(tag); | ||||||
|                                     getWorld().getName(), |                             sw.restoreTag(getWorld().getName(), blockVector3.getX(), blockVector3.getY(), blockVector3.getZ()); | ||||||
|                                     blockVector3.getX(), |  | ||||||
|                                     blockVector3.getY(), |  | ||||||
|                                     blockVector3.getZ(), |  | ||||||
|                                     tag |  | ||||||
|                             ); |  | ||||||
|                         } |                         } | ||||||
|                     }); |                     }); | ||||||
|                 } |                 } | ||||||
| @@ -301,7 +295,9 @@ 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.INSTANCE.restore(existing, Objects.requireNonNull(tag)); |                 StateWrapper sw = new StateWrapper(tag); | ||||||
|  |  | ||||||
|  |                 sw.restoreTag(existing); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -34,8 +34,6 @@ 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. | ||||||
|  * |  * | ||||||
| @@ -46,6 +44,7 @@ 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 | ||||||
| @@ -65,11 +64,20 @@ 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 { | ||||||
|                 StateWrapper.INSTANCE.restore(limitedRegion.getBlockState(x, y, z).getBlock(), Objects.requireNonNull(tag)); |                 if (useOtherRestoreTagMethod && getWorld() != null) { | ||||||
|  |                     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; | ||||||
| @@ -105,8 +113,9 @@ 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 StateWrapper.INSTANCE.restore(limitedRegion.getBlockState(x, y, z).getBlock(), tag); |             return 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; | ||||||
|   | |||||||
| @@ -27,8 +27,6 @@ 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. | ||||||
|  */ |  */ | ||||||
| @@ -41,8 +39,8 @@ public class BukkitSchematicHandler extends SchematicHandler { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean restoreTile(QueueCoordinator queue, CompoundTag tag, int x, int y, int z) { |     public boolean restoreTile(QueueCoordinator queue, CompoundTag ct, int x, int y, int z) { | ||||||
|         return StateWrapper.INSTANCE.restore(Objects.requireNonNull(queue.getWorld()).getName(), x, y, z, tag); |         return new StateWrapper(ct).restoreTag(queue.getWorld().getName(), x, y, z); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,80 +0,0 @@ | |||||||
| /* |  | ||||||
|  * PlotSquared, a land and world management plugin for Minecraft. |  | ||||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> |  | ||||||
|  * Copyright (C) IntellectualSites team and contributors |  | ||||||
|  * |  | ||||||
|  * This program is free software: you can redistribute it and/or modify |  | ||||||
|  * it under the terms of the GNU General Public License as published by |  | ||||||
|  * the Free Software Foundation, either version 3 of the License, or |  | ||||||
|  * (at your option) any later version. |  | ||||||
|  * |  | ||||||
|  * This program is distributed in the hope that it will be useful, |  | ||||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of |  | ||||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the |  | ||||||
|  * GNU General Public License for more details. |  | ||||||
|  * |  | ||||||
|  * You should have received a copy of the GNU General Public License |  | ||||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. |  | ||||||
|  */ |  | ||||||
| package com.plotsquared.bukkit.schematic; |  | ||||||
|  |  | ||||||
| import com.google.gson.JsonArray; |  | ||||||
| import com.google.gson.JsonElement; |  | ||||||
| import com.google.gson.JsonObject; |  | ||||||
| import com.google.gson.JsonPrimitive; |  | ||||||
| import com.google.gson.JsonSerializationContext; |  | ||||||
| import com.google.gson.JsonSerializer; |  | ||||||
| import com.sk89q.jnbt.ByteArrayTag; |  | ||||||
| import com.sk89q.jnbt.CompoundTag; |  | ||||||
| import com.sk89q.jnbt.IntArrayTag; |  | ||||||
| import com.sk89q.jnbt.ListTag; |  | ||||||
| import com.sk89q.jnbt.LongArrayTag; |  | ||||||
| import com.sk89q.jnbt.Tag; |  | ||||||
|  |  | ||||||
| import java.lang.reflect.Type; |  | ||||||
|  |  | ||||||
| final class NbtGsonSerializer implements JsonSerializer<Tag> { |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public JsonElement serialize(final Tag src, final Type typeOfSrc, final JsonSerializationContext context) { |  | ||||||
|         if (src instanceof CompoundTag compoundTag) { |  | ||||||
|             JsonObject object = new JsonObject(); |  | ||||||
|             compoundTag.getValue().forEach((s, tag) -> object.add(s, context.serialize(tag))); |  | ||||||
|             return object; |  | ||||||
|         } |  | ||||||
|         if (src instanceof ListTag listTag) { |  | ||||||
|             JsonArray array = new JsonArray(); |  | ||||||
|             listTag.getValue().forEach(tag -> array.add(context.serialize(tag))); |  | ||||||
|             return array; |  | ||||||
|         } |  | ||||||
|         if (src instanceof ByteArrayTag byteArrayTag) { |  | ||||||
|             JsonArray array = new JsonArray(); |  | ||||||
|             for (final byte b : byteArrayTag.getValue()) { |  | ||||||
|                 array.add(b); |  | ||||||
|             } |  | ||||||
|             return array; |  | ||||||
|         } |  | ||||||
|         if (src instanceof IntArrayTag intArrayTag) { |  | ||||||
|             JsonArray array = new JsonArray(); |  | ||||||
|             for (final int i : intArrayTag.getValue()) { |  | ||||||
|                 array.add(i); |  | ||||||
|             } |  | ||||||
|             return array; |  | ||||||
|         } |  | ||||||
|         if (src instanceof LongArrayTag longArrayTag) { |  | ||||||
|             JsonArray array = new JsonArray(); |  | ||||||
|             for (final long l : longArrayTag.getValue()) { |  | ||||||
|                 array.add(l); |  | ||||||
|             } |  | ||||||
|             return array; |  | ||||||
|         } |  | ||||||
|         if (src.getValue() instanceof Number number) { |  | ||||||
|             return new JsonPrimitive(number); |  | ||||||
|         } |  | ||||||
|         if (src.getValue() instanceof String string) { |  | ||||||
|             return new JsonPrimitive(string); |  | ||||||
|         } |  | ||||||
|         throw new IllegalArgumentException("Don't know how to serialize " + src); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -18,71 +18,332 @@ | |||||||
|  */ |  */ | ||||||
| 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.plotsquared.core.PlotSquared; | 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 io.papermc.lib.PaperLib; | ||||||
| import org.apache.logging.log4j.LogManager; | import org.apache.logging.log4j.LogManager; | ||||||
| import org.apache.logging.log4j.Logger; | 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; |  | ||||||
|  |  | ||||||
| @ApiStatus.Internal | import java.util.ArrayList; | ||||||
| public sealed interface StateWrapper permits StateWrapperSpigot, StateWrapper.Factory.NoopStateWrapper { | import java.util.HashMap; | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.Map; | ||||||
|  | import java.util.Map.Entry; | ||||||
|  | import java.util.Objects; | ||||||
|  | import java.util.UUID; | ||||||
|  |  | ||||||
|     StateWrapper INSTANCE = Factory.createStateWrapper(); | public class StateWrapper { | ||||||
|  |  | ||||||
|     boolean restore(final @NonNull Block block, final @NonNull CompoundTag data); |     public CompoundTag tag; | ||||||
|  |  | ||||||
|     default boolean restore(final String worldName, final int x, final int y, final int z, final CompoundTag data) { |     private boolean paperErrorTextureSent = false; | ||||||
|         final World world = BukkitUtil.getWorld(worldName); |     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + StateWrapper.class.getSimpleName()); | ||||||
|  |  | ||||||
|  |     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 this.restore(world.getBlockAt(x, y, z), data); |         return restoreTag(world.getBlockAt(x, y, z)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @ApiStatus.Internal |     /** | ||||||
|     final class Factory { |      * Restore the TileEntity data to the given block | ||||||
|  |      * | ||||||
|  |      * @param block Block to restore to | ||||||
|  |      * @return true if successful | ||||||
|  |      */ | ||||||
|  |     @SuppressWarnings("deprecation") // #setLine is needed for Spigot compatibility | ||||||
|  |     public boolean restoreTag(@NonNull Block block) { | ||||||
|  |         if (this.tag == null) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         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"); | ||||||
|  |  | ||||||
|         private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + StateWrapper.class.getSimpleName()); |                     if (player != null && !player.isEmpty()) { | ||||||
|  |  | ||||||
|         private static final String INITIALIZATION_ERROR_TEMPLATE = """ |  | ||||||
|                 Failed to initialize StateWrapper: {} |  | ||||||
|                 Block-/Tile-Entities, pasted by schematics for example, won't be updated with their respective block data. This affects things like sign text, banner patterns, skulls, etc. |  | ||||||
|                 Try updating your Server Software, PlotSquared and WorldEdit / FastAsyncWorldEdit first. If the issue persists, report it on the issue tracker. |  | ||||||
|                 """; |  | ||||||
|  |  | ||||||
|         private static StateWrapper createStateWrapper() { |  | ||||||
|             int[] serverVersion = PlotSquared.platform().serverVersion(); |  | ||||||
|             if (PaperLib.isPaper() && (serverVersion[1] == 21 && serverVersion[2] >= 5) || serverVersion[1] > 21) { |  | ||||||
|                         try { |                         try { | ||||||
|                     return new StateWrapperPaper1_21_5(); |                             skull.setOwningPlayer(Bukkit.getOfflinePlayer(player)); | ||||||
|  |                             skull.update(true); | ||||||
|                         } catch (Exception e) { |                         } catch (Exception e) { | ||||||
|                     LOGGER.error("Failed to initialize Paper-specific state wrapper, falling back to Spigot", e); |                             e.printStackTrace(); | ||||||
|                         } |                         } | ||||||
|             } |                         return true; | ||||||
|             try { |  | ||||||
|                 return new StateWrapperSpigot(); |  | ||||||
|             } catch (Exception e) { |  | ||||||
|                 LOGGER.error(INITIALIZATION_ERROR_TEMPLATE, StateWrapperSpigot.class.getSimpleName(), e); |  | ||||||
|             } |  | ||||||
|             return new NoopStateWrapper(); |  | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|  |                     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; | ||||||
|  |  | ||||||
|         @ApiStatus.Internal |                 } | ||||||
|         static final class NoopStateWrapper implements StateWrapper { |                 return false; | ||||||
|  |             } | ||||||
|             @Override |             case "banner" -> { | ||||||
|             public boolean restore(final @NonNull Block block, final @NonNull CompoundTag data) { |                 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; |         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; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,155 +0,0 @@ | |||||||
| /* |  | ||||||
|  * PlotSquared, a land and world management plugin for Minecraft. |  | ||||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> |  | ||||||
|  * Copyright (C) IntellectualSites team and contributors |  | ||||||
|  * |  | ||||||
|  * This program is free software: you can redistribute it and/or modify |  | ||||||
|  * it under the terms of the GNU General Public License as published by |  | ||||||
|  * the Free Software Foundation, either version 3 of the License, or |  | ||||||
|  * (at your option) any later version. |  | ||||||
|  * |  | ||||||
|  * This program is distributed in the hope that it will be useful, |  | ||||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of |  | ||||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the |  | ||||||
|  * GNU General Public License for more details. |  | ||||||
|  * |  | ||||||
|  * You should have received a copy of the GNU General Public License |  | ||||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. |  | ||||||
|  */ |  | ||||||
| package com.plotsquared.bukkit.schematic; |  | ||||||
|  |  | ||||||
| import com.google.gson.Gson; |  | ||||||
| import com.google.gson.GsonBuilder; |  | ||||||
| import com.google.gson.JsonElement; |  | ||||||
| import com.google.gson.JsonParser; |  | ||||||
| import com.google.gson.JsonSyntaxException; |  | ||||||
| import com.sk89q.jnbt.CompoundTag; |  | ||||||
| import com.sk89q.jnbt.Tag; |  | ||||||
| import org.apache.logging.log4j.LogManager; |  | ||||||
| import org.apache.logging.log4j.Logger; |  | ||||||
| import org.bukkit.DyeColor; |  | ||||||
| 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.util.Arrays; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.Locale; |  | ||||||
|  |  | ||||||
| final class StateWrapperPaper1_21_5 extends StateWrapperSpigot { |  | ||||||
|  |  | ||||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + StateWrapperPaper1_21_5.class.getSimpleName()); |  | ||||||
|  |  | ||||||
|     private static final Gson GSON = new GsonBuilder().registerTypeHierarchyAdapter(Tag.class, new NbtGsonSerializer()).create(); |  | ||||||
|     private static Object KYORI_GSON_SERIALIZER = null; |  | ||||||
|     private static MethodHandle GSON_SERIALIZER_DESERIALIZE_TREE = null; |  | ||||||
|     private static MethodHandle BUKKIT_SIGN_SIDE_LINE_SET = null; |  | ||||||
|  |  | ||||||
|     public StateWrapperPaper1_21_5() { |  | ||||||
|         super(); |  | ||||||
|         try { |  | ||||||
|             initializeSignHack(); |  | ||||||
|             LOGGER.info("Using {} for block data population", StateWrapperPaper1_21_5.class.getSimpleName()); |  | ||||||
|         } catch (Throwable e) { |  | ||||||
|             throw new RuntimeException("Failed to initialize sign hack", e); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void postEntityStateLoad(final @NonNull BlockState blockState, final @NonNull CompoundTag data) throws Throwable { |  | ||||||
|         // signs need special handling during generation |  | ||||||
|         if (blockState instanceof Sign sign) { |  | ||||||
|             if (data.getValue().get("front_text") instanceof CompoundTag textTag) { |  | ||||||
|                 setSignTextHack(sign.getSide(Side.FRONT), textTag); |  | ||||||
|             } |  | ||||||
|             if (data.getValue().get("back_text") instanceof CompoundTag textTag) { |  | ||||||
|                 setSignTextHack(sign.getSide(Side.BACK), textTag); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public Logger logger() { |  | ||||||
|         return StateWrapperPaper1_21_5.LOGGER; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Set sign content on the bukkit tile entity. The server does not load sign content applied via the main logic |  | ||||||
|      * (CraftBlockEntity#load), as the SignEntity needs to have a valid ServerLevel assigned to it. |  | ||||||
|      * That's not possible on worldgen; therefore, this hack has to be used additionally. |  | ||||||
|      * <br /> |  | ||||||
|      * Modern sign content (non-plain-text sign lines) require Paper. |  | ||||||
|      * |  | ||||||
|      * @param side The sign side to apply data onto. |  | ||||||
|      * @param text The compound tag containing the data for the sign side ({@code front_text} / {@code back_text}) |  | ||||||
|      * @throws Throwable if something went wrong when reflectively updating the sign. |  | ||||||
|      */ |  | ||||||
|     private static void setSignTextHack(SignSide side, CompoundTag text) throws Throwable { |  | ||||||
|         if (text.containsKey("color")) { |  | ||||||
|             //noinspection UnstableApiUsage |  | ||||||
|             side.setColor(DyeColor.legacyValueOf(text.getString("color").toUpperCase(Locale.ROOT))); |  | ||||||
|         } |  | ||||||
|         if (text.containsKey("has_glowing_text")) { |  | ||||||
|             side.setGlowingText(text.getByte("has_glowing_text") == 1); |  | ||||||
|         } |  | ||||||
|         List<Tag> lines = text.getList("messages"); |  | ||||||
|         if (lines != null) { |  | ||||||
|             for (int i = 0; i < Math.min(lines.size(), 3); i++) { |  | ||||||
|                 Tag line = lines.get(i); |  | ||||||
|                 Object content = line.getValue(); |  | ||||||
|                 // Minecraft uses mixed lists / arrays in their sign texts. One line can be a complex component, whereas |  | ||||||
|                 // the following line could simply be a string. Those simpler lines are represented as `{"": ""}` (only in |  | ||||||
|                 // SNBT those will be shown as a standard string). |  | ||||||
|                 if (line instanceof CompoundTag compoundTag && compoundTag.getValue().containsKey("")) { |  | ||||||
|                     content = compoundTag.getValue().get(""); |  | ||||||
|                 } |  | ||||||
|                 // absolute garbage way to try to handle stringified components (pre 1.21.5) |  | ||||||
|                 else if (content instanceof String contentAsString && (contentAsString.startsWith("{") || contentAsString.startsWith("["))) { |  | ||||||
|                     try { |  | ||||||
|                         content = JsonParser.parseString(contentAsString); |  | ||||||
|                     } catch (JsonSyntaxException e) { |  | ||||||
|                         // well, it wasn't JSON after all |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 // serializes the line content from JNBT to Gson JSON objects, passes that to adventure and deserializes |  | ||||||
|                 // into an adventure component. |  | ||||||
|                 // pass all possible types of content into the deserializer (Strings, Compounds, Arrays), even though Strings |  | ||||||
|                 // could be set directly via Sign#setLine(int, String). The overhead is minimal, the serializer can handle |  | ||||||
|                 // strings - and we don't have to use the deprecated method. |  | ||||||
|                 BUKKIT_SIGN_SIDE_LINE_SET.invoke( |  | ||||||
|                         side, i, GSON_SERIALIZER_DESERIALIZE_TREE.invoke( |  | ||||||
|                                 KYORI_GSON_SERIALIZER, |  | ||||||
|                                 content instanceof JsonElement ? content : GSON.toJsonTree(content) |  | ||||||
|                         ) |  | ||||||
|                 ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static void initializeSignHack() throws Throwable { |  | ||||||
|         char[] dontObfuscate = new char[]{ |  | ||||||
|                 'n', 'e', 't', '.', 'k', 'y', 'o', 'r', 'i', '.', 'a', 'd', 'v', 'e', 'n', 't', 'u', 'r', 'e', '.', |  | ||||||
|                 't', 'e', 'x', 't', '.', 's', 'e', 'r', 'i', 'a', 'l', 'i', 'z', 'e', 'r', '.', 'g', 's', 'o', 'n', '.', |  | ||||||
|                 'G', 's', 'o', 'n', 'C', 'o', 'm', 'p', 'o', 'n', 'e', 'n', 't', 'S', 'e', 'r', 'i', 'a', 'l', 'i', 'z', 'e', 'r' |  | ||||||
|         }; |  | ||||||
|         Class<?> gsonComponentSerializerClass = Class.forName(new String(dontObfuscate)); |  | ||||||
|         KYORI_GSON_SERIALIZER = Arrays.stream(gsonComponentSerializerClass.getMethods()) |  | ||||||
|                 .filter(method -> method.getName().equals("gson")) |  | ||||||
|                 .findFirst() |  | ||||||
|                 .orElseThrow().invoke(null); |  | ||||||
|         GSON_SERIALIZER_DESERIALIZE_TREE = LOOKUP.unreflect(Arrays |  | ||||||
|                 .stream(gsonComponentSerializerClass.getMethods()) |  | ||||||
|                 .filter(method -> method.getName().equals("deserializeFromTree") && method.getParameterCount() == 1) |  | ||||||
|                 .findFirst() |  | ||||||
|                 .orElseThrow()); |  | ||||||
|         BUKKIT_SIGN_SIDE_LINE_SET = LOOKUP.unreflect(Arrays.stream(SignSide.class.getMethods()) |  | ||||||
|                 .filter(method -> method.getName().equals("line") && method.getParameterCount() == 2) |  | ||||||
|                 .findFirst() |  | ||||||
|                 .orElseThrow()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,210 +0,0 @@ | |||||||
| /* |  | ||||||
|  * PlotSquared, a land and world management plugin for Minecraft. |  | ||||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> |  | ||||||
|  * Copyright (C) IntellectualSites team and contributors |  | ||||||
|  * |  | ||||||
|  * This program is free software: you can redistribute it and/or modify |  | ||||||
|  * it under the terms of the GNU General Public License as published by |  | ||||||
|  * the Free Software Foundation, either version 3 of the License, or |  | ||||||
|  * (at your option) any later version. |  | ||||||
|  * |  | ||||||
|  * This program is distributed in the hope that it will be useful, |  | ||||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of |  | ||||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the |  | ||||||
|  * GNU General Public License for more details. |  | ||||||
|  * |  | ||||||
|  * You should have received a copy of the GNU General Public License |  | ||||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. |  | ||||||
|  */ |  | ||||||
| package com.plotsquared.bukkit.schematic; |  | ||||||
|  |  | ||||||
| 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.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.checkerframework.checker.nullness.qual.NonNull; |  | ||||||
|  |  | ||||||
| import java.lang.invoke.MethodHandle; |  | ||||||
| import java.lang.invoke.MethodHandles; |  | ||||||
| import java.lang.invoke.MethodType; |  | ||||||
| import java.lang.reflect.Method; |  | ||||||
| import java.util.function.Consumer; |  | ||||||
|  |  | ||||||
| sealed class StateWrapperSpigot implements StateWrapper permits StateWrapperPaper1_21_5 { |  | ||||||
|  |  | ||||||
|     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()); |  | ||||||
|     static final MethodHandles.Lookup LOOKUP = MethodHandles.lookup(); |  | ||||||
|  |  | ||||||
|     private static BukkitImplAdapter ADAPTER = null; |  | ||||||
|     private static Class<?> LIN_TAG_CLASS = null; |  | ||||||
|     private static Class<?> JNBT_TAG_CLASS = null; |  | ||||||
|     private static Class<?> CRAFT_BLOCK_ENTITY_STATE_CLASS = 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 TO_LIN_TAG = null; |  | ||||||
|  |  | ||||||
|     public StateWrapperSpigot() { |  | ||||||
|         try { |  | ||||||
|             findNbtCompoundClassType(clazz -> LIN_TAG_CLASS = clazz, clazz -> JNBT_TAG_CLASS = clazz); |  | ||||||
|             ReflectionUtils.RefClass worldEditPluginRefClass = ReflectionUtils.getRefClass(WorldEditPlugin.class); |  | ||||||
|             WorldEditPlugin worldEditPlugin = (WorldEditPlugin) worldEditPluginRefClass |  | ||||||
|                     .getMethod("getInstance") |  | ||||||
|                     .of(null) |  | ||||||
|                     .call(); |  | ||||||
|             ADAPTER = (BukkitImplAdapter) worldEditPluginRefClass |  | ||||||
|                     .getMethod("getBukkitImplAdapter") |  | ||||||
|                     .of(worldEditPlugin) |  | ||||||
|                     .call(); |  | ||||||
|             PAPERWEIGHT_ADAPTER_FROM_NATIVE = findPaperweightAdapterFromNativeMethodHandle( |  | ||||||
|                     ADAPTER.getClass(), LIN_TAG_CLASS, JNBT_TAG_CLASS |  | ||||||
|             ); |  | ||||||
|             TO_LIN_TAG = findToLinTagMethodHandle(LIN_TAG_CLASS); |  | ||||||
|         } catch (NoSuchMethodException | ClassNotFoundException | IllegalAccessException | NoCapablePlatformException e) { |  | ||||||
|             throw new RuntimeException("Failed to access required WorldEdit methods", e); |  | ||||||
|         } |  | ||||||
|         try { |  | ||||||
|             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); |  | ||||||
|         } catch (ClassNotFoundException | NoSuchMethodException | IllegalAccessException 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, |  | ||||||
|                     LIN_TAG_CLASS == null ? data : TO_LIN_TAG.invoke(data) |  | ||||||
|             ); |  | ||||||
|             // load block entity data |  | ||||||
|             CRAFT_BLOCK_ENTITY_STATE_LOAD_DATA.invoke(blockState, nativeTag); |  | ||||||
|  |  | ||||||
|             postEntityStateLoad(blockState, data); |  | ||||||
|  |  | ||||||
|             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; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void postEntityStateLoad(final @NonNull BlockState blockState, final @NonNull CompoundTag data) throws Throwable { |  | ||||||
|  |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Logger logger() { |  | ||||||
|         return StateWrapperSpigot.LOGGER; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Initialize the used NBT tag class. For modern FAWE and WE that'll be Lin - for older ones JNBT. |  | ||||||
|      * |  | ||||||
|      * @throws ClassNotFoundException if neither can be found. |  | ||||||
|      */ |  | ||||||
|     private static void findNbtCompoundClassType(Consumer<Class<?>> linClass, Consumer<Class<?>> jnbtClass) throws |  | ||||||
|             ClassNotFoundException { |  | ||||||
|         try { |  | ||||||
|             linClass.accept(Class.forName("org.enginehub.linbus.tree.LinTag")); |  | ||||||
|         } catch (ClassNotFoundException e) { |  | ||||||
|             jnbtClass.accept(Class.forName("com.sk89q.jnbt.Tag")); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 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. |  | ||||||
|      * |  | ||||||
|      * @param linTagClass {@code Tag} class of lin-bus, or {@code null} if not available. |  | ||||||
|      * @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(Class<?> linTagClass) throws ClassNotFoundException, |  | ||||||
|             NoSuchMethodException, IllegalAccessException { |  | ||||||
|         if (linTagClass == null) { |  | ||||||
|             return null; |  | ||||||
|         } |  | ||||||
|         return LOOKUP.findVirtual( |  | ||||||
|                 Class.forName("org.enginehub.linbus.tree.ToLinTag"), |  | ||||||
|                 "toLinTag", |  | ||||||
|                 MethodType.methodType(linTagClass) |  | ||||||
|         ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * 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/FAWE version pre LinBus introduction: {@code fromNative(org.sk89q.jnbt.Tag)}</li> |  | ||||||
|      *     <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 |  | ||||||
|      * @param linTagClass  The lin-bus {@code Tag} class, if existing - otherwise {@code null} |  | ||||||
|      * @param jnbtTagClass The jnbt {@code Tag} class, if lin-bus was not found in classpath - otherwise {@code null} |  | ||||||
|      * @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, Class<?> linTagClass, Class<?> jnbtTagClass |  | ||||||
|     ) throws IllegalAccessException, NoSuchMethodException { |  | ||||||
|         final MethodHandles.Lookup lookup = MethodHandles.privateLookupIn(adapterClass, LOOKUP); |  | ||||||
|         if (jnbtTagClass != null) { |  | ||||||
|             // usage of JNBT = identical method signatures for WE and FAWE |  | ||||||
|             return lookup.findVirtual(adapterClass, "fromNative", MethodType.methodType(Object.class, jnbtTagClass)); |  | ||||||
|         } |  | ||||||
|         try { |  | ||||||
|             // FAWE |  | ||||||
|             return lookup.findVirtual(adapterClass, "fromNativeLin", MethodType.methodType(Object.class, linTagClass)); |  | ||||||
|         } catch (NoSuchMethodException e) { |  | ||||||
|             // WE |  | ||||||
|             return lookup.findVirtual(adapterClass, "fromNative", MethodType.methodType(Object.class, linTagClass)); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static MethodHandle findCraftBlockEntityStateLoadDataMethodHandle(Class<?> craftBlockEntityStateClass) throws |  | ||||||
|             NoSuchMethodException, IllegalAccessException, ClassNotFoundException { |  | ||||||
|         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, ClassNotFoundException { |  | ||||||
|         for (final Method method : craftBlockEntityStateClass.getMethods()) { |  | ||||||
|             if (method.getReturnType().equals(Boolean.TYPE) && method.getParameterCount() == 2 && |  | ||||||
|                     method.getParameterTypes()[0] == Boolean.TYPE && method.getParameterTypes()[1] == Boolean.TYPE) { |  | ||||||
|                 return LOOKUP.unreflect(method); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         throw new NoSuchMethodException("Couldn't find method for #update(boolean, boolean) in " + craftBlockEntityStateClass.getName()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -354,17 +354,13 @@ 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(type); |                 BukkitAdapter.adapt(entity.getType()); | ||||||
|  |  | ||||||
|         if (EntityCategories.PLAYER.contains(entityType)) { |         if (EntityCategories.PLAYER.contains(entityType)) { | ||||||
|             return false; |             return false; | ||||||
|   | |||||||
| @@ -120,14 +120,6 @@ 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. | ||||||
|      * |      * | ||||||
|   | |||||||
| @@ -713,8 +713,6 @@ public class PlotSquared { | |||||||
|                 case CREATION_DATE_TIMESTAMP -> toReturn.addAll(sortPlotsByTimestamp(map.get(area))); |                 case CREATION_DATE_TIMESTAMP -> toReturn.addAll(sortPlotsByTimestamp(map.get(area))); | ||||||
|                 case DISTANCE_FROM_ORIGIN -> toReturn.addAll(sortPlotsByHash(map.get(area))); |                 case DISTANCE_FROM_ORIGIN -> toReturn.addAll(sortPlotsByHash(map.get(area))); | ||||||
|                 case LAST_MODIFIED -> toReturn.addAll(sortPlotsByModified(map.get(area))); |                 case LAST_MODIFIED -> toReturn.addAll(sortPlotsByModified(map.get(area))); | ||||||
|                 default -> { |  | ||||||
|                 } |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         return toReturn; |         return toReturn; | ||||||
|   | |||||||
| @@ -18,8 +18,6 @@ | |||||||
|  */ |  */ | ||||||
| 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; | ||||||
| @@ -32,14 +30,12 @@ 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, @Nullable final Path file) { |     Backup(final BackupProfile owner, final long creationTime, final Path file) { | ||||||
|         this.owner = owner; |         this.owner = owner; | ||||||
|         this.creationTime = creationTime; |         this.creationTime = creationTime; | ||||||
|         this.file = file; |         this.file = file; | ||||||
| @@ -53,7 +49,7 @@ public class Backup { | |||||||
|             try { |             try { | ||||||
|                 Files.deleteIfExists(file); |                 Files.deleteIfExists(file); | ||||||
|             } catch (final IOException e) { |             } catch (final IOException e) { | ||||||
|                 LOGGER.error("Error deleting backup at {}", file, e); |                 e.printStackTrace(); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -21,15 +21,14 @@ 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.exception.PlotSquaredException; | import com.plotsquared.core.player.ConsolePlayer; | ||||||
| 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 org.apache.logging.log4j.LogManager; | import net.kyori.adventure.text.minimessage.MiniMessage; | ||||||
| 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; | ||||||
|  |  | ||||||
| @@ -52,7 +51,7 @@ import java.util.concurrent.CompletableFuture; | |||||||
|  */ |  */ | ||||||
| public class PlayerBackupProfile implements BackupProfile { | public class PlayerBackupProfile implements BackupProfile { | ||||||
|  |  | ||||||
|     private static final Logger LOGGER = LogManager.getLogger("PlotSquared/" + PlayerBackupProfile.class.getSimpleName()); |     static final MiniMessage MINI_MESSAGE = MiniMessage.builder().build(); | ||||||
|  |  | ||||||
|     private final UUID owner; |     private final UUID owner; | ||||||
|     private final Plot plot; |     private final Plot plot; | ||||||
| @@ -88,7 +87,7 @@ public class PlayerBackupProfile implements BackupProfile { | |||||||
|                 Files.createDirectory(path); |                 Files.createDirectory(path); | ||||||
|             } |             } | ||||||
|         } catch (final Exception e) { |         } catch (final Exception e) { | ||||||
|             LOGGER.error("Error resolving {} from {}", child, parent, e); |             e.printStackTrace(); | ||||||
|         } |         } | ||||||
|         return path; |         return path; | ||||||
|     } |     } | ||||||
| @@ -105,7 +104,7 @@ public class PlayerBackupProfile implements BackupProfile { | |||||||
|                     try { |                     try { | ||||||
|                         Files.createDirectories(path); |                         Files.createDirectories(path); | ||||||
|                     } catch (IOException e) { |                     } catch (IOException e) { | ||||||
|                         LOGGER.error("Error creating directory {}", path, e); |                         e.printStackTrace(); | ||||||
|                         return Collections.emptyList(); |                         return Collections.emptyList(); | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
| @@ -118,11 +117,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) { | ||||||
|                             LOGGER.error("Error getting attributes for file {} to create backup", file, e); |                             e.printStackTrace(); | ||||||
|                         } |                         } | ||||||
|                     }); |                     }); | ||||||
|                 } catch (IOException e) { |                 } catch (IOException e) { | ||||||
|                     LOGGER.error("Error walking files from {}", path, e); |                     e.printStackTrace(); | ||||||
|                 } |                 } | ||||||
|                 backups.sort(Comparator.comparingLong(Backup::getCreationTime).reversed()); |                 backups.sort(Comparator.comparingLong(Backup::getCreationTime).reversed()); | ||||||
|                 return (this.backupCache = backups); |                 return (this.backupCache = backups); | ||||||
| @@ -134,7 +133,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) { | ||||||
|                 LOGGER.error("Error while listing backups", error); |                 error.printStackTrace(); | ||||||
|             } |             } | ||||||
|             backups.forEach(Backup::delete); |             backups.forEach(Backup::delete); | ||||||
|             this.backupCache = null; |             this.backupCache = null; | ||||||
| @@ -142,12 +141,10 @@ public class PlayerBackupProfile implements BackupProfile { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     public @NonNull Path getBackupDirectory() { |     public @NonNull Path getBackupDirectory() { | ||||||
|         return resolve( |         return resolve(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 | ||||||
| @@ -159,8 +156,7 @@ 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( |                 final boolean result = this.schematicHandler.exportAll(plots, getBackupDirectory().toFile(), | ||||||
|                         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)) | ||||||
|                 ); |                 ); | ||||||
| @@ -184,7 +180,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) { | ||||||
|                     LOGGER.error("Unsupported format for backup {}", backup.getFile(), e); |                     e.printStackTrace(); | ||||||
|                 } |                 } | ||||||
|                 if (schematic == null) { |                 if (schematic == null) { | ||||||
|                     future.completeExceptionally(new IllegalArgumentException( |                     future.completeExceptionally(new IllegalArgumentException( | ||||||
| @@ -204,9 +200,10 @@ public class PlayerBackupProfile implements BackupProfile { | |||||||
|                                     if (value) { |                                     if (value) { | ||||||
|                                         future.complete(null); |                                         future.complete(null); | ||||||
|                                     } else { |                                     } else { | ||||||
|                                         future.completeExceptionally(new PlotSquaredException( |                                         future.completeExceptionally(new RuntimeException(MINI_MESSAGE.escapeTags( | ||||||
|                                                 TranslatableCaption |                                                 TranslatableCaption | ||||||
|                                                         .of("schematics.schematic_paste_failed"))); |                                                         .of("schematics.schematic_paste_failed") | ||||||
|  |                                                         .getComponent(ConsolePlayer.getConsole())))); | ||||||
|                                     } |                                     } | ||||||
|                                 } |                                 } | ||||||
|                             } |                             } | ||||||
|   | |||||||
| @@ -32,8 +32,6 @@ 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; | ||||||
|  |  | ||||||
| @@ -49,7 +47,6 @@ 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; | ||||||
| @@ -115,12 +112,7 @@ public class SimpleBackupManager implements BackupManager { | |||||||
|                                 TagResolver.resolver("reason", Tag.inserting(Component.text(throwable.getMessage()))) |                                 TagResolver.resolver("reason", Tag.inserting(Component.text(throwable.getMessage()))) | ||||||
|                         ); |                         ); | ||||||
|                     } |                     } | ||||||
|                     LOGGER.error( |                     throwable.printStackTrace(); | ||||||
|                             "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")); | ||||||
| @@ -136,7 +128,6 @@ 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,6 +21,7 @@ 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; | ||||||
| @@ -57,7 +58,8 @@ public class Alias extends SubCommand { | |||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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,7 +24,6 @@ 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; | ||||||
| @@ -33,8 +32,6 @@ 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; | ||||||
| @@ -60,8 +57,6 @@ 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 | ||||||
| @@ -331,28 +326,19 @@ 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( |                         TagResolver.resolver("plot", Tag.inserting( | ||||||
|                                 "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(reason)) |                                 TagResolver.resolver("reason", Tag.inserting(Component.text(throwable.getMessage()))) | ||||||
|                         ); |                         ); | ||||||
|                         LOGGER.error("Error loading player ({}) backup", player.getName(), throwable); |                         throwable.printStackTrace(); | ||||||
|                         return; |                     } else { | ||||||
|                     } |  | ||||||
|                         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"), | ||||||
| @@ -378,8 +364,7 @@ public final class Backup extends Command { | |||||||
|                                         ) |                                         ) | ||||||
|                                 ); |                                 ); | ||||||
|                             } else { |                             } else { | ||||||
|                             CmdConfirm.addPending( |                                 CmdConfirm.addPending(player, "/plot backup load " + number, | ||||||
|                                     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) { | ||||||
| @@ -397,6 +382,7 @@ public final class Backup extends Command { | |||||||
|                                 ); |                                 ); | ||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|  |                     } | ||||||
|                 }); |                 }); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -20,7 +20,6 @@ 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; | ||||||
| @@ -85,9 +84,8 @@ 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( | ||||||
|                 plotCount + plots.size() <= player.getAllowedPlots(), |                 player.getPlotCount() + 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()))) | ||||||
|         ); |         ); | ||||||
| @@ -146,7 +144,6 @@ 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,6 +26,7 @@ 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; | ||||||
| @@ -71,7 +72,8 @@ public class Claim extends SubCommand { | |||||||
|         if (args.length >= 1) { |         if (args.length >= 1) { | ||||||
|             schematic = args[0]; |             schematic = args[0]; | ||||||
|         } |         } | ||||||
|         Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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; | ||||||
| @@ -88,7 +90,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(plot.getWorldName()); |                 player.getPlotCount(location.getWorldName()); | ||||||
|  |  | ||||||
|         final PlotArea area = plot.getArea(); |         final PlotArea area = plot.getArea(); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -131,7 +131,6 @@ 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,7 +24,6 @@ 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; | ||||||
|  |  | ||||||
| @@ -99,14 +98,6 @@ 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(); | ||||||
|         int plotCount = Settings.Limit.GLOBAL ? player.getPlotCount() : player.getPlotCount(plot.getWorldName()); |         if (!Settings.Done.COUNTS_TOWARDS_LIMIT && (player.getAllowedPlots() | ||||||
|         if (!Settings.Done.COUNTS_TOWARDS_LIMIT && (player.getAllowedPlots() < plotCount + size)) { |                 < player.getPlotCount() + 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,6 +19,7 @@ | |||||||
| 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; | ||||||
| @@ -36,7 +37,8 @@ public class Copy extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(final PlotPlayer<?> player, String[] args) { |     public boolean onCommand(final PlotPlayer<?> player, String[] args) { | ||||||
|         Plot plot1 = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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,6 +22,7 @@ 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; | ||||||
| @@ -46,7 +47,8 @@ public class CreateRoadSchematic extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|         Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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; | ||||||
| @@ -55,7 +57,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 (!(plot.getArea() instanceof HybridPlotWorld)) { |         if (!(location.getPlotArea() 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,8 +86,7 @@ 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().serverBrand()).append(": ") |                 b.append("Server Version: ").append(PlotSquared.platform().serverImplementation()) | ||||||
|                         .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,6 +22,7 @@ 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; | ||||||
| @@ -56,7 +57,8 @@ public class DebugRoadRegen extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|         Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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"), | ||||||
| @@ -90,7 +92,8 @@ public class DebugRoadRegen extends SubCommand { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     public boolean regenPlot(PlotPlayer<?> player) { |     public boolean regenPlot(PlotPlayer<?> player) { | ||||||
|         PlotArea area = player.getContextualPlotArea(); |         Location location = player.getLocation(); | ||||||
|  |         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; | ||||||
| @@ -145,10 +148,10 @@ public class DebugRoadRegen extends SubCommand { | |||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         PlotArea area = player.getContextualPlotArea(); |         Location location = player.getLocation(); | ||||||
|  |         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,6 +23,7 @@ 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; | ||||||
| @@ -60,7 +61,8 @@ public class Delete extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(final PlotPlayer<?> player, String[] args) { |     public boolean onCommand(final PlotPlayer<?> player, String[] args) { | ||||||
|         final Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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; | ||||||
| @@ -90,7 +92,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(plot.getWorldName()); |                 player.getPlotCount(location.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")); | ||||||
| @@ -122,7 +124,6 @@ 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,7 +70,8 @@ public class Deny extends SubCommand { | |||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|  |  | ||||||
|         final Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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,6 +26,7 @@ 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; | ||||||
| @@ -60,7 +61,8 @@ public class Done extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(final PlotPlayer<?> player, String[] args) { |     public boolean onCommand(final PlotPlayer<?> player, String[] args) { | ||||||
|         final Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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.getCurrentPlot().getWorldName(); |         String world = player.getLocation().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,6 +27,7 @@ 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; | ||||||
| @@ -178,7 +179,8 @@ 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 Plot plot = player.getCurrentPlot(); |         final Location location = player.getLocation(); | ||||||
|  |         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; | ||||||
| @@ -342,7 +344,7 @@ public final class FlagCommand extends Command { | |||||||
|         if (plotFlag == null) { |         if (plotFlag == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Plot plot = player.getCurrentPlot(); |         Plot plot = player.getLocation().getPlotAbs(); | ||||||
|         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( | ||||||
| @@ -407,7 +409,7 @@ public final class FlagCommand extends Command { | |||||||
|         if (plotFlag == null) { |         if (plotFlag == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         Plot plot = player.getCurrentPlot(); |         Plot plot = player.getLocation().getPlotAbs(); | ||||||
|         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( | ||||||
| @@ -417,7 +419,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.getCurrentPlot().getFlagContainer() |         final PlotFlag localFlag = player.getLocation().getPlotAbs().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(",")) { | ||||||
| @@ -442,7 +444,7 @@ public final class FlagCommand extends Command { | |||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         boolean result = |         boolean result = | ||||||
|                 player.getCurrentPlot().setFlag(localFlag.merge(parsed.getValue())); |                 player.getLocation().getPlotAbs().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; | ||||||
| @@ -482,7 +484,7 @@ public final class FlagCommand extends Command { | |||||||
|         if (flag == null) { |         if (flag == null) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         final Plot plot = player.getCurrentPlot(); |         final Plot plot = player.getLocation().getPlotAbs(); | ||||||
|         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) { | ||||||
| @@ -685,7 +687,7 @@ public final class FlagCommand extends Command { | |||||||
|                             .build() |                             .build() | ||||||
|             ); |             ); | ||||||
|             // Default value |             // Default value | ||||||
|             final String defaultValue = player.getCurrentPlot().getArea().getFlagContainer() |             final String defaultValue = player.getLocation().getPlotArea().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,10 +101,6 @@ 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 { | ||||||
| @@ -177,14 +173,8 @@ 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,34 +113,38 @@ 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(TranslatableCaption.of("help.help_header").toComponent(player)); |                 builder.append(MINI_MESSAGE.deserialize(TranslatableCaption.of("help.help_header").getComponent(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()); |                     builder.append(Component.newline()).append(MINI_MESSAGE | ||||||
|                     builder.append(TranslatableCaption.of("help.help_info_item").toComponent( |                             .deserialize( | ||||||
|                             player, TagResolver.builder() |                                     TranslatableCaption.of("help.help_info_item").getComponent(player), | ||||||
|  |                                     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()); |                 builder.append(Component.newline()).append(MINI_MESSAGE | ||||||
|                 builder.append(TranslatableCaption.of("help.help_info_item").toComponent( |                         .deserialize( | ||||||
|                         player, TagResolver.builder() |                                 TranslatableCaption.of("help.help_info_item").getComponent(player), | ||||||
|  |                                 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", Tag.inserting(TranslatableCaption |                                                 "category_desc", | ||||||
|  |                                                 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()); |                 builder.append(Component.newline()).append(MINI_MESSAGE.deserialize(TranslatableCaption | ||||||
|                 builder.append(TranslatableCaption.of("help.help_footer").toComponent(player)); |                         .of("help.help_footer") | ||||||
|  |                         .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,6 +20,7 @@ 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; | ||||||
| @@ -130,9 +131,13 @@ 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(TranslatableCaption.of("info.plot_info_categories.use_likes")); |                     player.sendMessage(StaticCaption.of( | ||||||
|  |                             "&6Categories&7: &amembers&7, &aalias&7, &abiome&7, &aseen&7, &adenied&7, &aflags&7, &aid&7, &asize&7, &atrusted&7, " | ||||||
|  |                                     + "&aowner&7, " + " &alikes")); | ||||||
|                 } else { |                 } else { | ||||||
|                     player.sendMessage(TranslatableCaption.of("info.plot_info_categories.use_rating")); |                     player.sendMessage(StaticCaption.of( | ||||||
|  |                             "&6Categories&7: &amembers&7, &aalias&7, &abiome&7, &aseen&7, &adenied&7, &aflags&7, &aid&7, &asize&7, &atrusted&7, " | ||||||
|  |                                     + "&aowner&7, " + " &arating")); | ||||||
|                 } |                 } | ||||||
|                 return false; |                 return false; | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -65,7 +65,8 @@ public class Kick extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|         Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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; | ||||||
| @@ -123,7 +124,7 @@ public class Kick extends SubCommand { | |||||||
|                         ); |                         ); | ||||||
|                         return; |                         return; | ||||||
|                     } |                     } | ||||||
|                     Location spawn = this.worldUtil.getSpawn(plot.getWorldName()); |                     Location spawn = this.worldUtil.getSpawn(location.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]); | ||||||
| @@ -147,7 +148,8 @@ 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) { | ||||||
|         Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         PlotArea area = player.getContextualPlotArea(); |         String world = player.getLocation().getWorldName(); | ||||||
|         String world = area != null ? area.getWorldName() : ""; |         PlotArea area = player.getApplicablePlotArea(); | ||||||
|         String arg = args[0].toLowerCase(); |         String arg = args[0].toLowerCase(); | ||||||
|         final boolean[] sort = new boolean[]{true}; |         final boolean[] sort = new boolean[]{true}; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -68,6 +68,11 @@ 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,13 +44,8 @@ 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. | ||||||
|  */ |  */ | ||||||
| @@ -152,7 +147,8 @@ 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(), e); |                     LOGGER.error("Failed to register command {}", command.getCanonicalName()); | ||||||
|  |                     e.printStackTrace(); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
| @@ -240,179 +236,110 @@ public class MainCommand extends Command { | |||||||
|             RunnableVal3<Command, Runnable, Runnable> confirm, |             RunnableVal3<Command, Runnable, Runnable> confirm, | ||||||
|             RunnableVal2<Command, CommandResult> whenDone |             RunnableVal2<Command, CommandResult> whenDone | ||||||
|     ) { |     ) { | ||||||
|         prepareArguments(new CommandExecutionData(player, args, confirm, whenDone, null)) |         // Optional command scope // | ||||||
|                 .thenCompose(executionData -> { |         Location location = null; | ||||||
|                     if (executionData.isEmpty()) { |         Plot plot = null; | ||||||
|  |         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); | ||||||
|                 } |                 } | ||||||
|                     var data = executionData.get(); |                 // Trim command | ||||||
|                     try { |                 args = Arrays.copyOfRange(args, 1, args.length); | ||||||
|                         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")); |  | ||||||
|             } |             } | ||||||
|                     } finally { |             if (args.length >= 2 && !args[0].isEmpty() && args[0].charAt(0) == '-') { | ||||||
|                         if (data.postCommandData() != null) { |                 if ("f".equals(args[0].substring(1))) { | ||||||
|                             resetCommandScope(data.player(), data.postCommandData()); |                     confirm = new RunnableVal3<>() { | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                     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)) { | ||||||
|                         && Optional.of(area.getPrices().get(cmd.getFullId())) |                                 PlotExpression priceEval = | ||||||
|                         .map(priceEval -> priceEval.evaluate(0d)) |                                         area.getPrices().get(cmd.getFullId()); | ||||||
|                         .filter(price -> price != 0d) |                                 double price = priceEval != null ? priceEval.evaluate(0d) : 0d; | ||||||
|                         .filter(price -> PlotSquared.platform().econHandler().getMoney(player) < price) |                                 if (price != 0d | ||||||
|                         .isPresent()) { |                                         && PlotSquared.platform().econHandler().getMoney(player) < price) { | ||||||
|                                     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) { |         } | ||||||
|         Objects.requireNonNull(commandMeta.location()); |         try { | ||||||
|         Objects.requireNonNull(commandMeta.plot()); |             super.execute(player, args, confirm, whenDone); | ||||||
|         Location location; |         } catch (CommandException e) { | ||||||
|         Plot plot; |             throw e; | ||||||
|  |         } 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)) { | ||||||
|             location = locationMetaDataAccess.get().orElse(null); |                 if (location == 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(commandMeta.location()); |                     locationMetaDataAccess.set(location); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             try (final MetaDataAccess<Plot> plotMetaDataAccess |             try (final MetaDataAccess<Plot> plotMetaDataAccess | ||||||
|                          = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { |                          = player.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { | ||||||
|             if (commandMeta.plot() == null) { |                 if (plot == null) { | ||||||
|                     plotMetaDataAccess.remove(); |                     plotMetaDataAccess.remove(); | ||||||
|                 } else { |                 } else { | ||||||
|                 plotMetaDataAccess.set(commandMeta.plot()); |                     plotMetaDataAccess.set(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,6 +20,7 @@ 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; | ||||||
|  |  | ||||||
| @@ -35,7 +36,8 @@ public class Middle extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] arguments) { |     public boolean onCommand(PlotPlayer<?> player, String[] arguments) { | ||||||
|         Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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,6 +20,7 @@ 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; | ||||||
| @@ -54,7 +55,8 @@ public class Move extends SubCommand { | |||||||
|             RunnableVal3<Command, Runnable, Runnable> confirm, |             RunnableVal3<Command, Runnable, Runnable> confirm, | ||||||
|             RunnableVal2<Command, CommandResult> whenDone |             RunnableVal2<Command, CommandResult> whenDone | ||||||
|     ) { |     ) { | ||||||
|         Plot plot1 = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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,6 +23,7 @@ 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; | ||||||
| @@ -56,7 +57,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_tears", "music_disc_lava_chicken" |                     "music_disc_creator_music_box", "music_disc_precipice" | ||||||
|             ); |             ); | ||||||
|  |  | ||||||
|     // make sure all discs and the bedrock ("cancel") fit into the inventory |     // make sure all discs and the bedrock ("cancel") fit into the inventory | ||||||
| @@ -73,7 +74,8 @@ public class Music extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|         final Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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,6 +21,7 @@ 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; | ||||||
| @@ -55,7 +56,8 @@ public class Remove extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|         Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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; | ||||||
| @@ -130,7 +132,8 @@ 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) { | ||||||
|         Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         Plot plot = location.getPlotAbs(); | ||||||
|         if (plot == null) { |         if (plot == null) { | ||||||
|             return Collections.emptyList(); |             return Collections.emptyList(); | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -22,6 +22,7 @@ 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; | ||||||
| @@ -101,7 +102,8 @@ public class SchematicCmd extends SubCommand { | |||||||
|                     ); |                     ); | ||||||
|                     break; |                     break; | ||||||
|                 } |                 } | ||||||
|                 final Plot plot = player.getCurrentPlot(); |                 Location loc = player.getLocation(); | ||||||
|  |                 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; | ||||||
| @@ -245,7 +247,8 @@ 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; | ||||||
|                 } |                 } | ||||||
|                 Plot plot = player.getCurrentPlot(); |                 Location location = player.getLocation(); | ||||||
|  |                 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.getContextualPlotArea(); |                 final PlotArea plotArea = player.getLocation().getPlotArea(); | ||||||
|                 if (plotArea == null) { |                 if (plotArea == null) { | ||||||
|                     return false; |                     return false; | ||||||
|                 } |                 } | ||||||
|   | |||||||
| @@ -19,6 +19,7 @@ | |||||||
| 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; | ||||||
| @@ -31,7 +32,8 @@ public abstract class SetCommand extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(PlotPlayer<?> player, String[] args) { |     public boolean onCommand(PlotPlayer<?> player, String[] args) { | ||||||
|         Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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,6 +19,7 @@ | |||||||
| 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; | ||||||
| @@ -43,7 +44,8 @@ public class Swap extends SubCommand { | |||||||
|             RunnableVal3<Command, Runnable, Runnable> confirm, |             RunnableVal3<Command, Runnable, Runnable> confirm, | ||||||
|             RunnableVal2<Command, CommandResult> whenDone |             RunnableVal2<Command, CommandResult> whenDone | ||||||
|     ) { |     ) { | ||||||
|         Plot plot1 = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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); | ||||||
| @@ -77,10 +79,8 @@ 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( |         return plot1.getPlotModificationManager().move(plot2, player, () -> { | ||||||
|                 plot2, player, () -> { |         }, true).thenApply(result -> { | ||||||
|                 }, true |  | ||||||
|         ).thenApply(result -> { |  | ||||||
|             if (result) { |             if (result) { | ||||||
|                 player.sendMessage( |                 player.sendMessage( | ||||||
|                         TranslatableCaption.of("swap.swap_success"), |                         TranslatableCaption.of("swap.swap_success"), | ||||||
|   | |||||||
| @@ -22,6 +22,7 @@ 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; | ||||||
| @@ -50,7 +51,8 @@ public class Unlink extends SubCommand { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onCommand(final PlotPlayer<?> player, String[] args) { |     public boolean onCommand(final PlotPlayer<?> player, String[] args) { | ||||||
|         final Plot plot = player.getCurrentPlot(); |         Location location = player.getLocation(); | ||||||
|  |         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; | ||||||
|   | |||||||
| @@ -67,13 +67,10 @@ public class Visit extends Command { | |||||||
|  |  | ||||||
|     private void visit( |     private void visit( | ||||||
|             final @NonNull PlotPlayer<?> player, final @NonNull PlotQuery query, final PlotArea sortByArea, |             final @NonNull PlotPlayer<?> player, final @NonNull PlotQuery query, final PlotArea sortByArea, | ||||||
|             final RunnableVal3<Command, Runnable, Runnable> confirm, final RunnableVal2<Command, CommandResult> whenDone, int page |             final RunnableVal3<Command, Runnable, Runnable> confirm, final RunnableVal2<Command, CommandResult> whenDone, | ||||||
|  |             int page, final boolean isQueryingBasePlot | ||||||
|     ) { |     ) { | ||||||
|         // We get the query once, |         if (!isQueryingBasePlot && query.hasMinimumMatches(2)) { | ||||||
|         // then we get it another time further on |  | ||||||
|         final List<Plot> unsorted = query.asList(); |  | ||||||
|  |  | ||||||
|         if (unsorted.size() > 1) { |  | ||||||
|             query.whereBasePlot(); |             query.whereBasePlot(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -237,7 +234,8 @@ public class Visit extends Command { | |||||||
|                                     finalSortByArea, |                                     finalSortByArea, | ||||||
|                                     confirm, |                                     confirm, | ||||||
|                                     whenDone, |                                     whenDone, | ||||||
|                                     finalPage1 |                                     finalPage1, | ||||||
|  |                                     true | ||||||
|                             ); |                             ); | ||||||
|                         } |                         } | ||||||
|                     }); |                     }); | ||||||
| @@ -261,12 +259,9 @@ public class Visit extends Command { | |||||||
|                         if (throwable instanceof TimeoutException) { |                         if (throwable instanceof TimeoutException) { | ||||||
|                             // The request timed out |                             // The request timed out | ||||||
|                             player.sendMessage(TranslatableCaption.of("players.fetching_players_timeout")); |                             player.sendMessage(TranslatableCaption.of("players.fetching_players_timeout")); | ||||||
|                         } else if (uuid != null && (Settings.Teleport.VISIT_MERGED_OWNERS |                             return; | ||||||
|                                 ? !PlotQuery.newQuery().ownersInclude(uuid).anyMatch() |                         } | ||||||
|                                 : !PlotQuery.newQuery().ownedBy(uuid).anyMatch())) { |                         if (uuid == null){ | ||||||
|                             // It was a valid UUID but the player has no plots |  | ||||||
|                             player.sendMessage(TranslatableCaption.of("errors.player_no_plots")); |  | ||||||
|                         } else if (uuid == null) { |  | ||||||
|                             // player not found, so we assume it's an alias if no page was provided |                             // player not found, so we assume it's an alias if no page was provided | ||||||
|                             if (finalPage == Integer.MIN_VALUE) { |                             if (finalPage == Integer.MIN_VALUE) { | ||||||
|                                 this.visit( |                                 this.visit( | ||||||
| @@ -275,7 +270,8 @@ public class Visit extends Command { | |||||||
|                                         player.getApplicablePlotArea(), |                                         player.getApplicablePlotArea(), | ||||||
|                                         confirm, |                                         confirm, | ||||||
|                                         whenDone, |                                         whenDone, | ||||||
|                                         1 |                                         1, | ||||||
|  |                                         false | ||||||
|                                 ); |                                 ); | ||||||
|                             } else { |                             } else { | ||||||
|                                 player.sendMessage( |                                 player.sendMessage( | ||||||
| @@ -283,24 +279,23 @@ public class Visit extends Command { | |||||||
|                                         TagResolver.resolver("value", Tag.inserting(Component.text(finalArgs[0]))) |                                         TagResolver.resolver("value", Tag.inserting(Component.text(finalArgs[0]))) | ||||||
|                                 ); |                                 ); | ||||||
|                             } |                             } | ||||||
|                         } else { |                             return; | ||||||
|                             this.visit( |  | ||||||
|                                     player, |  | ||||||
|                                     Settings.Teleport.VISIT_MERGED_OWNERS |  | ||||||
|                                             ? PlotQuery.newQuery().ownersInclude(uuid).whereBasePlot() |  | ||||||
|                                             : PlotQuery.newQuery().ownedBy(uuid).whereBasePlot(), |  | ||||||
|                                     null, |  | ||||||
|                                     confirm, |  | ||||||
|                                     whenDone, |  | ||||||
|                                     finalPage |  | ||||||
|                             ); |  | ||||||
|                         } |                         } | ||||||
|  |                         final PlotQuery query = Settings.Teleport.VISIT_MERGED_OWNERS | ||||||
|  |                                 ? PlotQuery.newQuery().ownersInclude(uuid) | ||||||
|  |                                 : PlotQuery.newQuery().ownedBy(uuid); | ||||||
|  |                         if (!query.anyMatch()) { | ||||||
|  |                             // It was a valid UUID but the player has no plots | ||||||
|  |                             player.sendMessage(TranslatableCaption.of("errors.player_no_plots")); | ||||||
|  |                             return; | ||||||
|  |                         } | ||||||
|  |                         this.visit(player, query.whereBasePlot(), null, confirm, whenDone, finalPage, true); | ||||||
|                     }); |                     }); | ||||||
|                 } else { |                 } else { | ||||||
|                     // Try to parse a plot |                     // Try to parse a plot | ||||||
|                     final Plot plot = Plot.getPlotFromString(player, finalArgs[0], true); |                     final Plot plot = Plot.getPlotFromString(player, finalArgs[0], true); | ||||||
|                     if (plot != null) { |                     if (plot != null) { | ||||||
|                         this.visit(player, PlotQuery.newQuery().withPlot(plot), null, confirm, whenDone, 1); |                         this.visit(player, PlotQuery.newQuery().withPlot(plot), null, confirm, whenDone, 1, false); | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|                 break; |                 break; | ||||||
|   | |||||||
| @@ -20,7 +20,6 @@ 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; | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -45,16 +44,6 @@ 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,7 +21,6 @@ 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 { | ||||||
| @@ -52,14 +51,6 @@ 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,7 +27,6 @@ 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; | ||||||
|  |  | ||||||
| @@ -97,23 +96,13 @@ 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)); | ||||||
|         } |         } | ||||||
|         TagResolver[] finalResolvers = Arrays.copyOf(tagResolvers, tagResolvers.length + 1); |         return MiniMessage.miniMessage().deserialize(getComponent(localeHolder), TagResolver.resolver( | ||||||
|         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 | ||||||
|   | |||||||
| @@ -1,62 +0,0 @@ | |||||||
| /* |  | ||||||
|  * PlotSquared, a land and world management plugin for Minecraft. |  | ||||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> |  | ||||||
|  * Copyright (C) IntellectualSites team and contributors |  | ||||||
|  * |  | ||||||
|  * This program is free software: you can redistribute it and/or modify |  | ||||||
|  * it under the terms of the GNU General Public License as published by |  | ||||||
|  * the Free Software Foundation, either version 3 of the License, or |  | ||||||
|  * (at your option) any later version. |  | ||||||
|  * |  | ||||||
|  * This program is distributed in the hope that it will be useful, |  | ||||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of |  | ||||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the |  | ||||||
|  * GNU General Public License for more details. |  | ||||||
|  * |  | ||||||
|  * You should have received a copy of the GNU General Public License |  | ||||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. |  | ||||||
|  */ |  | ||||||
| package com.plotsquared.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 @Nullable Plot getCurrentPlot() { |     public 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(getContextualWorldName()); |             return getPlotCount(getLocation().getWorldName()); | ||||||
|         } |         } | ||||||
|         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(getContextualWorldName()); |             return getClusterCount(getLocation().getWorldName()); | ||||||
|         } |         } | ||||||
|         final AtomicInteger count = new AtomicInteger(0); |         final AtomicInteger count = new AtomicInteger(0); | ||||||
|         this.plotAreaManager.forEachPlotArea(value -> { |         this.plotAreaManager.forEachPlotArea(value -> { | ||||||
| @@ -352,34 +352,6 @@ 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. | ||||||
|      * |      * | ||||||
| @@ -436,12 +408,8 @@ 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() { | ||||||
| @@ -476,7 +444,7 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get this player's UUID. |      * Get this player's UUID. | ||||||
|      * <p>=== !IMPORTANT ===</p> |      * === !IMPORTANT ===<br> | ||||||
|      * 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) | ||||||
|      * |      * | ||||||
| @@ -646,16 +614,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(14); |             ByteBuffer buffer = ByteBuffer.allocate(13); | ||||||
|             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.putShort((short) location.getY()); |             buffer.put((byte) location.getY()); | ||||||
|             buffer.putInt(location.getZ()); |             buffer.putInt(location.getZ()); | ||||||
|             setPersistentMeta("quitLocV2", buffer.array()); |             setPersistentMeta("quitLoc", buffer.array()); | ||||||
|         } else if (hasPersistentMeta("quitLocV2")) { |         } else if (hasPersistentMeta("quitLoc")) { | ||||||
|             removePersistentMeta("quitLocV2"); |             removePersistentMeta("quitLoc"); | ||||||
|         } |         } | ||||||
|         if (plot != null) { |         if (plot != null) { | ||||||
|             this.eventDispatcher.callLeave(this, plot); |             this.eventDispatcher.callLeave(this, plot); | ||||||
| @@ -710,8 +678,7 @@ 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( |             DBFunc.getPersistentMeta(getUUID(), new RunnableVal<>() { | ||||||
|                     getUUID(), new RunnableVal<>() { |  | ||||||
|                 @Override |                 @Override | ||||||
|                 public void run(Map<String, byte[]> value) { |                 public void run(Map<String, byte[]> value) { | ||||||
|                     try { |                     try { | ||||||
| @@ -733,18 +700,11 @@ 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; | ||||||
| @@ -754,7 +714,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 = V2 ? quitWorld.getShort() : (quitWorld.get() & 0xFF); |                         int y = quitWorld.get() & 0xFF; | ||||||
|                         int z = quitWorld.getInt(); |                         int z = quitWorld.getInt(); | ||||||
|                         Plot plot = area.getOwnedPlot(id); |                         Plot plot = area.getOwnedPlot(id); | ||||||
|  |  | ||||||
| @@ -788,11 +748,10 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|                     } catch (Throwable e) { |                     } catch (Throwable e) { | ||||||
|                                 LOGGER.error("Error populating persistent meta for player {}", PlotPlayer.this.getName(), e); |                         e.printStackTrace(); | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|                     } |             }); | ||||||
|             ); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -860,8 +819,7 @@ public abstract class PlotPlayer<P> implements CommandCaller, OfflinePlotPlayer, | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @SuppressWarnings("unchecked") |     @SuppressWarnings("unchecked") | ||||||
|     @Nullable |     @Nullable <T> T getPersistentMeta(final @NonNull MetaDataKey<T> key) { | ||||||
|     <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; | ||||||
| @@ -1031,11 +989,9 @@ 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( |                         TagResolver.resolver("value", Tag.inserting( | ||||||
|                                 "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,8 +321,7 @@ public class Plot { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the plot from a string. Performs a check to ensure Plot#getBottomAbs is not outside world bounds |      * Get the plot from a string. | ||||||
|      * (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 | ||||||
| @@ -333,31 +332,6 @@ 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) { | ||||||
| @@ -415,51 +389,13 @@ public class Plot { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Gets a plot from a string e.g. [area];[id]. Performs a check to ensure Plot#getBottomAbs is not outside world bounds |      * Gets a plot from a string e.g. [area];[id] | ||||||
|      * (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) { | ||||||
| @@ -483,8 +419,7 @@ public class Plot { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Return a new/cached plot object at a given location. Does not check world bounds for potential crashes, these should be |      * Return a new/cached plot object at a given location. | ||||||
|      * 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. | ||||||
| @@ -541,7 +476,7 @@ public class Plot { | |||||||
|      * |      * | ||||||
|      * @return World name |      * @return World name | ||||||
|      */ |      */ | ||||||
|     public @NonNull String getWorldName() { |     public @Nullable String getWorldName() { | ||||||
|         return area.getWorldName(); |         return area.getWorldName(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -1782,7 +1717,6 @@ 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,7 +371,8 @@ 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,18 +63,4 @@ 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 @NonNull String getWorldName() { |     public String getWorldName() { | ||||||
|         return getId().toUnderscoreSeparatedString(); |         return getId().toUnderscoreSeparatedString(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -135,7 +135,6 @@ 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; | ||||||
| @@ -145,11 +144,10 @@ 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(uri).toURL(); |             url = URI.create(Settings.Web.URL + "?key=" + uuid + "&type=" + extension).toURL(); | ||||||
|         } catch (MalformedURLException e) { |         } catch (MalformedURLException e) { | ||||||
|             LOGGER.error("Malformed URI `{}`", uri, e); |             e.printStackTrace(); | ||||||
|             whenDone.run(); |             whenDone.run(); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| @@ -195,7 +193,7 @@ public abstract class SchematicHandler { | |||||||
|                 } |                 } | ||||||
|                 TaskManager.runTask(whenDone); |                 TaskManager.runTask(whenDone); | ||||||
|             } catch (IOException e) { |             } catch (IOException e) { | ||||||
|                 LOGGER.error("Error while uploading schematic for UUID {}", finalUuid, e); |                 e.printStackTrace(); | ||||||
|                 TaskManager.runTask(whenDone); |                 TaskManager.runTask(whenDone); | ||||||
|             } |             } | ||||||
|         }); |         }); | ||||||
| @@ -390,14 +388,8 @@ 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 |  | ||||||
|             ); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -464,7 +456,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) { | ||||||
|                 LOGGER.error("Error reading schematic from file {}", file.getAbsolutePath(), e); |                 e.printStackTrace(); | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             throw new UnsupportedFormatException("This schematic format is not recognised or supported."); |             throw new UnsupportedFormatException("This schematic format is not recognised or supported."); | ||||||
| @@ -478,7 +470,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) { | ||||||
|             LOGGER.error("Error reading schematic from {}", url, e); |             e.printStackTrace(); | ||||||
|         } |         } | ||||||
|         return null; |         return null; | ||||||
|     } |     } | ||||||
| @@ -494,7 +486,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) { | ||||||
|                 LOGGER.error("Error reading schematic", e); |                 e.printStackTrace(); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         return null; |         return null; | ||||||
| @@ -523,7 +515,7 @@ public abstract class SchematicHandler { | |||||||
|             } |             } | ||||||
|             return schematics; |             return schematics; | ||||||
|         } catch (JsonParseException | IOException e) { |         } catch (JsonParseException | IOException e) { | ||||||
|             LOGGER.error("Error retrieving saves for UUID {}", uuid, e); |             e.printStackTrace(); | ||||||
|         } |         } | ||||||
|         return null; |         return null; | ||||||
|     } |     } | ||||||
| @@ -540,7 +532,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) { | ||||||
|                     LOGGER.error("Error uploading schematic for UUID {}", uuid, e1); |                     e1.printStackTrace(); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }, whenDone); |         }, whenDone); | ||||||
| @@ -564,9 +556,9 @@ public abstract class SchematicHandler { | |||||||
|                 nbtStream.writeNamedTag("Schematic", tag); |                 nbtStream.writeNamedTag("Schematic", tag); | ||||||
|             } |             } | ||||||
|         } catch (FileNotFoundException e) { |         } catch (FileNotFoundException e) { | ||||||
|             LOGGER.error("Error saving schematic at {}", path, e); |             e.printStackTrace(); | ||||||
|         } catch (IOException e) { |         } catch (IOException e) { | ||||||
|             LOGGER.error("Error saving schematic at {}", path, e); |             e.printStackTrace(); | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|         return true; |         return true; | ||||||
| @@ -589,7 +581,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.isEmpty()) { |         if (biomeBuffer.size() == 0 || biomePalette.size() == 0) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -741,7 +733,10 @@ 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<>(block.getNbtData().getValue()); |                                         Map<String, Tag> values = new HashMap<>(); | ||||||
|  |                                         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"); | ||||||
|   | |||||||
| @@ -0,0 +1,50 @@ | |||||||
|  | /* | ||||||
|  |  * 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.comparator; | ||||||
|  |  | ||||||
|  | import com.plotsquared.core.plot.Plot; | ||||||
|  |  | ||||||
|  | import java.util.Comparator; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Sort plots by {@link Plot#temp} (being the auto increment id in database) in natural order for {@code temp > 0}. | ||||||
|  |  * For {@code temp < 1} sort by {@link Plot#hashCode()} | ||||||
|  |  */ | ||||||
|  | public class PlotByCreationDateComparator implements Comparator<Plot> { | ||||||
|  |  | ||||||
|  |     public static final Comparator<Plot> INSTANCE = new PlotByCreationDateComparator(); | ||||||
|  |  | ||||||
|  |     private PlotByCreationDateComparator() { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     @SuppressWarnings("deprecation") // Plot#temp | ||||||
|  |     public int compare(final Plot first, final Plot second) { | ||||||
|  |         if (first.temp > 0 && second.temp > 0) { | ||||||
|  |             return Integer.compare(first.temp, second.temp); | ||||||
|  |         } | ||||||
|  |         // second is implicitly `< 1` (due to previous condition) | ||||||
|  |         if (first.temp > 0) { | ||||||
|  |             return 1; | ||||||
|  |         } | ||||||
|  |         // sort dangling plots (temp < 1) by their hashcode | ||||||
|  |         return Integer.compare(first.hashCode(), second.hashCode()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,52 @@ | |||||||
|  | /* | ||||||
|  |  * 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.comparator; | ||||||
|  |  | ||||||
|  | import com.plotsquared.core.plot.Plot; | ||||||
|  | import com.plotsquared.core.plot.flag.implementations.DoneFlag; | ||||||
|  | import com.plotsquared.core.util.MathMan; | ||||||
|  |  | ||||||
|  | import java.util.Comparator; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Sort plots by their {@link DoneFlag} in reverse numeric natural order. (more recent "finished plots" first) | ||||||
|  |  * <br> | ||||||
|  |  * Non-finished plots last, unsorted. | ||||||
|  |  */ | ||||||
|  | public class PlotByDoneComparator implements Comparator<Plot> { | ||||||
|  |  | ||||||
|  |     public static final PlotByDoneComparator INSTANCE = new PlotByDoneComparator(); | ||||||
|  |  | ||||||
|  |     private PlotByDoneComparator() { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public int compare(final Plot first, final Plot second) { | ||||||
|  |         String firstDone = first.getFlag(DoneFlag.class); | ||||||
|  |         String lastDone = second.getFlag(DoneFlag.class); | ||||||
|  |         if (MathMan.isInteger(firstDone)) { | ||||||
|  |             if (MathMan.isInteger(lastDone)) { | ||||||
|  |                 return Integer.parseInt(lastDone) - Integer.parseInt(firstDone); | ||||||
|  |             } | ||||||
|  |             return -1; // only "first" is finished, so sort "second" after "first" | ||||||
|  |         } | ||||||
|  |         return 0; // neither is finished | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,61 @@ | |||||||
|  | /* | ||||||
|  |  * 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.comparator; | ||||||
|  |  | ||||||
|  | import com.plotsquared.core.plot.Plot; | ||||||
|  | import com.plotsquared.core.plot.Rating; | ||||||
|  |  | ||||||
|  | import java.util.Comparator; | ||||||
|  | import java.util.Map; | ||||||
|  | import java.util.UUID; | ||||||
|  |  | ||||||
|  | public class PlotByRatingComparator implements Comparator<Plot> { | ||||||
|  |  | ||||||
|  |     public static final PlotByRatingComparator INSTANCE = new PlotByRatingComparator(); | ||||||
|  |  | ||||||
|  |     PlotByRatingComparator() { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public int compare(final Plot p1, final Plot p2) { | ||||||
|  |         double v1 = 0; | ||||||
|  |         int p1s = p1.getSettings().getRatings().size(); | ||||||
|  |         int p2s = p2.getRatings().size(); | ||||||
|  |         if (!p1.getSettings().getRatings().isEmpty()) { | ||||||
|  |             v1 = p1.getRatings().values().stream().mapToDouble(Rating::getAverageRating) | ||||||
|  |                     .map(av -> av * av).sum(); | ||||||
|  |             v1 /= p1s; | ||||||
|  |             v1 += p1s; | ||||||
|  |         } | ||||||
|  |         double v2 = 0; | ||||||
|  |         if (!p2.getSettings().getRatings().isEmpty()) { | ||||||
|  |             for (Map.Entry<UUID, Rating> entry : p2.getRatings().entrySet()) { | ||||||
|  |                 double av = entry.getValue().getAverageRating(); | ||||||
|  |                 v2 += av * av; | ||||||
|  |             } | ||||||
|  |             v2 /= p2s; | ||||||
|  |             v2 += p2s; | ||||||
|  |         } | ||||||
|  |         if (v2 == v1 && v2 != 0) { | ||||||
|  |             return p2s - p1s; | ||||||
|  |         } | ||||||
|  |         return (int) Math.signum(v2 - v1); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,49 @@ | |||||||
|  | /* | ||||||
|  |  * 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.comparator; | ||||||
|  |  | ||||||
|  | import com.plotsquared.core.plot.Plot; | ||||||
|  | import com.plotsquared.core.plot.PlotArea; | ||||||
|  |  | ||||||
|  | import javax.annotation.Nullable; | ||||||
|  | import java.util.Comparator; | ||||||
|  |  | ||||||
|  | public class PlotInPrioritizedAreaComparator implements Comparator<Plot> { | ||||||
|  |  | ||||||
|  |     private final PlotArea priorityArea; | ||||||
|  |  | ||||||
|  |     public PlotInPrioritizedAreaComparator(@Nullable final PlotArea area) { | ||||||
|  |         this.priorityArea = area; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public int compare(final Plot first, final Plot second) { | ||||||
|  |         if (this.priorityArea == null) { | ||||||
|  |             return 0; // no defined priority? don't sort | ||||||
|  |         } | ||||||
|  |         if (this.priorityArea.equals(first.getArea())) { | ||||||
|  |             return -1; | ||||||
|  |         } | ||||||
|  |         if (this.priorityArea.equals(second.getArea())) { | ||||||
|  |             return 1; | ||||||
|  |         } | ||||||
|  |         return 0; // same area, don't sort | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -25,16 +25,21 @@ 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) { | ||||||
|         this.rendered = TranslatableCaption.of("help.help_item").toComponent(audience, TagResolver.builder() |         rendered = MINI_MESSAGE.deserialize( | ||||||
|  |                 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(), " | ") | ||||||
| @@ -42,7 +47,8 @@ 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,12 +49,10 @@ 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} | ||||||
| @@ -129,22 +127,6 @@ 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); | ||||||
|   | |||||||
| @@ -24,6 +24,7 @@ import com.plotsquared.core.plot.PlotArea; | |||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
| import java.util.LinkedList; | import java.util.LinkedList; | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  | import java.util.stream.Stream; | ||||||
|  |  | ||||||
| class AreaLimitedPlotProvider implements PlotProvider { | class AreaLimitedPlotProvider implements PlotProvider { | ||||||
|  |  | ||||||
| @@ -42,4 +43,9 @@ class AreaLimitedPlotProvider implements PlotProvider { | |||||||
|         return plots; |         return plots; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public Stream<Plot> streamPlots() { | ||||||
|  |         return streamPlotsInPlotAreas(this.areas.toArray(PlotArea[]::new)); | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -22,6 +22,7 @@ import com.plotsquared.core.PlotSquared; | |||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
|  |  | ||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
|  | import java.util.stream.Stream; | ||||||
|  |  | ||||||
| class ExpiredPlotProvider implements PlotProvider { | class ExpiredPlotProvider implements PlotProvider { | ||||||
|  |  | ||||||
| @@ -30,4 +31,9 @@ class ExpiredPlotProvider implements PlotProvider { | |||||||
|         return PlotSquared.platform().expireManager().getPendingExpired(); |         return PlotSquared.platform().expireManager().getPendingExpired(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public Stream<Plot> streamPlots() { | ||||||
|  |         return getPlots().stream(); | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -23,6 +23,7 @@ import org.checkerframework.checker.nullness.qual.NonNull; | |||||||
|  |  | ||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
| import java.util.Collections; | import java.util.Collections; | ||||||
|  | import java.util.stream.Stream; | ||||||
|  |  | ||||||
| class FixedPlotProvider implements PlotProvider { | class FixedPlotProvider implements PlotProvider { | ||||||
|  |  | ||||||
| @@ -37,4 +38,9 @@ class FixedPlotProvider implements PlotProvider { | |||||||
|         return Collections.singleton(plot); |         return Collections.singleton(plot); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public Stream<Plot> streamPlots() { | ||||||
|  |         return Stream.of(plot); | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -26,6 +26,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.List; | import java.util.List; | ||||||
|  | import java.util.stream.Stream; | ||||||
|  |  | ||||||
| class GlobalPlotProvider implements PlotProvider { | class GlobalPlotProvider implements PlotProvider { | ||||||
|  |  | ||||||
| @@ -44,4 +45,9 @@ class GlobalPlotProvider implements PlotProvider { | |||||||
|         return plots; |         return plots; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public Stream<Plot> streamPlots() { | ||||||
|  |         return streamPlotsInPlotAreas(this.plotAreaManager.getAllPlotAreas()); | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -22,6 +22,7 @@ import com.plotsquared.core.plot.Plot; | |||||||
|  |  | ||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
| import java.util.Collections; | import java.util.Collections; | ||||||
|  | import java.util.stream.Stream; | ||||||
|  |  | ||||||
| class NullProvider implements PlotProvider { | class NullProvider implements PlotProvider { | ||||||
|  |  | ||||||
| @@ -30,4 +31,9 @@ class NullProvider implements PlotProvider { | |||||||
|         return Collections.emptyList(); |         return Collections.emptyList(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public Stream<Plot> streamPlots() { | ||||||
|  |         return Stream.empty(); | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -23,17 +23,11 @@ import org.checkerframework.checker.nullness.qual.NonNull; | |||||||
|  |  | ||||||
| import java.util.UUID; | import java.util.UUID; | ||||||
|  |  | ||||||
| class OwnersIncludeFilter implements PlotFilter { | record OwnersIncludeFilter(UUID owner) implements PlotFilter { | ||||||
|  |  | ||||||
|     private final UUID owner; |  | ||||||
|  |  | ||||||
|     OwnersIncludeFilter(final @NonNull UUID owner) { |  | ||||||
|         this.owner = owner; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean accepts(final @NonNull Plot plot) { |     public boolean accepts(final @NonNull Plot plot) { | ||||||
|         return plot.isBasePlot() && plot.getOwners().size() > 0 && plot.getOwners().contains(owner); |         return plot.isBasePlot() && plot.isOwner(owner); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -19,12 +19,44 @@ | |||||||
| package com.plotsquared.core.util.query; | package com.plotsquared.core.util.query; | ||||||
|  |  | ||||||
| import com.plotsquared.core.plot.Plot; | import com.plotsquared.core.plot.Plot; | ||||||
|  | import com.plotsquared.core.plot.PlotArea; | ||||||
|  |  | ||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
|  | import java.util.Iterator; | ||||||
|  | import java.util.Spliterator; | ||||||
|  | import java.util.Spliterators; | ||||||
|  | import java.util.stream.Stream; | ||||||
|  | import java.util.stream.StreamSupport; | ||||||
|  |  | ||||||
| @FunctionalInterface |  | ||||||
| interface PlotProvider { | interface PlotProvider { | ||||||
|  |  | ||||||
|     Collection<Plot> getPlots(); |     Collection<Plot> getPlots(); | ||||||
|  |  | ||||||
|  |     Stream<Plot> streamPlots(); | ||||||
|  |  | ||||||
|  |     default Stream<Plot> streamPlotsInPlotAreas(PlotArea[] areas) { | ||||||
|  |         if (areas == null || areas.length == 0) { | ||||||
|  |             return Stream.of(); | ||||||
|  |         } | ||||||
|  |         return StreamSupport.stream(Spliterators.spliteratorUnknownSize(new Iterator<>() { | ||||||
|  |             private int areaIndex = -1; | ||||||
|  |             private Iterator<Plot> currentAreaPlots; | ||||||
|  |             @Override | ||||||
|  |             public boolean hasNext() { | ||||||
|  |                 if (currentAreaPlots == null || !currentAreaPlots.hasNext()) { | ||||||
|  |                     if (areaIndex >= areas.length - 1) { | ||||||
|  |                         return false; | ||||||
|  |                     } | ||||||
|  |                     currentAreaPlots = areas[++areaIndex].getPlots().iterator(); | ||||||
|  |                 } | ||||||
|  |                 return true; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             @Override | ||||||
|  |             public Plot next() { | ||||||
|  |                 return currentAreaPlots.next(); | ||||||
|  |             } | ||||||
|  |         }, Spliterator.IMMUTABLE), false); | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -23,10 +23,11 @@ import com.plotsquared.core.PlotSquared; | |||||||
| 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.Rating; |  | ||||||
| import com.plotsquared.core.plot.flag.implementations.DoneFlag; |  | ||||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | import com.plotsquared.core.plot.world.PlotAreaManager; | ||||||
| import com.plotsquared.core.util.MathMan; | import com.plotsquared.core.util.comparator.PlotByDoneComparator; | ||||||
|  | import com.plotsquared.core.util.comparator.PlotByRatingComparator; | ||||||
|  | import com.plotsquared.core.util.comparator.PlotByCreationDateComparator; | ||||||
|  | import com.plotsquared.core.util.comparator.PlotInPrioritizedAreaComparator; | ||||||
| import org.checkerframework.checker.nullness.qual.NonNull; | import org.checkerframework.checker.nullness.qual.NonNull; | ||||||
|  |  | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
| @@ -37,7 +38,6 @@ import java.util.HashSet; | |||||||
| import java.util.Iterator; | import java.util.Iterator; | ||||||
| import java.util.LinkedList; | import java.util.LinkedList; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Map; |  | ||||||
| import java.util.Set; | import java.util.Set; | ||||||
| import java.util.UUID; | import java.util.UUID; | ||||||
| import java.util.function.Predicate; | import java.util.function.Predicate; | ||||||
| @@ -229,6 +229,7 @@ public final class PlotQuery implements Iterable<Plot> { | |||||||
|     public @NonNull PlotQuery hasOwner() { |     public @NonNull PlotQuery hasOwner() { | ||||||
|         return this.addFilter(new HasOwnerFilter()); |         return this.addFilter(new HasOwnerFilter()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Query for plots with a specific alias |      * Query for plots with a specific alias | ||||||
|      * |      * | ||||||
| @@ -307,7 +308,15 @@ public final class PlotQuery implements Iterable<Plot> { | |||||||
|      * @return Matching plots |      * @return Matching plots | ||||||
|      */ |      */ | ||||||
|     public @NonNull Stream<Plot> asStream() { |     public @NonNull Stream<Plot> asStream() { | ||||||
|         return this.asList().stream(); |         Stream<Plot> stream = this.plotProvider.streamPlots().filter(testPlotAgainstFilters()); | ||||||
|  |         if (this.sortingStrategy == SortingStrategy.NO_SORTING) { | ||||||
|  |             return stream; | ||||||
|  |         } | ||||||
|  |         stream = stream.sorted(getConfiguredComparator()); | ||||||
|  |         if (this.priorityArea != null) { | ||||||
|  |             stream = stream.sorted(new PlotInPrioritizedAreaComparator(this.priorityArea)); | ||||||
|  |         } | ||||||
|  |         return stream; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -334,49 +343,10 @@ public final class PlotQuery implements Iterable<Plot> { | |||||||
|         } |         } | ||||||
|         if (this.sortingStrategy == SortingStrategy.NO_SORTING) { |         if (this.sortingStrategy == SortingStrategy.NO_SORTING) { | ||||||
|             return result; |             return result; | ||||||
|         } else if (this.sortingStrategy == SortingStrategy.SORT_BY_TEMP) { |  | ||||||
|             return PlotSquared.get().sortPlotsByTemp(result); |  | ||||||
|         } else if (this.sortingStrategy == SortingStrategy.SORT_BY_DONE) { |  | ||||||
|             result.sort((a, b) -> { |  | ||||||
|                 String va = a.getFlag(DoneFlag.class); |  | ||||||
|                 String vb = b.getFlag(DoneFlag.class); |  | ||||||
|                 if (MathMan.isInteger(va)) { |  | ||||||
|                     if (MathMan.isInteger(vb)) { |  | ||||||
|                         return Integer.parseInt(vb) - Integer.parseInt(va); |  | ||||||
|         } |         } | ||||||
|                     return -1; |         result.sort(getConfiguredComparator()); | ||||||
|                 } |         if (this.priorityArea != null) { | ||||||
|                 return 1; |             result.sort(new PlotInPrioritizedAreaComparator(this.priorityArea)); | ||||||
|             }); |  | ||||||
|         } else if (this.sortingStrategy == SortingStrategy.SORT_BY_RATING) { |  | ||||||
|             result.sort((p1, p2) -> { |  | ||||||
|                 double v1 = 0; |  | ||||||
|                 int p1s = p1.getSettings().getRatings().size(); |  | ||||||
|                 int p2s = p2.getRatings().size(); |  | ||||||
|                 if (!p1.getSettings().getRatings().isEmpty()) { |  | ||||||
|                     v1 = p1.getRatings().values().stream().mapToDouble(Rating::getAverageRating) |  | ||||||
|                             .map(av -> av * av).sum(); |  | ||||||
|                     v1 /= p1s; |  | ||||||
|                     v1 += p1s; |  | ||||||
|                 } |  | ||||||
|                 double v2 = 0; |  | ||||||
|                 if (!p2.getSettings().getRatings().isEmpty()) { |  | ||||||
|                     for (Map.Entry<UUID, Rating> entry : p2.getRatings().entrySet()) { |  | ||||||
|                         double av = entry.getValue().getAverageRating(); |  | ||||||
|                         v2 += av * av; |  | ||||||
|                     } |  | ||||||
|                     v2 /= p2s; |  | ||||||
|                     v2 += p2s; |  | ||||||
|                 } |  | ||||||
|                 if (v2 == v1 && v2 != 0) { |  | ||||||
|                     return p2s - p1s; |  | ||||||
|                 } |  | ||||||
|                 return (int) Math.signum(v2 - v1); |  | ||||||
|             }); |  | ||||||
|         } else if (this.sortingStrategy == SortingStrategy.SORT_BY_CREATION) { |  | ||||||
|             return PlotSquared.get().sortPlots(result, PlotSquared.SortType.CREATION_DATE, this.priorityArea); |  | ||||||
|         } else if (this.sortingStrategy == SortingStrategy.COMPARATOR) { |  | ||||||
|             result.sort(this.plotComparator); |  | ||||||
|         } |         } | ||||||
|         return result; |         return result; | ||||||
|     } |     } | ||||||
| @@ -427,28 +397,22 @@ public final class PlotQuery implements Iterable<Plot> { | |||||||
|      * @return {@code true} if any provided plot matches the filters. |      * @return {@code true} if any provided plot matches the filters. | ||||||
|      */ |      */ | ||||||
|     public boolean anyMatch() { |     public boolean anyMatch() { | ||||||
|         if (this.filters.isEmpty()) { |         return hasMinimumMatches(1); | ||||||
|             return !this.plotProvider.getPlots().isEmpty(); |  | ||||||
|         } else { |  | ||||||
|             final Collection<Plot> plots = this.plotProvider.getPlots(); |  | ||||||
|             outer: |  | ||||||
|             for (final Plot plot : plots) { |  | ||||||
|                 // a plot must pass all filters to match the criteria |  | ||||||
|                 for (final PlotFilter filter : this.filters) { |  | ||||||
|                     if (!filter.accepts(plot)) { |  | ||||||
|                         continue outer; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 return true; // a plot passed all filters, so we have a match |  | ||||||
|             } |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @NonNull |     /** | ||||||
|     private PlotQuery addFilter(final @NonNull PlotFilter filter) { |      * Get whether this query matches at least {@code minimum} plots. | ||||||
|         this.filters.add(filter); |      * <br /> | ||||||
|         return this; |      * Should be prioritized over {@link #count()}, if possible. | ||||||
|  |      * This method only queries as many plots and areas as required without applying sorting. (short-circuiting on the stream) | ||||||
|  |      * | ||||||
|  |      * @param minimum the minimum amount of matches plots expected (inclusive) | ||||||
|  |      * @return {@code true} if this query's result contains at least {@code minimum} plots (after optional filters). | ||||||
|  |      * | ||||||
|  |      * @since TODO | ||||||
|  |      */ | ||||||
|  |     public boolean hasMinimumMatches(int minimum) { | ||||||
|  |         return this.plotProvider.streamPlots().filter(testPlotAgainstFilters()).limit(minimum).count() == minimum; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @NonNull |     @NonNull | ||||||
| @@ -457,4 +421,34 @@ public final class PlotQuery implements Iterable<Plot> { | |||||||
|         return this.asCollection().iterator(); |         return this.asCollection().iterator(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @NonNull | ||||||
|  |     private PlotQuery addFilter(final @NonNull PlotFilter filter) { | ||||||
|  |         this.filters.add(filter); | ||||||
|  |         return this; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private Comparator<Plot> getConfiguredComparator() { | ||||||
|  |         return switch (sortingStrategy) { | ||||||
|  |             case NO_SORTING -> (p1, p2) -> 0; | ||||||
|  |             case SORT_BY_TEMP, SORT_BY_CREATION -> PlotByCreationDateComparator.INSTANCE; | ||||||
|  |             case SORT_BY_DONE -> PlotByDoneComparator.INSTANCE; | ||||||
|  |             case SORT_BY_RATING -> PlotByRatingComparator.INSTANCE; | ||||||
|  |             case COMPARATOR -> plotComparator; | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private Predicate<Plot> testPlotAgainstFilters() { | ||||||
|  |         if (this.filters.isEmpty()) { | ||||||
|  |             return plot -> true; | ||||||
|  |         } | ||||||
|  |         return plot -> { | ||||||
|  |             for (final PlotFilter filter : filters) { | ||||||
|  |                 if (!filter.accepts(plot)) { | ||||||
|  |                     return false; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             return true; | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -31,6 +31,7 @@ import java.util.List; | |||||||
| import java.util.UUID; | import java.util.UUID; | ||||||
| import java.util.stream.Collectors; | import java.util.stream.Collectors; | ||||||
| import java.util.stream.IntStream; | import java.util.stream.IntStream; | ||||||
|  | import java.util.stream.Stream; | ||||||
|  |  | ||||||
| class SearchPlotProvider implements PlotProvider { | class SearchPlotProvider implements PlotProvider { | ||||||
|  |  | ||||||
| @@ -115,4 +116,9 @@ class SearchPlotProvider implements PlotProvider { | |||||||
|         return getPlotsBySearch(this.searchTerm); |         return getPlotsBySearch(this.searchTerm); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public Stream<Plot> streamPlots() { | ||||||
|  |         return getPlots().stream(); // TODO: look into potentially optimizations here? | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -234,7 +234,6 @@ | |||||||
|   "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>", | ||||||
| @@ -354,8 +353,6 @@ | |||||||
|   "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>", | ||||||
| @@ -393,11 +390,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><world>;<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><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><world>;<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><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><world>;<plot></gold><dark_aqua>.</dark_aqua>", |   "working.claimed": "<prefix><dark_aqua>You successfully claimed the plot </dark_aqua><gold><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>", | ||||||
| @@ -415,7 +412,6 @@ | |||||||
|   "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>", | ||||||
| @@ -458,7 +454,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>1</gold> <gray>grant has been added. (<grants> total grants)</gray>", |   "grants.added": "<prefix><gold><grants></gold> <gray>grant(s) have been added.</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,7 +37,12 @@ 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").clickEvent(ClickEvent.openFile("World")); |         var component = Component.text("Hello") | ||||||
|  |                 .clickEvent(ClickEvent.clickEvent( | ||||||
|  |                                 commonAction, | ||||||
|  |                                 "World" | ||||||
|  |                         ) | ||||||
|  |                 ); | ||||||
|         var transformedComponent = transform.transform(component); |         var transformedComponent = transform.transform(component); | ||||||
|         Assertions.assertNull(transformedComponent.clickEvent()); |         Assertions.assertNull(transformedComponent.clickEvent()); | ||||||
|     } |     } | ||||||
| @@ -47,7 +52,10 @@ 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.changePage(1337); |         var originalClickEvent = ClickEvent.clickEvent( | ||||||
|  |                 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); | ||||||
| @@ -68,12 +76,12 @@ class ClickStripTransformTest { | |||||||
|                 .insertion("DEF"); |                 .insertion("DEF"); | ||||||
|         var component = Component.text("Hello ") |         var component = Component.text("Hello ") | ||||||
|                 .append( |                 .append( | ||||||
|                         inner.clickEvent(ClickEvent.openUrl("https://example.org")) |                         inner.clickEvent(ClickEvent.clickEvent(ClickEvent.Action.OPEN_URL, "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().getFirst()); // only the click event has changed |         Assertions.assertEquals(inner, transformedComponent.children().get(0)); // only the click event has changed | ||||||
|         Assertions.assertNull(transformedComponent.children().getFirst().clickEvent()); |         Assertions.assertNull(transformedComponent.children().get(0).clickEvent()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -2,16 +2,18 @@ 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.publish) |     alias(libs.plugins.nexus) | ||||||
|  |  | ||||||
|     eclipse |     eclipse | ||||||
|     idea |     idea | ||||||
| @@ -20,7 +22,7 @@ plugins { | |||||||
| } | } | ||||||
|  |  | ||||||
| group = "com.intellectualsites.plotsquared" | group = "com.intellectualsites.plotsquared" | ||||||
| version = "7.5.9-SNAPSHOT" | version = "7.5.1-SNAPSHOT" | ||||||
|  |  | ||||||
| if (!File("$rootDir/.git").exists()) { | if (!File("$rootDir/.git").exists()) { | ||||||
|     logger.lifecycle(""" |     logger.lifecycle(""" | ||||||
| @@ -39,6 +41,16 @@ 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") | ||||||
| @@ -56,7 +68,7 @@ subprojects { | |||||||
|     apply { |     apply { | ||||||
|         plugin<JavaPlugin>() |         plugin<JavaPlugin>() | ||||||
|         plugin<JavaLibraryPlugin>() |         plugin<JavaLibraryPlugin>() | ||||||
|         plugin<com.vanniktech.maven.publish.MavenPublishPlugin>() |         plugin<MavenPublishPlugin>() | ||||||
|         plugin<ShadowPlugin>() |         plugin<ShadowPlugin>() | ||||||
|         plugin<SpotlessPlugin>() |         plugin<SpotlessPlugin>() | ||||||
|         plugin<SigningPlugin>() |         plugin<SigningPlugin>() | ||||||
| @@ -65,16 +77,10 @@ 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:6.0.0") |         testImplementation("org.junit.jupiter:junit-jupiter:5.12.0") | ||||||
|         testRuntimeOnly("org.junit.platform:junit-platform-launcher:6.0.0") |         testRuntimeOnly("org.junit.platform:junit-platform-launcher:1.12.0") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     plugins.withId("java") { |     plugins.withId("java") { | ||||||
| @@ -101,16 +107,15 @@ subprojects { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     afterEvaluate { |     java { | ||||||
|  |         withSourcesJar() | ||||||
|  |         withJavadocJar() | ||||||
|  |     } | ||||||
|  |  | ||||||
|     val javaComponent = components["java"] as AdhocComponentWithVariants |     val javaComponent = components["java"] as AdhocComponentWithVariants | ||||||
|         configurations.findByName("shadowRuntimeElements")?.let { shadowRuntimeElements -> |     javaComponent.withVariantsFromConfiguration(configurations["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")) { | ||||||
| @@ -122,15 +127,14 @@ subprojects { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     mavenPublishing { |     publishing { | ||||||
|         coordinates( |         publications { | ||||||
|             groupId = "$group", |             create<MavenPublication>("maven") { | ||||||
|             artifactId = project.name, |                 from(components["java"]) | ||||||
|             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") | ||||||
|  |  | ||||||
| @@ -181,8 +185,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() |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -190,6 +194,7 @@ subprojects { | |||||||
|  |  | ||||||
|         compileJava { |         compileJava { | ||||||
|             options.compilerArgs.add("-parameters") |             options.compilerArgs.add("-parameters") | ||||||
|  |             options.isDeprecation = true | ||||||
|             options.encoding = "UTF-8" |             options.encoding = "UTF-8" | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -212,18 +217,27 @@ 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", "1.21.5", "1.21.6", "1.21.7", "1.21.8") | val supportedVersions = listOf("1.19.4", "1.20.6", "1.21.1", "1.21.3", "1.21.4") | ||||||
| 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("Paper") } |                 .first { it -> it.contains("Bukkit") } | ||||||
|         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.8" | spotbugs = "4.9.1" | ||||||
| checkerqual = "3.51.1" | checkerqual = "3.49.0" | ||||||
| gson = "2.10" | gson = "2.10" | ||||||
| guava = "31.1-jre" | guava = "31.1-jre" | ||||||
| snakeyaml = "2.0" | snakeyaml = "2.0" | ||||||
| adventure = "4.25.0" | adventure = "4.19.0" | ||||||
| adventure-bukkit = "4.4.1" | adventure-bukkit = "4.3.4" | ||||||
| log4j = "2.19.0" | log4j = "2.19.0" | ||||||
|  |  | ||||||
| # Plugins | # Plugins | ||||||
| worldedit = "7.2.20" | worldedit = "7.2.20" | ||||||
| fawe = "2.14.0" | fawe = "2.13.0" | ||||||
| placeholderapi = "2.11.6" | placeholderapi = "2.11.6" | ||||||
| luckperms = "5.5" | luckperms = "5.4" | ||||||
| essentialsx = "2.21.2" | essentialsx = "2.20.1" | ||||||
| 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.4" | arkitektonika = "2.1.3" | ||||||
| squirrelid = "0.3.2" | squirrelid = "0.3.2" | ||||||
| paster = "1.1.7" | paster = "1.1.6" | ||||||
| bstats = "3.1.0" | bstats = "3.1.0" | ||||||
| paperlib = "1.0.8" | paperlib = "1.0.8" | ||||||
| informative-annotations = "1.6" | informative-annotations = "1.5" | ||||||
| vault = "1.7.1" | vault = "1.7.1" | ||||||
| serverlib = "2.3.7" | serverlib = "2.3.6" | ||||||
|  |  | ||||||
| # Gradle plugins | # Gradle plugins | ||||||
| shadow = "9.2.2" | shadow = "8.3.6" | ||||||
| grgit = "4.1.1" | grgit = "4.1.1" | ||||||
| spotless = "8.0.0" | spotless = "7.0.2" | ||||||
| publish = "0.34.0" | nexus = "2.0.0" | ||||||
| runPaper = "3.0.2" | runPaper = "2.3.1" | ||||||
|  |  | ||||||
| [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" } | ||||||
| publish = { id = "com.vanniktech.maven.publish", version.ref = "publish" } | nexus = { id = "io.github.gradle-nexus.publish-plugin", version.ref = "nexus" } | ||||||
| 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-9.1.0-bin.zip | distributionUrl=https\://services.gradle.org/distributions/gradle-8.12.1-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 the original authors. | # Copyright © 2015-2021 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,6 +114,7 @@ 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. | ||||||
| @@ -171,6 +172,7 @@ 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" ) | ||||||
|  |  | ||||||
| @@ -203,14 +205,15 @@ 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, and optsEnvironmentVar are not allowed to contain shell fragments, | #   * DEFAULT_JVM_OPTS, JAVA_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" \ | ||||||
|         -jar "$APP_HOME/gradle/wrapper/gradle-wrapper.jar" \ |         -classpath "$CLASSPATH" \ | ||||||
|  |         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,10 +70,11 @@ 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%" -jar "%APP_HOME%\gradle\wrapper\gradle-wrapper.jar" %* | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* | ||||||
|  |  | ||||||
| :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