2023-08-03 05:58:22 +02:00
|
|
|
/*
|
|
|
|
* This file is part of ViaVersion - https://github.com/ViaVersion/ViaVersion
|
|
|
|
* Copyright (C) 2023 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.protocols.protocol1_20_2to1_20;
|
|
|
|
|
2023-09-18 16:02:06 +02:00
|
|
|
import com.github.steveice10.opennbt.tag.builtin.CompoundTag;
|
2023-09-19 03:54:03 +02:00
|
|
|
import com.google.gson.JsonElement;
|
2023-09-16 05:27:44 +02:00
|
|
|
import com.viaversion.viaversion.api.Via;
|
2023-09-25 08:40:13 +02:00
|
|
|
import com.viaversion.viaversion.api.connection.ProtocolInfo;
|
2023-08-03 05:58:22 +02:00
|
|
|
import com.viaversion.viaversion.api.connection.UserConnection;
|
2023-08-11 07:47:36 +02:00
|
|
|
import com.viaversion.viaversion.api.data.MappingData;
|
|
|
|
import com.viaversion.viaversion.api.data.MappingDataBase;
|
2023-10-19 01:28:12 +02:00
|
|
|
import com.viaversion.viaversion.api.minecraft.entities.EntityTypes1_19_4;
|
2023-08-03 05:58:22 +02:00
|
|
|
import com.viaversion.viaversion.api.protocol.AbstractProtocol;
|
2023-08-03 10:36:30 +02:00
|
|
|
import com.viaversion.viaversion.api.protocol.packet.Direction;
|
|
|
|
import com.viaversion.viaversion.api.protocol.packet.PacketWrapper;
|
2023-08-03 05:58:22 +02:00
|
|
|
import com.viaversion.viaversion.api.protocol.packet.State;
|
2023-08-04 04:22:18 +02:00
|
|
|
import com.viaversion.viaversion.api.protocol.remapper.PacketHandler;
|
2023-08-03 05:58:22 +02:00
|
|
|
import com.viaversion.viaversion.api.rewriter.EntityRewriter;
|
|
|
|
import com.viaversion.viaversion.api.rewriter.ItemRewriter;
|
|
|
|
import com.viaversion.viaversion.api.type.Type;
|
|
|
|
import com.viaversion.viaversion.data.entity.EntityTrackerBase;
|
2023-08-03 13:58:23 +02:00
|
|
|
import com.viaversion.viaversion.exception.CancelException;
|
|
|
|
import com.viaversion.viaversion.protocols.base.ClientboundLoginPackets;
|
2023-08-03 10:36:30 +02:00
|
|
|
import com.viaversion.viaversion.protocols.base.ServerboundLoginPackets;
|
2023-08-03 05:58:22 +02:00
|
|
|
import com.viaversion.viaversion.protocols.protocol1_19_4to1_19_3.ClientboundPackets1_19_4;
|
|
|
|
import com.viaversion.viaversion.protocols.protocol1_19_4to1_19_3.ServerboundPackets1_19_4;
|
2023-08-03 10:36:30 +02:00
|
|
|
import com.viaversion.viaversion.protocols.protocol1_20_2to1_20.packet.ClientboundConfigurationPackets1_20_2;
|
2023-08-03 05:58:22 +02:00
|
|
|
import com.viaversion.viaversion.protocols.protocol1_20_2to1_20.packet.ClientboundPackets1_20_2;
|
2023-08-03 10:36:30 +02:00
|
|
|
import com.viaversion.viaversion.protocols.protocol1_20_2to1_20.packet.ServerboundConfigurationPackets1_20_2;
|
2023-08-03 05:58:22 +02:00
|
|
|
import com.viaversion.viaversion.protocols.protocol1_20_2to1_20.packet.ServerboundPackets1_20_2;
|
2023-08-03 13:58:23 +02:00
|
|
|
import com.viaversion.viaversion.protocols.protocol1_20_2to1_20.rewriter.BlockItemPacketRewriter1_20_2;
|
|
|
|
import com.viaversion.viaversion.protocols.protocol1_20_2to1_20.rewriter.EntityPacketRewriter1_20_2;
|
2023-08-03 15:03:42 +02:00
|
|
|
import com.viaversion.viaversion.protocols.protocol1_20_2to1_20.storage.ConfigurationState;
|
2023-08-06 07:40:15 +02:00
|
|
|
import com.viaversion.viaversion.protocols.protocol1_20_2to1_20.storage.ConfigurationState.BridgePhase;
|
2023-09-19 03:54:03 +02:00
|
|
|
import com.viaversion.viaversion.protocols.protocol1_20_2to1_20.storage.LastResourcePack;
|
2023-10-10 17:09:36 +02:00
|
|
|
import com.viaversion.viaversion.protocols.protocol1_20_2to1_20.storage.LastTags;
|
2023-08-11 07:47:36 +02:00
|
|
|
import com.viaversion.viaversion.rewriter.SoundRewriter;
|
2023-10-24 02:01:33 +02:00
|
|
|
import com.viaversion.viaversion.util.Key;
|
2023-10-10 17:09:36 +02:00
|
|
|
import org.checkerframework.checker.nullness.qual.Nullable;
|
2023-10-08 05:05:25 +02:00
|
|
|
|
2023-10-24 02:01:33 +02:00
|
|
|
import java.util.UUID;
|
|
|
|
|
2023-08-03 05:58:22 +02:00
|
|
|
public final class Protocol1_20_2To1_20 extends AbstractProtocol<ClientboundPackets1_19_4, ClientboundPackets1_20_2, ServerboundPackets1_19_4, ServerboundPackets1_20_2> {
|
|
|
|
|
2023-08-11 07:47:36 +02:00
|
|
|
public static final MappingData MAPPINGS = new MappingDataBase("1.20", "1.20.2");
|
2023-08-03 10:38:18 +02:00
|
|
|
private final EntityPacketRewriter1_20_2 entityPacketRewriter = new EntityPacketRewriter1_20_2(this);
|
|
|
|
private final BlockItemPacketRewriter1_20_2 itemPacketRewriter = new BlockItemPacketRewriter1_20_2(this);
|
2023-08-03 05:58:22 +02:00
|
|
|
|
|
|
|
public Protocol1_20_2To1_20() {
|
|
|
|
super(ClientboundPackets1_19_4.class, ClientboundPackets1_20_2.class, ServerboundPackets1_19_4.class, ServerboundPackets1_20_2.class);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void registerPackets() {
|
2023-08-03 15:03:42 +02:00
|
|
|
// Close your eyes and turn around while you still can
|
2023-08-03 05:58:22 +02:00
|
|
|
super.registerPackets();
|
|
|
|
|
2023-08-11 07:47:36 +02:00
|
|
|
final SoundRewriter<ClientboundPackets1_19_4> soundRewriter = new SoundRewriter<>(this);
|
|
|
|
soundRewriter.register1_19_3Sound(ClientboundPackets1_19_4.SOUND);
|
|
|
|
soundRewriter.registerEntitySound(ClientboundPackets1_19_4.ENTITY_SOUND);
|
|
|
|
|
2023-09-24 02:48:47 +02:00
|
|
|
registerClientbound(ClientboundPackets1_19_4.PLUGIN_MESSAGE, wrapper -> {
|
2023-10-24 02:01:33 +02:00
|
|
|
final String channel = Key.namespaced(wrapper.passthrough(Type.STRING));
|
2023-09-24 02:48:47 +02:00
|
|
|
if (channel.equals("minecraft:brand")) {
|
2023-09-25 00:52:17 +02:00
|
|
|
wrapper.passthrough(Type.STRING);
|
2023-10-08 05:05:25 +02:00
|
|
|
wrapper.clearInputBuffer();
|
2023-09-25 00:52:17 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
registerServerbound(ServerboundPackets1_20_2.PLUGIN_MESSAGE, wrapper -> {
|
2023-10-24 02:01:33 +02:00
|
|
|
final String channel = Key.namespaced(wrapper.passthrough(Type.STRING));
|
2023-09-25 00:52:17 +02:00
|
|
|
if (channel.equals("minecraft:brand")) {
|
|
|
|
wrapper.passthrough(Type.STRING);
|
2023-10-08 05:05:25 +02:00
|
|
|
wrapper.clearInputBuffer();
|
2023-09-24 02:48:47 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2023-09-19 03:54:03 +02:00
|
|
|
registerClientbound(ClientboundPackets1_19_4.RESOURCE_PACK, wrapper -> {
|
|
|
|
final String url = wrapper.passthrough(Type.STRING);
|
|
|
|
final String hash = wrapper.passthrough(Type.STRING);
|
|
|
|
final boolean required = wrapper.passthrough(Type.BOOLEAN);
|
|
|
|
final JsonElement prompt = wrapper.passthrough(Type.OPTIONAL_COMPONENT);
|
|
|
|
wrapper.user().put(new LastResourcePack(url, hash, required, prompt));
|
|
|
|
});
|
|
|
|
|
2023-10-10 17:09:36 +02:00
|
|
|
registerClientbound(ClientboundPackets1_19_4.TAGS, wrapper -> wrapper.user().put(new LastTags(wrapper)));
|
|
|
|
registerClientbound(State.CONFIGURATION, ClientboundConfigurationPackets1_20_2.UPDATE_TAGS.getId(), ClientboundConfigurationPackets1_20_2.UPDATE_TAGS.getId(),
|
|
|
|
wrapper -> wrapper.user().put(new LastTags(wrapper)));
|
|
|
|
|
2023-08-31 03:49:54 +02:00
|
|
|
registerClientbound(ClientboundPackets1_19_4.DISPLAY_SCOREBOARD, wrapper -> {
|
2023-08-03 05:58:22 +02:00
|
|
|
final byte slot = wrapper.read(Type.BYTE);
|
|
|
|
wrapper.write(Type.VAR_INT, (int) slot);
|
|
|
|
});
|
|
|
|
|
2023-08-03 10:36:30 +02:00
|
|
|
registerServerbound(State.LOGIN, ServerboundLoginPackets.HELLO.getId(), ServerboundLoginPackets.HELLO.getId(), wrapper -> {
|
2023-08-03 05:58:22 +02:00
|
|
|
wrapper.passthrough(Type.STRING); // Name
|
2023-08-03 10:36:30 +02:00
|
|
|
|
|
|
|
final UUID uuid = wrapper.read(Type.UUID);
|
|
|
|
wrapper.write(Type.OPTIONAL_UUID, uuid);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Deal with the new CONFIGURATION protocol state the client expects
|
2023-08-03 15:03:42 +02:00
|
|
|
// After the game profile is received by the client, it will send its login ack,
|
2023-08-05 04:28:39 +02:00
|
|
|
// switch to the configuration protocol state and send its brand.
|
2023-08-03 15:03:42 +02:00
|
|
|
// We need to wait for it send the login ack before actually sending the play login,
|
|
|
|
// hence packets are added to a queue. With the data from the login packet, we sent what is needed
|
|
|
|
// during the configuration phase before finally transitioning to the play state with the client as well.
|
2023-08-03 13:58:23 +02:00
|
|
|
registerClientbound(State.LOGIN, ClientboundLoginPackets.GAME_PROFILE.getId(), ClientboundLoginPackets.GAME_PROFILE.getId(), wrapper -> {
|
2023-08-06 07:40:15 +02:00
|
|
|
wrapper.user().get(ConfigurationState.class).setBridgePhase(BridgePhase.PROFILE_SENT);
|
2023-09-28 03:12:46 +02:00
|
|
|
wrapper.user().getProtocolInfo().setServerState(State.PLAY);
|
2023-08-03 13:58:23 +02:00
|
|
|
});
|
|
|
|
|
2023-08-03 10:36:30 +02:00
|
|
|
registerServerbound(State.LOGIN, ServerboundLoginPackets.LOGIN_ACKNOWLEDGED.getId(), -1, wrapper -> {
|
|
|
|
wrapper.cancel();
|
2023-08-04 04:22:18 +02:00
|
|
|
|
2023-09-25 12:10:44 +02:00
|
|
|
// Overwrite the state set in the base protocol to what the server actually keeps sending
|
|
|
|
wrapper.user().getProtocolInfo().setServerState(State.PLAY);
|
|
|
|
|
2023-08-03 15:03:42 +02:00
|
|
|
final ConfigurationState configurationState = wrapper.user().get(ConfigurationState.class);
|
2023-08-06 07:40:15 +02:00
|
|
|
configurationState.setBridgePhase(BridgePhase.CONFIGURATION);
|
2023-08-04 04:22:18 +02:00
|
|
|
configurationState.sendQueuedPackets(wrapper.user());
|
2023-08-03 10:36:30 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
registerServerbound(State.CONFIGURATION, ServerboundConfigurationPackets1_20_2.FINISH_CONFIGURATION.getId(), -1, wrapper -> {
|
|
|
|
wrapper.cancel();
|
2023-08-03 15:03:42 +02:00
|
|
|
|
2023-09-25 08:40:13 +02:00
|
|
|
wrapper.user().getProtocolInfo().setClientState(State.PLAY);
|
|
|
|
|
2023-08-04 04:22:18 +02:00
|
|
|
final ConfigurationState configurationState = wrapper.user().get(ConfigurationState.class);
|
2023-08-06 07:40:15 +02:00
|
|
|
configurationState.setBridgePhase(BridgePhase.NONE);
|
2023-08-04 04:22:18 +02:00
|
|
|
configurationState.sendQueuedPackets(wrapper.user());
|
|
|
|
configurationState.clear();
|
2023-08-03 10:36:30 +02:00
|
|
|
});
|
2023-10-03 11:49:44 +02:00
|
|
|
registerServerbound(State.CONFIGURATION, ServerboundConfigurationPackets1_20_2.CLIENT_INFORMATION.getId(), -1, wrapper -> {
|
|
|
|
final ConfigurationState.ClientInformation clientInformation = new ConfigurationState.ClientInformation(
|
|
|
|
wrapper.read(Type.STRING), // Language
|
|
|
|
wrapper.read(Type.BYTE), // View distance
|
|
|
|
wrapper.read(Type.VAR_INT), // Chat visibility
|
|
|
|
wrapper.read(Type.BOOLEAN), // Chat colors
|
|
|
|
wrapper.read(Type.UNSIGNED_BYTE), // Model customization
|
|
|
|
wrapper.read(Type.VAR_INT), // Main hand
|
|
|
|
wrapper.read(Type.BOOLEAN), // Text filtering enabled
|
|
|
|
wrapper.read(Type.BOOLEAN) // Allow listing in server list preview
|
|
|
|
);
|
|
|
|
|
|
|
|
// Store it to re-send it when another ClientboundLoginPacket is sent, since the client will only send it
|
|
|
|
// once per connection right after the handshake
|
|
|
|
final ConfigurationState configurationState = wrapper.user().get(ConfigurationState.class);
|
|
|
|
configurationState.setClientInformation(clientInformation);
|
|
|
|
wrapper.cancel();
|
|
|
|
});
|
2023-08-04 04:22:18 +02:00
|
|
|
registerServerbound(State.CONFIGURATION, ServerboundConfigurationPackets1_20_2.CUSTOM_PAYLOAD.getId(), -1, queueServerboundPacket(ServerboundPackets1_20_2.PLUGIN_MESSAGE));
|
|
|
|
registerServerbound(State.CONFIGURATION, ServerboundConfigurationPackets1_20_2.KEEP_ALIVE.getId(), -1, queueServerboundPacket(ServerboundPackets1_20_2.KEEP_ALIVE));
|
|
|
|
registerServerbound(State.CONFIGURATION, ServerboundConfigurationPackets1_20_2.PONG.getId(), -1, queueServerboundPacket(ServerboundPackets1_20_2.PONG));
|
2023-10-03 11:49:44 +02:00
|
|
|
|
2023-09-19 03:54:03 +02:00
|
|
|
// Cancel this, as it will always just be the response to a re-sent pack from us
|
|
|
|
registerServerbound(State.CONFIGURATION, ServerboundConfigurationPackets1_20_2.RESOURCE_PACK.getId(), -1, PacketWrapper::cancel);
|
2023-08-03 10:36:30 +02:00
|
|
|
|
2023-08-04 04:22:18 +02:00
|
|
|
cancelClientbound(ClientboundPackets1_19_4.UPDATE_ENABLED_FEATURES); // TODO Sad emoji
|
2023-09-18 16:02:06 +02:00
|
|
|
registerServerbound(ServerboundPackets1_20_2.CONFIGURATION_ACKNOWLEDGED, null, wrapper -> {
|
|
|
|
wrapper.cancel();
|
|
|
|
|
|
|
|
final ConfigurationState configurationState = wrapper.user().get(ConfigurationState.class);
|
2023-09-19 13:14:45 +02:00
|
|
|
if (configurationState.bridgePhase() != BridgePhase.REENTERING_CONFIGURATION) {
|
2023-09-18 16:02:06 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reenter the configuration state
|
2023-09-25 08:40:13 +02:00
|
|
|
wrapper.user().getProtocolInfo().setClientState(State.CONFIGURATION);
|
2023-09-18 16:02:06 +02:00
|
|
|
configurationState.setBridgePhase(BridgePhase.CONFIGURATION);
|
2023-09-19 03:54:03 +02:00
|
|
|
|
|
|
|
final LastResourcePack lastResourcePack = wrapper.user().get(LastResourcePack.class);
|
2023-09-19 13:14:45 +02:00
|
|
|
sendConfigurationPackets(wrapper.user(), configurationState.lastDimensionRegistry(), lastResourcePack);
|
2023-09-18 16:02:06 +02:00
|
|
|
});
|
2023-08-03 10:36:30 +02:00
|
|
|
cancelServerbound(ServerboundPackets1_20_2.CHUNK_BATCH_RECEIVED);
|
2023-08-10 05:35:47 +02:00
|
|
|
|
|
|
|
registerServerbound(ServerboundPackets1_20_2.PING_REQUEST, null, wrapper -> {
|
|
|
|
wrapper.cancel();
|
|
|
|
final long time = wrapper.read(Type.LONG);
|
|
|
|
|
|
|
|
final PacketWrapper responsePacket = wrapper.create(ClientboundPackets1_20_2.PONG_RESPONSE);
|
|
|
|
responsePacket.write(Type.LONG, time);
|
|
|
|
responsePacket.sendFuture(Protocol1_20_2To1_20.class);
|
|
|
|
});
|
2023-08-03 10:36:30 +02:00
|
|
|
}
|
|
|
|
|
2023-08-04 04:22:18 +02:00
|
|
|
private PacketHandler queueServerboundPacket(final ServerboundPackets1_20_2 packetType) {
|
|
|
|
return wrapper -> {
|
|
|
|
wrapper.setPacketType(packetType);
|
|
|
|
wrapper.user().get(ConfigurationState.class).addPacketToQueue(wrapper, false);
|
|
|
|
wrapper.cancel();
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-08-03 10:36:30 +02:00
|
|
|
@Override
|
2023-08-03 13:58:23 +02:00
|
|
|
public void transform(final Direction direction, final State state, final PacketWrapper packetWrapper) throws Exception {
|
2023-09-25 08:40:13 +02:00
|
|
|
if (direction == Direction.SERVERBOUND) {
|
|
|
|
// The client will have the correct state set
|
|
|
|
super.transform(direction, state, packetWrapper);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-08-03 15:03:42 +02:00
|
|
|
final ConfigurationState configurationBridge = packetWrapper.user().get(ConfigurationState.class);
|
2023-09-18 16:02:06 +02:00
|
|
|
if (configurationBridge == null) {
|
|
|
|
// Bad state during an unexpected disconnect
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-08-06 07:40:15 +02:00
|
|
|
final BridgePhase phase = configurationBridge.bridgePhase();
|
|
|
|
if (phase == BridgePhase.NONE) {
|
2023-08-03 15:03:42 +02:00
|
|
|
super.transform(direction, state, packetWrapper);
|
2023-08-04 04:22:18 +02:00
|
|
|
return;
|
2023-08-03 10:36:30 +02:00
|
|
|
}
|
|
|
|
|
2023-10-10 17:09:36 +02:00
|
|
|
final int unmappedId = packetWrapper.getId();
|
2023-09-19 03:54:03 +02:00
|
|
|
if (phase == BridgePhase.PROFILE_SENT || phase == BridgePhase.REENTERING_CONFIGURATION) {
|
2023-10-10 17:09:36 +02:00
|
|
|
if (unmappedId == ClientboundPackets1_19_4.TAGS.getId()) {
|
|
|
|
// Don't re-send old tags during config phase
|
|
|
|
packetWrapper.user().remove(LastTags.class);
|
|
|
|
}
|
|
|
|
|
2023-08-06 07:40:15 +02:00
|
|
|
// Queue packets sent by the server while we wait for the client to transition to the configuration state
|
2023-08-04 04:22:18 +02:00
|
|
|
configurationBridge.addPacketToQueue(packetWrapper, true);
|
2023-08-03 15:03:42 +02:00
|
|
|
throw CancelException.generate();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (packetWrapper.getPacketType() == null || packetWrapper.getPacketType().state() != State.CONFIGURATION) {
|
2023-08-06 07:40:15 +02:00
|
|
|
// Map some of them to their configuration state counterparts, but make sure to let join game through
|
2023-08-04 04:22:18 +02:00
|
|
|
if (unmappedId == ClientboundPackets1_19_4.JOIN_GAME.getId()) {
|
2023-09-25 10:59:15 +02:00
|
|
|
super.transform(direction, State.PLAY, packetWrapper);
|
2023-08-04 04:22:18 +02:00
|
|
|
return;
|
2023-08-03 10:36:30 +02:00
|
|
|
}
|
2023-08-03 15:03:42 +02:00
|
|
|
|
2023-08-04 04:22:18 +02:00
|
|
|
if (configurationBridge.queuedOrSentJoinGame()) {
|
2023-09-16 05:27:44 +02:00
|
|
|
if (!packetWrapper.user().isClientSide() && !Via.getPlatform().isProxy() && unmappedId == ClientboundPackets1_19_4.SYSTEM_CHAT.getId()) {
|
|
|
|
// Cancelling this on the Vanilla server will cause it to exceptionally resend a message
|
|
|
|
// Assume that we have already sent the login packet and just let it through
|
|
|
|
super.transform(direction, State.PLAY, packetWrapper);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-08-04 04:22:18 +02:00
|
|
|
configurationBridge.addPacketToQueue(packetWrapper, true);
|
|
|
|
throw CancelException.generate();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unmappedId == ClientboundPackets1_19_4.PLUGIN_MESSAGE.getId()) {
|
|
|
|
packetWrapper.setPacketType(ClientboundConfigurationPackets1_20_2.CUSTOM_PAYLOAD);
|
|
|
|
} else if (unmappedId == ClientboundPackets1_19_4.DISCONNECT.getId()) {
|
|
|
|
packetWrapper.setPacketType(ClientboundConfigurationPackets1_20_2.DISCONNECT);
|
|
|
|
} else if (unmappedId == ClientboundPackets1_19_4.KEEP_ALIVE.getId()) {
|
|
|
|
packetWrapper.setPacketType(ClientboundConfigurationPackets1_20_2.KEEP_ALIVE);
|
|
|
|
} else if (unmappedId == ClientboundPackets1_19_4.PING.getId()) {
|
|
|
|
packetWrapper.setPacketType(ClientboundConfigurationPackets1_20_2.PING);
|
|
|
|
} else if (unmappedId == ClientboundPackets1_19_4.UPDATE_ENABLED_FEATURES.getId()) {
|
|
|
|
packetWrapper.setPacketType(ClientboundConfigurationPackets1_20_2.UPDATE_ENABLED_FEATURES);
|
|
|
|
} else if (unmappedId == ClientboundPackets1_19_4.TAGS.getId()) {
|
|
|
|
packetWrapper.setPacketType(ClientboundConfigurationPackets1_20_2.UPDATE_TAGS);
|
|
|
|
} else {
|
|
|
|
// Not a packet that can be mapped to the configuration protocol
|
2023-09-19 03:54:03 +02:00
|
|
|
// Includes resource pack packets to make sure it is not applied sooner than the server expects
|
2023-08-04 04:22:18 +02:00
|
|
|
configurationBridge.addPacketToQueue(packetWrapper, true);
|
|
|
|
throw CancelException.generate();
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2023-08-03 15:03:42 +02:00
|
|
|
|
|
|
|
// Redirect packets during the fake configuration phase
|
|
|
|
// This might mess up people using Via API/other protocols down the line, but such is life. We can't have different states for server and client
|
|
|
|
super.transform(direction, State.CONFIGURATION, packetWrapper);
|
2023-08-03 10:36:30 +02:00
|
|
|
}
|
|
|
|
|
2023-09-19 03:54:03 +02:00
|
|
|
public static void sendConfigurationPackets(final UserConnection connection, final CompoundTag dimensionRegistry, @Nullable final LastResourcePack lastResourcePack) throws Exception {
|
2023-09-25 08:40:13 +02:00
|
|
|
final ProtocolInfo protocolInfo = connection.getProtocolInfo();
|
|
|
|
protocolInfo.setServerState(State.CONFIGURATION);
|
|
|
|
|
2023-09-18 16:02:06 +02:00
|
|
|
final PacketWrapper registryDataPacket = PacketWrapper.create(ClientboundConfigurationPackets1_20_2.REGISTRY_DATA, connection);
|
2023-10-06 13:37:52 +02:00
|
|
|
registryDataPacket.write(Type.COMPOUND_TAG, dimensionRegistry);
|
2023-09-18 16:02:06 +02:00
|
|
|
registryDataPacket.send(Protocol1_20_2To1_20.class);
|
|
|
|
|
|
|
|
// Enabling features is only possible during the configuration phase
|
|
|
|
// TODO Sad emoji
|
|
|
|
final PacketWrapper enableFeaturesPacket = PacketWrapper.create(ClientboundConfigurationPackets1_20_2.UPDATE_ENABLED_FEATURES, connection);
|
|
|
|
enableFeaturesPacket.write(Type.VAR_INT, 1);
|
|
|
|
enableFeaturesPacket.write(Type.STRING, "minecraft:vanilla");
|
|
|
|
enableFeaturesPacket.send(Protocol1_20_2To1_20.class);
|
|
|
|
|
2023-10-10 17:09:36 +02:00
|
|
|
final LastTags lastTags = connection.get(LastTags.class);
|
|
|
|
if (lastTags != null) {
|
|
|
|
// The server might still follow up with a tags packet, but we wouldn't know
|
|
|
|
lastTags.sendLastTags(connection);
|
|
|
|
}
|
|
|
|
|
2023-09-19 03:54:03 +02:00
|
|
|
if (lastResourcePack != null) {
|
|
|
|
// The client for some reason drops the resource pack when reentering the configuration state
|
|
|
|
final PacketWrapper resourcePackPacket = PacketWrapper.create(ClientboundConfigurationPackets1_20_2.RESOURCE_PACK, connection);
|
|
|
|
resourcePackPacket.write(Type.STRING, lastResourcePack.url());
|
|
|
|
resourcePackPacket.write(Type.STRING, lastResourcePack.hash());
|
|
|
|
resourcePackPacket.write(Type.BOOLEAN, lastResourcePack.required());
|
|
|
|
resourcePackPacket.write(Type.OPTIONAL_COMPONENT, lastResourcePack.prompt());
|
|
|
|
resourcePackPacket.send(Protocol1_20_2To1_20.class);
|
|
|
|
}
|
|
|
|
|
2023-09-18 16:02:06 +02:00
|
|
|
final PacketWrapper finishConfigurationPacket = PacketWrapper.create(ClientboundConfigurationPackets1_20_2.FINISH_CONFIGURATION, connection);
|
|
|
|
finishConfigurationPacket.send(Protocol1_20_2To1_20.class);
|
2023-09-25 08:40:13 +02:00
|
|
|
|
|
|
|
protocolInfo.setServerState(State.PLAY);
|
2023-09-18 16:02:06 +02:00
|
|
|
}
|
|
|
|
|
2023-08-11 07:47:36 +02:00
|
|
|
@Override
|
|
|
|
public MappingData getMappingData() {
|
|
|
|
return MAPPINGS;
|
|
|
|
}
|
|
|
|
|
2023-08-03 10:36:30 +02:00
|
|
|
@Override
|
2023-09-25 08:40:13 +02:00
|
|
|
protected void registerConfigurationChangeHandlers() {
|
|
|
|
// Don't handle it in the transitioning protocol
|
2023-08-03 05:58:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void init(final UserConnection user) {
|
2023-08-03 15:03:42 +02:00
|
|
|
user.put(new ConfigurationState());
|
2023-10-19 01:28:12 +02:00
|
|
|
addEntityTracker(user, new EntityTrackerBase(user, EntityTypes1_19_4.PLAYER));
|
2023-08-03 05:58:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public EntityRewriter<Protocol1_20_2To1_20> getEntityRewriter() {
|
2023-08-03 10:38:18 +02:00
|
|
|
return entityPacketRewriter;
|
2023-08-03 05:58:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public ItemRewriter<Protocol1_20_2To1_20> getItemRewriter() {
|
2023-08-03 10:38:18 +02:00
|
|
|
return itemPacketRewriter;
|
2023-08-03 05:58:22 +02:00
|
|
|
}
|
2023-09-25 00:52:17 +02:00
|
|
|
}
|