All Downloads are FREE. Search and download functionalities are using the official Maven repository.

cn.nukkit.blockentity.BlockEntity Maven / Gradle / Ivy

There is a newer version: 1.20.40-r1
Show newest version
package cn.nukkit.blockentity;

import cn.nukkit.Server;
import cn.nukkit.api.*;
import cn.nukkit.block.Block;
import cn.nukkit.block.BlockID;
import cn.nukkit.level.Position;
import cn.nukkit.level.format.FullChunk;
import cn.nukkit.math.Vector3;
import cn.nukkit.nbt.tag.CompoundTag;
import cn.nukkit.scheduler.Task;
import cn.nukkit.utils.ChunkException;
import co.aikar.timings.Timing;
import co.aikar.timings.Timings;
import com.google.common.collect.BiMap;
import com.google.common.collect.HashBiMap;
import lombok.extern.log4j.Log4j2;

import javax.annotation.Nullable;
import java.lang.reflect.Constructor;
import java.util.ArrayList;
import java.util.List;

/**
 * @author MagicDroidX
 */
@Log4j2
public abstract class BlockEntity extends Position {
    //WARNING: DO NOT CHANGE ANY NAME HERE, OR THE CLIENT WILL CRASH
    public static final String CHEST = "Chest";
    public static final String ENDER_CHEST = "EnderChest";
    public static final String FURNACE = "Furnace";
    @PowerNukkitOnly public static final String BLAST_FURNACE = "BlastFurnace";
    @PowerNukkitOnly
    public static final String SMOKER = "Smoker";
    public static final String SIGN = "Sign";
    public static final String MOB_SPAWNER = "MobSpawner";
    public static final String ENCHANT_TABLE = "EnchantTable";
    public static final String SKULL = "Skull";
    public static final String FLOWER_POT = "FlowerPot";
    public static final String BREWING_STAND = "BrewingStand";
    public static final String DAYLIGHT_DETECTOR = "DaylightDetector";
    public static final String MUSIC = "Music";
    public static final String ITEM_FRAME = "ItemFrame";
    @PowerNukkitXOnly
    @Since("1.19.60-r1")
    public static final String GLOW_ITEM_FRAME = "GlowItemFrame";
    public static final String CAULDRON = "Cauldron";
    public static final String BEACON = "Beacon";
    public static final String PISTON_ARM = "PistonArm";
    public static final String MOVING_BLOCK = "MovingBlock";
    public static final String COMPARATOR = "Comparator";
    public static final String HOPPER = "Hopper";
    public static final String BED = "Bed";
    public static final String JUKEBOX = "Jukebox";
    public static final String SHULKER_BOX = "ShulkerBox";
    public static final String BANNER = "Banner";
    @PowerNukkitOnly public static final String LECTERN = "Lectern";
    @PowerNukkitOnly public static final String BEEHIVE = "Beehive";
    @PowerNukkitOnly public static final String CONDUIT = "Conduit";
    @PowerNukkitOnly public static final String BARREL = "Barrel";
    @PowerNukkitOnly public static final String CAMPFIRE = "Campfire";
    @PowerNukkitOnly public static final String BELL = "Bell";
    @PowerNukkitOnly public static final String DISPENSER = "Dispenser";
    @PowerNukkitOnly public static final String DROPPER = "Dropper";
    @PowerNukkitOnly @Since("1.4.0.0-PN") public static final String NETHER_REACTOR = "NetherReactor";
    @PowerNukkitOnly @Since("1.4.0.0-PN") public static final String LODESTONE = "Lodestone";
    @PowerNukkitOnly @Since("1.4.0.0-PN") public static final String TARGET = "Target";
    @PowerNukkitOnly @Since("FUTURE") public static final String END_PORTAL = "EndPortal";
    @PowerNukkitOnly @Since("FUTURE") public static final String END_GATEWAY = "EndGateway";
    @PowerNukkitOnly @Since("1.6.0.0-PNX") public static final String COMMAND_BLOCK = "CommandBlock";
    @PowerNukkitOnly @Since("1.6.0.0-PNX") public static final String SCULK_SENSOR = "SculkSensor";
    @PowerNukkitOnly @Since("1.6.0.0-PNX") public static final String SCULK_CATALYST = "SculkCatalyst";
    @PowerNukkitOnly
    @Since("1.6.0.0-PNX")
    public static final String SCULK_SHRIEKER = "SculkShrieker";
    @PowerNukkitXOnly
    @Since("1.19.21-r2")
    public static final String STRUCTURE_BLOCK = "StructureBlock";

    public static long count = 1;

    private static final BiMap> knownBlockEntities = HashBiMap.create(35);

    public FullChunk chunk;
    public String name;
    public long id;

    public boolean movable;

    public boolean closed = false;
    public CompoundTag namedTag;
    @Deprecated @DeprecationDetails(since = "1.3.1.2-PN", reason = "Not necessary and causes slowdown")
    @PowerNukkitDifference(info = "Not updated anymore", since = "1.3.1.2-PN")
    protected long lastUpdate;
    protected Server server;
    protected Timing timing;

