mirror of
				https://github.com/IntellectualSites/PlotSquared.git
				synced 2025-10-24 23:23:44 +02:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			renovate/f
			...
			fix/disabl
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | defb84e7c4 | 
							
								
								
									
										5
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							| @@ -24,14 +24,13 @@ body: | ||||
|   - type: dropdown | ||||
|     attributes: | ||||
|       label: Server Version | ||||
|       description: Which server version are you using? If your server version is not listed, it is not supported. Update to a supported version first. | ||||
|       description: Which server version version you using? If your server version is not listed, it is not supported. Update to a supported version first. | ||||
|       multiple: false | ||||
|       options: | ||||
|         - '1.21.1' | ||||
|         - '1.20.6' | ||||
|         - '1.20.4' | ||||
|         - '1.20' | ||||
|         - '1.19.4' | ||||
|         - '1.18.2' | ||||
|     validations: | ||||
|       required: true | ||||
|  | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/build-pr.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/build-pr.yml
									
									
									
									
										vendored
									
									
								
							| @@ -11,7 +11,7 @@ jobs: | ||||
|       - name: Checkout Repository | ||||
|         uses: actions/checkout@v4 | ||||
|       - name: Validate Gradle Wrapper | ||||
|         uses: gradle/actions/wrapper-validation@v4 | ||||
|         uses: gradle/actions/wrapper-validation@v3 | ||||
|       - name: Setup Java | ||||
|         uses: actions/setup-java@v4 | ||||
|         with: | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -11,7 +11,7 @@ jobs: | ||||
|       - name: Checkout Repository | ||||
|         uses: actions/checkout@v4 | ||||
|       - name: Validate Gradle Wrapper | ||||
|         uses: gradle/actions/wrapper-validation@v4 | ||||
|         uses: gradle/actions/wrapper-validation@v3 | ||||
|       - name: Setup Java | ||||
|         uses: actions/setup-java@v4 | ||||
|         with: | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/label-merge-conflicts.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/label-merge-conflicts.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -15,7 +15,7 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Label conflicting PRs | ||||
|         uses: eps1lon/actions-label-merge-conflict@v3.0.2 | ||||
|         uses: eps1lon/actions-label-merge-conflict@v3.0.0 | ||||
|         with: | ||||
|           dirtyLabel: "unresolved-merge-conflict" | ||||
|           repoToken: "${{ secrets.GITHUB_TOKEN }}" | ||||
|   | ||||
| @@ -34,7 +34,6 @@ import com.plotsquared.bukkit.listener.BlockEventListener117; | ||||
| import com.plotsquared.bukkit.listener.ChunkListener; | ||||
| import com.plotsquared.bukkit.listener.EntityEventListener; | ||||
| import com.plotsquared.bukkit.listener.EntitySpawnListener; | ||||
| import com.plotsquared.bukkit.listener.HighFreqBlockEventListener; | ||||
| import com.plotsquared.bukkit.listener.PaperListener; | ||||
| import com.plotsquared.bukkit.listener.PlayerEventListener; | ||||
| import com.plotsquared.bukkit.listener.PlayerEventListener1201; | ||||
| @@ -363,9 +362,6 @@ public final class BukkitPlatform extends JavaPlugin implements Listener, PlotPl | ||||
|                 getServer().getPluginManager().registerEvents(injector().getInstance(PlayerEventListener1201.class), this); | ||||
|             } | ||||
|             getServer().getPluginManager().registerEvents(injector().getInstance(BlockEventListener.class), this); | ||||
|             if (Settings.HIGH_FREQUENCY_LISTENER) { | ||||
|                 getServer().getPluginManager().registerEvents(injector().getInstance(HighFreqBlockEventListener.class), this); | ||||
|             } | ||||
|             if (serverVersion()[1] >= 17) { | ||||
|                 getServer().getPluginManager().registerEvents(injector().getInstance(BlockEventListener117.class), this); | ||||
|             } | ||||
|   | ||||
| @@ -24,6 +24,7 @@ import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.configuration.caption.TranslatableCaption; | ||||
| import com.plotsquared.core.database.DBFunc; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.permissions.Permission; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| @@ -47,6 +48,7 @@ import com.plotsquared.core.plot.flag.implementations.LeafDecayFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.LiquidFlowFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.MycelGrowFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.PlaceFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.RedstoneFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.SnowFormFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.SnowMeltFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.SoilDryFlag; | ||||
| @@ -60,6 +62,7 @@ import com.plotsquared.core.util.task.TaskTime; | ||||
| import com.sk89q.worldedit.WorldEdit; | ||||
| import com.sk89q.worldedit.bukkit.BukkitAdapter; | ||||
| import com.sk89q.worldedit.world.block.BlockType; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.minimessage.tag.Tag; | ||||
| import net.kyori.adventure.text.minimessage.tag.resolver.TagResolver; | ||||
| import org.bukkit.Bukkit; | ||||
| @@ -89,9 +92,11 @@ import org.bukkit.event.block.BlockFromToEvent; | ||||
| import org.bukkit.event.block.BlockGrowEvent; | ||||
| import org.bukkit.event.block.BlockIgniteEvent; | ||||
| import org.bukkit.event.block.BlockMultiPlaceEvent; | ||||
| import org.bukkit.event.block.BlockPhysicsEvent; | ||||
| import org.bukkit.event.block.BlockPistonExtendEvent; | ||||
| import org.bukkit.event.block.BlockPistonRetractEvent; | ||||
| import org.bukkit.event.block.BlockPlaceEvent; | ||||
| import org.bukkit.event.block.BlockRedstoneEvent; | ||||
| import org.bukkit.event.block.BlockSpreadEvent; | ||||
| import org.bukkit.event.block.CauldronLevelChangeEvent; | ||||
| import org.bukkit.event.block.EntityBlockFormEvent; | ||||
| @@ -107,6 +112,7 @@ import java.util.Iterator; | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
| import java.util.Set; | ||||
| import java.util.UUID; | ||||
| import java.util.stream.Collectors; | ||||
| import java.util.stream.Stream; | ||||
|  | ||||
| @@ -117,6 +123,14 @@ import static org.bukkit.Tag.WALL_CORALS; | ||||
| @SuppressWarnings("unused") | ||||
| public class BlockEventListener implements Listener { | ||||
|  | ||||
|     private static final Set<Material> PISTONS = Set.of( | ||||
|             Material.PISTON, | ||||
|             Material.STICKY_PISTON | ||||
|     ); | ||||
|     private static final Set<Material> PHYSICS_BLOCKS = Set.of( | ||||
|             Material.TURTLE_EGG, | ||||
|             Material.TURTLE_SPAWN_EGG | ||||
|     ); | ||||
|     private static final Set<Material> SNOW = Stream.of(Material.values()) // needed as Tag.SNOW isn't present in 1.16.5 | ||||
|             .filter(material -> material.name().contains("SNOW")) | ||||
|             .filter(Material::isBlock) | ||||
| @@ -150,6 +164,114 @@ public class BlockEventListener implements Listener { | ||||
|         }, TaskTime.ticks(3L)); | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onRedstoneEvent(BlockRedstoneEvent event) { | ||||
|         Block block = event.getBlock(); | ||||
|         Location location = BukkitUtil.adapt(block.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot == null) { | ||||
|             if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, RedstoneFlag.class, false)) { | ||||
|                 event.setNewCurrent(0); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (!plot.getFlag(RedstoneFlag.class)) { | ||||
|             event.setNewCurrent(0); | ||||
|             plot.debug("Redstone event was cancelled because redstone = false"); | ||||
|             return; | ||||
|         } | ||||
|         if (Settings.Redstone.DISABLE_OFFLINE) { | ||||
|             boolean disable = false; | ||||
|             if (!DBFunc.SERVER.equals(plot.getOwner())) { | ||||
|                 if (plot.isMerged()) { | ||||
|                     disable = true; | ||||
|                     for (UUID owner : plot.getOwners()) { | ||||
|                         if (PlotSquared.platform().playerManager().getPlayerIfExists(owner) != null) { | ||||
|                             disable = false; | ||||
|                             break; | ||||
|                         } | ||||
|                     } | ||||
|                 } else { | ||||
|                     disable = PlotSquared.platform().playerManager().getPlayerIfExists(plot.getOwnerAbs()) == null; | ||||
|                 } | ||||
|             } | ||||
|             if (disable) { | ||||
|                 for (UUID trusted : plot.getTrusted()) { | ||||
|                     if (PlotSquared.platform().playerManager().getPlayerIfExists(trusted) != null) { | ||||
|                         disable = false; | ||||
|                         break; | ||||
|                     } | ||||
|                 } | ||||
|                 if (disable) { | ||||
|                     event.setNewCurrent(0); | ||||
|                     plot.debug("Redstone event was cancelled because no trusted player was in the plot"); | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         if (Settings.Redstone.DISABLE_UNOCCUPIED) { | ||||
|             for (final PlotPlayer<?> player : PlotSquared.platform().playerManager().getPlayers()) { | ||||
|                 if (plot.equals(player.getCurrentPlot())) { | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|             event.setNewCurrent(0); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(ignoreCancelled = true, priority = EventPriority.HIGHEST) | ||||
|     public void onPhysicsEvent(BlockPhysicsEvent event) { | ||||
|         Block block = event.getBlock(); | ||||
|         Location location = BukkitUtil.adapt(block.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = area.getOwnedPlotAbs(location); | ||||
|         if (plot == null) { | ||||
|             return; | ||||
|         } | ||||
|         if (event.getChangedType().hasGravity() && plot.getFlag(DisablePhysicsFlag.class)) { | ||||
|             event.setCancelled(true); | ||||
|             sendBlockChange(event.getBlock().getLocation(), event.getBlock().getBlockData()); | ||||
|             plot.debug("Prevented block physics and resent block change because disable-physics = true"); | ||||
|             return; | ||||
|         } | ||||
|         if (event.getChangedType() == Material.COMPARATOR) { | ||||
|             if (!plot.getFlag(RedstoneFlag.class)) { | ||||
|                 event.setCancelled(true); | ||||
|                 plot.debug("Prevented comparator update because redstone = false"); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (PHYSICS_BLOCKS.contains(event.getChangedType())) { | ||||
|             if (plot.getFlag(DisablePhysicsFlag.class)) { | ||||
|                 event.setCancelled(true); | ||||
|                 plot.debug("Prevented block physics because disable-physics = true"); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (Settings.Redstone.DETECT_INVALID_EDGE_PISTONS) { | ||||
|             if (PISTONS.contains(block.getType())) { | ||||
|                 org.bukkit.block.data.Directional piston = (org.bukkit.block.data.Directional) block.getBlockData(); | ||||
|                 final BlockFace facing = piston.getFacing(); | ||||
|                 location = location.add(facing.getModX(), facing.getModY(), facing.getModZ()); | ||||
|                 Plot newPlot = area.getOwnedPlotAbs(location); | ||||
|                 if (plot.equals(newPlot)) { | ||||
|                     return; | ||||
|                 } | ||||
|                 if (!plot.isMerged() || !plot.getConnectedPlots().contains(newPlot)) { | ||||
|                     event.setCancelled(true); | ||||
|                     plot.debug("Prevented piston update because of invalid edge piston detection"); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void blockCreate(BlockPlaceEvent event) { | ||||
|         Location location = BukkitUtil.adapt(event.getBlock().getLocation()); | ||||
| @@ -163,6 +285,13 @@ public class BlockEventListener implements Listener { | ||||
|         if (plot != null) { | ||||
|             if (area.notifyIfOutsideBuildArea(pp, location.getY())) { | ||||
|                 event.setCancelled(true); | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("height.height_limit"), | ||||
|                         TagResolver.builder() | ||||
|                                 .tag("minheight", Tag.inserting(Component.text(area.getMinBuildHeight()))) | ||||
|                                 .tag("maxheight", Tag.inserting(Component.text(area.getMaxBuildHeight()))) | ||||
|                                 .build() | ||||
|                 ); | ||||
|                 return; | ||||
|             } | ||||
|             if (!plot.hasOwner()) { | ||||
| @@ -254,6 +383,13 @@ public class BlockEventListener implements Listener { | ||||
|                 } | ||||
|             } else if (area.notifyIfOutsideBuildArea(plotPlayer, location.getY())) { | ||||
|                 event.setCancelled(true); | ||||
|                 plotPlayer.sendMessage( | ||||
|                         TranslatableCaption.of("height.height_limit"), | ||||
|                         TagResolver.builder() | ||||
|                                 .tag("minheight", Tag.inserting(Component.text(area.getMinBuildHeight()))) | ||||
|                                 .tag("maxheight", Tag.inserting(Component.text(area.getMaxBuildHeight()))) | ||||
|                                 .build() | ||||
|                 ); | ||||
|                 return; | ||||
|             } | ||||
|             if (!plot.hasOwner()) { | ||||
| @@ -1209,9 +1345,18 @@ public class BlockEventListener implements Listener { | ||||
|             if (pp.hasPermission(Permission.PERMISSION_ADMIN_BUILD_HEIGHT_LIMIT)) { | ||||
|                 continue; | ||||
|             } | ||||
|             if (area.notifyIfOutsideBuildArea(pp, currentLocation.getY())) { | ||||
|                 event.setCancelled(true); | ||||
|                 break; | ||||
|             if (currentLocation.getY() >= area.getMaxBuildHeight() || currentLocation.getY() < area.getMinBuildHeight()) { | ||||
|                 pp.sendMessage( | ||||
|                         TranslatableCaption.of("height.height_limit"), | ||||
|                         TagResolver.builder() | ||||
|                                 .tag("minheight", Tag.inserting(Component.text(area.getMinBuildHeight()))) | ||||
|                                 .tag("maxheight", Tag.inserting(Component.text(area.getMaxBuildHeight()))) | ||||
|                                 .build() | ||||
|                 ); | ||||
|                 if (area.notifyIfOutsideBuildArea(pp, currentLocation.getY())) { | ||||
|                     event.setCancelled(true); | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -1,201 +0,0 @@ | ||||
| /* | ||||
|  * PlotSquared, a land and world management plugin for Minecraft. | ||||
|  * Copyright (C) IntellectualSites <https://intellectualsites.com> | ||||
|  * Copyright (C) IntellectualSites team and contributors | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| package com.plotsquared.bukkit.listener; | ||||
|  | ||||
| import com.google.inject.Inject; | ||||
| import com.plotsquared.bukkit.player.BukkitPlayer; | ||||
| import com.plotsquared.bukkit.util.BukkitUtil; | ||||
| import com.plotsquared.core.PlotSquared; | ||||
| import com.plotsquared.core.configuration.Settings; | ||||
| import com.plotsquared.core.database.DBFunc; | ||||
| import com.plotsquared.core.location.Location; | ||||
| import com.plotsquared.core.player.PlotPlayer; | ||||
| import com.plotsquared.core.plot.Plot; | ||||
| import com.plotsquared.core.plot.PlotArea; | ||||
| import com.plotsquared.core.plot.flag.implementations.DisablePhysicsFlag; | ||||
| import com.plotsquared.core.plot.flag.implementations.RedstoneFlag; | ||||
| import com.plotsquared.core.plot.world.PlotAreaManager; | ||||
| import com.plotsquared.core.util.PlotFlagUtil; | ||||
| import com.plotsquared.core.util.task.TaskManager; | ||||
| import com.plotsquared.core.util.task.TaskTime; | ||||
| import com.sk89q.worldedit.WorldEdit; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.Material; | ||||
| import org.bukkit.block.Block; | ||||
| import org.bukkit.block.BlockFace; | ||||
| import org.bukkit.block.data.BlockData; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.EventPriority; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.event.block.BlockPhysicsEvent; | ||||
| import org.bukkit.event.block.BlockRedstoneEvent; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.util.Set; | ||||
| import java.util.UUID; | ||||
|  | ||||
| @SuppressWarnings("unused") | ||||
| public class HighFreqBlockEventListener implements Listener { | ||||
|  | ||||
|     private static final Set<Material> PISTONS = Set.of( | ||||
|             Material.PISTON, | ||||
|             Material.STICKY_PISTON | ||||
|     ); | ||||
|     private static final Set<Material> PHYSICS_BLOCKS = Set.of( | ||||
|             Material.TURTLE_EGG, | ||||
|             Material.TURTLE_SPAWN_EGG | ||||
|     ); | ||||
|  | ||||
|     private final PlotAreaManager plotAreaManager; | ||||
|     private final WorldEdit worldEdit; | ||||
|  | ||||
|     @Inject | ||||
|     public HighFreqBlockEventListener(final @NonNull PlotAreaManager plotAreaManager, final @NonNull WorldEdit worldEdit) { | ||||
|         this.plotAreaManager = plotAreaManager; | ||||
|         this.worldEdit = worldEdit; | ||||
|     } | ||||
|  | ||||
|     public static void sendBlockChange(final org.bukkit.Location bloc, final BlockData data) { | ||||
|         TaskManager.runTaskLater(() -> { | ||||
|             String world = bloc.getWorld().getName(); | ||||
|             int x = bloc.getBlockX(); | ||||
|             int z = bloc.getBlockZ(); | ||||
|             int distance = Bukkit.getViewDistance() * 16; | ||||
|  | ||||
|             for (final PlotPlayer<?> player : PlotSquared.platform().playerManager().getPlayers()) { | ||||
|                 Location location = player.getLocation(); | ||||
|                 if (location.getWorldName().equals(world)) { | ||||
|                     if (16 * Math.abs(location.getX() - x) / 16 > distance || 16 * Math.abs(location.getZ() - z) / 16 > distance) { | ||||
|                         continue; | ||||
|                     } | ||||
|                     ((BukkitPlayer) player).player.sendBlockChange(bloc, data); | ||||
|                 } | ||||
|             } | ||||
|         }, TaskTime.ticks(3L)); | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onRedstoneEvent(BlockRedstoneEvent event) { | ||||
|         Block block = event.getBlock(); | ||||
|         Location location = BukkitUtil.adapt(block.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = location.getOwnedPlot(); | ||||
|         if (plot == null) { | ||||
|             if (PlotFlagUtil.isAreaRoadFlagsAndFlagEquals(area, RedstoneFlag.class, false)) { | ||||
|                 event.setNewCurrent(0); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (!plot.getFlag(RedstoneFlag.class)) { | ||||
|             event.setNewCurrent(0); | ||||
|             plot.debug("Redstone event was cancelled because redstone = false"); | ||||
|             return; | ||||
|         } | ||||
|         if (Settings.Redstone.DISABLE_OFFLINE) { | ||||
|             boolean disable = false; | ||||
|             if (!DBFunc.SERVER.equals(plot.getOwner())) { | ||||
|                 if (plot.isMerged()) { | ||||
|                     disable = true; | ||||
|                     for (UUID owner : plot.getOwners()) { | ||||
|                         if (PlotSquared.platform().playerManager().getPlayerIfExists(owner) != null) { | ||||
|                             disable = false; | ||||
|                             break; | ||||
|                         } | ||||
|                     } | ||||
|                 } else { | ||||
|                     disable = PlotSquared.platform().playerManager().getPlayerIfExists(plot.getOwnerAbs()) == null; | ||||
|                 } | ||||
|             } | ||||
|             if (disable) { | ||||
|                 for (UUID trusted : plot.getTrusted()) { | ||||
|                     if (PlotSquared.platform().playerManager().getPlayerIfExists(trusted) != null) { | ||||
|                         disable = false; | ||||
|                         break; | ||||
|                     } | ||||
|                 } | ||||
|                 if (disable) { | ||||
|                     event.setNewCurrent(0); | ||||
|                     plot.debug("Redstone event was cancelled because no trusted player was in the plot"); | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         if (Settings.Redstone.DISABLE_UNOCCUPIED) { | ||||
|             for (final PlotPlayer<?> player : PlotSquared.platform().playerManager().getPlayers()) { | ||||
|                 if (plot.equals(player.getCurrentPlot())) { | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|             event.setNewCurrent(0); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(ignoreCancelled = true, priority = EventPriority.HIGHEST) | ||||
|     public void onPhysicsEvent(BlockPhysicsEvent event) { | ||||
|         Block block = event.getBlock(); | ||||
|         Location location = BukkitUtil.adapt(block.getLocation()); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (area == null) { | ||||
|             return; | ||||
|         } | ||||
|         Plot plot = area.getOwnedPlotAbs(location); | ||||
|         if (plot == null) { | ||||
|             return; | ||||
|         } | ||||
|         if (event.getChangedType().hasGravity() && plot.getFlag(DisablePhysicsFlag.class)) { | ||||
|             event.setCancelled(true); | ||||
|             sendBlockChange(event.getBlock().getLocation(), event.getBlock().getBlockData()); | ||||
|             plot.debug("Prevented block physics and resent block change because disable-physics = true"); | ||||
|             return; | ||||
|         } | ||||
|         if (event.getChangedType() == Material.COMPARATOR) { | ||||
|             if (!plot.getFlag(RedstoneFlag.class)) { | ||||
|                 event.setCancelled(true); | ||||
|                 plot.debug("Prevented comparator update because redstone = false"); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (PHYSICS_BLOCKS.contains(event.getChangedType())) { | ||||
|             if (plot.getFlag(DisablePhysicsFlag.class)) { | ||||
|                 event.setCancelled(true); | ||||
|                 plot.debug("Prevented block physics because disable-physics = true"); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|         if (Settings.Redstone.DETECT_INVALID_EDGE_PISTONS) { | ||||
|             if (PISTONS.contains(block.getType())) { | ||||
|                 org.bukkit.block.data.Directional piston = (org.bukkit.block.data.Directional) block.getBlockData(); | ||||
|                 final BlockFace facing = piston.getFacing(); | ||||
|                 location = location.add(facing.getModX(), facing.getModY(), facing.getModZ()); | ||||
|                 Plot newPlot = area.getOwnedPlotAbs(location); | ||||
|                 if (plot.equals(newPlot)) { | ||||
|                     return; | ||||
|                 } | ||||
|                 if (!plot.isMerged() || !plot.getConnectedPlots().contains(newPlot)) { | ||||
|                     event.setCancelled(true); | ||||
|                     plot.debug("Prevented piston update because of invalid edge piston detection"); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -533,9 +533,9 @@ public class PlayerEventListener implements Listener { | ||||
|                     // to is identical to the plot's home location, and untrusted-visit is true | ||||
|                     // i.e. untrusted-visit can override deny-teleport | ||||
|                     // this is acceptable, because otherwise it wouldn't make sense to have both flags set | ||||
|                     if (result || (plot.getFlag(UntrustedVisitFlag.class) && plot.getHomeSynchronous().equals(BukkitUtil.adaptComplete(to)))) { | ||||
|                         plotListener.plotEntry(pp, plot); | ||||
|                     } else { | ||||
|                     if (!result && !(plot.getFlag(UntrustedVisitFlag.class) && plot | ||||
|                             .getHomeSynchronous() | ||||
|                             .equals(BukkitUtil.adaptComplete(to)))) { | ||||
|                         pp.sendMessage( | ||||
|                                 TranslatableCaption.of("deny.no_enter"), | ||||
|                                 TagResolver.resolver("plot", Tag.inserting(Component.text(plot.toString()))) | ||||
| @@ -548,19 +548,6 @@ public class PlayerEventListener implements Listener { | ||||
|         playerMove(event); | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onWorldChanged(PlayerChangedWorldEvent event) { | ||||
|         Player player = event.getPlayer(); | ||||
|         BukkitPlayer pp = BukkitUtil.adapt(player); | ||||
|         if (this.worldEdit != null) { | ||||
|             if (!pp.hasPermission(Permission.PERMISSION_WORLDEDIT_BYPASS)) { | ||||
|                 if (pp.getAttribute("worldedit")) { | ||||
|                     pp.removeAttribute("worldedit"); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void vehicleMove(VehicleMoveEvent event) | ||||
|             throws IllegalAccessException { | ||||
| @@ -900,6 +887,40 @@ public class PlayerEventListener implements Listener { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onWorldChanged(PlayerChangedWorldEvent event) { | ||||
|         Player player = event.getPlayer(); | ||||
|         BukkitPlayer pp = BukkitUtil.adapt(player); | ||||
|         // Delete last location | ||||
|         Plot plot; | ||||
|         try (final MetaDataAccess<Plot> lastPlotAccess = | ||||
|                      pp.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LAST_PLOT)) { | ||||
|             plot = lastPlotAccess.remove(); | ||||
|         } | ||||
|         try (final MetaDataAccess<Location> lastLocationAccess = | ||||
|                      pp.accessTemporaryMetaData(PlayerMetaDataKeys.TEMPORARY_LOCATION)) { | ||||
|             lastLocationAccess.remove(); | ||||
|         } | ||||
|         if (plot != null) { | ||||
|             plotListener.plotExit(pp, plot); | ||||
|         } | ||||
|         if (this.worldEdit != null) { | ||||
|             if (!pp.hasPermission(Permission.PERMISSION_WORLDEDIT_BYPASS)) { | ||||
|                 if (pp.getAttribute("worldedit")) { | ||||
|                     pp.removeAttribute("worldedit"); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         Location location = pp.getLocation(); | ||||
|         PlotArea area = location.getPlotArea(); | ||||
|         if (location.isPlotArea()) { | ||||
|             plot = location.getPlot(); | ||||
|             if (plot != null) { | ||||
|                 plotListener.plotEntry(pp, plot); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("deprecation") | ||||
|     @EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true) | ||||
|     public void onInventoryClick(InventoryClickEvent event) { | ||||
|   | ||||
| @@ -23,15 +23,11 @@ import com.plotsquared.core.location.World; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
|  | ||||
| import java.lang.ref.SoftReference; | ||||
| import java.lang.ref.WeakReference; | ||||
| import java.util.Map; | ||||
|  | ||||
| public class BukkitWorld implements World<org.bukkit.World> { | ||||
|  | ||||
|     // Upon world unload we should probably have the P2 BukkitWorld be GCed relatively eagerly, thus freeing the bukkit world. | ||||
|     //  We also want to avoid circumstances where a bukkit world has been GCed, but a P2 BukkitWorld has not | ||||
|     private static final Map<String, WeakReference<BukkitWorld>> worldMap = Maps.newHashMap(); | ||||
|     private static final Map<String, BukkitWorld> worldMap = Maps.newHashMap(); | ||||
|     private static final boolean HAS_MIN_Y; | ||||
|  | ||||
|     static { | ||||
| @@ -45,11 +41,10 @@ public class BukkitWorld implements World<org.bukkit.World> { | ||||
|         HAS_MIN_Y = temp; | ||||
|     } | ||||
|  | ||||
|     // We want to allow GC to remove bukkit worlds, but not too eagerly | ||||
|     private final SoftReference<org.bukkit.World> world; | ||||
|     private final org.bukkit.World world; | ||||
|  | ||||
|     private BukkitWorld(final org.bukkit.World world) { | ||||
|         this.world = new SoftReference<>(world); | ||||
|         this.world = world; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -73,13 +68,12 @@ public class BukkitWorld implements World<org.bukkit.World> { | ||||
|      * @return World instance | ||||
|      */ | ||||
|     public static @NonNull BukkitWorld of(final org.bukkit.World world) { | ||||
|         WeakReference<BukkitWorld> bukkitWorldRef = worldMap.get(world.getName()); | ||||
|         BukkitWorld bukkitWorld; | ||||
|         if (bukkitWorldRef != null && (bukkitWorld = bukkitWorldRef.get()) != null && world.equals(bukkitWorld.world.get())) { | ||||
|         BukkitWorld bukkitWorld = worldMap.get(world.getName()); | ||||
|         if (bukkitWorld != null && bukkitWorld.getPlatformWorld().equals(world)) { | ||||
|             return bukkitWorld; | ||||
|         } | ||||
|         bukkitWorld = new BukkitWorld(world); | ||||
|         worldMap.put(world.getName(), new WeakReference<>(bukkitWorld)); | ||||
|         worldMap.put(world.getName(), bukkitWorld); | ||||
|         return bukkitWorld; | ||||
|     } | ||||
|  | ||||
| @@ -103,26 +97,22 @@ public class BukkitWorld implements World<org.bukkit.World> { | ||||
|  | ||||
|     @Override | ||||
|     public org.bukkit.World getPlatformWorld() { | ||||
|         org.bukkit.World world = this.world.get(); | ||||
|         if (world == null) { | ||||
|             throw new IllegalStateException("Bukkit platform world was unloaded from memory"); | ||||
|         } | ||||
|         return world; | ||||
|         return this.world; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public @NonNull String getName() { | ||||
|         return this.getPlatformWorld().getName(); | ||||
|         return this.world.getName(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public int getMinHeight() { | ||||
|         return getMinWorldHeight(getPlatformWorld()); | ||||
|         return getMinWorldHeight(world); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public int getMaxHeight() { | ||||
|         return getMaxWorldHeight(getPlatformWorld()) - 1; | ||||
|         return getMaxWorldHeight(world) - 1; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|   | ||||
| @@ -43,11 +43,6 @@ public class Settings extends Config { | ||||
|             "Leave it off if you don't need it, it can spam your console."}) | ||||
|     public static boolean DEBUG = true; | ||||
|  | ||||
|     @Comment({"The activity of high-frequency event listener can be deactivated here to improve the server performance. ", | ||||
|             "Affected settings: 'redstone' settings here below. Affected flags: 'disable-physics', 'redstone'. ", | ||||
|             "Only deactivate this setting if you do not need any of the mentioned settings or flags."}) | ||||
|     public static boolean HIGH_FREQUENCY_LISTENER = true; | ||||
|  | ||||
|     @Create // This value will be generated automatically | ||||
|     public static ConfigBlock<Auto_Clear> AUTO_CLEAR = null; | ||||
|     // A ConfigBlock is a section that can have multiple instances e.g. multiple expiry tasks | ||||
|   | ||||
| @@ -24,7 +24,7 @@ import com.plotsquared.core.plot.Plot; | ||||
| public class PlayerEnterPlotEvent extends PlotPlayerEvent { | ||||
|  | ||||
|     /** | ||||
|      * PlayerEnterPlotEvent: Called when a player enters a plot | ||||
|      * Called when a player leaves a plot. | ||||
|      * | ||||
|      * @param player Player that entered the plot | ||||
|      * @param plot   Plot that was entered | ||||
|   | ||||
| @@ -643,22 +643,35 @@ public class Plot { | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets an immutable set of owner UUIDs for a plot (supports multi-owner mega-plots). | ||||
|      * Gets a immutable set of owner UUIDs for a plot (supports multi-owner mega-plots). | ||||
|      * <p> | ||||
|      * This method cannot be used to add or remove owners from a plot. | ||||
|      * </p> | ||||
|      * | ||||
|      * @return Immutable set of plot owners | ||||
|      * @return Immutable view of plot owners | ||||
|      */ | ||||
|     public @NonNull Set<UUID> getOwners() { | ||||
|         ImmutableSet.Builder<UUID> owners = ImmutableSet.builder(); | ||||
|         for (Plot plot : getConnectedPlots()) { | ||||
|             UUID owner = plot.getOwner(); | ||||
|             if (owner != null) { | ||||
|                 owners.add(owner); | ||||
|             } | ||||
|         if (this.getOwner() == null) { | ||||
|             return ImmutableSet.of(); | ||||
|         } | ||||
|         return owners.build(); | ||||
|         if (isMerged()) { | ||||
|             Set<Plot> plots = getConnectedPlots(); | ||||
|             Plot[] array = plots.toArray(new Plot[0]); | ||||
|             ImmutableSet.Builder<UUID> owners = ImmutableSet.builder(); | ||||
|             UUID last = this.getOwner(); | ||||
|             owners.add(this.getOwner()); | ||||
|             for (final Plot current : array) { | ||||
|                 if (current.getOwner() == null) { | ||||
|                     continue; | ||||
|                 } | ||||
|                 if (last == null || current.getOwner().getMostSignificantBits() != last.getMostSignificantBits()) { | ||||
|                     owners.add(current.getOwner()); | ||||
|                     last = current.getOwner(); | ||||
|                 } | ||||
|             } | ||||
|             return owners.build(); | ||||
|         } | ||||
|         return ImmutableSet.of(this.getOwner()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|   | ||||
| @@ -679,8 +679,10 @@ public abstract class PlotArea implements ComponentLike { | ||||
|                     TranslatableCaption.of("height.height_limit"), | ||||
|                     TagResolver.builder() | ||||
|                             .tag("minheight", Tag.inserting(Component.text(minBuildHeight))) | ||||
|                             .tag("maxheight", Tag.inserting(Component.text(maxBuildHeight))) | ||||
|                             .build() | ||||
|                             .tag( | ||||
|                                     "maxheight", | ||||
|                                     Tag.inserting(Component.text(maxBuildHeight)) | ||||
|                             ).build() | ||||
|             ); | ||||
|             // Return true if "failed" as the method will always be inverted otherwise | ||||
|             return true; | ||||
|   | ||||
| @@ -31,8 +31,6 @@ import com.sk89q.worldedit.world.registry.LegacyMapper; | ||||
| import org.checkerframework.checker.nullness.qual.NonNull; | ||||
| import org.checkerframework.checker.nullness.qual.Nullable; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| /** | ||||
|  * {@link BlockState} related utility methods | ||||
|  */ | ||||
| @@ -45,8 +43,7 @@ public final class BlockUtil { | ||||
|         PARSER_CONTEXT.setRestricted(false); | ||||
|         PARSER_CONTEXT.setPreferringWildcard(false); | ||||
|         PARSER_CONTEXT.setTryLegacy(true); | ||||
|         List<InputParser<BaseBlock>> parsers = WorldEdit.getInstance().getBlockFactory().getParsers(); | ||||
|         PARSER = parsers.get(parsers.size() - 1); // Default parser is always at the end | ||||
|         PARSER = WorldEdit.getInstance().getBlockFactory().getParsers().get(0); | ||||
|     } | ||||
|  | ||||
|     private BlockUtil() { | ||||
|   | ||||
| @@ -22,7 +22,7 @@ plugins { | ||||
| } | ||||
|  | ||||
| group = "com.intellectualsites.plotsquared" | ||||
| version = "7.3.10-SNAPSHOT" | ||||
| version = "7.3.9-SNAPSHOT" | ||||
|  | ||||
| if (!File("$rootDir/.git").exists()) { | ||||
|     logger.lifecycle(""" | ||||
| @@ -79,8 +79,8 @@ subprojects { | ||||
|  | ||||
|     dependencies { | ||||
|         // Tests | ||||
|         testImplementation("org.junit.jupiter:junit-jupiter:5.10.3") | ||||
|         testRuntimeOnly("org.junit.platform:junit-platform-launcher:1.10.3") | ||||
|         testImplementation("org.junit.jupiter:junit-jupiter:5.10.2") | ||||
|         testRuntimeOnly("org.junit.platform:junit-platform-launcher:1.10.2") | ||||
|     } | ||||
|  | ||||
|     plugins.withId("java") { | ||||
| @@ -94,7 +94,7 @@ subprojects { | ||||
|     } | ||||
|  | ||||
|     configurations.all { | ||||
|         attributes.attribute(TargetJvmVersion.TARGET_JVM_VERSION_ATTRIBUTE, 21) | ||||
|         attributes.attribute(TargetJvmVersion.TARGET_JVM_VERSION_ATTRIBUTE, 17) | ||||
|     } | ||||
|  | ||||
|     spotless { | ||||
| @@ -230,7 +230,7 @@ tasks.getByName<Jar>("jar") { | ||||
|     enabled = false | ||||
| } | ||||
|  | ||||
| val supportedVersions = listOf("1.18.2", "1.19.4", "1.20.6", "1.21.1") | ||||
| val supportedVersions = listOf("1.18.2", "1.19.4", "1.20.1", "1.20.4") | ||||
| tasks { | ||||
|     register("cacheLatestFaweArtifact") { | ||||
|         val lastSuccessfulBuildUrl = uri("https://ci.athion.net/job/FastAsyncWorldEdit/lastSuccessfulBuild/api/json").toURL() | ||||
|   | ||||
| @@ -1,20 +1,20 @@ | ||||
| [versions] | ||||
| # Platform expectations | ||||
| paper = "1.20.4-R0.1-SNAPSHOT" | ||||
| paper = "1.20.2-R0.1-SNAPSHOT" | ||||
| guice = "7.0.0" | ||||
| spotbugs = "4.8.6" | ||||
| checkerqual = "3.46.0" | ||||
| spotbugs = "4.8.4" | ||||
| checkerqual = "3.43.0" | ||||
| gson = "2.10" | ||||
| guava = "31.1-jre" | ||||
| snakeyaml = "2.0" | ||||
| adventure = "4.17.0" | ||||
| adventure-bukkit = "4.3.4" | ||||
| adventure = "4.16.0" | ||||
| adventure-bukkit = "4.3.2" | ||||
| log4j = "2.19.0" | ||||
|  | ||||
| # Plugins | ||||
| worldedit = "7.2.20" | ||||
| fawe = "2.11.1" | ||||
| placeholderapi = "2.11.6" | ||||
| fawe = "2.9.2" | ||||
| placeholderapi = "2.11.5" | ||||
| luckperms = "5.4" | ||||
| essentialsx = "2.20.1" | ||||
| mvdwapi = "3.1.1" | ||||
| @@ -71,6 +71,7 @@ mvdwapi = { group = "com.intellectualsites.mvdwplaceholderapi", name = "MVdWPlac | ||||
| squirrelid = { group = "org.enginehub", name = "squirrelid", version.ref = "squirrelid" } | ||||
| arkitektonika = { group = "com.intellectualsites.arkitektonika", name = "Arkitektonika-Client", version.ref = "arkitektonika" } | ||||
| paster = { group = "com.intellectualsites.paster", name = "Paster", version.ref = "paster" } | ||||
| bstatsBase = { group = "org.bstats", name = "bstats-base", version.ref = "bstats" } | ||||
| bstatsBukkit = { group = "org.bstats", name = "bstats-bukkit", version.ref = "bstats" } | ||||
| informativeAnnotations = { group = "com.intellectualsites.informative-annotations", name = "informative-annotations", version.ref = "informative-annotations" } | ||||
| paperlib = { group = "io.papermc", name = "paperlib", version.ref = "paperlib" } | ||||
|   | ||||
							
								
								
									
										
											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 | ||||
| distributionPath=wrapper/dists | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.9-bin.zip | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.7-bin.zip | ||||
| networkTimeout=10000 | ||||
| validateDistributionUrl=true | ||||
| zipStoreBase=GRADLE_USER_HOME | ||||
|   | ||||
							
								
								
									
										7
									
								
								gradlew
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								gradlew
									
									
									
									
										vendored
									
									
								
							| @@ -15,8 +15,6 @@ | ||||
| # See the License for the specific language governing permissions and | ||||
| # limitations under the License. | ||||
| # | ||||
| # SPDX-License-Identifier: Apache-2.0 | ||||
| # | ||||
|  | ||||
| ############################################################################## | ||||
| # | ||||
| @@ -57,7 +55,7 @@ | ||||
| #       Darwin, MinGW, and NonStop. | ||||
| # | ||||
| #   (3) This script is generated from the Groovy template | ||||
| #       https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt | ||||
| #       https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt | ||||
| #       within the Gradle project. | ||||
| # | ||||
| #       You can find Gradle at https://github.com/gradle/gradle/. | ||||
| @@ -86,8 +84,7 @@ done | ||||
| # shellcheck disable=SC2034 | ||||
| APP_BASE_NAME=${0##*/} | ||||
| # Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036) | ||||
| APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s | ||||
| ' "$PWD" ) || exit | ||||
| APP_HOME=$( cd "${APP_HOME:-./}" > /dev/null && pwd -P ) || exit | ||||
|  | ||||
| # Use the maximum available, or set MAX_FD != -1 to use that value. | ||||
| MAX_FD=maximum | ||||
|   | ||||
							
								
								
									
										2
									
								
								gradlew.bat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								gradlew.bat
									
									
									
									
										vendored
									
									
								
							| @@ -13,8 +13,6 @@ | ||||
| @rem See the License for the specific language governing permissions and | ||||
| @rem limitations under the License. | ||||
| @rem | ||||
| @rem SPDX-License-Identifier: Apache-2.0 | ||||
| @rem | ||||
|  | ||||
| @if "%DEBUG%"=="" @echo off | ||||
| @rem ########################################################################## | ||||
|   | ||||
		Reference in New Issue
	
	Block a user