mirror of https://github.com/GeyserMC/Geyser.git
291 lines
13 KiB
Java
291 lines
13 KiB
Java
/*
|
|
* Copyright (c) 2019-2021 GeyserMC. http://geysermc.org
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
*
|
|
* @author GeyserMC
|
|
* @link https://github.com/GeyserMC/Geyser
|
|
*/
|
|
|
|
package org.geysermc.connector.network.translators.java.entity;
|
|
|
|
import com.github.steveice10.mc.protocol.packet.ingame.server.entity.ServerEntitySetPassengersPacket;
|
|
import com.nukkitx.math.vector.Vector3f;
|
|
import com.nukkitx.protocol.bedrock.data.entity.EntityData;
|
|
import com.nukkitx.protocol.bedrock.data.entity.EntityFlag;
|
|
import com.nukkitx.protocol.bedrock.data.entity.EntityLinkData;
|
|
import com.nukkitx.protocol.bedrock.packet.SetEntityLinkPacket;
|
|
import com.nukkitx.protocol.bedrock.v428.Bedrock_v428;
|
|
import it.unimi.dsi.fastutil.longs.LongOpenHashSet;
|
|
import org.geysermc.connector.entity.Entity;
|
|
import org.geysermc.connector.entity.living.ArmorStandEntity;
|
|
import org.geysermc.connector.entity.living.animal.AnimalEntity;
|
|
import org.geysermc.connector.entity.type.EntityType;
|
|
import org.geysermc.connector.network.session.GeyserSession;
|
|
import org.geysermc.connector.network.translators.PacketTranslator;
|
|
import org.geysermc.connector.network.translators.Translator;
|
|
|
|
import java.util.Arrays;
|
|
|
|
@Translator(packet = ServerEntitySetPassengersPacket.class)
|
|
public class JavaEntitySetPassengersTranslator extends PacketTranslator<ServerEntitySetPassengersPacket> {
|
|
|
|
@Override
|
|
public void translate(ServerEntitySetPassengersPacket packet, GeyserSession session) {
|
|
Entity entity;
|
|
if (packet.getEntityId() == session.getPlayerEntity().getEntityId()) {
|
|
entity = session.getPlayerEntity();
|
|
} else {
|
|
entity = session.getEntityCache().getEntityByJavaId(packet.getEntityId());
|
|
}
|
|
|
|
if (entity == null) return;
|
|
|
|
LongOpenHashSet passengers = entity.getPassengers().clone();
|
|
boolean rider = true;
|
|
for (long passengerId : packet.getPassengerIds()) {
|
|
Entity passenger = session.getEntityCache().getEntityByJavaId(passengerId);
|
|
if (passengerId == session.getPlayerEntity().getEntityId()) {
|
|
passenger = session.getPlayerEntity();
|
|
session.setRidingVehicleEntity(entity);
|
|
// We need to confirm teleports before entering a vehicle, or else we will likely exit right out
|
|
if (session.getConnector().getConfig().isCacheChunks()) {
|
|
session.confirmTeleport(passenger.getPosition().sub(0, EntityType.PLAYER.getOffset(), 0).toDouble());
|
|
}
|
|
}
|
|
// Passenger hasn't loaded in (likely since we're waiting for a skin response)
|
|
// and entity link needs to be set later
|
|
if (passenger == null && passengerId != 0) {
|
|
session.getEntityCache().addCachedPlayerEntityLink(passengerId, packet.getEntityId());
|
|
}
|
|
if (passenger == null) {
|
|
continue;
|
|
}
|
|
|
|
EntityLinkData.Type type = rider ? EntityLinkData.Type.RIDER : EntityLinkData.Type.PASSENGER;
|
|
SetEntityLinkPacket linkPacket = new SetEntityLinkPacket();
|
|
linkPacket.setEntityLink(new EntityLinkData(entity.getGeyserId(), passenger.getGeyserId(), type, false));
|
|
session.sendUpstreamPacket(linkPacket);
|
|
passengers.add(passengerId);
|
|
|
|
// Head rotation on boats
|
|
if (entity.getEntityType() == EntityType.BOAT) {
|
|
passenger.getMetadata().put(EntityData.RIDER_ROTATION_LOCKED, (byte) 1);
|
|
passenger.getMetadata().put(EntityData.RIDER_MAX_ROTATION, 90f);
|
|
// Can be removed once 1.16.200 to 1.16.201 support is dropped
|
|
if (session.getUpstream().getSession().getPacketCodec().getProtocolVersion() >= Bedrock_v428.V428_CODEC.getProtocolVersion()) {
|
|
passenger.getMetadata().put(EntityData.RIDER_MIN_ROTATION, 1f);
|
|
passenger.getMetadata().put(EntityData.RIDER_ROTATION_OFFSET, -90f);
|
|
} else {
|
|
passenger.getMetadata().put(EntityData.RIDER_MIN_ROTATION, -90f);
|
|
}
|
|
} else {
|
|
passenger.getMetadata().put(EntityData.RIDER_ROTATION_LOCKED, (byte) 0);
|
|
passenger.getMetadata().put(EntityData.RIDER_MAX_ROTATION, 0f);
|
|
passenger.getMetadata().put(EntityData.RIDER_MIN_ROTATION, 0f);
|
|
}
|
|
|
|
passenger.updateBedrockMetadata(session);
|
|
rider = false;
|
|
}
|
|
|
|
entity.setPassengers(passengers);
|
|
|
|
for (long passengerId : entity.getPassengers()) {
|
|
Entity passenger = session.getEntityCache().getEntityByJavaId(passengerId);
|
|
if (passengerId == session.getPlayerEntity().getEntityId()) {
|
|
passenger = session.getPlayerEntity();
|
|
}
|
|
if (passenger == null) {
|
|
continue;
|
|
}
|
|
if (Arrays.stream(packet.getPassengerIds()).noneMatch(id -> id == passengerId)) {
|
|
SetEntityLinkPacket linkPacket = new SetEntityLinkPacket();
|
|
linkPacket.setEntityLink(new EntityLinkData(entity.getGeyserId(), passenger.getGeyserId(), EntityLinkData.Type.REMOVE, false));
|
|
session.sendUpstreamPacket(linkPacket);
|
|
passengers.remove(passenger.getEntityId());
|
|
passenger.getMetadata().put(EntityData.RIDER_ROTATION_LOCKED, (byte) 0);
|
|
passenger.getMetadata().put(EntityData.RIDER_MAX_ROTATION, 0f);
|
|
passenger.getMetadata().put(EntityData.RIDER_MIN_ROTATION, 0f);
|
|
if (session.getUpstream().getSession().getPacketCodec().getProtocolVersion() >= Bedrock_v428.V428_CODEC.getProtocolVersion()) {
|
|
passenger.getMetadata().put(EntityData.RIDER_ROTATION_OFFSET, 0f);
|
|
}
|
|
|
|
this.updateOffset(passenger, entity, session, false, false, (packet.getPassengerIds().length > 1));
|
|
} else {
|
|
this.updateOffset(passenger, entity, session, (packet.getPassengerIds()[0] == passengerId), true, (packet.getPassengerIds().length > 1));
|
|
}
|
|
|
|
// Force an update to the passenger metadata
|
|
passenger.updateBedrockMetadata(session);
|
|
}
|
|
|
|
switch (entity.getEntityType()) {
|
|
case HORSE:
|
|
case SKELETON_HORSE:
|
|
case DONKEY:
|
|
case MULE:
|
|
case RAVAGER:
|
|
entity.getMetadata().put(EntityData.RIDER_MAX_ROTATION, 181.0f);
|
|
entity.updateBedrockMetadata(session);
|
|
break;
|
|
}
|
|
}
|
|
|
|
private float getMountedHeightOffset(Entity mount) {
|
|
float height = mount.getMetadata().getFloat(EntityData.BOUNDING_BOX_HEIGHT);
|
|
float mountedHeightOffset = height * 0.75f;
|
|
switch (mount.getEntityType()) {
|
|
case CHICKEN:
|
|
case SPIDER:
|
|
mountedHeightOffset = height * 0.5f;
|
|
break;
|
|
case DONKEY:
|
|
case MULE:
|
|
mountedHeightOffset -= 0.25f;
|
|
break;
|
|
case LLAMA:
|
|
mountedHeightOffset = height * 0.67f;
|
|
break;
|
|
case MINECART:
|
|
case MINECART_HOPPER:
|
|
case MINECART_TNT:
|
|
case MINECART_CHEST:
|
|
case MINECART_FURNACE:
|
|
case MINECART_SPAWNER:
|
|
case MINECART_COMMAND_BLOCK:
|
|
mountedHeightOffset = 0;
|
|
break;
|
|
case BOAT:
|
|
mountedHeightOffset = -0.1f;
|
|
break;
|
|
case HOGLIN:
|
|
case ZOGLIN:
|
|
boolean isBaby = mount.getMetadata().getFlags().getFlag(EntityFlag.BABY);
|
|
mountedHeightOffset = height - (isBaby ? 0.2f : 0.15f);
|
|
break;
|
|
case PIGLIN:
|
|
mountedHeightOffset = height * 0.92f;
|
|
break;
|
|
case RAVAGER:
|
|
mountedHeightOffset = 2.1f;
|
|
break;
|
|
case SKELETON_HORSE:
|
|
mountedHeightOffset -= 0.1875f;
|
|
break;
|
|
case STRIDER:
|
|
mountedHeightOffset = height - 0.19f;
|
|
break;
|
|
}
|
|
return mountedHeightOffset;
|
|
}
|
|
|
|
private float getHeightOffset(Entity passenger) {
|
|
boolean isBaby;
|
|
switch (passenger.getEntityType()) {
|
|
case SKELETON:
|
|
case STRAY:
|
|
case WITHER_SKELETON:
|
|
return -0.6f;
|
|
case ARMOR_STAND:
|
|
if (((ArmorStandEntity) passenger).isMarker()) {
|
|
return 0.0f;
|
|
} else {
|
|
return 0.1f;
|
|
}
|
|
case ENDERMITE:
|
|
case SILVERFISH:
|
|
return 0.1f;
|
|
case PIGLIN:
|
|
case PIGLIN_BRUTE:
|
|
case ZOMBIFIED_PIGLIN:
|
|
isBaby = passenger.getMetadata().getFlags().getFlag(EntityFlag.BABY);
|
|
return isBaby ? -0.05f : -0.45f;
|
|
case ZOMBIE:
|
|
isBaby = passenger.getMetadata().getFlags().getFlag(EntityFlag.BABY);
|
|
return isBaby ? 0.0f : -0.45f;
|
|
case EVOKER:
|
|
case ILLUSIONER:
|
|
case PILLAGER:
|
|
case RAVAGER:
|
|
case VINDICATOR:
|
|
case WITCH:
|
|
return -0.45f;
|
|
case PLAYER:
|
|
return -0.35f;
|
|
}
|
|
if (passenger instanceof AnimalEntity) {
|
|
return 0.14f;
|
|
}
|
|
return 0f;
|
|
}
|
|
|
|
private void updateOffset(Entity passenger, Entity mount, GeyserSession session, boolean rider, boolean riding, boolean moreThanOneEntity) {
|
|
passenger.getMetadata().getFlags().setFlag(EntityFlag.RIDING, riding);
|
|
if (riding) {
|
|
// Without the Y offset, Bedrock players will find themselves in the floor when mounting
|
|
float mountedHeightOffset = getMountedHeightOffset(mount);
|
|
float heightOffset = getHeightOffset(passenger);
|
|
|
|
float xOffset = 0;
|
|
float yOffset = mountedHeightOffset + heightOffset;
|
|
float zOffset = 0;
|
|
switch (mount.getEntityType()) {
|
|
case BOAT:
|
|
// Without the X offset, more than one entity on a boat is stacked on top of each other
|
|
if (rider && moreThanOneEntity) {
|
|
xOffset = 0.2f;
|
|
} else if (moreThanOneEntity) {
|
|
xOffset = -0.6f;
|
|
}
|
|
break;
|
|
case CHICKEN:
|
|
zOffset = -0.1f;
|
|
break;
|
|
case LLAMA:
|
|
zOffset = -0.3f;
|
|
break;
|
|
}
|
|
/*
|
|
* Bedrock Differences
|
|
* Zoglin & Hoglin seem to be taller in Bedrock edition
|
|
* Horses are tinier
|
|
* Players, Minecarts, and Boats have different origins
|
|
*/
|
|
if (passenger.getEntityType() == EntityType.PLAYER && mount.getEntityType() != EntityType.PLAYER) {
|
|
yOffset += EntityType.PLAYER.getOffset();
|
|
}
|
|
switch (mount.getEntityType()) {
|
|
case MINECART:
|
|
case MINECART_HOPPER:
|
|
case MINECART_TNT:
|
|
case MINECART_CHEST:
|
|
case MINECART_FURNACE:
|
|
case MINECART_SPAWNER:
|
|
case MINECART_COMMAND_BLOCK:
|
|
case BOAT:
|
|
yOffset -= mount.getEntityType().getHeight() * 0.5f;
|
|
}
|
|
Vector3f offset = Vector3f.from(xOffset, yOffset, zOffset);
|
|
passenger.getMetadata().put(EntityData.RIDER_SEAT_POSITION, offset);
|
|
}
|
|
passenger.updateBedrockMetadata(session);
|
|
}
|
|
}
|