This repository has been archived on 2023-11-26. You can view files and clone it, but cannot push or open issues or pull requests.
EnergonRelics/src/main/java/com/thebrokenrail/energonrelics/energy/core/EnergyProviderBlockEntity.java

195 lines
6.2 KiB
Java
Raw Normal View History

2020-07-13 20:37:21 +00:00
package com.thebrokenrail.energonrelics.energy.core;
import com.thebrokenrail.energonrelics.config.HardcodedConfig;
2020-07-13 20:37:21 +00:00
import com.thebrokenrail.energonrelics.EnergonRelics;
import com.thebrokenrail.energonrelics.component.NetworkComponent;
import com.thebrokenrail.energonrelics.energy.core.util.Action;
import com.thebrokenrail.energonrelics.energy.core.util.EnergyProvider;
2020-07-21 02:39:30 +00:00
import com.thebrokenrail.energonrelics.util.BlockPosWithDimension;
2020-07-13 20:37:21 +00:00
import net.fabricmc.fabric.api.block.entity.BlockEntityClientSerializable;
import net.minecraft.block.BlockState;
import net.minecraft.block.Blocks;
import net.minecraft.block.entity.BlockEntity;
import net.minecraft.block.entity.BlockEntityType;
import net.minecraft.item.ItemStack;
import net.minecraft.nbt.CompoundTag;
import net.minecraft.server.world.ServerWorld;
import net.minecraft.util.Tickable;
import net.minecraft.util.math.BlockPos;
import net.minecraft.util.math.Vec3d;
2020-07-21 02:39:30 +00:00
import net.minecraft.util.registry.RegistryKey;
2020-07-19 19:12:39 +00:00
import net.minecraft.world.World;
2020-07-13 20:37:21 +00:00
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Objects;
public class EnergyProviderBlockEntity extends BlockEntity implements EnergyProvider, BlockEntityClientSerializable, Tickable {
2020-07-22 23:55:32 +00:00
private static volatile boolean isTicking = false;
2020-07-13 20:37:21 +00:00
private static final List<EnergyProviderBlockEntity> scheduledTicks = new ArrayList<>();
2020-07-22 23:55:32 +00:00
private static final List<EnergyProviderBlockEntity> scheduledTicksNext = new ArrayList<>();
2020-07-13 20:37:21 +00:00
public static void tickScheduled() {
2020-07-22 23:55:32 +00:00
isTicking = true;
2020-07-13 20:37:21 +00:00
Collections.shuffle(scheduledTicks);
for (EnergyProviderBlockEntity provider : scheduledTicks) {
provider.serverTick();
}
scheduledTicks.clear();
2020-07-22 23:55:32 +00:00
isTicking = false;
for (EnergyProviderBlockEntity entity : scheduledTicksNext) {
entity.schedule();
}
scheduledTicksNext.clear();
2020-07-13 20:37:21 +00:00
}
2020-07-22 23:51:42 +00:00
private void schedule() {
2020-07-22 23:55:32 +00:00
List<EnergyProviderBlockEntity> list;
if (isTicking) {
list = scheduledTicksNext;
} else {
list = scheduledTicks;
}
if (!list.contains(this)) {
list.add(this);
2020-07-22 23:51:42 +00:00
}
}
2020-07-13 20:37:21 +00:00
public EnergyProviderBlockEntity(BlockEntityType<?> type) {
super(type);
}
@Override
public final void addPropagatedAction(Action.PropagatedAction action) {
if (isEnergyProvider()) {
2020-07-22 18:07:29 +00:00
if (!completedActions.contains(action)) {
handlePropagatedAction(action);
completedActions.add(action);
}
2020-07-13 20:37:21 +00:00
} else {
throw new UnsupportedOperationException();
}
}
@Override
public final boolean isNetwork(int network) {
return isEnergyProvider() && EnergonRelics.NETWORK_CHIP_ITEM.getID(stack) == network;
}
@Override
public final boolean isWithinDistance(Vec3d pos) {
return getPos().isWithinDistance(pos, HardcodedConfig.POWER_RANGE);
2020-07-13 20:37:21 +00:00
}
2020-07-21 02:39:30 +00:00
@Override
public RegistryKey<World> getRegistryKey() {
return Objects.requireNonNull(getWorld()).getRegistryKey();
}
2020-07-13 20:37:21 +00:00
private ItemStack stack = ItemStack.EMPTY;
@Override
public CompoundTag toTag(CompoundTag tag) {
super.toTag(tag);
if (isEnergyProvider()) {
tag.put("NetworkChip", stack.toTag(new CompoundTag()));
}
return tag;
}
@Override
public void fromTag(BlockState state, CompoundTag tag) {
super.fromTag(state, tag);
if (isEnergyProvider()) {
stack = ItemStack.fromTag(tag.getCompound("NetworkChip"));
} else {
stack = ItemStack.EMPTY;
}
}
public boolean isEnergyProvider() {
return false;
}
2020-07-22 18:07:29 +00:00
private final List<Action.PropagatedAction> completedActions = new ArrayList<>();
2020-07-13 20:37:21 +00:00
protected void handlePropagatedAction(Action.PropagatedAction action) {
2020-07-22 23:51:42 +00:00
if (!isEnergyProvider()) {
throw new UnsupportedOperationException();
} else {
schedule();
}
2020-07-13 20:37:21 +00:00
}
private void tickPropagatedActions() {
2020-07-22 18:07:29 +00:00
completedActions.clear();
2020-07-13 20:37:21 +00:00
if (isEnergyProvider()) {
NetworkComponent component = NetworkComponent.getInstance((ServerWorld) Objects.requireNonNull(getWorld()));
2020-07-21 02:39:30 +00:00
List<BlockPosWithDimension> sources = component.getSourcePos(EnergonRelics.NETWORK_CHIP_ITEM.getID(stack));
if (!sources.contains(new BlockPosWithDimension(getPos(), getWorld().getRegistryKey()))) {
takeStack(getWorld());
2020-07-13 20:37:21 +00:00
}
}
}
protected void serverTick() {
tickPropagatedActions();
}
@Override
public final void tick() {
if (hasWorld() && !Objects.requireNonNull(getWorld()).isClient()) {
2020-07-22 23:51:42 +00:00
schedule();
2020-07-13 20:37:21 +00:00
}
}
2020-07-19 19:12:39 +00:00
private static void setEnergyProviderSource(World world, BlockPos pos, ItemStack stack, boolean remove) {
ServerWorld serverWorld = (ServerWorld) world;
NetworkComponent component = NetworkComponent.getInstance(serverWorld);
2020-07-21 02:39:30 +00:00
BlockPosWithDimension newPos = new BlockPosWithDimension(pos, world.getRegistryKey());
2020-07-19 19:12:39 +00:00
if (remove) {
2020-07-21 02:39:30 +00:00
component.removeSource(EnergonRelics.NETWORK_CHIP_ITEM.getID(stack), newPos);
2020-07-19 19:12:39 +00:00
} else {
2020-07-21 02:39:30 +00:00
component.addSource(EnergonRelics.NETWORK_CHIP_ITEM.getID(stack), newPos);
2020-07-19 19:12:39 +00:00
}
}
2020-07-21 02:39:30 +00:00
public ItemStack takeStack(World world) {
2020-07-13 20:37:21 +00:00
ItemStack newStack = stack.copy();
2020-07-21 02:39:30 +00:00
setEnergyProviderSource(world, getPos(), newStack, true);
2020-07-13 20:37:21 +00:00
stack = ItemStack.EMPTY;
markDirty();
return newStack;
}
2020-07-21 02:39:30 +00:00
public void placeStack(ItemStack newStack, World world) {
setEnergyProviderSource(world, getPos(), newStack, false);
2020-07-13 20:37:21 +00:00
stack = newStack.copy();
markDirty();
}
public boolean hasStack() {
return !stack.isEmpty();
}
@Override
public void fromClientTag(CompoundTag compoundTag) {
fromTag(Blocks.AIR.getDefaultState(), compoundTag);
}
@Override
public CompoundTag toClientTag(CompoundTag compoundTag) {
return toTag(compoundTag);
}
@Override
public void markDirty() {
super.markDirty();
2020-07-21 02:39:30 +00:00
if (hasWorld()) {
sync();
}
2020-07-13 20:37:21 +00:00
}
}