net.minecraftforge.common.util.BlockSnapshot.java Source code

Java tutorial

Introduction

Here is the source code for net.minecraftforge.common.util.BlockSnapshot.java

Source

/*
 * Minecraft Forge
 * Copyright (c) 2016-2019.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation version 2.1
 * of the License.
 *
 * This library 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 */

package net.minecraftforge.common.util;

import java.lang.ref.WeakReference;
import java.util.Objects;

import net.minecraft.block.BlockState;
import net.minecraft.nbt.CompoundNBT;
import net.minecraft.tileentity.TileEntity;
import net.minecraft.util.ResourceLocation;
import net.minecraft.util.math.BlockPos;
import net.minecraft.world.IWorld;
import net.minecraft.world.World;
import net.minecraft.world.dimension.DimensionType;
import net.minecraftforge.fml.server.ServerLifecycleHooks;
import net.minecraftforge.registries.ForgeRegistries;

import javax.annotation.Nullable;

/**
 * Represents a captured snapshot of a block which will not change
 * automatically.
 * <p>
 * Unlike Block, which only one object can exist per coordinate, BlockSnapshot
 * can exist multiple times for any given Block.
 */
public class BlockSnapshot {
    private static final boolean DEBUG = Boolean
            .parseBoolean(System.getProperty("forge.debugBlockSnapshot", "false"));

    private final BlockPos pos;
    private final int dimId;
    @Nullable
    private BlockState replacedBlock;
    private int flag;
    @Nullable
    private final CompoundNBT nbt;
    @Nullable
    private WeakReference<IWorld> world;
    private final ResourceLocation registryName;
    private final int meta = 0; // TODO BlockSnapshot needs a total refactor for the absence of metadata

    public BlockSnapshot(IWorld world, BlockPos pos, BlockState state) {
        this(world, pos, state, getTileNBT(world.getTileEntity(pos)));
    }

    public BlockSnapshot(IWorld world, BlockPos pos, BlockState state, @Nullable CompoundNBT nbt) {
        this.setWorld(world);
        this.dimId = world.getDimension().getType().getId();
        this.pos = pos.toImmutable();
        this.setReplacedBlock(state);
        this.registryName = state.getBlock().getRegistryName();
        this.setFlag(3);
        this.nbt = nbt;
        if (DEBUG) {
            System.out.printf("Created BlockSnapshot - [World: %s ][Location: %d,%d,%d ][Block: %s ][Meta: %d ]",
                    world.getWorldInfo().getWorldName(), pos.getX(), pos.getY(), pos.getZ(), getRegistryName(),
                    getMeta());
        }
    }

    public BlockSnapshot(IWorld world, BlockPos pos, BlockState state, int flag) {
        this(world, pos, state);
        this.setFlag(flag);
    }

    /**
     * Raw constructor designed for serialization usages.
     */
    public BlockSnapshot(int dimension, BlockPos pos, ResourceLocation registryName, int meta, int flag,
            @Nullable CompoundNBT nbt) {
        this.dimId = dimension;
        this.pos = pos.toImmutable();
        this.setFlag(flag);
        this.registryName = registryName;
        this.nbt = nbt;
    }

    public static BlockSnapshot getBlockSnapshot(IWorld world, BlockPos pos) {
        return new BlockSnapshot(world, pos, world.getBlockState(pos));
    }

    public static BlockSnapshot getBlockSnapshot(IWorld world, BlockPos pos, int flag) {
        return new BlockSnapshot(world, pos, world.getBlockState(pos), flag);
    }

    public static BlockSnapshot readFromNBT(CompoundNBT tag) {
        return new BlockSnapshot(tag.getInt("dimension"),
                new BlockPos(tag.getInt("posX"), tag.getInt("posY"), tag.getInt("posZ")),
                new ResourceLocation(tag.getString("blockMod"), tag.getString("blockName")), tag.getInt("metadata"),
                tag.getInt("flag"), tag.getBoolean("hasTE") ? tag.getCompound("tileEntity") : null);
    }

    @Nullable
    private static CompoundNBT getTileNBT(@Nullable TileEntity te) {
        if (te == null)
            return null;
        CompoundNBT nbt = new CompoundNBT();
        te.write(nbt);
        return nbt;
    }

    public BlockState getCurrentBlock() {
        return getWorld().getBlockState(getPos());
    }

