Encased belt blocks

This commit is contained in:
reidbhuntley 2022-01-11 13:04:08 -05:00
parent 2768123b52
commit b1a9993b16
14 changed files with 174 additions and 173 deletions

View file

@ -75,7 +75,7 @@ public class KineticTileEntity extends SmartTileEntity
updateInitialConnections(state); updateInitialConnections(state);
} }
protected void updateInitialConnections(BlockState state) { public void updateInitialConnections(BlockState state) {
if (state.getBlock() instanceof IRotate rotate) { if (state.getBlock() instanceof IRotate rotate) {
initialConnections = rotate.buildInitialConnections(ConnectionsBuilder.builder(), state).build(); initialConnections = rotate.buildInitialConnections(ConnectionsBuilder.builder(), state).build();
if (getLevel() != null && !getLevel().isClientSide) { if (getLevel() != null && !getLevel().isClientSide) {

View file

@ -130,15 +130,6 @@ public class GantryCarriageBlock extends DirectionalAxisKineticBlock implements
return Axis.Y; return Axis.Y;
} }
public static Axis getValidGantryPinionAxis(BlockState state, Axis shaftAxis) {
Axis facingAxis = state.getValue(FACING)
.getAxis();
for (Axis axis : Iterate.axes)
if (axis != shaftAxis && axis != facingAxis)
return axis;
return Axis.Y;
}
@Override @Override
public Class<GantryCarriageTileEntity> getTileEntityClass() { public Class<GantryCarriageTileEntity> getTileEntityClass() {
return GantryCarriageTileEntity.class; return GantryCarriageTileEntity.class;

View file

@ -8,6 +8,7 @@ import com.simibubi.create.AllBlocks;
import com.simibubi.create.AllShapes; import com.simibubi.create.AllShapes;
import com.simibubi.create.AllTileEntities; import com.simibubi.create.AllTileEntities;
import com.simibubi.create.content.contraptions.base.DirectionalKineticBlock; import com.simibubi.create.content.contraptions.base.DirectionalKineticBlock;
import com.simibubi.create.content.contraptions.solver.ConnectionsBuilder;
import com.simibubi.create.foundation.block.ITE; import com.simibubi.create.foundation.block.ITE;
import com.simibubi.create.foundation.utility.Iterate; import com.simibubi.create.foundation.utility.Iterate;
import com.simibubi.create.foundation.utility.Lang; import com.simibubi.create.foundation.utility.Lang;
@ -175,12 +176,17 @@ public class GantryShaftBlock extends DirectionalKineticBlock implements ITE<Gan
public void onPlace(BlockState state, Level worldIn, BlockPos pos, BlockState oldState, boolean isMoving) { public void onPlace(BlockState state, Level worldIn, BlockPos pos, BlockState oldState, boolean isMoving) {
super.onPlace(state, worldIn, pos, oldState, isMoving); super.onPlace(state, worldIn, pos, oldState, isMoving);
if (!worldIn.isClientSide() && oldState.is(AllBlocks.GANTRY_SHAFT.get())) { if (oldState.is(AllBlocks.GANTRY_SHAFT.get())) {
if (!worldIn.isClientSide()) {
Part oldPart = oldState.getValue(PART), part = state.getValue(PART); Part oldPart = oldState.getValue(PART), part = state.getValue(PART);
if ((oldPart != Part.MIDDLE && part == Part.MIDDLE) || (oldPart == Part.SINGLE && part != Part.SINGLE)) { if ((oldPart != Part.MIDDLE && part == Part.MIDDLE) || (oldPart == Part.SINGLE && part != Part.SINGLE)) {
BlockEntity te = worldIn.getBlockEntity(pos); withTileEntityDo(worldIn, pos, GantryShaftTileEntity::checkAttachedCarriageBlocks);
if (te instanceof GantryShaftTileEntity) }
((GantryShaftTileEntity) te).checkAttachedCarriageBlocks(); }
boolean oldPowered = oldState.getValue(POWERED), powered = state.getValue(POWERED);
if (oldPowered != powered) {
withTileEntityDo(worldIn, pos, kte -> kte.updateConnections(state));
} }
} }
} }
@ -227,10 +233,11 @@ public class GantryShaftBlock extends DirectionalKineticBlock implements ITE<Gan
toUpdate.add(pos); toUpdate.add(pos);
for (BlockPos blockPos : toUpdate) { for (BlockPos blockPos : toUpdate) {
BlockState blockState = worldIn.getBlockState(blockPos); BlockState blockState = worldIn.getBlockState(blockPos);
if (blockState.getBlock() instanceof GantryShaftBlock) if (blockState.getBlock() instanceof GantryShaftBlock gsb) {
worldIn.setBlock(blockPos, blockState.setValue(POWERED, shouldPower), 2); worldIn.setBlock(blockPos, blockState.setValue(POWERED, shouldPower), 2);
} }
} }
}
protected boolean shouldBePowered(BlockState state, Level worldIn, BlockPos pos) { protected boolean shouldBePowered(BlockState state, Level worldIn, BlockPos pos) {
boolean shouldPower = worldIn.hasNeighborSignal(pos); boolean shouldPower = worldIn.hasNeighborSignal(pos);

View file

@ -22,17 +22,32 @@ public class GantryShaftTileEntity extends KineticTileEntity {
super(typeIn, pos, state); super(typeIn, pos, state);
} }
@Override private KineticConnections connections;
public KineticConnections getConnections() {
BlockState state = getBlockState();
Direction facing = state.getValue(GantryShaftBlock.FACING);
public void updateConnections(BlockState state) {
if (!AllBlocks.GANTRY_SHAFT.has(state)) {
connections = KineticConnections.empty();
return;
}
Direction facing = state.getValue(GantryShaftBlock.FACING);
ConnectionsBuilder builder = ConnectionsBuilder.builder().withFullShaft(facing.getAxis()); ConnectionsBuilder builder = ConnectionsBuilder.builder().withFullShaft(facing.getAxis());
if (!AllBlocks.GANTRY_SHAFT.has(state) || !state.getValue(GantryShaftBlock.POWERED)) if (state.getValue(GantryShaftBlock.POWERED))
return builder.build(); builder = builder.withDirectional(AllConnections.Directional.GANTRY_RACK, facing);
return builder.withDirectional(AllConnections.Directional.GANTRY_RACK, facing).build(); connections = builder.build();
}
@Override
public KineticConnections getConnections() {
return connections;
}
@Override
public void initialize() {
updateConnections(getBlockState());
super.initialize();
} }
public void checkAttachedCarriageBlocks() { public void checkAttachedCarriageBlocks() {

View file

@ -79,10 +79,10 @@ public class Instruction {
return 0; return 0;
} }
Optional<InstructionSpeedModifiers> getSpeedModifier() { float getSpeedModifier() {
return switch (instruction) { return switch (instruction) {
case TURN_ANGLE, TURN_DISTANCE -> Optional.of(speedModifier); case TURN_ANGLE, TURN_DISTANCE -> speedModifier.value;
default -> Optional.empty(); default -> 0;
}; };
} }

View file

@ -162,23 +162,17 @@ public class SequencedGearshiftTileEntity extends KineticTileEntity {
ConnectionsBuilder builder = ConnectionsBuilder.builder(); ConnectionsBuilder builder = ConnectionsBuilder.builder();
if (isVirtual()) return builder.withFullShaft(facing.getAxis()).build(); if (isVirtual()) return builder.withFullShaft(facing.getAxis()).build();
builder = builder.withHalfShaft(facing.getOpposite()); builder = builder.withHalfShaft(facing.getOpposite(), 1);
Optional<InstructionSpeedModifiers> modifier = getModifier(); float modifier = getModifier();
if (modifier.isEmpty() || isRemoved()) return builder.build(); if (modifier == 0 || isRemoved()) return builder.build();
AllConnections.Shafts shaft = switch(modifier.get()) { return builder.withHalfShaft(facing, modifier).build();
case FORWARD_FAST -> AllConnections.Shafts.SHAFT_X2;
case FORWARD -> AllConnections.Shafts.SHAFT;
case BACK -> AllConnections.Shafts.SHAFT_REV;
case BACK_FAST -> AllConnections.Shafts.SHAFT_REV_X2;
};
return builder.withHalfShaft(shaft, facing).build();
} }
public Optional<InstructionSpeedModifiers> getModifier() { public float getModifier() {
if (currentInstruction >= instructions.size() || isIdle()) if (currentInstruction >= instructions.size() || isIdle())
return Optional.empty(); return 0;
return instructions.get(currentInstruction).getSpeedModifier(); return instructions.get(currentInstruction).getSpeedModifier();
} }

View file

@ -2,6 +2,9 @@ package com.simibubi.create.content.contraptions.relays.encased;
import com.simibubi.create.content.contraptions.base.KineticTileEntity; import com.simibubi.create.content.contraptions.base.KineticTileEntity;
import com.simibubi.create.content.contraptions.solver.ConnectionsBuilder;
import com.simibubi.create.content.contraptions.solver.KineticConnections;
import net.minecraft.core.BlockPos; import net.minecraft.core.BlockPos;
import net.minecraft.nbt.CompoundTag; import net.minecraft.nbt.CompoundTag;
import net.minecraft.world.level.block.entity.BlockEntityType; import net.minecraft.world.level.block.entity.BlockEntityType;
@ -11,10 +14,12 @@ public class AdjustablePulleyTileEntity extends KineticTileEntity {
int signal; int signal;
boolean signalChanged; boolean signalChanged;
KineticConnections connections;
public AdjustablePulleyTileEntity(BlockEntityType<?> type, BlockPos pos, BlockState state) { public AdjustablePulleyTileEntity(BlockEntityType<?> type, BlockPos pos, BlockState state) {
super(type, pos, state); super(type, pos, state);
signal = 0; signal = 0;
updateConnections(state);
setLazyTickRate(40); setLazyTickRate(40);
} }
@ -26,12 +31,14 @@ public class AdjustablePulleyTileEntity extends KineticTileEntity {
@Override @Override
protected void read(CompoundTag compound, boolean clientPacket) { protected void read(CompoundTag compound, boolean clientPacket) {
signal = compound.getInt("Signal"); analogSignalChanged(compound.getInt("Signal"));
super.read(compound, clientPacket); super.read(compound, clientPacket);
} }
public float getModifier() { protected float getModifier() {
return getModifierForSignal(signal); if (signal == 0)
return 1;
return 1 + ((signal + 1) / 16f);
} }
public void neighbourChanged() { public void neighbourChanged() {
@ -60,16 +67,16 @@ public class AdjustablePulleyTileEntity extends KineticTileEntity {
} }
protected void analogSignalChanged(int newSignal) { protected void analogSignalChanged(int newSignal) {
//detachKinetics();
//removeSource();
signal = newSignal; signal = newSignal;
//attachKinetics(); updateConnections(getBlockState());
} }
protected float getModifierForSignal(int newPower) { private void updateConnections(BlockState state) {
if (newPower == 0) connections = EncasedBeltBlock.encasedBeltConnections(state, getModifier()).build();
return 1;
return 1 + ((newPower + 1) / 16f);
} }
@Override
public KineticConnections getConnections() {
return connections;
}
} }

View file

@ -4,6 +4,8 @@ import com.simibubi.create.AllTileEntities;
import com.simibubi.create.content.contraptions.base.DirectionalAxisKineticBlock; import com.simibubi.create.content.contraptions.base.DirectionalAxisKineticBlock;
import com.simibubi.create.content.contraptions.base.KineticTileEntity; import com.simibubi.create.content.contraptions.base.KineticTileEntity;
import com.simibubi.create.content.contraptions.base.RotatedPillarKineticBlock; import com.simibubi.create.content.contraptions.base.RotatedPillarKineticBlock;
import com.simibubi.create.content.contraptions.solver.AllConnections;
import com.simibubi.create.content.contraptions.solver.ConnectionsBuilder;
import com.simibubi.create.foundation.block.ITE; import com.simibubi.create.foundation.block.ITE;
import com.simibubi.create.foundation.utility.Iterate; import com.simibubi.create.foundation.utility.Iterate;
import com.simibubi.create.foundation.utility.Lang; import com.simibubi.create.foundation.utility.Lang;
@ -121,8 +123,12 @@ public class EncasedBeltBlock extends RotatedPillarKineticBlock implements ITE<K
if ((part == Part.START) != positive) if ((part == Part.START) != positive)
part = Part.MIDDLE; part = Part.MIDDLE;
return stateIn.setValue(PART, part) BlockState newState = stateIn.setValue(PART, part)
.setValue(CONNECTED_ALONG_FIRST_COORDINATE, connectionAlongFirst); .setValue(CONNECTED_ALONG_FIRST_COORDINATE, connectionAlongFirst);
withTileEntityDo(worldIn, currentPos, kte -> kte.updateInitialConnections(newState));
return newState;
} }
@Override @Override
@ -164,39 +170,30 @@ public class EncasedBeltBlock extends RotatedPillarKineticBlock implements ITE<K
return state.getValue(AXIS); return state.getValue(AXIS);
} }
public static boolean areBlocksConnected(BlockState state, BlockState other, Direction facing) { public static ConnectionsBuilder encasedBeltConnections(BlockState state, float mod) {
Part part = state.getValue(PART); Part part = state.getValue(PART);
Axis shaftAxis = state.getValue(AXIS);
Axis connectionAxis = getConnectionAxis(state); Axis connectionAxis = getConnectionAxis(state);
Axis otherConnectionAxis = getConnectionAxis(other);
if (otherConnectionAxis != connectionAxis) ConnectionsBuilder builder = ConnectionsBuilder.builder().withFullShaft(shaftAxis);
return false;
if (facing.getAxis() != connectionAxis)
return false;
if (facing.getAxisDirection() == AxisDirection.POSITIVE && (part == Part.MIDDLE || part == Part.START))
return true;
if (facing.getAxisDirection() == AxisDirection.NEGATIVE && (part == Part.MIDDLE || part == Part.END))
return true;
return false; if (part == Part.START || part == Part.MIDDLE)
builder.withDirectional(AllConnections.Directional.ENCASED_BELT, AllConnections.pos(connectionAxis), mod);
if (part == Part.MIDDLE || part == Part.END)
builder.withDirectional(AllConnections.Directional.ENCASED_BELT, AllConnections.neg(connectionAxis), mod);
return builder;
}
@Override
public ConnectionsBuilder buildInitialConnections(ConnectionsBuilder builder, BlockState state) {
return encasedBeltConnections(state, 1);
} }
protected static Axis getConnectionAxis(BlockState state) { protected static Axis getConnectionAxis(BlockState state) {
Axis axis = state.getValue(AXIS); Axis axis = state.getValue(AXIS);
boolean connectionAlongFirst = state.getValue(CONNECTED_ALONG_FIRST_COORDINATE); boolean alongFirst = state.getValue(CONNECTED_ALONG_FIRST_COORDINATE);
Axis connectionAxis = return alongFirst ? (axis == Axis.X ? Axis.Y : Axis.X) : (axis == Axis.Z ? Axis.Y : Axis.Z);
connectionAlongFirst ? (axis == Axis.X ? Axis.Y : Axis.X) : (axis == Axis.Z ? Axis.Y : Axis.Z);
return connectionAxis;
}
public static float getRotationSpeedModifier(KineticTileEntity from, KineticTileEntity to) {
float fromMod = 1;
float toMod = 1;
if (from instanceof AdjustablePulleyTileEntity)
fromMod = ((AdjustablePulleyTileEntity) from).getModifier();
if (to instanceof AdjustablePulleyTileEntity)
toMod = ((AdjustablePulleyTileEntity) to).getModifier();
return fromMod / toMod;
} }
public enum Part implements StringRepresentable { public enum Part implements StringRepresentable {

View file

@ -80,11 +80,11 @@ public class GearboxBlock extends RotatedPillarKineticBlock implements ITE<Gearb
return state.getValue(AXIS); return state.getValue(AXIS);
} }
private static boolean isReversed(Direction from, Direction to) { private static float getModifier(Direction from, Direction to) {
if (from == to) return false; if (from == to) return 1;
Axis fromAxis = from.getAxis(), toAxis = to.getAxis(); Axis fromAxis = from.getAxis(), toAxis = to.getAxis();
if (fromAxis == toAxis) return true; if (fromAxis == toAxis) return -1;
return AllConnections.perpendicularRatios(to.getNormal().subtract(from.getNormal()), fromAxis, toAxis) == -1; return AllConnections.perpendicularRatios(to.getNormal().subtract(from.getNormal()), fromAxis, toAxis);
} }
@Override @Override
@ -92,10 +92,7 @@ public class GearboxBlock extends RotatedPillarKineticBlock implements ITE<Gearb
Axis axis = state.getValue(AXIS); Axis axis = state.getValue(AXIS);
Direction start = DirectionHelper.getPositivePerpendicular(axis); Direction start = DirectionHelper.getPositivePerpendicular(axis);
for (Direction cur : Iterate.directionsPerpendicularTo(axis)) { for (Direction cur : Iterate.directionsPerpendicularTo(axis)) {
AllConnections.Shafts shaft = isReversed(start, cur) builder = builder.withHalfShaft(cur, getModifier(start, cur));
? AllConnections.Shafts.SHAFT_REV
: AllConnections.Shafts.SHAFT;
builder = builder.withHalfShaft(shaft, cur);
} }
return builder; return builder;
} }

