forked from GeyserMC/Geyser
Start block breaking animations (inventory is broken)
This commit is contained in:
parent
1bcee8d36f
commit
650a1e2ab1
7 changed files with 181 additions and 9 deletions
|
@ -52,6 +52,7 @@ public class Inventory {
|
||||||
@Setter
|
@Setter
|
||||||
protected String title;
|
protected String title;
|
||||||
|
|
||||||
|
@Getter
|
||||||
@Setter
|
@Setter
|
||||||
protected ItemStack[] items;
|
protected ItemStack[] items;
|
||||||
|
|
||||||
|
|
|
@ -50,6 +50,6 @@ public class PlayerInventory extends Inventory {
|
||||||
}
|
}
|
||||||
|
|
||||||
public ItemStack getItemInHand() {
|
public ItemStack getItemInHand() {
|
||||||
return items[heldItemSlot];
|
return items[36 + heldItemSlot];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,9 +32,9 @@ import com.nukkitx.nbt.NbtUtils;
|
||||||
import com.nukkitx.nbt.stream.NBTInputStream;
|
import com.nukkitx.nbt.stream.NBTInputStream;
|
||||||
import com.nukkitx.nbt.tag.CompoundTag;
|
import com.nukkitx.nbt.tag.CompoundTag;
|
||||||
import com.nukkitx.nbt.tag.ListTag;
|
import com.nukkitx.nbt.tag.ListTag;
|
||||||
import gnu.trove.map.TObjectIntMap;
|
|
||||||
import gnu.trove.map.hash.TObjectIntHashMap;
|
|
||||||
import it.unimi.dsi.fastutil.ints.*;
|
import it.unimi.dsi.fastutil.ints.*;
|
||||||
|
import it.unimi.dsi.fastutil.objects.Object2IntMap;
|
||||||
|
import it.unimi.dsi.fastutil.objects.Object2IntOpenHashMap;
|
||||||
import org.geysermc.connector.GeyserConnector;
|
import org.geysermc.connector.GeyserConnector;
|
||||||
import org.geysermc.connector.utils.Toolbox;
|
import org.geysermc.connector.utils.Toolbox;
|
||||||
|
|
||||||
|
@ -52,6 +52,8 @@ public class BlockTranslator {
|
||||||
private static final IntSet WATERLOGGED = new IntOpenHashSet();
|
private static final IntSet WATERLOGGED = new IntOpenHashSet();
|
||||||
|
|
||||||
private static final Map<BlockState, String> JAVA_ID_TO_BLOCK_ENTITY_MAP = new HashMap<>();
|
private static final Map<BlockState, String> JAVA_ID_TO_BLOCK_ENTITY_MAP = new HashMap<>();
|
||||||
|
public static final Int2FloatMap JAVA_RUNTIME_ID_TO_HARDNESS = new Int2FloatOpenHashMap();
|
||||||
|
|
||||||
|
|
||||||
private static final int BLOCK_STATE_VERSION = 17760256;
|
private static final int BLOCK_STATE_VERSION = 17760256;
|
||||||
|
|
||||||
|
@ -81,7 +83,9 @@ public class BlockTranslator {
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw new AssertionError("Unable to load Java block mappings", e);
|
throw new AssertionError("Unable to load Java block mappings", e);
|
||||||
}
|
}
|
||||||
TObjectIntMap<CompoundTag> addedStatesMap = new TObjectIntHashMap<>(512, 0.5f, -1);
|
|
||||||
|
Object2IntMap<CompoundTag> addedStatesMap = new Object2IntOpenHashMap<>();
|
||||||
|
|
||||||
List<CompoundTag> paletteList = new ArrayList<>();
|
List<CompoundTag> paletteList = new ArrayList<>();
|
||||||
|
|
||||||
int waterRuntimeId = -1;
|
int waterRuntimeId = -1;
|
||||||
|
@ -94,6 +98,11 @@ public class BlockTranslator {
|
||||||
String javaId = entry.getKey();
|
String javaId = entry.getKey();
|
||||||
BlockState javaBlockState = new BlockState(javaRuntimeId);
|
BlockState javaBlockState = new BlockState(javaRuntimeId);
|
||||||
CompoundTag blockTag = buildBedrockState(entry.getValue());
|
CompoundTag blockTag = buildBedrockState(entry.getValue());
|
||||||
|
// TODO fix this, (no block should have a null hardness)
|
||||||
|
JsonNode hardnessNode = entry.getValue().get("block_hardness");
|
||||||
|
if (hardnessNode != null) {
|
||||||
|
JAVA_RUNTIME_ID_TO_HARDNESS.put(javaRuntimeId, hardnessNode.floatValue());
|
||||||
|
}
|
||||||
|
|
||||||
JAVA_ID_BLOCK_MAP.put(javaId, javaBlockState);
|
JAVA_ID_BLOCK_MAP.put(javaId, javaBlockState);
|
||||||
|
|
||||||
|
@ -118,7 +127,7 @@ public class BlockTranslator {
|
||||||
addedStatesMap.put(blockTag, bedrockRuntimeId);
|
addedStatesMap.put(blockTag, bedrockRuntimeId);
|
||||||
paletteList.add(runtimeTag);
|
paletteList.add(runtimeTag);
|
||||||
} else {
|
} else {
|
||||||
int duplicateRuntimeId = addedStatesMap.get(blockTag);
|
int duplicateRuntimeId = addedStatesMap.getOrDefault(blockTag, -1);
|
||||||
if (duplicateRuntimeId == -1) {
|
if (duplicateRuntimeId == -1) {
|
||||||
GeyserConnector.getInstance().getLogger().debug("Mapping " + javaId + " was not found for bedrock edition!");
|
GeyserConnector.getInstance().getLogger().debug("Mapping " + javaId + " was not found for bedrock edition!");
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -49,11 +49,17 @@ public abstract class BaseInventoryTranslator extends InventoryTranslator{
|
||||||
if (action.getSource().getContainerId() == ContainerId.INVENTORY) {
|
if (action.getSource().getContainerId() == ContainerId.INVENTORY) {
|
||||||
//hotbar
|
//hotbar
|
||||||
if (slotnum >= 9) {
|
if (slotnum >= 9) {
|
||||||
|
// TODO REMOVE
|
||||||
|
System.out.println(slotnum);
|
||||||
return slotnum + this.size - 9;
|
return slotnum + this.size - 9;
|
||||||
} else {
|
} else {
|
||||||
|
// TODO REMOVE
|
||||||
|
System.out.println(slotnum);
|
||||||
return slotnum + this.size + 27;
|
return slotnum + this.size + 27;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// TODO REMOVE
|
||||||
|
System.out.println(slotnum);
|
||||||
return slotnum;
|
return slotnum;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -62,11 +68,17 @@ public abstract class BaseInventoryTranslator extends InventoryTranslator{
|
||||||
if (slot >= this.size) {
|
if (slot >= this.size) {
|
||||||
final int tmp = slot - this.size;
|
final int tmp = slot - this.size;
|
||||||
if (tmp < 27) {
|
if (tmp < 27) {
|
||||||
|
// TODO REMOVE
|
||||||
|
System.out.println(slot);
|
||||||
return tmp + 9;
|
return tmp + 9;
|
||||||
} else {
|
} else {
|
||||||
|
// TODO REMOVE
|
||||||
|
System.out.println(slot);
|
||||||
return tmp - 27;
|
return tmp - 27;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// TODO REMOVE
|
||||||
|
System.out.println(slot);
|
||||||
return slot;
|
return slot;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -32,7 +32,7 @@ import lombok.Getter;
|
||||||
@AllArgsConstructor
|
@AllArgsConstructor
|
||||||
public class ItemEntry {
|
public class ItemEntry {
|
||||||
|
|
||||||
public static ItemEntry AIR = new ItemEntry("minecraft:air", 0, 0, 0);
|
public static ItemEntry AIR = new ItemEntry("minecraft:air", 0, 0, 0/*, "none", "none"*/);
|
||||||
|
|
||||||
private String javaIdentifier;
|
private String javaIdentifier;
|
||||||
private int javaId;
|
private int javaId;
|
||||||
|
@ -40,6 +40,9 @@ public class ItemEntry {
|
||||||
private int bedrockId;
|
private int bedrockId;
|
||||||
private int bedrockData;
|
private int bedrockData;
|
||||||
|
|
||||||
|
//private String toolType;
|
||||||
|
//private String toolTier;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object obj) {
|
public boolean equals(Object obj) {
|
||||||
return obj == this || (obj instanceof ItemEntry && ((ItemEntry) obj).getBedrockId() == this.getBedrockId() && ((ItemEntry) obj).getJavaIdentifier().equals(this.getJavaIdentifier()));
|
return obj == this || (obj instanceof ItemEntry && ((ItemEntry) obj).getBedrockId() == this.getBedrockId() && ((ItemEntry) obj).getJavaIdentifier().equals(this.getJavaIdentifier()));
|
||||||
|
|
|
@ -25,10 +25,22 @@
|
||||||
|
|
||||||
package org.geysermc.connector.network.translators.java.entity.player;
|
package org.geysermc.connector.network.translators.java.entity.player;
|
||||||
|
|
||||||
|
import com.github.steveice10.mc.protocol.data.game.entity.metadata.ItemStack;
|
||||||
import com.github.steveice10.mc.protocol.packet.ingame.server.entity.player.ServerPlayerActionAckPacket;
|
import com.github.steveice10.mc.protocol.packet.ingame.server.entity.player.ServerPlayerActionAckPacket;
|
||||||
|
import com.github.steveice10.opennbt.tag.builtin.CompoundTag;
|
||||||
|
import com.nukkitx.math.vector.Vector3f;
|
||||||
|
import com.nukkitx.protocol.bedrock.data.LevelEventType;
|
||||||
|
import com.nukkitx.protocol.bedrock.packet.LevelEventPacket;
|
||||||
|
import org.geysermc.connector.inventory.Inventory;
|
||||||
|
import org.geysermc.connector.inventory.PlayerInventory;
|
||||||
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.block.BlockTranslator;
|
||||||
|
import org.geysermc.connector.network.translators.item.ItemEntry;
|
||||||
import org.geysermc.connector.utils.ChunkUtils;
|
import org.geysermc.connector.utils.ChunkUtils;
|
||||||
|
import org.geysermc.connector.utils.Toolbox;
|
||||||
|
|
||||||
|
import java.util.Arrays;
|
||||||
|
|
||||||
public class JavaPlayerActionAckTranslator extends PacketTranslator<ServerPlayerActionAckPacket> {
|
public class JavaPlayerActionAckTranslator extends PacketTranslator<ServerPlayerActionAckPacket> {
|
||||||
|
|
||||||
|
@ -38,6 +50,137 @@ public class JavaPlayerActionAckTranslator extends PacketTranslator<ServerPlayer
|
||||||
case FINISH_DIGGING:
|
case FINISH_DIGGING:
|
||||||
ChunkUtils.updateBlock(session, packet.getNewState(), packet.getPosition());
|
ChunkUtils.updateBlock(session, packet.getNewState(), packet.getPosition());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case START_DIGGING: {
|
||||||
|
LevelEventPacket levelEvent = new LevelEventPacket();
|
||||||
|
levelEvent.setType(LevelEventType.BLOCK_START_BREAK);
|
||||||
|
levelEvent.setPosition(Vector3f.from(
|
||||||
|
packet.getPosition().getX(),
|
||||||
|
packet.getPosition().getY(),
|
||||||
|
packet.getPosition().getZ()
|
||||||
|
));
|
||||||
|
float blockHardness = BlockTranslator.JAVA_RUNTIME_ID_TO_HARDNESS.get(packet.getNewState().getId());
|
||||||
|
|
||||||
|
PlayerInventory inventory = session.getInventory();
|
||||||
|
ItemStack[] items = inventory.getItems();
|
||||||
|
System.out.println(Arrays.deepToString(items));
|
||||||
|
for (int i = 0; i != items.length; i++) {
|
||||||
|
System.out.println("(" + i + ") " + items[i]);
|
||||||
|
}
|
||||||
|
int itemHandSlot = inventory.getHeldItemSlot();
|
||||||
|
System.out.println(itemHandSlot);
|
||||||
|
ItemStack item = inventory.getItemInHand();
|
||||||
|
System.out.println(item);
|
||||||
|
/*if (itemStack.getNbt() == null) {
|
||||||
|
itemStack = new ItemStack(itemStack.getId(), itemStack.getAmount(), new CompoundTag(""));
|
||||||
|
}*/
|
||||||
|
//ItemEntry item = Toolbox.ITEM_ENTRIES.get(itemStack.getId());
|
||||||
|
//System.out.println(item);
|
||||||
|
double breakTime = Math.ceil(getBreakTime(blockHardness, packet.getNewState().getId()/*, item*/) * 20);
|
||||||
|
levelEvent.setData((int) (65535 / breakTime));
|
||||||
|
session.getUpstream().sendPacket(levelEvent);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case CANCEL_DIGGING: {
|
||||||
|
LevelEventPacket levelEvent = new LevelEventPacket();
|
||||||
|
levelEvent.setType(LevelEventType.BLOCK_STOP_BREAK);
|
||||||
|
levelEvent.setPosition(Vector3f.from(
|
||||||
|
packet.getPosition().getX(),
|
||||||
|
packet.getPosition().getY(),
|
||||||
|
packet.getPosition().getZ()
|
||||||
|
));
|
||||||
|
levelEvent.setData(0);
|
||||||
|
session.getUpstream().sendPacket(levelEvent);
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private double toolBreakTimeBonus0(
|
||||||
|
/*String toolType,*/ String toolTier/*, boolean isWoolBlock, boolean isCobweb*/) {
|
||||||
|
//if (toolType == ItemTool.TYPE_SWORD) return isCobweb ? 15.0 : 1.0;
|
||||||
|
//if (toolType == ItemTool.TYPE_SHEARS) return isWoolBlock ? 5.0 : 15.0;
|
||||||
|
//if (toolType == ItemTool.TYPE_NONE) return 1.0;
|
||||||
|
switch (toolTier) {
|
||||||
|
case "wooden":
|
||||||
|
return 2.0;
|
||||||
|
case "stone":
|
||||||
|
return 4.0;
|
||||||
|
case "iron":
|
||||||
|
return 6.0;
|
||||||
|
case "diamond":
|
||||||
|
return 8.0;
|
||||||
|
case "golden":
|
||||||
|
return 12.0;
|
||||||
|
default:
|
||||||
|
return 1.0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*private static double speedBonusByEfficiencyLore0(int efficiencyLoreLevel) {
|
||||||
|
if (efficiencyLoreLevel == 0) return 0;
|
||||||
|
return efficiencyLoreLevel * efficiencyLoreLevel + 1;
|
||||||
|
}*/
|
||||||
|
|
||||||
|
/*private static double speedRateByHasteLore0(int hasteLoreLevel) {
|
||||||
|
return 1.0 + (0.2 * hasteLoreLevel);
|
||||||
|
}*/
|
||||||
|
|
||||||
|
/*private static int toolType0(Item item) {
|
||||||
|
if (item.isSword()) return ItemTool.TYPE_SWORD;
|
||||||
|
if (item.isShovel()) return ItemTool.TYPE_SHOVEL;
|
||||||
|
if (item.isPickaxe()) return ItemTool.TYPE_PICKAXE;
|
||||||
|
if (item.isAxe()) return ItemTool.TYPE_AXE;
|
||||||
|
if (item.isShears()) return ItemTool.TYPE_SHEARS;
|
||||||
|
return ItemTool.TYPE_NONE;
|
||||||
|
}*/
|
||||||
|
|
||||||
|
/*private static boolean correctTool0(int blockToolType, Item item) {
|
||||||
|
return (blockToolType == ItemTool.TYPE_SWORD && item.isSword()) ||
|
||||||
|
(blockToolType == ItemTool.TYPE_SHOVEL && item.isShovel()) ||
|
||||||
|
(blockToolType == ItemTool.TYPE_PICKAXE && item.isPickaxe()) ||
|
||||||
|
(blockToolType == ItemTool.TYPE_AXE && item.isAxe()) ||
|
||||||
|
(blockToolType == ItemTool.TYPE_SHEARS && item.isShears()) ||
|
||||||
|
blockToolType == ItemTool.TYPE_NONE;
|
||||||
|
}*/
|
||||||
|
|
||||||
|
//http://minecraft.gamepedia.com/Breaking
|
||||||
|
private double breakTime0(double blockHardness/*, String toolTier*/
|
||||||
|
/*double blockHardness, boolean correctTool, boolean canHarvestWithHand,
|
||||||
|
int blockId, int toolType, String toolTier, int efficiencyLoreLevel, int hasteEffectLevel,
|
||||||
|
boolean insideOfWaterWithoutAquaAffinity, boolean outOfWaterButNotOnGround*/) {
|
||||||
|
double baseTime = (/*(correctTool || canHarvestWithHand)*/true ? 1.5 : 5.0) * blockHardness;
|
||||||
|
double speed = 1.0 / baseTime;
|
||||||
|
//boolean isWoolBlock = blockId == Block.WOOL, isCobweb = blockId == Block.COBWEB;
|
||||||
|
//if (correctTool)
|
||||||
|
//speed *= toolBreakTimeBonus0(toolTier/*toolType, toolTier, isWoolBlock, isCobweb*/);
|
||||||
|
//speed += speedBonusByEfficiencyLore0(efficiencyLoreLevel);
|
||||||
|
//speed *= speedRateByHasteLore0(hasteEffectLevel);
|
||||||
|
//if (insideOfWaterWithoutAquaAffinity) speed *= 0.2;
|
||||||
|
//if (outOfWaterButNotOnGround) speed *= 0.2;
|
||||||
|
return 1.0 / speed;
|
||||||
|
}
|
||||||
|
|
||||||
|
private double getBreakTime(double blockHardness, int blockId/*, ItemEntry item*/) {
|
||||||
|
//Objects.requireNonNull(item, "getBreakTime: Item can not be null");
|
||||||
|
//Objects.requireNonNull(player, "getBreakTime: Player can not be null");
|
||||||
|
//boolean correctTool = correctTool0(getToolType(), item);
|
||||||
|
//boolean canHarvestWithHand = canHarvestWithHand();
|
||||||
|
//int blockId = getId();
|
||||||
|
//int itemToolType = toolType0(item);
|
||||||
|
//int itemTier = item.getTier();
|
||||||
|
//int efficiencyLoreLevel = Optional.ofNullable(item.getEnchantment(Enchantment.ID_EFFICIENCY))
|
||||||
|
// .map(Enchantment::getLevel).orElse(0);
|
||||||
|
//int hasteEffectLevel = Optional.ofNullable(player.getEffect(Effect.HASTE))
|
||||||
|
// .map(Effect::getAmplifier).orElse(0);
|
||||||
|
//boolean insideOfWaterWithoutAquaAffinity = player.isInsideOfWater() &&
|
||||||
|
// Optional.ofNullable(player.getInventory().getHelmet().getEnchantment(Enchantment.ID_WATER_WORKER))
|
||||||
|
// .map(Enchantment::getLevel).map(l -> l >= 1).orElse(false);
|
||||||
|
//boolean outOfWaterButNotOnGround = (!player.isInsideOfWater()) && (!player.isOnGround());
|
||||||
|
//return breakTime0(blockHardness, correctTool, canHarvestWithHand, blockId, itemToolType, itemTier,
|
||||||
|
// efficiencyLoreLevel, hasteEffectLevel, insideOfWaterWithoutAquaAffinity, outOfWaterButNotOnGround);
|
||||||
|
return breakTime0(blockHardness/*, item.getToolTier()*/);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -52,7 +52,7 @@ public class Toolbox {
|
||||||
public static final CompoundTag BIOMES;
|
public static final CompoundTag BIOMES;
|
||||||
public static final ItemData[] CREATIVE_ITEMS;
|
public static final ItemData[] CREATIVE_ITEMS;
|
||||||
|
|
||||||
public static final Collection<StartGamePacket.ItemEntry> ITEMS = new ArrayList<>();
|
public static final List<StartGamePacket.ItemEntry> ITEMS = new ArrayList<>();
|
||||||
|
|
||||||
public static final Int2ObjectMap<ItemEntry> ITEM_ENTRIES = new Int2ObjectOpenHashMap<>();
|
public static final Int2ObjectMap<ItemEntry> ITEM_ENTRIES = new Int2ObjectOpenHashMap<>();
|
||||||
|
|
||||||
|
@ -103,8 +103,12 @@ public class Toolbox {
|
||||||
Iterator<Map.Entry<String, JsonNode>> iterator = items.fields();
|
Iterator<Map.Entry<String, JsonNode>> iterator = items.fields();
|
||||||
while (iterator.hasNext()) {
|
while (iterator.hasNext()) {
|
||||||
Map.Entry<String, JsonNode> entry = iterator.next();
|
Map.Entry<String, JsonNode> entry = iterator.next();
|
||||||
ITEM_ENTRIES.put(itemIndex, new ItemEntry(entry.getKey(), itemIndex,
|
ITEM_ENTRIES.put(itemIndex, new ItemEntry(
|
||||||
entry.getValue().get("bedrock_id").intValue(), entry.getValue().get("bedrock_data").intValue()));
|
entry.getKey(), itemIndex,
|
||||||
|
entry.getValue().get("bedrock_id").intValue(),
|
||||||
|
entry.getValue().get("bedrock_data").intValue()/*,
|
||||||
|
entry.getValue().get("tool_type").textValue(),
|
||||||
|
entry.getValue().get("tool_tier").textValue()*/));
|
||||||
itemIndex++;
|
itemIndex++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue