/* * This file is part of ViaVersion - https://github.com/ViaVersion/ViaVersion * 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 . */ package com.viaversion.viaversion.protocols.protocol1_16to1_15_2; import com.google.common.base.Joiner; import com.google.gson.JsonArray; import com.google.gson.JsonElement; import com.google.gson.JsonObject; import com.viaversion.viaversion.api.Via; import com.viaversion.viaversion.api.connection.UserConnection; import com.viaversion.viaversion.api.minecraft.RegistryType; import com.viaversion.viaversion.api.minecraft.entities.EntityTypes1_16; import com.viaversion.viaversion.api.platform.providers.ViaProviders; import com.viaversion.viaversion.api.protocol.AbstractProtocol; import com.viaversion.viaversion.api.protocol.packet.State; import com.viaversion.viaversion.api.protocol.remapper.PacketHandlers; import com.viaversion.viaversion.api.type.Type; import com.viaversion.viaversion.api.type.types.misc.ParticleType; import com.viaversion.viaversion.api.type.types.version.Types1_16; import com.viaversion.viaversion.data.entity.EntityTrackerBase; import com.viaversion.viaversion.protocols.protocol1_14to1_13_2.ServerboundPackets1_14; import com.viaversion.viaversion.protocols.protocol1_15to1_14_4.ClientboundPackets1_15; import com.viaversion.viaversion.protocols.protocol1_16to1_15_2.data.MappingData; import com.viaversion.viaversion.protocols.protocol1_16to1_15_2.data.TranslationMappings; import com.viaversion.viaversion.protocols.protocol1_16to1_15_2.metadata.MetadataRewriter1_16To1_15_2; import com.viaversion.viaversion.protocols.protocol1_16to1_15_2.packets.EntityPackets; import com.viaversion.viaversion.protocols.protocol1_16to1_15_2.packets.InventoryPackets; import com.viaversion.viaversion.protocols.protocol1_16to1_15_2.packets.WorldPackets; import com.viaversion.viaversion.protocols.protocol1_16to1_15_2.provider.PlayerAbilitiesProvider; import com.viaversion.viaversion.protocols.protocol1_16to1_15_2.storage.InventoryTracker1_16; import com.viaversion.viaversion.rewriter.SoundRewriter; import com.viaversion.viaversion.rewriter.StatisticsRewriter; import com.viaversion.viaversion.rewriter.TagRewriter; import com.viaversion.viaversion.util.GsonUtil; import com.viaversion.viaversion.util.Key; import java.nio.charset.StandardCharsets; import java.util.ArrayList; import java.util.List; import java.util.UUID; public class Protocol1_16To1_15_2 extends AbstractProtocol { private static final UUID ZERO_UUID = new UUID(0, 0); public static final MappingData MAPPINGS = new MappingData(); private final MetadataRewriter1_16To1_15_2 metadataRewriter = new MetadataRewriter1_16To1_15_2(this); private final InventoryPackets itemRewriter = new InventoryPackets(this); private final TranslationMappings componentRewriter = new TranslationMappings(this); private TagRewriter tagRewriter; public Protocol1_16To1_15_2() { super(ClientboundPackets1_15.class, ClientboundPackets1_16.class, ServerboundPackets1_14.class, ServerboundPackets1_16.class); } @Override protected void registerPackets() { super.registerPackets(); EntityPackets.register(this); WorldPackets.register(this); tagRewriter = new TagRewriter<>(this); tagRewriter.register(ClientboundPackets1_15.TAGS, RegistryType.ENTITY); new StatisticsRewriter<>(this).register(ClientboundPackets1_15.STATISTICS); // Login Success registerClientbound(State.LOGIN, 0x02, 0x02, wrapper -> { // Transform string to a uuid UUID uuid = UUID.fromString(wrapper.read(Type.STRING)); wrapper.write(Type.UUID, uuid); }); // Motd Status - line breaks are no longer allowed for player samples registerClientbound(State.STATUS, 0x00, 0x00, wrapper -> { String original = wrapper.passthrough(Type.STRING); JsonObject object = GsonUtil.getGson().fromJson(original, JsonObject.class); JsonObject players = object.getAsJsonObject("players"); if (players == null) return; JsonArray sample = players.getAsJsonArray("sample"); if (sample == null) return; JsonArray splitSamples = new JsonArray(); for (JsonElement element : sample) { JsonObject playerInfo = element.getAsJsonObject(); String name = playerInfo.getAsJsonPrimitive("name").getAsString(); if (name.indexOf('\n') == -1) { splitSamples.add(playerInfo); continue; } String id = playerInfo.getAsJsonPrimitive("id").getAsString(); for (String s : name.split("\n")) { JsonObject newSample = new JsonObject(); newSample.addProperty("name", s); newSample.addProperty("id", id); splitSamples.add(newSample); } } // Replace data if changed if (splitSamples.size() != sample.size()) { players.add("sample", splitSamples); wrapper.set(Type.STRING, 0, object.toString()); } }); // Handle (relevant) component cases for translatable and score changes registerClientbound(ClientboundPackets1_15.CHAT_MESSAGE, new PacketHandlers() { @Override public void register() { map(Type.COMPONENT); map(Type.BYTE); handler(wrapper -> { componentRewriter.processText(wrapper.get(Type.COMPONENT, 0)); wrapper.write(Type.UUID, ZERO_UUID); // Sender uuid - always send as 'system' }); } }); componentRewriter.registerBossBar(ClientboundPackets1_15.BOSSBAR); componentRewriter.registerTitle(ClientboundPackets1_15.TITLE); componentRewriter.registerCombatEvent(ClientboundPackets1_15.COMBAT_EVENT); SoundRewriter soundRewriter = new SoundRewriter<>(this); soundRewriter.registerSound(ClientboundPackets1_15.SOUND); soundRewriter.registerSound(ClientboundPackets1_15.ENTITY_SOUND); registerServerbound(ServerboundPackets1_16.INTERACT_ENTITY, wrapper -> { wrapper.passthrough(Type.VAR_INT); // Entity Id int action = wrapper.passthrough(Type.VAR_INT); if (action == 0 || action == 2) { if (action == 2) { // Location wrapper.passthrough(Type.FLOAT); wrapper.passthrough(Type.FLOAT); wrapper.passthrough(Type.FLOAT); } wrapper.passthrough(Type.VAR_INT); // Hand } // New boolean: Whether the client is sneaking/pressing shift wrapper.read(Type.BOOLEAN); }); if (Via.getConfig().isIgnoreLong1_16ChannelNames()) { registerServerbound(ServerboundPackets1_16.PLUGIN_MESSAGE, new PacketHandlers() { @Override public void register() { handler(wrapper -> { String channel = wrapper.passthrough(Type.STRING); final String namespacedChannel = Key.namespaced(channel); if (channel.length() > 32) { if (!Via.getConfig().isSuppressConversionWarnings()) { Via.getPlatform().getLogger().warning("Ignoring incoming plugin channel, as it is longer than 32 characters: " + channel); } wrapper.cancel(); } else if (namespacedChannel.equals("minecraft:register") || namespacedChannel.equals("minecraft:unregister")) { String[] channels = new String(wrapper.read(Type.REMAINING_BYTES), StandardCharsets.UTF_8).split("\0"); List checkedChannels = new ArrayList<>(channels.length); for (String registeredChannel : channels) { if (registeredChannel.length() > 32) { if (!Via.getConfig().isSuppressConversionWarnings()) { Via.getPlatform().getLogger().warning("Ignoring incoming plugin channel register of '" + registeredChannel + "', as it is longer than 32 characters"); } continue; } checkedChannels.add(registeredChannel); } if (checkedChannels.isEmpty()) { wrapper.cancel(); return; } wrapper.write(Type.REMAINING_BYTES, Joiner.on('\0').join(checkedChannels).getBytes(StandardCharsets.UTF_8)); } }); } }); } registerServerbound(ServerboundPackets1_16.PLAYER_ABILITIES, wrapper -> { wrapper.passthrough(Type.BYTE); // Flags final PlayerAbilitiesProvider playerAbilities = Via.getManager().getProviders().get(PlayerAbilitiesProvider.class); wrapper.write(Type.FLOAT, playerAbilities.getFlyingSpeed(wrapper.user())); wrapper.write(Type.FLOAT, playerAbilities.getWalkingSpeed(wrapper.user())); }); cancelServerbound(ServerboundPackets1_16.GENERATE_JIGSAW); cancelServerbound(ServerboundPackets1_16.UPDATE_JIGSAW_BLOCK); } @Override protected void onMappingDataLoaded() { int[] wallPostOverrideTag = new int[47]; int arrayIndex = 0; wallPostOverrideTag[arrayIndex++] = 140; wallPostOverrideTag[arrayIndex++] = 179; wallPostOverrideTag[arrayIndex++] = 264; for (int i = 153; i <= 158; i++) { wallPostOverrideTag[arrayIndex++] = i; } for (int i = 163; i <= 168; i++) { wallPostOverrideTag[arrayIndex++] = i; } for (int i = 408; i <= 439; i++) { wallPostOverrideTag[arrayIndex++] = i; } tagRewriter.addTag(RegistryType.BLOCK, "minecraft:wall_post_override", wallPostOverrideTag); tagRewriter.addTag(RegistryType.BLOCK, "minecraft:beacon_base_blocks", 133, 134, 148, 265); tagRewriter.addTag(RegistryType.BLOCK, "minecraft:climbable", 160, 241, 658); tagRewriter.addTag(RegistryType.BLOCK, "minecraft:fire", 142); tagRewriter.addTag(RegistryType.BLOCK, "minecraft:campfires", 679); tagRewriter.addTag(RegistryType.BLOCK, "minecraft:fence_gates", 242, 467, 468, 469, 470, 471); tagRewriter.addTag(RegistryType.BLOCK, "minecraft:unstable_bottom_center", 242, 467, 468, 469, 470, 471); tagRewriter.addTag(RegistryType.BLOCK, "minecraft:wooden_trapdoors", 193, 194, 195, 196, 197, 198); tagRewriter.addTag(RegistryType.ITEM, "minecraft:wooden_trapdoors", 215, 216, 217, 218, 219, 220); tagRewriter.addTag(RegistryType.ITEM, "minecraft:beacon_payment_items", 529, 530, 531, 760); tagRewriter.addTag(RegistryType.ENTITY, "minecraft:impact_projectiles", 2, 72, 71, 37, 69, 79, 83, 15, 93); // The client crashes if we don't send all tags it may use tagRewriter.addEmptyTag(RegistryType.BLOCK, "minecraft:guarded_by_piglins"); tagRewriter.addEmptyTag(RegistryType.BLOCK, "minecraft:soul_speed_blocks"); tagRewriter.addEmptyTag(RegistryType.BLOCK, "minecraft:soul_fire_base_blocks"); tagRewriter.addEmptyTag(RegistryType.BLOCK, "minecraft:non_flammable_wood"); tagRewriter.addEmptyTag(RegistryType.ITEM, "minecraft:non_flammable_wood"); // The rest of not accessed tags added in older versions; #1830 tagRewriter.addEmptyTags(RegistryType.BLOCK, "minecraft:bamboo_plantable_on", "minecraft:beds", "minecraft:bee_growables", "minecraft:beehives", "minecraft:coral_plants", "minecraft:crops", "minecraft:dragon_immune", "minecraft:flowers", "minecraft:portals", "minecraft:shulker_boxes", "minecraft:small_flowers", "minecraft:tall_flowers", "minecraft:trapdoors", "minecraft:underwater_bonemeals", "minecraft:wither_immune", "minecraft:wooden_fences", "minecraft:wooden_trapdoors"); tagRewriter.addEmptyTags(RegistryType.ENTITY, "minecraft:arrows", "minecraft:beehive_inhabitors", "minecraft:raiders", "minecraft:skeletons"); tagRewriter.addEmptyTags(RegistryType.ITEM, "minecraft:beds", "minecraft:coals", "minecraft:fences", "minecraft:flowers", "minecraft:lectern_books", "minecraft:music_discs", "minecraft:small_flowers", "minecraft:tall_flowers", "minecraft:trapdoors", "minecraft:walls", "minecraft:wooden_fences"); Types1_16.PARTICLE.filler(this) .reader("block", ParticleType.Readers.BLOCK) .reader("dust", ParticleType.Readers.DUST) .reader("falling_dust", ParticleType.Readers.BLOCK) .reader("item", ParticleType.Readers.ITEM1_13_2); } @Override public void register(ViaProviders providers) { providers.register(PlayerAbilitiesProvider.class, new PlayerAbilitiesProvider()); } @Override public void init(UserConnection userConnection) { userConnection.addEntityTracker(this.getClass(), new EntityTrackerBase(userConnection, EntityTypes1_16.PLAYER)); userConnection.put(new InventoryTracker1_16()); } @Override public MappingData getMappingData() { return MAPPINGS; } @Override public MetadataRewriter1_16To1_15_2 getEntityRewriter() { return metadataRewriter; } @Override public InventoryPackets getItemRewriter() { return itemRewriter; } public TranslationMappings getComponentRewriter() { return componentRewriter; } }