ViaVersion/bukkit/src/main/java/com/viaversion/viaversion/bukkit/listeners/multiversion/PlayerSneakListener.java

155 lines
6.3 KiB
Java
Raw Normal View History

/*
* This file is part of ViaVersion - https://github.com/ViaVersion/ViaVersion
2024-01-01 12:39:45 +01:00
* Copyright (C) 2016-2024 ViaVersion 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 <http://www.gnu.org/licenses/>.
*/
package com.viaversion.viaversion.bukkit.listeners.multiversion;
2019-04-22 14:38:22 +02:00
import com.viaversion.viaversion.ViaVersionPlugin;
import com.viaversion.viaversion.api.Via;
import com.viaversion.viaversion.api.connection.ProtocolInfo;
import com.viaversion.viaversion.api.connection.UserConnection;
import com.viaversion.viaversion.api.protocol.version.ProtocolVersion;
import com.viaversion.viaversion.bukkit.listeners.ViaBukkitListener;
2019-04-22 14:38:22 +02:00
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import java.util.UUID;
import java.util.WeakHashMap;
import java.util.logging.Level;
import org.bukkit.entity.EntityType;
import org.bukkit.entity.Player;
import org.bukkit.event.EventHandler;
import org.bukkit.event.entity.EntityDamageEvent;
import org.bukkit.event.player.PlayerQuitEvent;
import org.bukkit.event.player.PlayerToggleSneakEvent;
2019-04-22 14:38:22 +02:00
public class PlayerSneakListener extends ViaBukkitListener {
2019-04-22 16:17:34 +02:00
private static final float STANDING_HEIGHT = 1.8F;
private static final float HEIGHT_1_14 = 1.5F;
private static final float HEIGHT_1_9 = 1.6F;
private static final float DEFAULT_WIDTH = 0.6F;
private final boolean is1_9Fix;
private final boolean is1_14Fix;
2019-04-22 15:55:38 +02:00
private Map<Player, Boolean> sneaking; // true = 1.14+, else false
private Set<UUID> sneakingUuids;
private final Method getHandle;
2019-04-22 14:38:22 +02:00
private Method setSize;
2019-04-22 15:55:38 +02:00
private boolean useCache;
2019-04-22 14:38:22 +02:00
2019-09-19 11:22:06 +02:00
public PlayerSneakListener(ViaVersionPlugin plugin, boolean is1_9Fix, boolean is1_14Fix) throws ReflectiveOperationException {
2019-04-22 14:38:22 +02:00
super(plugin, null);
this.is1_9Fix = is1_9Fix;
this.is1_14Fix = is1_14Fix;
2019-09-19 11:22:06 +02:00
final String packageName = plugin.getServer().getClass().getPackage().getName();
getHandle = Class.forName(packageName + ".entity.CraftPlayer").getMethod("getHandle");
final Class<?> entityPlayerClass = Class.forName(packageName
.replace("org.bukkit.craftbukkit", "net.minecraft.server") + ".EntityPlayer");
2019-04-22 14:38:22 +02:00
try {
2019-09-19 11:22:06 +02:00
setSize = entityPlayerClass.getMethod("setSize", Float.TYPE, Float.TYPE);
} catch (NoSuchMethodException e) {
// Don't catch this one
setSize = entityPlayerClass.getMethod("a", Float.TYPE, Float.TYPE);
2019-04-22 14:38:22 +02:00
}
2019-09-19 11:22:06 +02:00
2019-04-22 15:55:38 +02:00
// From 1.9 upwards the server hitbox is set in every entity tick, so we have to reset it everytime
2021-03-26 12:51:38 +01:00
if (Via.getAPI().getServerVersion().lowestSupportedVersion() >= ProtocolVersion.v1_9.getVersion()) {
2019-05-21 19:02:03 +02:00
sneaking = new WeakHashMap<>();
2019-04-22 15:55:38 +02:00
useCache = true;
2023-10-11 15:14:41 +02:00
plugin.getServer().getScheduler().runTaskTimer(plugin, () -> {
for (Map.Entry<Player, Boolean> entry : sneaking.entrySet()) {
setHeight(entry.getKey(), entry.getValue() ? HEIGHT_1_14 : HEIGHT_1_9);
2019-04-22 14:38:22 +02:00
}
2019-04-22 15:55:38 +02:00
}, 1, 1);
}
// Suffocation removal only required for 1.14+ clients.
if (is1_14Fix) {
sneakingUuids = new HashSet<>();
}
2019-04-22 14:38:22 +02:00
}
@EventHandler(ignoreCancelled = true)
2019-04-22 15:55:38 +02:00
public void playerToggleSneak(PlayerToggleSneakEvent event) {
Player player = event.getPlayer();
2019-04-23 23:23:37 +02:00
UserConnection userConnection = getUserConnection(player);
if (userConnection == null) return;
2020-06-07 12:19:36 +02:00
ProtocolInfo info = userConnection.getProtocolInfo();
2019-04-27 18:36:01 +02:00
if (info == null) return;
2019-04-23 23:23:37 +02:00
2019-04-27 18:36:01 +02:00
int protocolVersion = info.getProtocolVersion();
2020-10-16 18:21:45 +02:00
if (is1_14Fix && protocolVersion >= ProtocolVersion.v1_14.getVersion()) {
2019-04-22 16:17:34 +02:00
setHeight(player, event.isSneaking() ? HEIGHT_1_14 : STANDING_HEIGHT);
if (event.isSneaking())
sneakingUuids.add(player.getUniqueId());
else
sneakingUuids.remove(player.getUniqueId());
2019-04-22 15:55:38 +02:00
if (!useCache) return;
if (event.isSneaking())
sneaking.put(player, true);
else
sneaking.remove(player);
2020-10-16 18:21:45 +02:00
} else if (is1_9Fix && protocolVersion >= ProtocolVersion.v1_9.getVersion()) {
2019-04-22 16:17:34 +02:00
setHeight(player, event.isSneaking() ? HEIGHT_1_9 : STANDING_HEIGHT);
2019-04-22 15:55:38 +02:00
if (!useCache) return;
if (event.isSneaking())
sneaking.put(player, false);
else
sneaking.remove(player);
2019-04-22 14:38:22 +02:00
}
}
@EventHandler(ignoreCancelled = true)
public void playerDamage(EntityDamageEvent event) {
if (!is1_14Fix) return;
if (event.getCause() != EntityDamageEvent.DamageCause.SUFFOCATION) return;
if (event.getEntityType() != EntityType.PLAYER) return;
Player player = (Player) event.getEntity();
if (!sneakingUuids.contains(player.getUniqueId())) return;
2019-05-20 10:30:42 +02:00
// Don't cancel when they should actually be suffocating; Essentially cancel when the head is in the top block only ever so slightly
// ~0.041 should suffice, but gotta stay be safe
double y = player.getEyeLocation().getY() + 0.045;
y -= (int) y;
if (y < 0.09) {
event.setCancelled(true);
}
}
@EventHandler
public void playerQuit(PlayerQuitEvent event) {
if (sneaking != null)
sneaking.remove(event.getPlayer());
if (sneakingUuids != null)
sneakingUuids.remove(event.getPlayer().getUniqueId());
}
2019-04-22 16:17:34 +02:00
private void setHeight(Player player, float height) {
2019-04-22 14:38:22 +02:00
try {
2019-04-22 16:17:34 +02:00
setSize.invoke(getHandle.invoke(player), DEFAULT_WIDTH, height);
2019-04-22 14:38:22 +02:00
} catch (IllegalAccessException | InvocationTargetException e) {
Via.getPlatform().getLogger().log(Level.SEVERE, "Failed to set player height", e);
2019-04-22 14:38:22 +02:00
}
}
}