View file

@ -28,8 +28,8 @@ public class GearshiftTileEntity extends KineticTileEntity {
if (!state.getValue(BlockStateProperties.POWERED)) if (!state.getValue(BlockStateProperties.POWERED))
return builder.withFullShaft(dir.getAxis()).build(); return builder.withFullShaft(dir.getAxis()).build();
return builder return builder
.withHalfShaft(AllConnections.Shafts.SHAFT_REV, dir) .withHalfShaft(dir, -1)
.withHalfShaft(AllConnections.Shafts.SHAFT, dir.getOpposite()) .withHalfShaft(dir.getOpposite(), 1)
.build(); .build();
} }

View file

@ -16,12 +16,10 @@ import java.util.Optional;
public class AllConnections { public class AllConnections {
public static void register() { public static void register() {
Shafts.registerTypes();
Directional.registerTypes(); Directional.registerTypes();
Axial.registerTypes(); Axial.registerTypes();
DirAxial.registerTypes(); DirAxial.registerTypes();
Shafts.registerRatios();
Directional.registerRatios(); Directional.registerRatios();
Axial.registerRatios(); Axial.registerRatios();
DirAxial.registerRatios(); DirAxial.registerRatios();
@ -29,48 +27,21 @@ public class AllConnections {
private static record Entry(Vec3i offset, String to, float ratio) {} private static record Entry(Vec3i offset, String to, float ratio) {}
public enum Shafts {
SHAFT("shaft", 1),
SHAFT_REV("shaft_rev", -1),
SHAFT_X2("shaft_x2", 2),
SHAFT_REV_X2("shaft_rev_x2", -2);
public final String prefix;
public final float ratio;
Shafts(String prefix, float ratio) {
this.prefix = prefix;
this.ratio = ratio;
}
public String type(Direction dir) {
return prefix + "." + dir;
}
public static void registerTypes() {
for (Shafts value : values()) {
for (Direction dir : Direction.values()) {
KineticConnectionsRegistry.registerConnectionType(value.type(dir));
}
}
}
public static void registerRatios() {
for (Shafts from : values()) {
for (Shafts to : values()) {
for (Direction dir : Direction.values()) {
KineticConnectionsRegistry.registerConnectionRatio(
KineticConnectionsRegistry.getConnectionType(from.type(dir)).get(),
KineticConnectionsRegistry.getConnectionType(to.type(dir.getOpposite())).get(),
dir.getNormal(),
from.ratio / to.ratio
);
}
}
}
}
}
public enum Directional { public enum Directional {
SHAFT("shaft") {
@Override
public List<Entry> genConnections(Direction dir) {
return List.of(new Entry(dir.getNormal(), SHAFT.type(dir.getOpposite()), 1));
}
},
ENCASED_BELT("encased_belt") {
@Override
public List<Entry> genConnections(Direction dir) {
return List.of(new Entry(dir.getNormal(), ENCASED_BELT.type(dir.getOpposite()), 1));
}
},
GANTRY_RACK("gantry_rack") { GANTRY_RACK("gantry_rack") {
@Override @Override
public List<Entry> genConnections(Direction dir) { return List.of(); } public List<Entry> genConnections(Direction dir) { return List.of(); }

View file

@ -2,11 +2,13 @@ package com.simibubi.create.content.contraptions.solver;
import net.minecraft.core.Direction; import net.minecraft.core.Direction;
import java.util.HashMap;
import java.util.HashSet; import java.util.HashSet;
import java.util.Map;
import java.util.Set; import java.util.Set;
public class ConnectionsBuilder { public class ConnectionsBuilder {
private final Set<KineticConnection> connections = new HashSet<>(); private final Map<KineticConnection, Float> connections = new HashMap<>();
public static ConnectionsBuilder builder() { public static ConnectionsBuilder builder() {
return new ConnectionsBuilder(); return new ConnectionsBuilder();
@ -16,32 +18,31 @@ public class ConnectionsBuilder {
return new KineticConnections(connections); return new KineticConnections(connections);
} }
public ConnectionsBuilder withHalfShaft(AllConnections.Shafts shaft, Direction dir) { public ConnectionsBuilder withDirectional(AllConnections.Directional directional, Direction dir, float mod) {
connections.add(KineticConnectionsRegistry.getConnectionType(shaft.type(dir)).get()); connections.put(KineticConnectionsRegistry.getConnectionType(directional.type(dir)).get(), mod);
return this; return this;
} }
public ConnectionsBuilder withHalfShaft(Direction dir) {
return withHalfShaft(AllConnections.Shafts.SHAFT, dir);
}
public ConnectionsBuilder withFullShaft(Direction.Axis axis) {
return this.withHalfShaft(AllConnections.pos(axis)).withHalfShaft(AllConnections.neg(axis));
}
public ConnectionsBuilder withDirectional(AllConnections.Directional directional, Direction dir) { public ConnectionsBuilder withDirectional(AllConnections.Directional directional, Direction dir) {
connections.add(KineticConnectionsRegistry.getConnectionType(directional.type(dir)).get()); return withDirectional(directional, dir, 1);
}
public ConnectionsBuilder withAxial(AllConnections.Axial axial, Direction.Axis axis, float mod) {
connections.put(KineticConnectionsRegistry.getConnectionType(axial.type(axis)).get(), mod);
return this; return this;
} }
public ConnectionsBuilder withAxial(AllConnections.Axial axial, Direction.Axis axis) { public ConnectionsBuilder withAxial(AllConnections.Axial axial, Direction.Axis axis) {
connections.add(KineticConnectionsRegistry.getConnectionType(axial.type(axis)).get()); return withAxial(axial, axis, 1);
}
public ConnectionsBuilder withDirAxial(AllConnections.DirAxial dirAxial, Direction dir, boolean first, float mod) {
connections.put(KineticConnectionsRegistry.getConnectionType(dirAxial.type(dir, first)).get(), mod);
return this; return this;
} }
public ConnectionsBuilder withDirAxial(AllConnections.DirAxial dirAxial, Direction dir, boolean first) { public ConnectionsBuilder withDirAxial(AllConnections.DirAxial dirAxial, Direction dir, boolean first) {
connections.add(KineticConnectionsRegistry.getConnectionType(dirAxial.type(dir, first)).get()); return withDirAxial(dirAxial, dir, first, 1);
return this;
} }
public ConnectionsBuilder withLargeCog(Direction.Axis axis) { public ConnectionsBuilder withLargeCog(Direction.Axis axis) {
@ -51,4 +52,16 @@ public class ConnectionsBuilder {
public ConnectionsBuilder withSmallCog(Direction.Axis axis) { public ConnectionsBuilder withSmallCog(Direction.Axis axis) {
return withAxial(AllConnections.Axial.SMALL_COG, axis); return withAxial(AllConnections.Axial.SMALL_COG, axis);
} }
public ConnectionsBuilder withHalfShaft(Direction dir, float mod) {
return withDirectional(AllConnections.Directional.SHAFT, dir, mod);
}
public ConnectionsBuilder withHalfShaft(Direction dir) {
return withHalfShaft(dir, 1);
}
public ConnectionsBuilder withFullShaft(Direction.Axis axis) {
return this.withHalfShaft(AllConnections.pos(axis)).withHalfShaft(AllConnections.neg(axis));
}
} }

View file

@ -1,51 +1,49 @@
package com.simibubi.create.content.contraptions.solver; package com.simibubi.create.content.contraptions.solver;
import com.simibubi.create.foundation.utility.Pair;
import net.minecraft.core.Direction; import net.minecraft.core.Direction;
import net.minecraft.core.Vec3i; import net.minecraft.core.Vec3i;
import net.minecraft.nbt.CompoundTag; import net.minecraft.nbt.CompoundTag;
import net.minecraft.nbt.ListTag; import net.minecraft.nbt.ListTag;
import net.minecraft.nbt.StringTag; import net.minecraft.nbt.StringTag;
import net.minecraft.nbt.Tag; import net.minecraft.nbt.Tag;
import java.util.HashMap;
import java.util.HashSet; import java.util.HashSet;
import java.util.Map;
import java.util.Objects; import java.util.Objects;
import java.util.Optional; import java.util.Optional;
import java.util.Set; import java.util.Set;
import java.util.stream.Stream; import java.util.stream.Stream;
public class KineticConnections { public class KineticConnections {
private final Set<KineticConnection> connections; private final Map<KineticConnection, Float> connections;
public KineticConnections(Set<KineticConnection> connections) { public KineticConnections(Map<KineticConnection, Float> connections) {
this.connections = connections; this.connections = connections;
} }
public static KineticConnections empty() { public static KineticConnections empty() {
return new KineticConnections(Set.of()); return new KineticConnections(Map.of());
} }
public Stream<KineticConnection> stream() { public Set<Map.Entry<KineticConnection, Float>> entries() {
return connections.stream(); return connections.entrySet();
} }
public boolean hasStressOnlyConnections() { public boolean hasStressOnlyConnections() {
return connections.stream() return entries().stream()
.flatMap(c -> c.getRatios().values().stream() .flatMap(c -> c.getKey().getRatios().values().stream()
.flatMap(m -> m.values().stream())) .flatMap(m -> m.values().stream()))
.anyMatch(r -> r == 0); .anyMatch(r -> r == 0);
} }
public float getShaftSpeedModifier(Direction face) { public float getShaftSpeedModifier(Direction face) {
Vec3i offset = face.getNormal();
KineticConnection shaft = KineticConnectionsRegistry KineticConnection shaft = KineticConnectionsRegistry
.getConnectionType(AllConnections.Shafts.SHAFT.type(face.getOpposite())).get(); .getConnectionType(AllConnections.Directional.SHAFT.type(face)).get();
return stream() return Optional.ofNullable(connections.get(shaft)).orElse(0f);
.flatMap(c -> Optional.ofNullable(c.getRatios().get(offset))
.flatMap(r -> Optional.ofNullable(r.get(shaft)))
.stream())
.findFirst()
.orElse(0f);
} }
@Override @Override
@ -63,18 +61,25 @@ public class KineticConnections {
public CompoundTag save(CompoundTag tag) { public CompoundTag save(CompoundTag tag) {
ListTag connectionsTags = new ListTag(); ListTag connectionsTags = new ListTag();
for (KineticConnection connection : connections) { for (Map.Entry<KineticConnection, Float> entry : connections.entrySet()) {
connectionsTags.add(StringTag.valueOf(connection.name())); CompoundTag entryTag = new CompoundTag();
entryTag.putString("Name", entry.getKey().name());
if (entry.getValue() != 1) {
entryTag.putFloat("Mod", entry.getValue());
}
connectionsTags.add(entryTag);
} }
tag.put("Connections", connectionsTags); tag.put("Connections", connectionsTags);
return tag; return tag;
} }
public static KineticConnections load(CompoundTag tag) { public static KineticConnections load(CompoundTag tag) {
Set<KineticConnection> connections = new HashSet<>(); Map<KineticConnection, Float> connections = new HashMap<>();
tag.getList("Connections", Tag.TAG_STRING).forEach(t -> { tag.getList("Connections", Tag.TAG_COMPOUND).forEach(t -> {
KineticConnectionsRegistry.getConnectionType(t.getAsString()) CompoundTag ct = (CompoundTag) t;
.ifPresent(connections::add); String name = ct.getString("Name");
float mod = ct.contains("Mod") ? ct.getFloat("Mod") : 1;
KineticConnectionsRegistry.getConnectionType(name).ifPresent(c -> connections.put(c, mod));
}); });
return new KineticConnections(connections); return new KineticConnections(connections);
} }

View file

@ -160,17 +160,21 @@ public class KineticNode {
* the connecting node and the second value is the speed ratio of the connection * the connecting node and the second value is the speed ratio of the connection
*/ */
private Stream<Pair<KineticNode, Float>> getAllActiveConnections() { private Stream<Pair<KineticNode, Float>> getAllActiveConnections() {
return connections.stream() return connections.entries().stream()
.flatMap(from -> from.getRatios().entrySet().stream() .flatMap(from -> from.getKey().getRatios().entrySet().stream()
.map(e -> { .map(e -> {
Vec3i offset = e.getKey(); Vec3i offset = e.getKey();
return solver.getNode(pos.offset(offset)).flatMap(node -> { return solver.getNode(pos.offset(offset)).flatMap(node -> {
Map<KineticConnection, Float> ratios = e.getValue(); Map<KineticConnection, Float> ratios = e.getValue();
return node.getConnections().stream() for (Map.Entry<KineticConnection, Float> to : node.getConnections().entries()) {
.map(ratios::get) Float ratio = ratios.get(to.getKey());
.filter(Objects::nonNull) if (ratio != null) {
.findFirst() float fromMod = from.getValue();
.map(r -> Pair.of(node, r)); float toMod = to.getValue();
return Optional.of(Pair.of(node, ratio * fromMod / toMod));
}
}
return Optional.empty();
}); });
}) })
) )