2017-10-04 17:05:04 +02:00
|
|
|
/*
|
|
|
|
* This file is part of Industrial Wires.
|
2018-02-28 21:06:33 +01:00
|
|
|
* Copyright (C) 2016-2018 malte0811
|
2017-10-04 17:05:04 +02:00
|
|
|
* Industrial Wires is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
* Industrial Wires 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 General Public License for more details.
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with Industrial Wires. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package malte0811.industrialWires.converter;
|
|
|
|
|
|
|
|
import blusunrize.immersiveengineering.api.MultiblockHandler;
|
|
|
|
import blusunrize.immersiveengineering.api.crafting.IngredientStack;
|
|
|
|
import malte0811.industrialWires.IndustrialWires;
|
|
|
|
import malte0811.industrialWires.blocks.converter.MechanicalMBBlockType;
|
|
|
|
import malte0811.industrialWires.blocks.converter.TileEntityMultiblockConverter;
|
|
|
|
import malte0811.industrialWires.util.LocalSidedWorld;
|
2018-03-23 16:27:32 +01:00
|
|
|
import malte0811.industrialWires.util.MiscUtils;
|
2017-10-04 17:05:04 +02:00
|
|
|
import net.minecraft.block.state.IBlockState;
|
|
|
|
import net.minecraft.entity.player.EntityPlayer;
|
|
|
|
import net.minecraft.item.ItemStack;
|
|
|
|
import net.minecraft.tileentity.TileEntity;
|
|
|
|
import net.minecraft.util.EnumFacing;
|
|
|
|
import net.minecraft.util.math.BlockPos;
|
|
|
|
import net.minecraft.world.World;
|
|
|
|
|
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.List;
|
|
|
|
import java.util.function.BiConsumer;
|
|
|
|
import java.util.function.Consumer;
|
|
|
|
|
|
|
|
import static malte0811.industrialWires.blocks.converter.MechanicalMBBlockType.*;
|
|
|
|
|
|
|
|
public class MultiblockConverter implements MultiblockHandler.IMultiblock {
|
|
|
|
public static MultiblockConverter INSTANCE;
|
|
|
|
@Override
|
|
|
|
public String getUniqueName() {
|
|
|
|
return "iw:converter";
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean isBlockTrigger(IBlockState state) {
|
2018-02-17 20:37:53 +01:00
|
|
|
return MechMBPart.isValidDefaultCenter(state);
|
2017-10-04 17:05:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private boolean checkEnd(LocalSidedWorld w, BlockPos.PooledMutableBlockPos p) {
|
|
|
|
p.setPos(0, 0, 0);
|
2018-02-17 20:37:53 +01:00
|
|
|
if (!MechMBPart.isValidDefaultCenter(w.getBlockState(p))) {
|
2017-10-04 17:05:04 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
p.setPos(0, -1, 0);
|
2018-02-17 20:37:53 +01:00
|
|
|
return MechMBPart.isValidDefaultCenter(w.getBlockState(p));
|
2017-10-04 17:05:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private void formEnd(LocalSidedWorld w, BlockPos.PooledMutableBlockPos p, MechanicalMBBlockType type,
|
|
|
|
BiConsumer<TileEntityMultiblockConverter, Boolean> init) {
|
|
|
|
p.setPos(0, 0, 0);
|
|
|
|
w.setBlockState(p, IndustrialWires.mechanicalMB.getStateFromMeta(type.ordinal()));
|
|
|
|
TileEntity te = w.getTileEntity(p);
|
|
|
|
if (te instanceof TileEntityMultiblockConverter) {
|
|
|
|
init.accept((TileEntityMultiblockConverter) te, true);
|
|
|
|
}
|
|
|
|
p.setPos(0, -1, 0);
|
|
|
|
w.setBlockState(p, IndustrialWires.mechanicalMB.getStateFromMeta(NO_MODEL.ordinal()));
|
|
|
|
te = w.getTileEntity(p);
|
|
|
|
if (te instanceof TileEntityMultiblockConverter) {
|
|
|
|
init.accept((TileEntityMultiblockConverter) te, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean createStructure(World world, BlockPos pos, EnumFacing side, EntityPlayer player) {
|
|
|
|
BlockPos.PooledMutableBlockPos mutPos = BlockPos.PooledMutableBlockPos.retain();
|
|
|
|
try {
|
|
|
|
LocalSidedWorld w = new LocalSidedWorld(world, pos, side.getOpposite(), false);
|
|
|
|
if (!checkEnd(w, mutPos)) {
|
|
|
|
return false;
|
|
|
|
}
|
2018-02-28 21:03:26 +01:00
|
|
|
boolean foundAll = false;
|
|
|
|
List<MechMBPart> parts = new ArrayList<>();
|
|
|
|
int lastLength = 1;
|
|
|
|
double weight = 0;
|
|
|
|
while (!foundAll) {
|
|
|
|
mutPos.setPos(0, 0, lastLength);
|
|
|
|
w.setOrigin(w.getRealPos(mutPos));
|
|
|
|
MechMBPart next = null;
|
2018-03-23 16:27:32 +01:00
|
|
|
List<MechMBPart> instances = new ArrayList<>(MechMBPart.INSTANCES.values());
|
|
|
|
instances.sort(MechMBPart.SORT_BY_COUNT);
|
|
|
|
int lastCount = 0;
|
|
|
|
for (MechMBPart part:instances) {
|
|
|
|
int newCount = MiscUtils.count1Bits(part.getFormPattern());
|
|
|
|
if (newCount==1&&lastCount>1&&checkEnd(w, mutPos)) {
|
|
|
|
foundAll = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
lastCount = newCount;
|
|
|
|
if (part.canForm(w)) {
|
|
|
|
next = part;
|
|
|
|
String key = MechMBPart.REGISTRY.inverse().get(part.getClass());
|
2018-02-28 21:03:26 +01:00
|
|
|
MechMBPart.cacheNewInstance(key);
|
|
|
|
break;
|
2017-10-04 17:05:04 +02:00
|
|
|
}
|
|
|
|
}
|
2018-02-28 21:03:26 +01:00
|
|
|
if (next != null) {
|
|
|
|
parts.add(next);
|
|
|
|
lastLength = next.getLength();
|
|
|
|
weight += next.getInertia();
|
2018-03-23 16:27:32 +01:00
|
|
|
} else if (!foundAll) {
|
|
|
|
return false;
|
2017-10-04 17:05:04 +02:00
|
|
|
}
|
2018-02-28 21:03:26 +01:00
|
|
|
}
|
|
|
|
double finalWeight = weight;
|
|
|
|
w.setOrigin(pos);
|
|
|
|
formEnd(w, mutPos, END, (te, master) -> {
|
|
|
|
if (master) {
|
|
|
|
te.offset = BlockPos.ORIGIN;
|
|
|
|
te.setMechanical(parts.toArray(new MechMBPart[parts.size()]), 0);
|
|
|
|
te.energyState = new MechEnergy(finalWeight, 0);
|
|
|
|
} else {
|
|
|
|
te.offset = new BlockPos(0, -1, 0);
|
|
|
|
}
|
|
|
|
te.facing = side;
|
|
|
|
te.formed = true;
|
|
|
|
});
|
|
|
|
lastLength = 1;
|
|
|
|
Consumer<TileEntityMultiblockConverter> init = (te) -> {
|
|
|
|
te.offset = te.getPos().subtract(pos);
|
|
|
|
te.facing = side;
|
|
|
|
te.formed = true;
|
|
|
|
};
|
|
|
|
for (MechMBPart part : parts) {
|
2017-10-04 17:05:04 +02:00
|
|
|
mutPos.setPos(0, 0, lastLength);
|
2018-02-28 21:03:26 +01:00
|
|
|
w = new LocalSidedWorld(world, w.getRealPos(mutPos), w.getFacing(), w.isMirrored());
|
|
|
|
part.form(w, init);
|
|
|
|
lastLength = part.getLength();
|
|
|
|
}
|
|
|
|
mutPos.setPos(0, 0, lastLength);
|
|
|
|
w = new LocalSidedWorld(w.getWorld(), w.getRealPos(mutPos), w.getFacing(), w.isMirrored());
|
|
|
|
formEnd(w, mutPos, OTHER_END, (te, __) -> init.accept(te));
|
|
|
|
return true;
|
2017-10-04 17:05:04 +02:00
|
|
|
} finally {
|
|
|
|
mutPos.release();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public ItemStack[][][] getStructureManual() {
|
|
|
|
return new ItemStack[0][][];
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public IngredientStack[] getTotalMaterials() {
|
|
|
|
return new IngredientStack[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean overwriteBlockRender(ItemStack stack, int iterator) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public float getManualScale() {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean canRenderFormedStructure() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void renderFormedStructure() {
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|