use of net.minecraft.server.level.ServerLevel in project SpongeCommon by SpongePowered.
the class ServerGamePacketListenerImplMixin method impl$usePlayerDimensionForRespawn.
@Redirect(method = "handleClientCommand", at = @At(value = "INVOKE", target = "Lnet/minecraft/server/players/PlayerList;respawn(Lnet/minecraft/server/level/ServerPlayer;Z)Lnet/minecraft/server/level/ServerPlayer;"))
private net.minecraft.server.level.ServerPlayer impl$usePlayerDimensionForRespawn(final PlayerList playerList, final net.minecraft.server.level.ServerPlayer player, final boolean keepAllPlayerData) {
// A few changes to Vanilla logic here that, by default, still preserve game mechanics:
// - If we have conquered The End then keep the dimension type we're headed to (which is Overworld as of 1.15)
// - Otherwise, check the platform hooks for which dimension to respawn to. In Sponge, this is the Player's dimension they
// are already in if we can respawn there which is only true for Overworld dimensions
final ResourceKey<Level> respawnDimension = player.getRespawnDimension();
@Nullable final ServerLevel destinationWorld = this.server.getLevel(respawnDimension);
final ServerLevel overworld = this.server.getLevel(Level.OVERWORLD);
if (overworld == null) {
throw new IllegalStateException("Somehow the Overworld is not retrievable while trying to respawn player " + player.getGameProfile().getName());
}
final ServerLevel destination = destinationWorld == null ? overworld : destinationWorld;
final RespawnPlayerEvent.SelectWorld event = SpongeEventFactory.createRespawnPlayerEventSelectWorld(PhaseTracker.getCauseStackManager().currentCause(), (org.spongepowered.api.world.server.ServerWorld) destination, (org.spongepowered.api.world.server.ServerWorld) player.getLevel(), (org.spongepowered.api.world.server.ServerWorld) overworld, (ServerPlayer) player);
SpongeCommon.post(event);
((PlayerListBridge) this.server.getPlayerList()).bridge$setOriginalDestinationDimension(((ServerLevel) event.originalDestinationWorld()).dimension());
((PlayerListBridge) this.server.getPlayerList()).bridge$setNewDestinationDimension(((ServerLevel) event.destinationWorld()).dimension());
// The key is reset to null in the overwrite
return playerList.respawn(player, keepAllPlayerData);
}
use of net.minecraft.server.level.ServerLevel in project SpongeCommon by SpongePowered.
the class PlayerListMixin method impl$onInitPlayer_getWorld.
@Redirect(method = "placeNewPlayer", at = @At(value = "INVOKE", target = "Lnet/minecraft/server/MinecraftServer;getLevel(Lnet/minecraft/resources/ResourceKey;)Lnet/minecraft/server/level/ServerLevel;"))
private net.minecraft.server.level.ServerLevel impl$onInitPlayer_getWorld(final MinecraftServer minecraftServer, final ResourceKey<Level> dimension, final Connection networkManager, final net.minecraft.server.level.ServerPlayer mcPlayer) {
@Nullable final net.minecraft.network.chat.Component kickReason = ((ConnectionBridge) networkManager).bridge$getKickReason();
final Component disconnectMessage;
if (kickReason != null) {
disconnectMessage = SpongeAdventure.asAdventure(kickReason);
} else {
disconnectMessage = Component.text("You are not allowed to log in to this server.");
}
net.minecraft.server.level.ServerLevel mcWorld = minecraftServer.getLevel(dimension);
if (mcWorld == null) {
SpongeCommon.logger().warn("The player '{}' was located in a world that isn't loaded or doesn't exist. This is not safe so " + "the player will be moved to the spawn of the default world.", mcPlayer.getGameProfile().getName());
mcWorld = minecraftServer.overworld();
final BlockPos spawnPoint = mcWorld.getSharedSpawnPos();
mcPlayer.setPos(spawnPoint.getX() + 0.5, spawnPoint.getY() + 0.5, spawnPoint.getZ() + 0.5);
}
mcPlayer.setLevel(mcWorld);
final ServerPlayer player = (ServerPlayer) mcPlayer;
final ServerLocation location = player.serverLocation();
final Vector3d rotation = player.rotation();
// player.connection() cannot be used here, because it's still be null at this point
final ServerSideConnection connection = (ServerSideConnection) networkManager.getPacketListener();
// The user is not yet in the player list, so we need to make special provision.
final User user = SpongeUserView.createLoginEventUser(player);
final Cause cause = Cause.of(EventContext.empty(), connection, user);
final ServerSideConnectionEvent.Login event = SpongeEventFactory.createServerSideConnectionEventLogin(cause, disconnectMessage, disconnectMessage, location, location, rotation, rotation, connection, user);
if (kickReason != null) {
event.setCancelled(true);
}
if (SpongeCommon.post(event)) {
this.impl$disconnectClient(networkManager, event.message(), player.profile());
return null;
}
final ServerLocation toLocation = event.toLocation();
final Vector3d toRotation = event.toRotation();
mcPlayer.absMoveTo(toLocation.x(), toLocation.y(), toLocation.z(), (float) toRotation.y(), (float) toRotation.x());
return (net.minecraft.server.level.ServerLevel) toLocation.world();
}
use of net.minecraft.server.level.ServerLevel in project SpongeCommon by SpongePowered.
the class EntityMixin method bridge$changeDimension.
/**
* This is effectively an overwrite of changeDimension: required due to
* Forge changing the signature.
*
* @author dualspiral - 18th December 2020 - 1.16.4
* @author dualspiral - 8th August 2021 - 1.16.5 (adjusted for SpongeForge)
*
* @param originalDestinationWorld The original target world
* @param originalPortalLogic performs additional teleportation logic, as required.
* @return The {@link Entity} that is either this one, or replaces this one
*/
@SuppressWarnings("ConstantConditions")
@org.checkerframework.checker.nullness.qual.Nullable
public Entity bridge$changeDimension(final net.minecraft.server.level.ServerLevel originalDestinationWorld, final PortalLogic originalPortalLogic) {
// Sponge Start
if (this.shadow$getCommandSenderWorld().isClientSide || this.removed) {
return null;
}
final boolean isPlayer = ((Object) this) instanceof ServerPlayer;
final TeleportContext contextToSwitchTo = EntityPhase.State.PORTAL_DIMENSION_CHANGE.createPhaseContext(PhaseTracker.getInstance()).worldChange();
if (isPlayer) {
contextToSwitchTo.player();
}
try (final TeleportContext context = contextToSwitchTo.buildAndSwitch();
final CauseStackManager.StackFrame frame = PhaseTracker.getCauseStackManager().pushCauseFrame()) {
frame.pushCause(this);
frame.pushCause(originalPortalLogic.getPortalType());
frame.addContext(EventContextKeys.MOVEMENT_TYPE, originalPortalLogic.getMovementType());
this.impl$originalDestinationWorld = new WeakReference<>((ServerWorld) originalDestinationWorld);
final ChangeEntityWorldEvent.Pre preChangeEvent = PlatformHooks.INSTANCE.getEventHooks().callChangeEntityWorldEventPre((Entity) (Object) this, originalDestinationWorld);
if (preChangeEvent.isCancelled()) {
this.impl$onPreWorldChangeCanceled();
return null;
}
this.impl$customPortal = preChangeEvent.originalDestinationWorld() != preChangeEvent.destinationWorld();
final PortalLogic finalPortalLogic;
if (this.impl$customPortal && originalPortalLogic == originalDestinationWorld.getPortalForcer()) {
finalPortalLogic = (PortalLogic) ((ServerLevel) preChangeEvent.destinationWorld()).getPortalForcer();
} else {
finalPortalLogic = originalPortalLogic;
}
final net.minecraft.server.level.ServerLevel targetWorld = (net.minecraft.server.level.ServerLevel) preChangeEvent.destinationWorld();
final Vector3d currentPosition = VecHelper.toVector3d(this.shadow$position());
// If a player, set the fact they are changing dimensions
this.impl$onChangingDimension(targetWorld);
final net.minecraft.server.level.ServerLevel serverworld = (net.minecraft.server.level.ServerLevel) this.level;
final ResourceKey<Level> registrykey = serverworld.dimension();
if (isPlayer && registrykey == Level.END && targetWorld.dimension() == Level.OVERWORLD && finalPortalLogic.isVanilla()) {
// avoids modded dimensions
return this.impl$postProcessChangeDimension(this.impl$performGameWinLogic());
} else {
// Sponge Start: Redirect the find portal call to the teleporter.
// If this is vanilla, this will house our Reposition Event and return an appropriate
// portal info
final PortalInfo portalinfo = originalPortalLogic.getPortalInfo((Entity) (Object) this, targetWorld, // don't make this a method reference, it'll crash vanilla.
x -> this.shadow$findDimensionEntryPoint(x));
// Sponge End
if (portalinfo != null) {
if (portalinfo instanceof SpongePortalInfo) {
frame.addContext(EventContextKeys.PORTAL, ((SpongePortalInfo) portalinfo).portal());
}
// Only start teleporting if we have somewhere to go.
this.impl$prepareForPortalTeleport(serverworld, targetWorld);
try {
// Sponge Start: wrap the teleportation logic within a function to allow for modification
// of the teleporter
final Vector3d originalDestination = new Vector3d(portalinfo.pos.x, portalinfo.pos.y, portalinfo.pos.z);
// Note that impl$portalRepositioning is the lambda. As this will be different in ServerPlayer,
// we transfer it to a method instead so we can override it.
final Entity transportedEntity = originalPortalLogic.placeEntity((Entity) (Object) this, serverworld, targetWorld, this.yRot, createEndPlatform -> this.impl$portalRepositioning(createEndPlatform, serverworld, targetWorld, portalinfo));
// Make sure the right object was returned
this.impl$validateEntityAfterTeleport(transportedEntity, originalPortalLogic);
// If we need to reposition: well... reposition.
// Downside: portals won't come with us, but with how it's implemented in Forge,
// not sure how we'd do this.
//
// If this is vanilla, we've already fired and dealt with the event
final Cause cause = PhaseTracker.getCauseStackManager().currentCause();
if (transportedEntity != null && this.impl$shouldFireRepositionEvent) {
final Vector3d destination = VecHelper.toVector3d(this.shadow$position());
final ChangeEntityWorldEvent.Reposition reposition = SpongeEventFactory.createChangeEntityWorldEventReposition(cause, (org.spongepowered.api.entity.Entity) transportedEntity, (org.spongepowered.api.world.server.ServerWorld) serverworld, currentPosition, destination, (org.spongepowered.api.world.server.ServerWorld) originalDestinationWorld, originalDestination, (org.spongepowered.api.world.server.ServerWorld) targetWorld);
final Vector3d finalPosition;
if (reposition.isCancelled()) {
// send them back to the original destination
finalPosition = originalDestination;
} else if (reposition.destinationPosition() != destination) {
finalPosition = reposition.destinationPosition();
} else {
finalPosition = null;
}
if (finalPosition != null) {
// TODO: Rollback captures during phase - anything generated needs to vanish here
// Issue chunk ticket of type Portal, even if a portal isn't being created here.
final BlockPos ticketPos = VecHelper.toBlockPos(finalPosition);
targetWorld.getChunkSource().addRegionTicket(TicketType.PORTAL, new ChunkPos(ticketPos), 3, ticketPos);
this.shadow$absMoveTo(finalPosition.x(), finalPosition.y(), finalPosition.z());
}
}
// Used to perform player specific tasks.
this.impl$postPortalForceChangeTasks(transportedEntity, targetWorld, originalPortalLogic.getPortalType() instanceof NetherPortalType);
// Call post event
Sponge.eventManager().post(SpongeEventFactory.createChangeEntityWorldEventPost(cause, (org.spongepowered.api.entity.Entity) this, (ServerWorld) serverworld, (ServerWorld) originalDestinationWorld, (ServerWorld) targetWorld));
} catch (final RuntimeException e) {
// just in case a mod does something less than clever.
if ((Object) this instanceof ServerPlayer) {
this.impl$postPortalForceChangeTasks((Entity) (Object) this, (net.minecraft.server.level.ServerLevel) this.level, false);
}
throw e;
}
// Sponge End
} else {
// Didn't work out.
return null;
}
}
return this.impl$postProcessChangeDimension((Entity) (Object) this);
} finally {
// Reset for the next attempt.
this.impl$shouldFireRepositionEvent = true;
this.impl$originalDestinationWorld = null;
this.impl$customPortal = false;
}
}
use of net.minecraft.server.level.ServerLevel in project SpongeCommon by SpongePowered.
the class EntityMixin method impl$setLocation.
protected boolean impl$setLocation(final boolean isChangeOfWorld, final ServerLevel originalDestinationWorld, final ServerLevel destinationWorld, final Vector3d destinationPosition) {
((Entity) (Object) this).unRide();
if (isChangeOfWorld) {
final net.minecraft.server.level.ServerLevel originalWorld = (net.minecraft.server.level.ServerLevel) this.shadow$getCommandSenderWorld();
((PlatformServerLevelBridge) this.shadow$getCommandSenderWorld()).bridge$removeEntity((Entity) (Object) this, true);
this.bridge$revive();
this.shadow$setLevel(destinationWorld);
destinationWorld.addFromAnotherDimension((Entity) (Object) this);
originalWorld.resetEmptyTime();
destinationWorld.resetEmptyTime();
}
return this.impl$teleportToWithTicket(destinationPosition.x(), destinationPosition.y(), destinationPosition.z(), false);
}
use of net.minecraft.server.level.ServerLevel in project SpongeCommon by SpongePowered.
the class EntityMixin method bridge$setLocation.
@Override
public boolean bridge$setLocation(final ServerLocation location) {
if (this.removed || ((LevelBridge) location.world()).bridge$isFake()) {
return false;
}
try (final CauseStackManager.StackFrame frame = PhaseTracker.getCauseStackManager().pushCauseFrame()) {
frame.addContext(EventContextKeys.MOVEMENT_TYPE, MovementTypes.PLUGIN);
final net.minecraft.server.level.ServerLevel originalWorld = (ServerLevel) this.shadow$getCommandSenderWorld();
final net.minecraft.server.level.ServerLevel originalDestinationWorld = (net.minecraft.server.level.ServerLevel) location.world();
final net.minecraft.server.level.ServerLevel destinationWorld;
@org.checkerframework.checker.nullness.qual.Nullable final Vector3d destinationPosition;
final boolean isChangeOfWorld = this.shadow$getCommandSenderWorld() != originalDestinationWorld;
if (isChangeOfWorld) {
final ChangeEntityWorldEvent.Pre event = PlatformHooks.INSTANCE.getEventHooks().callChangeEntityWorldEventPre((Entity) (Object) this, originalDestinationWorld);
if (event.isCancelled() || ((LevelBridge) event.destinationWorld()).bridge$isFake()) {
return false;
}
destinationWorld = (ServerLevel) event.destinationWorld();
final ChangeEntityWorldEvent.Reposition repositionEvent = this.bridge$fireRepositionEvent(event.originalDestinationWorld(), event.destinationWorld(), location.position());
if (repositionEvent.isCancelled()) {
return false;
}
destinationPosition = repositionEvent.destinationPosition();
} else {
destinationWorld = (ServerLevel) this.level;
destinationPosition = this.impl$fireMoveEvent(PhaseTracker.SERVER, location.position());
if (destinationPosition == null) {
return false;
}
}
final boolean completed = this.impl$setLocation(isChangeOfWorld, originalDestinationWorld, destinationWorld, destinationPosition);
if (isChangeOfWorld) {
Sponge.eventManager().post(SpongeEventFactory.createChangeEntityWorldEventPost(PhaseTracker.getCauseStackManager().currentCause(), (org.spongepowered.api.entity.Entity) this, (ServerWorld) originalWorld, (ServerWorld) originalDestinationWorld, (ServerWorld) destinationWorld));
}
return completed;
}
}
Aggregations