    public IWorld getWorld() {
        IWorld world = this.world != null ? this.world.get() : null;
        if (world == null) {
            world = ServerLifecycleHooks.getCurrentServer().getWorld(DimensionType.getById(getDimId()));
            this.world = new WeakReference<IWorld>(world);
        }
        return world;
    }

    public BlockState getReplacedBlock() {
        if (this.replacedBlock == null) {
            this.replacedBlock = ForgeRegistries.BLOCKS.getValue(getRegistryName()).getStateById(getMeta());
        }
        return this.replacedBlock;
    }

    @Nullable
    public TileEntity getTileEntity() {
        return getNbt() != null ? TileEntity.create(getNbt()) : null;
    }

    public boolean restore() {
        return restore(false);
    }

    public boolean restore(boolean force) {
        return restore(force, true);
    }

    public boolean restore(boolean force, boolean notifyNeighbors) {
        return restoreToLocation(getWorld(), getPos(), force, notifyNeighbors);
    }

    public boolean restoreToLocation(IWorld world, BlockPos pos, boolean force, boolean notifyNeighbors) {
        BlockState current = getCurrentBlock();
        BlockState replaced = getReplacedBlock();

        int flags = notifyNeighbors ? Constants.BlockFlags.DEFAULT : Constants.BlockFlags.BLOCK_UPDATE;

        if (current != replaced) {
            if (force) {
                world.setBlockState(pos, replaced, flags);
            } else {
                return false;
            }
        }

        world.setBlockState(pos, replaced, flags);
        if (world instanceof World)
            ((World) world).notifyBlockUpdate(pos, current, replaced, flags);

        TileEntity te = null;
        if (getNbt() != null) {
            te = world.getTileEntity(pos);
            if (te != null) {
                te.read(getNbt());
                te.markDirty();
            }
        }

        if (DEBUG) {
            System.out.printf(
                    "Restored BlockSnapshot with data [World: %s ][Location: %d,%d,%d ][State: %s ][Block: %s ][TileEntity: %s ][force: %s ][notifyNeighbors: %s]",
                    world.getWorldInfo().getWorldName(), pos.getX(), pos.getY(), pos.getZ(), replaced,
                    replaced.getBlock().delegate.name(), te, force, notifyNeighbors);
        }
        return true;
    }

    public void writeToNBT(CompoundNBT compound) {
        compound.putString("blockMod", getRegistryName().getNamespace());
        compound.putString("blockName", getRegistryName().getPath());
        compound.putInt("posX", getPos().getX());
        compound.putInt("posY", getPos().getY());
        compound.putInt("posZ", getPos().getZ());
        compound.putInt("flag", getFlag());
        compound.putInt("dimension", getDimId());
        compound.putInt("metadata", getMeta());

        compound.putBoolean("hasTE", getNbt() != null);

        if (getNbt() != null) {
            compound.put("tileEntity", getNbt());
        }
    }

    @Override
    public boolean equals(Object obj) {
        if (obj == this)
            return true;
        if (obj == null || getClass() != obj.getClass())
            return false;

        final BlockSnapshot other = (BlockSnapshot) obj;
        if (this.getMeta() != other.getMeta() || this.getDimId() != other.getDimId()
                || !this.getPos().equals(other.getPos()) || !this.getRegistryName().equals(other.getRegistryName())
                || !Objects.equals(this.getNbt(), other.getNbt()))
            return false;
        return true;
    }

    @Override
    public int hashCode() {
        int hash = 7;
        hash = 73 * hash + this.getMeta();
        hash = 73 * hash + this.getDimId();
        hash = 73 * hash + this.getPos().hashCode();
        hash = 73 * hash + this.getRegistryName().hashCode();
        hash = 73 * hash + Objects.hashCode(this.getNbt());
        return hash;
    }

    public BlockPos getPos() {
        return pos;
    }

    public int getDimId() {
        return dimId;
    }

    public void setReplacedBlock(BlockState replacedBlock) {
        this.replacedBlock = replacedBlock;
    }

    public int getFlag() {
        return flag;
    }

    public void setFlag(int flag) {
        this.flag = flag;
    }

    @Nullable
    public CompoundNBT getNbt() {
        return nbt;
    }

    public void setWorld(IWorld world) {
        this.world = new WeakReference<>(world);
    }

    public ResourceLocation getRegistryName() {
        return registryName;
    }

    public int getMeta() {
        return meta;
    }
}