    public BlockEntity(FullChunk chunk, CompoundTag nbt) {
        if (chunk == null || chunk.getProvider() == null) {
            throw new ChunkException("Invalid garbage Chunk given to Block Entity");
        }

        this.timing = Timings.getBlockEntityTiming(this);
        this.server = chunk.getProvider().getLevel().getServer();
        this.chunk = chunk;
        this.setLevel(chunk.getProvider().getLevel());
        this.namedTag = nbt;
        this.name = "";
        this.id = BlockEntity.count++;
        this.x = this.namedTag.getInt("x");
        this.y = this.namedTag.getInt("y");
        this.z = this.namedTag.getInt("z");

        if (namedTag.contains("isMovable")) {
            this.movable = this.namedTag.getBoolean("isMovable");
        } else {
            this.movable = true;
            namedTag.putBoolean("isMovable", true);
        }

        this.initBlockEntity();
        
        if (closed) {
            throw new IllegalStateException("Could not create the entity "+getClass().getName()+", the initializer closed it on construction.");
        }
        
        this.chunk.addBlockEntity(this);
        this.getLevel().addBlockEntity(this);
    }

    @PowerNukkitXInternal
    public static void init() {
        registerBlockEntity(FURNACE, BlockEntityFurnace.class);
        registerBlockEntity(CHEST, BlockEntityChest.class);
        registerBlockEntity(SIGN, BlockEntitySign.class);
        registerBlockEntity(ENCHANT_TABLE, BlockEntityEnchantTable.class);
        registerBlockEntity(SKULL, BlockEntitySkull.class);
        registerBlockEntity(FLOWER_POT, BlockEntityFlowerPot.class);
        registerBlockEntity(BREWING_STAND, BlockEntityBrewingStand.class);
        registerBlockEntity(ITEM_FRAME, BlockEntityItemFrame.class);
        registerBlockEntity(CAULDRON, BlockEntityCauldron.class);
        registerBlockEntity(ENDER_CHEST, BlockEntityEnderChest.class);
        registerBlockEntity(BEACON, BlockEntityBeacon.class);
        registerBlockEntity(PISTON_ARM, BlockEntityPistonArm.class);
        registerBlockEntity(COMPARATOR, BlockEntityComparator.class);
        registerBlockEntity(HOPPER, BlockEntityHopper.class);
        registerBlockEntity(BED, BlockEntityBed.class);
        registerBlockEntity(JUKEBOX, BlockEntityJukebox.class);
        registerBlockEntity(SHULKER_BOX, BlockEntityShulkerBox.class);
        registerBlockEntity(BANNER, BlockEntityBanner.class);
        registerBlockEntity(MUSIC, BlockEntityMusic.class);
        registerBlockEntity(LECTERN, BlockEntityLectern.class);
        registerBlockEntity(BLAST_FURNACE, BlockEntityBlastFurnace.class);
        registerBlockEntity(SMOKER, BlockEntitySmoker.class);
        registerBlockEntity(BEEHIVE, BlockEntityBeehive.class);
        registerBlockEntity(CONDUIT, BlockEntityConduit.class);
        registerBlockEntity(BARREL, BlockEntityBarrel.class);
        registerBlockEntity(CAMPFIRE, BlockEntityCampfire.class);
        registerBlockEntity(BELL, BlockEntityBell.class);
        registerBlockEntity(DAYLIGHT_DETECTOR, BlockEntityDaylightDetector.class);
        registerBlockEntity(DISPENSER, BlockEntityDispenser.class);
        registerBlockEntity(DROPPER, BlockEntityDropper.class);
        registerBlockEntity(MOVING_BLOCK, BlockEntityMovingBlock.class);
        registerBlockEntity(NETHER_REACTOR, BlockEntityNetherReactor.class);
        registerBlockEntity(LODESTONE, BlockEntityLodestone.class);
        registerBlockEntity(TARGET, BlockEntityTarget.class);
        registerBlockEntity(END_PORTAL, BlockEntityEndPortal.class);
        registerBlockEntity(END_GATEWAY, BlockEntityEndGateway.class);
        //powernukkitx only
        registerBlockEntity(COMMAND_BLOCK, BlockEntityCommandBlock.class);
        registerBlockEntity(SCULK_SENSOR, BlockEntitySculkSensor.class);
        registerBlockEntity(SCULK_CATALYST, BlockEntitySculkCatalyst.class);
        registerBlockEntity(SCULK_SHRIEKER, BlockEntitySculkShrieker.class);
        registerBlockEntity(STRUCTURE_BLOCK, BlockEntityStructBlock.class);
        registerBlockEntity(GLOW_ITEM_FRAME, BlockEntityGlowItemFrame.class);
    }

    protected void initBlockEntity() {
        loadNBT();
    }

    @PowerNukkitOnly
    public static BlockEntity createBlockEntity(String type, Position position, Object... args) {
        return createBlockEntity(type, position, BlockEntity.getDefaultCompound(position, type), args);
    }

