forked from GeyserMC/Geyser
		
	Merge branch 'master' into inventory
This commit is contained in:
		
						commit
						1cb2e658e0
					
				
					 10 changed files with 283 additions and 13 deletions
				
			
		|  | @ -0,0 +1,82 @@ | ||||||
|  | /* | ||||||
|  |  * Copyright (c) 2019-2020 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.entity; | ||||||
|  | 
 | ||||||
|  | import com.github.steveice10.mc.protocol.data.game.entity.metadata.EntityMetadata; | ||||||
|  | import com.github.steveice10.mc.protocol.data.game.entity.metadata.Position; | ||||||
|  | import com.nukkitx.math.vector.Vector3f; | ||||||
|  | import com.nukkitx.math.vector.Vector3i; | ||||||
|  | import com.nukkitx.protocol.bedrock.data.EntityData; | ||||||
|  | import com.nukkitx.protocol.bedrock.data.EntityFlag; | ||||||
|  | import com.nukkitx.protocol.bedrock.packet.AddEntityPacket; | ||||||
|  | import org.geysermc.connector.entity.type.EntityType; | ||||||
|  | import org.geysermc.connector.network.session.GeyserSession; | ||||||
|  | 
 | ||||||
|  | public class EnderCrystalEntity extends Entity { | ||||||
|  | 
 | ||||||
|  |     public EnderCrystalEntity(long entityId, long geyserId, EntityType entityType, Vector3f position, Vector3f motion, Vector3f rotation) { | ||||||
|  |         super(entityId, geyserId, entityType, position, motion, rotation); | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public void updateBedrockMetadata(EntityMetadata entityMetadata, GeyserSession session) { | ||||||
|  |         // Show beam | ||||||
|  |         // Usually performed client-side on Bedrock except for Ender Dragon respawn event | ||||||
|  |         if (entityMetadata.getId() == 7) { | ||||||
|  |             if (entityMetadata.getValue() instanceof Position) { | ||||||
|  |                 Position pos = (Position) entityMetadata.getValue(); | ||||||
|  |                 metadata.put(EntityData.BLOCK_TARGET, Vector3i.from(pos.getX(), pos.getY(), pos.getZ())); | ||||||
|  |             } else { | ||||||
|  |                 metadata.put(EntityData.BLOCK_TARGET, Vector3i.ZERO); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         // There is a base located on the ender crystal | ||||||
|  |         if (entityMetadata.getId() == 8) { | ||||||
|  |             metadata.getFlags().setFlag(EntityFlag.SHOW_BOTTOM, (boolean) entityMetadata.getValue()); | ||||||
|  |         } | ||||||
|  |         super.updateBedrockMetadata(entityMetadata, session); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public void spawnEntity(GeyserSession session) { | ||||||
|  |         AddEntityPacket addEntityPacket = new AddEntityPacket(); | ||||||
|  |         // Not end crystal but ender crystal | ||||||
|  |         addEntityPacket.setIdentifier("minecraft:ender_crystal"); | ||||||
|  |         addEntityPacket.setRuntimeEntityId(geyserId); | ||||||
|  |         addEntityPacket.setUniqueEntityId(geyserId); | ||||||
|  |         addEntityPacket.setPosition(position); | ||||||
|  |         addEntityPacket.setMotion(motion); | ||||||
|  |         addEntityPacket.setRotation(getBedrockRotation()); | ||||||
|  |         addEntityPacket.setEntityType(entityType.getType()); | ||||||
|  |         addEntityPacket.getMetadata().putAll(metadata); | ||||||
|  | 
 | ||||||
|  |         valid = true; | ||||||
|  |         session.getUpstream().sendPacket(addEntityPacket); | ||||||
|  | 
 | ||||||
|  |         session.getConnector().getLogger().debug("Spawned entity " + entityType + " at location " + position + " with id " + geyserId + " (java id " + entityId + ")"); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,86 @@ | ||||||
|  | /* | ||||||
|  |  * Copyright (c) 2019-2020 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.entity.living.monster; | ||||||
|  | 
 | ||||||
|  | import com.github.steveice10.mc.protocol.data.game.entity.metadata.EntityMetadata; | ||||||
|  | import com.nukkitx.math.vector.Vector3f; | ||||||
|  | import com.nukkitx.protocol.bedrock.data.Attribute; | ||||||
|  | import com.nukkitx.protocol.bedrock.data.EntityEventType; | ||||||
|  | import com.nukkitx.protocol.bedrock.data.EntityFlag; | ||||||
|  | import com.nukkitx.protocol.bedrock.packet.AddEntityPacket; | ||||||
|  | import com.nukkitx.protocol.bedrock.packet.EntityEventPacket; | ||||||
|  | import org.geysermc.connector.entity.living.InsentientEntity; | ||||||
|  | import org.geysermc.connector.entity.type.EntityType; | ||||||
|  | import org.geysermc.connector.network.session.GeyserSession; | ||||||
|  | 
 | ||||||
|  | public class EnderDragonEntity extends InsentientEntity { | ||||||
|  | 
 | ||||||
|  |     public EnderDragonEntity(long entityId, long geyserId, EntityType entityType, Vector3f position, Vector3f motion, Vector3f rotation) { | ||||||
|  |         super(entityId, geyserId, entityType, position, motion, rotation); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public void updateBedrockMetadata(EntityMetadata entityMetadata, GeyserSession session) { | ||||||
|  |         if (entityMetadata.getId() == 15) { | ||||||
|  |             metadata.getFlags().setFlag(EntityFlag.FIRE_IMMUNE, true); | ||||||
|  |             switch ((int) entityMetadata.getValue()) { | ||||||
|  |                 // Performing breath attack | ||||||
|  |                 case 5: | ||||||
|  |                     EntityEventPacket entityEventPacket = new EntityEventPacket(); | ||||||
|  |                     entityEventPacket.setType(EntityEventType.DRAGON_FLAMING); | ||||||
|  |                     entityEventPacket.setRuntimeEntityId(geyserId); | ||||||
|  |                     entityEventPacket.setData(0); | ||||||
|  |                     session.getUpstream().sendPacket(entityEventPacket); | ||||||
|  |                 case 6: | ||||||
|  |                 case 7: | ||||||
|  |                     metadata.getFlags().setFlag(EntityFlag.SITTING, true); | ||||||
|  |                     break; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         super.updateBedrockMetadata(entityMetadata, session); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public void spawnEntity(GeyserSession session) { | ||||||
|  |         AddEntityPacket addEntityPacket = new AddEntityPacket(); | ||||||
|  |         addEntityPacket.setIdentifier("minecraft:" + entityType.name().toLowerCase()); | ||||||
|  |         addEntityPacket.setRuntimeEntityId(geyserId); | ||||||
|  |         addEntityPacket.setUniqueEntityId(geyserId); | ||||||
|  |         addEntityPacket.setPosition(position); | ||||||
|  |         addEntityPacket.setMotion(motion); | ||||||
|  |         addEntityPacket.setRotation(getBedrockRotation()); | ||||||
|  |         addEntityPacket.setEntityType(entityType.getType()); | ||||||
|  |         addEntityPacket.getMetadata().putAll(metadata); | ||||||
|  | 
 | ||||||
|  |         // Otherwise dragon is always 'dying' | ||||||
|  |         addEntityPacket.getAttributes().add(new Attribute("minecraft:health", 0.0f, 200f, 200f, 200f)); | ||||||
|  | 
 | ||||||
|  |         valid = true; | ||||||
|  |         session.getUpstream().sendPacket(addEntityPacket); | ||||||
|  | 
 | ||||||
|  |         session.getConnector().getLogger().debug("Spawned entity " + entityType + " at location " + position + " with id " + geyserId + " (java id " + entityId + ")"); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,56 @@ | ||||||
|  | /* | ||||||
|  |  * Copyright (c) 2019-2020 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.entity.living.monster; | ||||||
|  | 
 | ||||||
|  | import com.github.steveice10.mc.protocol.data.game.entity.metadata.EntityMetadata; | ||||||
|  | import com.github.steveice10.mc.protocol.data.game.world.block.BlockState; | ||||||
|  | import com.nukkitx.math.vector.Vector3f; | ||||||
|  | import com.nukkitx.protocol.bedrock.data.EntityData; | ||||||
|  | import com.nukkitx.protocol.bedrock.data.EntityFlag; | ||||||
|  | import org.geysermc.connector.entity.type.EntityType; | ||||||
|  | import org.geysermc.connector.network.session.GeyserSession; | ||||||
|  | import org.geysermc.connector.network.translators.block.BlockTranslator; | ||||||
|  | 
 | ||||||
|  | public class EndermanEntity extends MonsterEntity { | ||||||
|  | 
 | ||||||
|  |     public EndermanEntity(long entityId, long geyserId, EntityType entityType, Vector3f position, Vector3f motion, Vector3f rotation) { | ||||||
|  |         super(entityId, geyserId, entityType, position, motion, rotation); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public void updateBedrockMetadata(EntityMetadata entityMetadata, GeyserSession session) { | ||||||
|  |         // Held block | ||||||
|  |         if (entityMetadata.getId() == 15) { | ||||||
|  |             metadata.put(EntityData.ENDERMAN_HELD_ITEM_ID, BlockTranslator.getBedrockBlockId((BlockState) entityMetadata.getValue())); | ||||||
|  |         } | ||||||
|  |         // 'Angry' - mouth open | ||||||
|  |         if (entityMetadata.getId() == 16) { | ||||||
|  |             metadata.getFlags().setFlag(EntityFlag.ANGRY, (boolean) entityMetadata.getValue()); | ||||||
|  |         } | ||||||
|  |         // TODO: ID 17 is stared at but I don't believe it's used - maybe only for the sound effect. Check after particle merge | ||||||
|  |         super.updateBedrockMetadata(entityMetadata, session); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -69,7 +69,7 @@ public enum EntityType { | ||||||
|     SPIDER(SpiderEntity.class, 35, 0.9f, 1.4f, 1.4f, 1f), |     SPIDER(SpiderEntity.class, 35, 0.9f, 1.4f, 1.4f, 1f), | ||||||
|     ZOMBIE_PIGMAN(MonsterEntity.class, 36, 1.8f, 0.6f, 0.6f, 1.62f), |     ZOMBIE_PIGMAN(MonsterEntity.class, 36, 1.8f, 0.6f, 0.6f, 1.62f), | ||||||
|     SLIME(InsentientEntity.class, 37, 0.51f), |     SLIME(InsentientEntity.class, 37, 0.51f), | ||||||
|     ENDERMAN(MonsterEntity.class, 38, 2.9f, 0.6f), |     ENDERMAN(EndermanEntity.class, 38, 2.9f, 0.6f), | ||||||
|     SILVERFISH(MonsterEntity.class, 39, 0.3f, 0.4f), |     SILVERFISH(MonsterEntity.class, 39, 0.3f, 0.4f), | ||||||
|     CAVE_SPIDER(MonsterEntity.class, 40, 0.5f, 0.7f), |     CAVE_SPIDER(MonsterEntity.class, 40, 0.5f, 0.7f), | ||||||
|     GHAST(FlyingEntity.class, 41, 4.0f), |     GHAST(FlyingEntity.class, 41, 4.0f), | ||||||
|  | @ -84,7 +84,7 @@ public enum EntityType { | ||||||
|     ELDER_GUARDIAN(GuardianEntity.class, 50, 1.9975f), |     ELDER_GUARDIAN(GuardianEntity.class, 50, 1.9975f), | ||||||
|     NPC(PlayerEntity.class, 51, 1.8f, 0.6f, 0.6f, 1.62f), |     NPC(PlayerEntity.class, 51, 1.8f, 0.6f, 0.6f, 1.62f), | ||||||
|     WITHER(MonsterEntity.class, 52, 3.5f, 0.9f), |     WITHER(MonsterEntity.class, 52, 3.5f, 0.9f), | ||||||
|     ENDER_DRAGON(InsentientEntity.class, 53, 4f, 13f), |     ENDER_DRAGON(EnderDragonEntity.class, 53, 4f, 13f), | ||||||
|     SHULKER(GolemEntity.class, 54, 1f, 1f), |     SHULKER(GolemEntity.class, 54, 1f, 1f), | ||||||
|     ENDERMITE(MonsterEntity.class, 55, 0.3f, 0.4f), |     ENDERMITE(MonsterEntity.class, 55, 0.3f, 0.4f), | ||||||
|     AGENT(Entity.class, 56, 0f), |     AGENT(Entity.class, 56, 0f), | ||||||
|  | @ -104,7 +104,7 @@ public enum EntityType { | ||||||
|     EXPERIENCE_BOTTLE(ThrowableEntity.class, 68, 0.25f, 0.25f), |     EXPERIENCE_BOTTLE(ThrowableEntity.class, 68, 0.25f, 0.25f), | ||||||
|     EXPERIENCE_ORB(ExpOrbEntity.class, 69, 0f), |     EXPERIENCE_ORB(ExpOrbEntity.class, 69, 0f), | ||||||
|     EYE_OF_ENDER(Entity.class, 70, 0f), |     EYE_OF_ENDER(Entity.class, 70, 0f), | ||||||
|     END_CRYSTAL(Entity.class, 71, 0f), |     END_CRYSTAL(EnderCrystalEntity.class, 71, 0f), | ||||||
|     FIREWORK_ROCKET(Entity.class, 72, 0f), |     FIREWORK_ROCKET(Entity.class, 72, 0f), | ||||||
|     TRIDENT(ArrowEntity.class, 73, 0f), |     TRIDENT(ArrowEntity.class, 73, 0f), | ||||||
|     TURTLE(AnimalEntity.class, 74, 0.4f, 1.2f), |     TURTLE(AnimalEntity.class, 74, 0.4f, 1.2f), | ||||||
|  |  | ||||||
|  | @ -80,6 +80,13 @@ public class ConnectorServerEventHandler implements BedrockServerEventHandler { | ||||||
|             pong.setMotd(config.getBedrock().getMotd1()); |             pong.setMotd(config.getBedrock().getMotd1()); | ||||||
|             pong.setMotd(config.getBedrock().getMotd2()); |             pong.setMotd(config.getBedrock().getMotd2()); | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|  |         //Bedrock will not even attempt a connection if the client thinks the server is full | ||||||
|  |         //so we have to fake it not being full | ||||||
|  |         if (pong.getPlayerCount() >= pong.getMaximumPlayerCount()) { | ||||||
|  |             pong.setMaximumPlayerCount(pong.getPlayerCount() + 1); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         return pong; |         return pong; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -288,6 +288,9 @@ public class GeyserSession implements CommandSender { | ||||||
|                         loggingIn = false; |                         loggingIn = false; | ||||||
|                         loggedIn = false; |                         loggedIn = false; | ||||||
|                         connector.getLogger().info(authData.getName() + " has disconnected from remote java server on address " + remoteServer.getAddress() + " because of " + event.getReason()); |                         connector.getLogger().info(authData.getName() + " has disconnected from remote java server on address " + remoteServer.getAddress() + " because of " + event.getReason()); | ||||||
|  |                         if (event.getCause() != null) { | ||||||
|  |                             event.getCause().printStackTrace(); | ||||||
|  |                         } | ||||||
|                         upstream.disconnect(event.getReason()); |                         upstream.disconnect(event.getReason()); | ||||||
|                     } |                     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -99,8 +99,10 @@ public class ItemTranslator { | ||||||
|         CompoundTag tag = stack.getNbt(); |         CompoundTag tag = stack.getNbt(); | ||||||
|         IntTag mapId = tag.get("map"); |         IntTag mapId = tag.get("map"); | ||||||
| 
 | 
 | ||||||
|         if (mapId != null) |         if (mapId != null) { | ||||||
|             tag.put(new StringTag("map_uuid", mapId.getValue().toString())); |             tag.put(new StringTag("map_uuid", mapId.getValue().toString())); | ||||||
|  |             tag.put(new IntTag("map_name_index", mapId.getValue())); | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|         return ItemData.of(bedrockItem.getBedrockId(), (short) bedrockItem.getBedrockData(), stack.getAmount(), translateToBedrockNBT(tag)); |         return ItemData.of(bedrockItem.getBedrockId(), (short) bedrockItem.getBedrockData(), stack.getAmount(), translateToBedrockNBT(tag)); | ||||||
|  |  | ||||||
|  | @ -25,6 +25,10 @@ | ||||||
| 
 | 
 | ||||||
| package org.geysermc.connector.network.translators.java.world; | package org.geysermc.connector.network.translators.java.world; | ||||||
| 
 | 
 | ||||||
|  | import com.nukkitx.protocol.bedrock.data.GameRuleData; | ||||||
|  | import com.nukkitx.protocol.bedrock.packet.GameRulesChangedPacket; | ||||||
|  | import it.unimi.dsi.fastutil.longs.Long2LongMap; | ||||||
|  | import it.unimi.dsi.fastutil.longs.Long2LongOpenHashMap; | ||||||
| import org.geysermc.connector.network.session.GeyserSession; | import org.geysermc.connector.network.session.GeyserSession; | ||||||
| import org.geysermc.connector.network.translators.PacketTranslator; | import org.geysermc.connector.network.translators.PacketTranslator; | ||||||
| import org.geysermc.connector.network.translators.Translator; | import org.geysermc.connector.network.translators.Translator; | ||||||
|  | @ -35,11 +39,37 @@ import com.nukkitx.protocol.bedrock.packet.SetTimePacket; | ||||||
| @Translator(packet = ServerUpdateTimePacket.class) | @Translator(packet = ServerUpdateTimePacket.class) | ||||||
| public class JavaUpdateTimeTranslator extends PacketTranslator<ServerUpdateTimePacket> { | public class JavaUpdateTimeTranslator extends PacketTranslator<ServerUpdateTimePacket> { | ||||||
| 
 | 
 | ||||||
|  |     // If negative, the last time is stored so we know it's not some plugin behavior doing weird things. | ||||||
|  |     // Per-player for multi-world support | ||||||
|  |     private static final Long2LongMap LAST_RECORDED_TIMES = new Long2LongOpenHashMap(); | ||||||
|  | 
 | ||||||
|     @Override |     @Override | ||||||
|     public void translate(ServerUpdateTimePacket packet, GeyserSession session) { |     public void translate(ServerUpdateTimePacket packet, GeyserSession session) { | ||||||
|  | 
 | ||||||
|  |         // Bedrock sends a GameRulesChangedPacket if there is no daylight cycle | ||||||
|  |         // Java just sends a negative long if there is no daylight cycle | ||||||
|  |         long lastTime = LAST_RECORDED_TIMES.getOrDefault(session.getPlayerEntity().getEntityId(), 0); | ||||||
|  |         long time = packet.getTime(); | ||||||
|  | 
 | ||||||
|  |         if (lastTime != time) { | ||||||
|             // https://minecraft.gamepedia.com/Day-night_cycle#24-hour_Minecraft_day |             // https://minecraft.gamepedia.com/Day-night_cycle#24-hour_Minecraft_day | ||||||
|             SetTimePacket setTimePacket = new SetTimePacket(); |             SetTimePacket setTimePacket = new SetTimePacket(); | ||||||
|         setTimePacket.setTime((int) Math.abs(packet.getTime()) % 24000); |             setTimePacket.setTime((int) Math.abs(time) % 24000); | ||||||
|             session.getUpstream().sendPacket(setTimePacket); |             session.getUpstream().sendPacket(setTimePacket); | ||||||
|  |             // TODO: Performance efficient to always do this? | ||||||
|  |             LAST_RECORDED_TIMES.put(session.getPlayerEntity().getEntityId(), time); | ||||||
|  |         } | ||||||
|  |         if (lastTime < 0 && time >= 0) { | ||||||
|  |             setDoDaylightCycleGamerule(session, true); | ||||||
|  |         } else if (lastTime != time && time < 0) { | ||||||
|  |             setDoDaylightCycleGamerule(session, false); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     private void setDoDaylightCycleGamerule(GeyserSession session, boolean doCycle) { | ||||||
|  |         GameRulesChangedPacket gameRulesChangedPacket = new GameRulesChangedPacket(); | ||||||
|  |         gameRulesChangedPacket.getGameRules().add(new GameRuleData<>("dodaylightcycle", doCycle)); | ||||||
|  |         session.getUpstream().sendPacket(gameRulesChangedPacket); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @ -227,7 +227,7 @@ public enum MapColor { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static MapColor fromId(int id) { |     public static MapColor fromId(int id) { | ||||||
|         return Arrays.stream(values()).filter(color -> color.getId() == id).findFirst().get(); |         return Arrays.stream(values()).filter(color -> color.getId() == id).findFirst().orElse(COLOR_0); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public int toARGB() { |     public int toARGB() { | ||||||
|  |  | ||||||
|  | @ -58,7 +58,7 @@ public class MessageUtils { | ||||||
|                     strings.add(" - no permission or invalid command!"); |                     strings.add(" - no permission or invalid command!"); | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 List<String> furtherParams = getTranslationParams(translation.getTranslationParams()); |                 List<String> furtherParams = getTranslationParams(translation.getTranslationParams(), locale); | ||||||
|                 if (locale != null) { |                 if (locale != null) { | ||||||
|                     strings.add(insertParams(LocaleUtils.getLocaleString(translation.getTranslationKey(), locale), furtherParams)); |                     strings.add(insertParams(LocaleUtils.getLocaleString(translation.getTranslationKey(), locale), furtherParams)); | ||||||
|                 }else{ |                 }else{ | ||||||
|  | @ -96,7 +96,11 @@ public class MessageUtils { | ||||||
|             messageText = LocaleUtils.getLocaleString(messageText, locale); |             messageText = LocaleUtils.getLocaleString(messageText, locale); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         StringBuilder builder = new StringBuilder(messageText); |         StringBuilder builder = new StringBuilder(); | ||||||
|  |         builder.append(getFormat(message.getStyle().getFormats())); | ||||||
|  |         builder.append(getColorOrParent(message.getStyle())); | ||||||
|  |         builder.append(messageText); | ||||||
|  | 
 | ||||||
|         for (Message msg : message.getExtra()) { |         for (Message msg : message.getExtra()) { | ||||||
|             builder.append(getFormat(msg.getStyle().getFormats())); |             builder.append(getFormat(msg.getStyle().getFormats())); | ||||||
|             builder.append(getColorOrParent(msg.getStyle())); |             builder.append(getColorOrParent(msg.getStyle())); | ||||||
|  | @ -139,8 +143,8 @@ public class MessageUtils { | ||||||
|     private static String getColorOrParent(MessageStyle style) { |     private static String getColorOrParent(MessageStyle style) { | ||||||
|         ChatColor chatColor = style.getColor(); |         ChatColor chatColor = style.getColor(); | ||||||
| 
 | 
 | ||||||
|         if (chatColor == ChatColor.NONE) { |         if (chatColor == ChatColor.NONE && style.getParent() != null) { | ||||||
|             return getColor(style.getParent().getColor()); |             return getColorOrParent(style.getParent()); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         return getColor(chatColor); |         return getColor(chatColor); | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue