ActuallyAdditions/src/main/java/de/ellpeck/actuallyadditions/mod/tile/TileEntityCrusher.java
Mrbysco ef6e52f8a6 Fix up the Crusher and Double Crusher
Datagen Crushing recipes
2024-03-11 00:55:22 +01:00

291 lines
12 KiB
Java

/*
* This file ("TileEntityGrinder.java") is part of the Actually Additions mod for Minecraft.
* It is created and owned by Ellpeck and distributed
* under the Actually Additions License to be found at
* http://ellpeck.de/actaddlicense
* View the source code at https://github.com/Ellpeck/ActuallyAdditions
*
* © 2015-2017 Ellpeck
*/
package de.ellpeck.actuallyadditions.mod.tile;
import de.ellpeck.actuallyadditions.api.ActuallyAdditionsAPI;
import de.ellpeck.actuallyadditions.mod.AASounds;
import de.ellpeck.actuallyadditions.mod.blocks.ActuallyBlocks;
import de.ellpeck.actuallyadditions.mod.crafting.CrushingRecipe;
import de.ellpeck.actuallyadditions.mod.inventory.CrusherContainer;
import de.ellpeck.actuallyadditions.mod.network.gui.IButtonReactor;
import de.ellpeck.actuallyadditions.mod.util.ItemStackHandlerAA.IAcceptor;
import de.ellpeck.actuallyadditions.mod.util.ItemStackHandlerAA.IRemover;
import de.ellpeck.actuallyadditions.mod.util.StackUtil;
import net.minecraft.core.BlockPos;
import net.minecraft.core.Direction;
import net.minecraft.nbt.CompoundTag;
import net.minecraft.network.chat.Component;
import net.minecraft.sounds.SoundSource;
import net.minecraft.world.MenuProvider;
import net.minecraft.world.entity.player.Inventory;
import net.minecraft.world.entity.player.Player;
import net.minecraft.world.inventory.AbstractContainerMenu;
import net.minecraft.world.item.ItemStack;
import net.minecraft.world.item.crafting.RecipeHolder;
import net.minecraft.world.level.Level;
import net.minecraft.world.level.block.entity.BlockEntity;
import net.minecraft.world.level.block.entity.BlockEntityType;
import net.minecraft.world.level.block.state.BlockState;
import net.minecraft.world.level.block.state.properties.BlockStateProperties;
import net.neoforged.neoforge.energy.IEnergyStorage;
import javax.annotation.Nullable;
import java.util.Optional;
public class TileEntityCrusher extends TileEntityInventoryBase implements IButtonReactor, MenuProvider {
public static final int SLOT_INPUT_1 = 0;
public static final int SLOT_OUTPUT_1_1 = 1;
public static final int SLOT_OUTPUT_1_2 = 2;
public static final int SLOT_INPUT_2 = 3;
public static final int SLOT_OUTPUT_2_1 = 4;
public static final int SLOT_OUTPUT_2_2 = 5;
public static final int ENERGY_USE = 40;
public final CustomEnergyStorage storage = new CustomEnergyStorage(60000, 100, 0);
public int firstCrushTime;
public int secondCrushTime;
public boolean isDouble;
public boolean isAutoSplit;
private int lastEnergy;
private int lastFirstCrush;
private int lastSecondCrush;
private boolean lastAutoSplit;
private boolean lastCrushed;
public TileEntityCrusher(BlockEntityType<?> type, BlockPos pos, BlockState state, int slots) {
super(type, pos, state, slots);
}
public TileEntityCrusher(BlockPos pos, BlockState state) {
super(ActuallyBlocks.CRUSHER.getTileEntityType(), pos, state, 3);
this.isDouble = false;
}
@Override
public void writeSyncableNBT(CompoundTag compound, NBTType type) {
if (type != NBTType.SAVE_BLOCK) {
compound.putInt("FirstCrushTime", this.firstCrushTime);
compound.putInt("SecondCrushTime", this.secondCrushTime);
compound.putBoolean("IsAutoSplit", this.isAutoSplit);
}
this.storage.writeToNBT(compound);
super.writeSyncableNBT(compound, type);
}
@Override
public void readSyncableNBT(CompoundTag compound, NBTType type) {
if (type != NBTType.SAVE_BLOCK) {
this.firstCrushTime = compound.getInt("FirstCrushTime");
this.secondCrushTime = compound.getInt("SecondCrushTime");
this.isAutoSplit = compound.getBoolean("IsAutoSplit");
}
this.storage.readFromNBT(compound);
super.readSyncableNBT(compound, type);
}
public static <T extends BlockEntity> void clientTick(Level level, BlockPos pos, BlockState state, T t) {
if (t instanceof TileEntityCrusher tile) {
tile.clientTick();
}
}
public static <T extends BlockEntity> void serverTick(Level level, BlockPos pos, BlockState state, T t) {
if (t instanceof TileEntityCrusher tile) {
tile.serverTick();
}
}
@Override
protected void serverTick() {
super.serverTick();
if (isDouble && isAutoSplit) {
TileEntityPoweredFurnace.autoSplit(inv, SLOT_INPUT_1, SLOT_INPUT_2);
}
boolean crushed = false;
boolean canCrushOnFirst = canCrushOn(SLOT_INPUT_1, SLOT_OUTPUT_1_1, SLOT_OUTPUT_1_2);
boolean canCrushOnSecond = false;
if (isDouble) {
canCrushOnSecond = canCrushOn(SLOT_INPUT_2, SLOT_OUTPUT_2_1, SLOT_OUTPUT_2_2);
}
boolean shouldPlaySound = false;
if (canCrushOnFirst) {
if (storage.getEnergyStored() >= ENERGY_USE) {
if (firstCrushTime % 20 == 0) {
shouldPlaySound = true;
}
firstCrushTime++;
if (firstCrushTime >= getMaxCrushTime()) {
finishCrushing(SLOT_INPUT_1, SLOT_OUTPUT_1_1, SLOT_OUTPUT_1_2);
firstCrushTime = 0;
}
storage.extractEnergyInternal(ENERGY_USE, false);
}
crushed = storage.getEnergyStored() >= ENERGY_USE;
} else {
firstCrushTime = 0;
}
if (isDouble) {
if (canCrushOnSecond) {
if (storage.getEnergyStored() >= ENERGY_USE) {
if (secondCrushTime % 20 == 0) {
shouldPlaySound = true;
}
secondCrushTime++;
if (secondCrushTime >= getMaxCrushTime()) {
finishCrushing(SLOT_INPUT_2, SLOT_OUTPUT_2_1, SLOT_OUTPUT_2_2);
secondCrushTime = 0;
}
storage.extractEnergyInternal(ENERGY_USE, false);
}
crushed = storage.getEnergyStored() >= ENERGY_USE;
} else {
secondCrushTime = 0;
}
}
boolean current = getBlockState().getValue(BlockStateProperties.LIT);
boolean changeTo = current;
if (lastCrushed != crushed) {
changeTo = crushed;
}
if (isRedstonePowered) {
changeTo = true;
}
if (!crushed && !isRedstonePowered) {
changeTo = false;
}
if (changeTo != current) {
level.setBlockAndUpdate(getBlockPos(), getBlockState().setValue(BlockStateProperties.LIT, changeTo));
}
lastCrushed = crushed;
if ((lastEnergy != storage.getEnergyStored() || lastFirstCrush != firstCrushTime || lastSecondCrush != secondCrushTime || isAutoSplit != lastAutoSplit) && sendUpdateWithInterval()) {
lastEnergy = storage.getEnergyStored();
lastFirstCrush = firstCrushTime;
lastSecondCrush = secondCrushTime;
lastAutoSplit = isAutoSplit;
}
if (shouldPlaySound) {
level.playSound(null, getBlockPos().getX(), getBlockPos().getY(), getBlockPos().getZ(), AASounds.CRUSHER.get(), SoundSource.BLOCKS, 0.025F, 1.0F);
}
}
@Override
public IAcceptor getAcceptor() {
return (slot, stack, automation) -> !automation || (slot == SLOT_INPUT_1 || slot == SLOT_INPUT_2); /*CrusherRecipeRegistry.getRecipeFromInput(stack) != null*/ //TODO
}
@Override
public IRemover getRemover() {
return (slot, automation) -> !automation || slot == SLOT_OUTPUT_1_1 || slot == SLOT_OUTPUT_1_2 || slot == SLOT_OUTPUT_2_1 || slot == SLOT_OUTPUT_2_2;
}
public static Optional<RecipeHolder<CrushingRecipe>> getRecipeForInput(ItemStack itemStack) {
return ActuallyAdditionsAPI.CRUSHER_RECIPES.stream().filter($ -> $.value().matches(itemStack)).findFirst();
}
public boolean canCrushOn(int theInput, int theFirstOutput, int theSecondOutput) {
ItemStack inputStack = this.inv.getStackInSlot(theInput);
if (!inputStack.isEmpty()) {
Optional<RecipeHolder<CrushingRecipe>> recipeOpt = getRecipeForInput(inputStack);
if (recipeOpt.isEmpty()) {
return false;
}
CrushingRecipe recipe = recipeOpt.get().value();
ItemStack outputOne = recipe.getOutputOne();
ItemStack outputTwo = recipe.getOutputTwo();
if (!outputOne.isEmpty()) {
return (this.inv.getStackInSlot(theFirstOutput).isEmpty() || ItemStack.isSameItem(this.inv.getStackInSlot(theFirstOutput), outputOne) && this.inv.getStackInSlot(theFirstOutput).getCount() <= this.inv.getStackInSlot(theFirstOutput).getMaxStackSize() - outputOne.getCount()) && (outputTwo.isEmpty() || this.inv.getStackInSlot(theSecondOutput).isEmpty() || ItemStack.isSameItem(this.inv.getStackInSlot(theSecondOutput), outputTwo) && this.inv.getStackInSlot(theSecondOutput).getCount() <= this.inv.getStackInSlot(theSecondOutput).getMaxStackSize() - outputTwo.getCount());
}
}
return false;
}
private int getMaxCrushTime() {
return this.isDouble
? 150
: 100;
}
public void finishCrushing(int theInput, int theFirstOutput, int theSecondOutput) {
Optional<RecipeHolder<CrushingRecipe>> recipeOpt = getRecipeForInput(this.inv.getStackInSlot(theInput));
if (recipeOpt.isEmpty()) {
return;
}
CrushingRecipe recipe = recipeOpt.get().value();
ItemStack outputOne = recipe.getOutputOne();
if (!outputOne.isEmpty()) {
if (this.inv.getStackInSlot(theFirstOutput).isEmpty()) {
this.inv.setStackInSlot(theFirstOutput, outputOne.copy());
} else if (this.inv.getStackInSlot(theFirstOutput).getItem() == outputOne.getItem()) {
this.inv.setStackInSlot(theFirstOutput, StackUtil.grow(this.inv.getStackInSlot(theFirstOutput), outputOne.getCount()));
}
}
ItemStack outputTwo = recipe.getOutputTwo();
if (!outputTwo.isEmpty()) {
float rand = this.level.random.nextFloat();
if (rand <= recipe.getSecondChance()) {
if (this.inv.getStackInSlot(theSecondOutput).isEmpty()) {
this.inv.setStackInSlot(theSecondOutput, outputTwo.copy());
} else if (this.inv.getStackInSlot(theSecondOutput).getItem() == outputTwo.getItem()) {
this.inv.setStackInSlot(theSecondOutput, StackUtil.grow(this.inv.getStackInSlot(theSecondOutput), outputTwo.getCount()));
}
}
}
this.inv.getStackInSlot(theInput).shrink(1);
}
public int getEnergyScaled(int i) {
return this.storage.getEnergyStored() * i / this.storage.getMaxEnergyStored();
}
public int getFirstTimeToScale(int i) {
return this.firstCrushTime * i / this.getMaxCrushTime();
}
public int getSecondTimeToScale(int i) {
return this.secondCrushTime * i / this.getMaxCrushTime();
}
@Override
public void onButtonPressed(int buttonID, Player player) {
if (buttonID == 0) {
this.isAutoSplit = !this.isAutoSplit;
this.setChanged();
}
}
@Override
public IEnergyStorage getEnergyStorage(Direction facing) {
return this.storage;
}
@Override
public Component getDisplayName() {
return Component.translatable("container.actuallyadditions.crusher");
}
@Nullable
@Override
public AbstractContainerMenu createMenu(int windowId, Inventory playerInventory, Player player) {
return new CrusherContainer(windowId, playerInventory, this);
}
}