    @PowerNukkitOnly
    public static BlockEntity createBlockEntity(String type, Position pos, CompoundTag nbt, Object... args) {
        return createBlockEntity(type, pos.getLevel().getChunk(pos.getFloorX() >> 4, pos.getFloorZ() >> 4), nbt, args);
    }

    public static BlockEntity createBlockEntity(String type, FullChunk chunk, CompoundTag nbt, Object... args) {
        BlockEntity blockEntity = null;

        Class clazz = knownBlockEntities.get(type);
        if (clazz != null) {
            List exceptions = null;

            for (Constructor constructor : clazz.getConstructors()) {
                if (blockEntity != null) {
                    break;
                }

                if (constructor.getParameterCount() != (args == null ? 2 : args.length + 2)) {
                    continue;
                }

                try {
                    if (args == null || args.length == 0) {
                        blockEntity = (BlockEntity) constructor.newInstance(chunk, nbt);
                    } else {
                        Object[] objects = new Object[args.length + 2];

                        objects[0] = chunk;
                        objects[1] = nbt;
                        System.arraycopy(args, 0, objects, 2, args.length);
                        blockEntity = (BlockEntity) constructor.newInstance(objects);

                    }
                } catch (Exception e) {
                    if (exceptions == null) {
                        exceptions = new ArrayList<>();
                    }
                    exceptions.add(e);
                }

            }
            if (blockEntity == null) {
                Exception cause = new IllegalArgumentException("Could not create a block entity of type "+type, exceptions != null && exceptions.size() > 0? exceptions.get(0) : null);
                if (exceptions != null && exceptions.size() > 1) {
                    for (int i = 1; i < exceptions.size(); i++) {
                        cause.addSuppressed(exceptions.get(i));
                    }
                }
                log.error("Could not create a block entity of type {} with {} args", type, args == null? 0 : args.length, cause);
            }
        } else {
            log.debug("Block entity type {} is unknown", type);
        }


        return blockEntity;
    }

    public static boolean registerBlockEntity(String name, Class c) {
        if (c == null) {
            return false;
        }

        knownBlockEntities.put(name, c);
        return true;
    }

    public final String getSaveId() {
        return knownBlockEntities.inverse().get(getClass());
    }

    public long getId() {
        return id;
    }

    /**
     * 存储方块实体数据到namedtag
     */
    public void saveNBT() {
        this.namedTag.putString("id", this.getSaveId());
        this.namedTag.putInt("x", (int) this.getX());
        this.namedTag.putInt("y", (int) this.getY());
        this.namedTag.putInt("z", (int) this.getZ());
        this.namedTag.putBoolean("isMovable", this.movable);
    }

    /**
     * 从方块实体的namedtag中读取数据
     */
    @PowerNukkitXOnly
    @Since("1.19.60-r1")
    public void loadNBT() {}

    public CompoundTag getCleanedNBT() {
        this.saveNBT();
        CompoundTag tag = this.namedTag.clone();
        tag.remove("x").remove("y").remove("z").remove("id");
        if (tag.getTags().size() > 0) {
            return tag;
        } else {
            return null;
        }
    }

    public Block getBlock() {
        return this.getLevelBlock();
    }

    public abstract boolean isBlockEntityValid();

    public boolean onUpdate() {
        return false;
    }

    public final void scheduleUpdate() {
        this.level.scheduleBlockEntityUpdate(this);
    }

    public void close() {
        if (!this.closed) {
            this.closed = true;
            if (this.chunk != null) {
                this.chunk.removeBlockEntity(this);
            }
            if (this.level != null) {
                this.level.removeBlockEntity(this);
            }
            this.level = null;
        }
    }

    public void onBreak() {

    }

    @PowerNukkitOnly
    public void onBreak(boolean isSilkTouch) {
        onBreak();
    }

    public void setDirty() {
        chunk.setChanged();

        if (this.getLevelBlock().getId() != BlockID.AIR) {
            getLevel().getServer().getScheduler().scheduleTask(new Task() {
                @Override
                public void onRun(int currentTick) {
                    if (isValid() && isBlockEntityValid()) {
                        getLevel().updateComparatorOutputLevelSelective(BlockEntity.this, isObservable());
                    }
                }
            });
        }
    }

    /**
     * Indicates if an observer blocks that are looking at this block should blink when {@link #setDirty()} is called.
     */
    @PowerNukkitOnly
    @Since("1.4.0.0-PN")
    public boolean isObservable() {
        return true;
    }

    public String getName() {
        return name;
    }

    public boolean isMovable() {
        return movable;
    }

    public static CompoundTag getDefaultCompound(Vector3 pos, String id) {
        return new CompoundTag()
                .putString("id", id)
                .putInt("x", pos.getFloorX())
                .putInt("y", pos.getFloorY())
                .putInt("z", pos.getFloorZ());
    }

    @PowerNukkitOnly
    @Nullable
    @Override
    public final BlockEntity getLevelBlockEntity() {
        return super.getLevelBlockEntity();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy