mirror of
https://github.com/Creators-of-Create/Create.git
synced 2025-01-16 08:06:12 +01:00
Entering glue zone
- Reimplemented super glue as area-entities instead of individual connections
This commit is contained in:
parent
6ae6c4878f
commit
e1c3ad1a9b
25 changed files with 455 additions and 868 deletions
|
@ -6,7 +6,6 @@ import com.simibubi.create.content.contraptions.components.structureMovement.Con
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.OrientedContraptionEntity;
|
import com.simibubi.create.content.contraptions.components.structureMovement.OrientedContraptionEntity;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.OrientedContraptionEntityRenderer;
|
import com.simibubi.create.content.contraptions.components.structureMovement.OrientedContraptionEntityRenderer;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.gantry.GantryContraptionEntity;
|
import com.simibubi.create.content.contraptions.components.structureMovement.gantry.GantryContraptionEntity;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.glue.GlueInstance;
|
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.glue.SuperGlueEntity;
|
import com.simibubi.create.content.contraptions.components.structureMovement.glue.SuperGlueEntity;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.glue.SuperGlueRenderer;
|
import com.simibubi.create.content.contraptions.components.structureMovement.glue.SuperGlueRenderer;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.render.ContraptionEntityRenderer;
|
import com.simibubi.create.content.contraptions.components.structureMovement.render.ContraptionEntityRenderer;
|
||||||
|
@ -34,25 +33,20 @@ import net.minecraft.world.entity.MobCategory;
|
||||||
public class AllEntityTypes {
|
public class AllEntityTypes {
|
||||||
|
|
||||||
public static final EntityEntry<OrientedContraptionEntity> ORIENTED_CONTRAPTION = contraption("contraption",
|
public static final EntityEntry<OrientedContraptionEntity> ORIENTED_CONTRAPTION = contraption("contraption",
|
||||||
OrientedContraptionEntity::new, () -> OrientedContraptionEntityRenderer::new, 5, 3, true)
|
OrientedContraptionEntity::new, () -> OrientedContraptionEntityRenderer::new, 5, 3, true).register();
|
||||||
.register();
|
|
||||||
public static final EntityEntry<ControlledContraptionEntity> CONTROLLED_CONTRAPTION =
|
public static final EntityEntry<ControlledContraptionEntity> CONTROLLED_CONTRAPTION =
|
||||||
contraption("stationary_contraption", ControlledContraptionEntity::new, () -> ContraptionEntityRenderer::new,
|
contraption("stationary_contraption", ControlledContraptionEntity::new, () -> ContraptionEntityRenderer::new,
|
||||||
20, 40, false)
|
20, 40, false).register();
|
||||||
.register();
|
|
||||||
public static final EntityEntry<GantryContraptionEntity> GANTRY_CONTRAPTION = contraption("gantry_contraption",
|
public static final EntityEntry<GantryContraptionEntity> GANTRY_CONTRAPTION = contraption("gantry_contraption",
|
||||||
GantryContraptionEntity::new, () -> ContraptionEntityRenderer::new, 10, 40, false)
|
GantryContraptionEntity::new, () -> ContraptionEntityRenderer::new, 10, 40, false).register();
|
||||||
.register();
|
|
||||||
public static final EntityEntry<CarriageContraptionEntity> CARRIAGE_CONTRAPTION =
|
public static final EntityEntry<CarriageContraptionEntity> CARRIAGE_CONTRAPTION =
|
||||||
contraption("carriage_contraption", CarriageContraptionEntity::new,
|
contraption("carriage_contraption", CarriageContraptionEntity::new,
|
||||||
() -> CarriageContraptionEntityRenderer::new, 15, 3, true)
|
() -> CarriageContraptionEntityRenderer::new, 15, 3, true).instance(() -> CarriageContraptionInstance::new)
|
||||||
.instance(() -> CarriageContraptionInstance::new)
|
|
||||||
.register();
|
.register();
|
||||||
|
|
||||||
public static final EntityEntry<SuperGlueEntity> SUPER_GLUE =
|
public static final EntityEntry<SuperGlueEntity> SUPER_GLUE =
|
||||||
register("super_glue", SuperGlueEntity::new, () -> SuperGlueRenderer::new, MobCategory.MISC, 10,
|
register("super_glue", SuperGlueEntity::new, () -> SuperGlueRenderer::new, MobCategory.MISC, 10,
|
||||||
Integer.MAX_VALUE, false, true, SuperGlueEntity::build).instance(() -> GlueInstance::new, false)
|
Integer.MAX_VALUE, false, true, SuperGlueEntity::build).register();
|
||||||
.register();
|
|
||||||
|
|
||||||
public static final EntityEntry<BlueprintEntity> CRAFTING_BLUEPRINT =
|
public static final EntityEntry<BlueprintEntity> CRAFTING_BLUEPRINT =
|
||||||
register("crafting_blueprint", BlueprintEntity::new, () -> BlueprintRenderer::new, MobCategory.MISC, 10,
|
register("crafting_blueprint", BlueprintEntity::new, () -> BlueprintRenderer::new, MobCategory.MISC, 10,
|
||||||
|
|
|
@ -12,6 +12,7 @@ public enum AllSpecialTextures {
|
||||||
CUTOUT_CHECKERED("cutout_checkerboard.png"),
|
CUTOUT_CHECKERED("cutout_checkerboard.png"),
|
||||||
HIGHLIGHT_CHECKERED("highlighted_checkerboard.png"),
|
HIGHLIGHT_CHECKERED("highlighted_checkerboard.png"),
|
||||||
SELECTION("selection.png"),
|
SELECTION("selection.png"),
|
||||||
|
GLUE("glue.png"),
|
||||||
|
|
||||||
;
|
;
|
||||||
|
|
||||||
|
|
|
@ -11,7 +11,6 @@ import com.simibubi.create.api.behaviour.BlockSpoutingBehaviour;
|
||||||
import com.simibubi.create.content.CreateItemGroup;
|
import com.simibubi.create.content.CreateItemGroup;
|
||||||
import com.simibubi.create.content.contraptions.TorquePropagator;
|
import com.simibubi.create.content.contraptions.TorquePropagator;
|
||||||
import com.simibubi.create.content.contraptions.components.flywheel.engine.FurnaceEngineInteractions;
|
import com.simibubi.create.content.contraptions.components.flywheel.engine.FurnaceEngineInteractions;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.glue.GlueQueue;
|
|
||||||
import com.simibubi.create.content.curiosities.weapons.BuiltinPotatoProjectileTypes;
|
import com.simibubi.create.content.curiosities.weapons.BuiltinPotatoProjectileTypes;
|
||||||
import com.simibubi.create.content.logistics.RedstoneLinkNetworkHandler;
|
import com.simibubi.create.content.logistics.RedstoneLinkNetworkHandler;
|
||||||
import com.simibubi.create.content.logistics.block.data.AllDataGathererBehaviours;
|
import com.simibubi.create.content.logistics.block.data.AllDataGathererBehaviours;
|
||||||
|
@ -77,7 +76,6 @@ public class Create {
|
||||||
public static final TorquePropagator TORQUE_PROPAGATOR = new TorquePropagator();
|
public static final TorquePropagator TORQUE_PROPAGATOR = new TorquePropagator();
|
||||||
public static final GlobalRailwayManager RAILWAYS = new GlobalRailwayManager();
|
public static final GlobalRailwayManager RAILWAYS = new GlobalRailwayManager();
|
||||||
public static final ServerLagger LAGGER = new ServerLagger();
|
public static final ServerLagger LAGGER = new ServerLagger();
|
||||||
public static final GlueQueue GLUE_QUEUE = new GlueQueue();
|
|
||||||
public static final Random RANDOM = new Random();
|
public static final Random RANDOM = new Random();
|
||||||
|
|
||||||
private static final NonNullSupplier<CreateRegistrate> REGISTRATE = CreateRegistrate.lazy(ID);
|
private static final NonNullSupplier<CreateRegistrate> REGISTRATE = CreateRegistrate.lazy(ID);
|
||||||
|
|
|
@ -42,7 +42,6 @@ import com.simibubi.create.content.contraptions.components.structureMovement.cha
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.chassis.StickerBlock;
|
import com.simibubi.create.content.contraptions.components.structureMovement.chassis.StickerBlock;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.gantry.GantryCarriageBlock;
|
import com.simibubi.create.content.contraptions.components.structureMovement.gantry.GantryCarriageBlock;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.glue.SuperGlueEntity;
|
import com.simibubi.create.content.contraptions.components.structureMovement.glue.SuperGlueEntity;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.glue.SuperGlueHandler;
|
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.interaction.controls.ControlsBlock;
|
import com.simibubi.create.content.contraptions.components.structureMovement.interaction.controls.ControlsBlock;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.piston.MechanicalPistonBlock;
|
import com.simibubi.create.content.contraptions.components.structureMovement.piston.MechanicalPistonBlock;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.piston.MechanicalPistonBlock.PistonState;
|
import com.simibubi.create.content.contraptions.components.structureMovement.piston.MechanicalPistonBlock.PistonState;
|
||||||
|
@ -101,6 +100,7 @@ import net.minecraft.world.level.block.state.properties.BlockStateProperties;
|
||||||
import net.minecraft.world.level.block.state.properties.ChestType;
|
import net.minecraft.world.level.block.state.properties.ChestType;
|
||||||
import net.minecraft.world.level.block.state.properties.PistonType;
|
import net.minecraft.world.level.block.state.properties.PistonType;
|
||||||
import net.minecraft.world.level.chunk.HashMapPalette;
|
import net.minecraft.world.level.chunk.HashMapPalette;
|
||||||
|
import net.minecraft.world.level.levelgen.structure.BoundingBox;
|
||||||
import net.minecraft.world.level.levelgen.structure.templatesystem.StructureTemplate.StructureBlockInfo;
|
import net.minecraft.world.level.levelgen.structure.templatesystem.StructureTemplate.StructureBlockInfo;
|
||||||
import net.minecraft.world.level.material.FluidState;
|
import net.minecraft.world.level.material.FluidState;
|
||||||
import net.minecraft.world.level.material.Fluids;
|
import net.minecraft.world.level.material.Fluids;
|
||||||
|
@ -137,12 +137,12 @@ public abstract class Contraption {
|
||||||
protected Map<BlockPos, MountedFluidStorage> fluidStorage;
|
protected Map<BlockPos, MountedFluidStorage> fluidStorage;
|
||||||
protected List<MutablePair<StructureBlockInfo, MovementContext>> actors;
|
protected List<MutablePair<StructureBlockInfo, MovementContext>> actors;
|
||||||
protected Map<BlockPos, MovingInteractionBehaviour> interactors;
|
protected Map<BlockPos, MovingInteractionBehaviour> interactors;
|
||||||
protected Set<Pair<BlockPos, Direction>> superglue;
|
protected List<AABB> superglue;
|
||||||
protected List<BlockPos> seats;
|
protected List<BlockPos> seats;
|
||||||
protected Map<UUID, Integer> seatMapping;
|
protected Map<UUID, Integer> seatMapping;
|
||||||
protected Map<UUID, BlockFace> stabilizedSubContraptions;
|
protected Map<UUID, BlockFace> stabilizedSubContraptions;
|
||||||
|
|
||||||
private List<SuperGlueEntity> glueToRemove;
|
private Set<SuperGlueEntity> glueToRemove;
|
||||||
private Map<BlockPos, Entity> initialPassengers;
|
private Map<BlockPos, Entity> initialPassengers;
|
||||||
private List<BlockFace> pendingSubContraptions;
|
private List<BlockFace> pendingSubContraptions;
|
||||||
|
|
||||||
|
@ -161,10 +161,10 @@ public abstract class Contraption {
|
||||||
seats = new ArrayList<>();
|
seats = new ArrayList<>();
|
||||||
actors = new ArrayList<>();
|
actors = new ArrayList<>();
|
||||||
interactors = new HashMap<>();
|
interactors = new HashMap<>();
|
||||||
superglue = new HashSet<>();
|
superglue = new ArrayList<>();
|
||||||
seatMapping = new HashMap<>();
|
seatMapping = new HashMap<>();
|
||||||
fluidStorage = new HashMap<>();
|
fluidStorage = new HashMap<>();
|
||||||
glueToRemove = new ArrayList<>();
|
glueToRemove = new HashSet<>();
|
||||||
initialPassengers = new HashMap<>();
|
initialPassengers = new HashMap<>();
|
||||||
presentTileEntities = new HashMap<>();
|
presentTileEntities = new HashMap<>();
|
||||||
maybeInstancedTileEntities = new ArrayList<>();
|
maybeInstancedTileEntities = new ArrayList<>();
|
||||||
|
@ -389,8 +389,6 @@ public abstract class Contraption {
|
||||||
if (!visited.contains(posDown) && AllBlocks.CART_ASSEMBLER.has(stateBelow))
|
if (!visited.contains(posDown) && AllBlocks.CART_ASSEMBLER.has(stateBelow))
|
||||||
frontier.add(posDown);
|
frontier.add(posDown);
|
||||||
|
|
||||||
Map<Direction, SuperGlueEntity> superglue = SuperGlueHandler.gatherGlue(world, pos);
|
|
||||||
|
|
||||||
// Slime blocks and super glue drag adjacent blocks if possible
|
// Slime blocks and super glue drag adjacent blocks if possible
|
||||||
for (Direction offset : Iterate.directions) {
|
for (Direction offset : Iterate.directions) {
|
||||||
BlockPos offsetPos = pos.relative(offset);
|
BlockPos offsetPos = pos.relative(offset);
|
||||||
|
@ -404,7 +402,7 @@ public abstract class Contraption {
|
||||||
}
|
}
|
||||||
|
|
||||||
boolean wasVisited = visited.contains(offsetPos);
|
boolean wasVisited = visited.contains(offsetPos);
|
||||||
boolean faceHasGlue = superglue.containsKey(offset);
|
boolean faceHasGlue = SuperGlueEntity.isGlued(world, pos, offset, glueToRemove);
|
||||||
boolean blockAttachedTowardsFace =
|
boolean blockAttachedTowardsFace =
|
||||||
BlockMovementChecks.isBlockAttachedTowards(blockState, world, offsetPos, offset.getOpposite());
|
BlockMovementChecks.isBlockAttachedTowards(blockState, world, offsetPos, offset.getOpposite());
|
||||||
boolean brittle = BlockMovementChecks.isBrittle(blockState);
|
boolean brittle = BlockMovementChecks.isBrittle(blockState);
|
||||||
|
@ -425,8 +423,6 @@ public abstract class Contraption {
|
||||||
if (!wasVisited && (canStick || blockAttachedTowardsFace || faceHasGlue
|
if (!wasVisited && (canStick || blockAttachedTowardsFace || faceHasGlue
|
||||||
|| (offset == forcedDirection && !BlockMovementChecks.isNotSupportive(state, forcedDirection))))
|
|| (offset == forcedDirection && !BlockMovementChecks.isNotSupportive(state, forcedDirection))))
|
||||||
frontier.add(offsetPos);
|
frontier.add(offsetPos);
|
||||||
if (faceHasGlue)
|
|
||||||
addGlue(superglue.get(offset));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
addBlock(pos, capture(world, pos));
|
addBlock(pos, capture(world, pos));
|
||||||
|
@ -678,13 +674,6 @@ public abstract class Contraption {
|
||||||
return nbt;
|
return nbt;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void addGlue(SuperGlueEntity entity) {
|
|
||||||
BlockPos pos = entity.getHangingPosition();
|
|
||||||
Direction direction = entity.getFacingDirection();
|
|
||||||
this.superglue.add(Pair.of(toLocalPos(pos), direction));
|
|
||||||
glueToRemove.add(entity);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected BlockPos toLocalPos(BlockPos globalPos) {
|
protected BlockPos toLocalPos(BlockPos globalPos) {
|
||||||
return globalPos.subtract(anchor);
|
return globalPos.subtract(anchor);
|
||||||
}
|
}
|
||||||
|
@ -718,8 +707,8 @@ public abstract class Contraption {
|
||||||
});
|
});
|
||||||
|
|
||||||
superglue.clear();
|
superglue.clear();
|
||||||
NBTHelper.iterateCompoundList(nbt.getList("Superglue", Tag.TAG_COMPOUND), c -> superglue.add(
|
NBTHelper.iterateCompoundList(nbt.getList("Superglue", Tag.TAG_COMPOUND),
|
||||||
Pair.of(NbtUtils.readBlockPos(c.getCompound("Pos")), Direction.from3DDataValue(c.getByte("Direction")))));
|
c -> superglue.add(SuperGlueEntity.readBoundingBox(c)));
|
||||||
|
|
||||||
seats.clear();
|
seats.clear();
|
||||||
NBTHelper.iterateCompoundList(nbt.getList("Seats", Tag.TAG_COMPOUND), c -> seats.add(NbtUtils.readBlockPos(c)));
|
NBTHelper.iterateCompoundList(nbt.getList("Seats", Tag.TAG_COMPOUND), c -> seats.add(NbtUtils.readBlockPos(c)));
|
||||||
|
@ -802,11 +791,9 @@ public abstract class Contraption {
|
||||||
ListTag superglueNBT = new ListTag();
|
ListTag superglueNBT = new ListTag();
|
||||||
ListTag storageNBT = new ListTag();
|
ListTag storageNBT = new ListTag();
|
||||||
if (!spawnPacket) {
|
if (!spawnPacket) {
|
||||||
for (Pair<BlockPos, Direction> glueEntry : superglue) {
|
for (AABB glueEntry : superglue) {
|
||||||
CompoundTag c = new CompoundTag();
|
CompoundTag c = new CompoundTag();
|
||||||
c.put("Pos", NbtUtils.writeBlockPos(glueEntry.getKey()));
|
SuperGlueEntity.writeBoundingBox(c, glueEntry);
|
||||||
c.putByte("Direction", (byte) glueEntry.getValue()
|
|
||||||
.get3DDataValue());
|
|
||||||
superglueNBT.add(c);
|
superglueNBT.add(c);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -978,7 +965,17 @@ public abstract class Contraption {
|
||||||
.forEach(MountedStorage::removeStorageFromWorld);
|
.forEach(MountedStorage::removeStorageFromWorld);
|
||||||
fluidStorage.values()
|
fluidStorage.values()
|
||||||
.forEach(MountedFluidStorage::removeStorageFromWorld);
|
.forEach(MountedFluidStorage::removeStorageFromWorld);
|
||||||
glueToRemove.forEach(SuperGlueEntity::discard);
|
|
||||||
|
glueToRemove.forEach(glue -> {
|
||||||
|
superglue.add(glue.getBoundingBox()
|
||||||
|
.move(Vec3.atLowerCornerOf(offset.offset(anchor))
|
||||||
|
.scale(-1)));
|
||||||
|
glue.discard();
|
||||||
|
});
|
||||||
|
|
||||||
|
List<BoundingBox> minimisedGlue = new ArrayList<>();
|
||||||
|
for (int i = 0; i < superglue.size(); i++)
|
||||||
|
minimisedGlue.add(null);
|
||||||
|
|
||||||
for (boolean brittles : Iterate.trueAndFalse) {
|
for (boolean brittles : Iterate.trueAndFalse) {
|
||||||
for (Iterator<StructureBlockInfo> iterator = blocks.values()
|
for (Iterator<StructureBlockInfo> iterator = blocks.values()
|
||||||
|
@ -987,6 +984,18 @@ public abstract class Contraption {
|
||||||
if (brittles != BlockMovementChecks.isBrittle(block.state))
|
if (brittles != BlockMovementChecks.isBrittle(block.state))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
for (int i = 0; i < superglue.size(); i++) {
|
||||||
|
AABB aabb = superglue.get(i);
|
||||||
|
if (aabb == null
|
||||||
|
|| !aabb.contains(block.pos.getX() + .5, block.pos.getY() + .5, block.pos.getZ() + .5))
|
||||||
|
continue;
|
||||||
|
if (minimisedGlue.get(i) == null)
|
||||||
|
minimisedGlue.set(i, new BoundingBox(block.pos));
|
||||||
|
else
|
||||||
|
minimisedGlue.get(i)
|
||||||
|
.encapsulate(block.pos);
|
||||||
|
}
|
||||||
|
|
||||||
BlockPos add = block.pos.offset(anchor)
|
BlockPos add = block.pos.offset(anchor)
|
||||||
.offset(offset);
|
.offset(offset);
|
||||||
if (customBlockRemoval(world, add, block.state))
|
if (customBlockRemoval(world, add, block.state))
|
||||||
|
@ -1008,6 +1017,16 @@ public abstract class Contraption {
|
||||||
world.setBlock(add, Blocks.AIR.defaultBlockState(), flags);
|
world.setBlock(add, Blocks.AIR.defaultBlockState(), flags);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
superglue.clear();
|
||||||
|
for (BoundingBox box : minimisedGlue) {
|
||||||
|
if (box == null)
|
||||||
|
continue;
|
||||||
|
AABB bb = new AABB(box.minX(), box.minY(), box.minZ(), box.maxX() + 1, box.maxY() + 1, box.maxZ() + 1);
|
||||||
|
if (bb.getSize() > 1.01)
|
||||||
|
superglue.add(bb);
|
||||||
|
}
|
||||||
|
|
||||||
for (StructureBlockInfo block : blocks.values()) {
|
for (StructureBlockInfo block : blocks.values()) {
|
||||||
BlockPos add = block.pos.offset(anchor)
|
BlockPos add = block.pos.offset(anchor)
|
||||||
.offset(offset);
|
.offset(offset);
|
||||||
|
@ -1137,15 +1156,11 @@ public abstract class Contraption {
|
||||||
for (int i = 0; i < fluidInventory.getTanks(); i++)
|
for (int i = 0; i < fluidInventory.getTanks(); i++)
|
||||||
fluidInventory.drain(fluidInventory.getFluidInTank(i), FluidAction.EXECUTE);
|
fluidInventory.drain(fluidInventory.getFluidInTank(i), FluidAction.EXECUTE);
|
||||||
|
|
||||||
for (Pair<BlockPos, Direction> pair : superglue) {
|
for (AABB box : superglue) {
|
||||||
BlockPos targetPos = transform.apply(pair.getKey());
|
box = new AABB(transform.apply(new Vec3(box.minX, box.minY, box.minZ)),
|
||||||
Direction targetFacing = transform.transformFacing(pair.getValue());
|
transform.apply(new Vec3(box.maxX, box.maxY, box.maxZ)));
|
||||||
|
|
||||||
SuperGlueEntity entity = new SuperGlueEntity(world, targetPos, targetFacing);
|
|
||||||
if (entity.onValidSurface()) {
|
|
||||||
if (!world.isClientSide)
|
if (!world.isClientSide)
|
||||||
world.addFreshEntity(entity);
|
world.addFreshEntity(new SuperGlueEntity(world, box));
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,89 +0,0 @@
|
||||||
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.Instancer;
|
|
||||||
import com.jozufozu.flywheel.api.MaterialGroup;
|
|
||||||
import com.jozufozu.flywheel.api.MaterialManager;
|
|
||||||
import com.jozufozu.flywheel.api.instance.TickableInstance;
|
|
||||||
import com.jozufozu.flywheel.backend.instancing.entity.EntityInstance;
|
|
||||||
import com.jozufozu.flywheel.core.Materials;
|
|
||||||
import com.jozufozu.flywheel.core.instancing.ConditionalInstance;
|
|
||||||
import com.jozufozu.flywheel.core.materials.oriented.OrientedData;
|
|
||||||
import com.mojang.math.Quaternion;
|
|
||||||
import com.simibubi.create.AllItems;
|
|
||||||
import com.simibubi.create.Create;
|
|
||||||
import com.simibubi.create.foundation.utility.AngleHelper;
|
|
||||||
|
|
||||||
import net.minecraft.client.Minecraft;
|
|
||||||
import net.minecraft.client.renderer.RenderType;
|
|
||||||
import net.minecraft.core.BlockPos;
|
|
||||||
import net.minecraft.core.Direction;
|
|
||||||
import net.minecraft.resources.ResourceLocation;
|
|
||||||
import net.minecraft.world.entity.player.Player;
|
|
||||||
import net.minecraft.world.level.LightLayer;
|
|
||||||
|
|
||||||
public class GlueInstance extends EntityInstance<SuperGlueEntity> implements TickableInstance {
|
|
||||||
|
|
||||||
private static final ResourceLocation TEXTURE = Create.asResource("textures/entity/super_glue/slime.png");
|
|
||||||
|
|
||||||
private final Quaternion rotation;
|
|
||||||
protected ConditionalInstance<OrientedData> model;
|
|
||||||
|
|
||||||
public GlueInstance(MaterialManager materialManager, SuperGlueEntity entity) {
|
|
||||||
super(materialManager, entity);
|
|
||||||
|
|
||||||
Instancer<OrientedData> instancer = getInstancer(materialManager, entity);
|
|
||||||
|
|
||||||
Direction face = entity.getFacingDirection();
|
|
||||||
rotation = new Quaternion(AngleHelper.verticalAngle(face), AngleHelper.horizontalAngle(face), 0, true);
|
|
||||||
|
|
||||||
model = new ConditionalInstance<>(instancer)
|
|
||||||
.withCondition(this::shouldShow)
|
|
||||||
.withSetupFunc(this::positionModel)
|
|
||||||
.update();
|
|
||||||
}
|
|
||||||
|
|
||||||
private Instancer<OrientedData> getInstancer(MaterialManager materialManager, SuperGlueEntity entity) {
|
|
||||||
MaterialGroup group = GlueModel.USE_ATLAS ? materialManager.defaultCutout() : materialManager.cutout(RenderType.entityCutout(TEXTURE));
|
|
||||||
|
|
||||||
return group.material(Materials.ORIENTED).model(entity.getType(), GlueModel::get);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void tick() {
|
|
||||||
model.update();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void remove() {
|
|
||||||
model.delete();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void positionModel(OrientedData model) {
|
|
||||||
|
|
||||||
model.setPosition(getInstancePosition())
|
|
||||||
.setPivot(0, 0, 0)
|
|
||||||
.setRotation(rotation);
|
|
||||||
|
|
||||||
updateLight(model);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void updateLight() {
|
|
||||||
model.get().ifPresent(this::updateLight);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void updateLight(OrientedData model) {
|
|
||||||
BlockPos pos = entity.getHangingPosition();
|
|
||||||
model.setBlockLight(world.getBrightness(LightLayer.BLOCK, pos))
|
|
||||||
.setSkyLight(world.getBrightness(LightLayer.SKY, pos));
|
|
||||||
}
|
|
||||||
|
|
||||||
private boolean shouldShow() {
|
|
||||||
Player player = Minecraft.getInstance().player;
|
|
||||||
|
|
||||||
return entity.isVisible()
|
|
||||||
|| AllItems.SUPER_GLUE.isIn(player.getMainHandItem())
|
|
||||||
|| AllItems.SUPER_GLUE.isIn(player.getOffhandItem());
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,99 +0,0 @@
|
||||||
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.vertex.VertexList;
|
|
||||||
import com.jozufozu.flywheel.core.Formats;
|
|
||||||
import com.jozufozu.flywheel.core.model.Model;
|
|
||||||
import com.jozufozu.flywheel.core.vertex.PosTexNormalWriterUnsafe;
|
|
||||||
import com.mojang.blaze3d.platform.MemoryTracker;
|
|
||||||
import com.simibubi.create.AllStitchedTextures;
|
|
||||||
import com.simibubi.create.foundation.utility.VecHelper;
|
|
||||||
|
|
||||||
import net.minecraft.client.renderer.texture.TextureAtlasSprite;
|
|
||||||
import net.minecraft.core.Direction;
|
|
||||||
import net.minecraft.world.phys.Vec3;
|
|
||||||
|
|
||||||
public class GlueModel implements Model {
|
|
||||||
|
|
||||||
public static final GlueModel INSTANCE = new GlueModel();
|
|
||||||
static final boolean USE_ATLAS = false;
|
|
||||||
|
|
||||||
public static GlueModel get() {
|
|
||||||
return INSTANCE;
|
|
||||||
}
|
|
||||||
|
|
||||||
private final VertexList reader;
|
|
||||||
|
|
||||||
private GlueModel() {
|
|
||||||
PosTexNormalWriterUnsafe writer = Formats.POS_TEX_NORMAL.createWriter(MemoryTracker.create(size()));
|
|
||||||
createGlueModel(writer);
|
|
||||||
reader = writer.intoReader();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String name() {
|
|
||||||
return "glue";
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int vertexCount() {
|
|
||||||
return 8;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public VertexList getReader() {
|
|
||||||
return reader;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void createGlueModel(PosTexNormalWriterUnsafe buffer) {
|
|
||||||
Vec3 diff = Vec3.atLowerCornerOf(Direction.SOUTH.getNormal());
|
|
||||||
Vec3 extension = diff.normalize()
|
|
||||||
.scale(1 / 32f - 1 / 128f);
|
|
||||||
|
|
||||||
Vec3 plane = VecHelper.axisAlingedPlaneOf(diff);
|
|
||||||
Direction.Axis axis = Direction.getNearest(diff.x, diff.y, diff.z)
|
|
||||||
.getAxis();
|
|
||||||
|
|
||||||
Vec3 start = Vec3.ZERO.subtract(extension);
|
|
||||||
Vec3 end = Vec3.ZERO.add(extension);
|
|
||||||
|
|
||||||
plane = plane.scale(1 / 2f);
|
|
||||||
Vec3 a1 = plane.add(start);
|
|
||||||
Vec3 b1 = plane.add(end);
|
|
||||||
plane = VecHelper.rotate(plane, -90, axis);
|
|
||||||
Vec3 a2 = plane.add(start);
|
|
||||||
Vec3 b2 = plane.add(end);
|
|
||||||
plane = VecHelper.rotate(plane, -90, axis);
|
|
||||||
Vec3 a3 = plane.add(start);
|
|
||||||
Vec3 b3 = plane.add(end);
|
|
||||||
plane = VecHelper.rotate(plane, -90, axis);
|
|
||||||
Vec3 a4 = plane.add(start);
|
|
||||||
Vec3 b4 = plane.add(end);
|
|
||||||
|
|
||||||
float minU;
|
|
||||||
float maxU;
|
|
||||||
float minV;
|
|
||||||
float maxV;
|
|
||||||
|
|
||||||
if (USE_ATLAS) {
|
|
||||||
TextureAtlasSprite sprite = AllStitchedTextures.SUPER_GLUE.get();
|
|
||||||
minU = sprite.getU0();
|
|
||||||
maxU = sprite.getU1();
|
|
||||||
minV = sprite.getV0();
|
|
||||||
maxV = sprite.getV1();
|
|
||||||
} else {
|
|
||||||
minU = minV = 0;
|
|
||||||
maxU = maxV = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
// inside quad
|
|
||||||
buffer.putVertex((float) a1.x, (float) a1.y, (float) a1.z, 0, 0, -1, maxU, minV);
|
|
||||||
buffer.putVertex((float) a2.x, (float) a2.y, (float) a2.z, 0, 0, -1, maxU, maxV);
|
|
||||||
buffer.putVertex((float) a3.x, (float) a3.y, (float) a3.z, 0, 0, -1, minU, maxV);
|
|
||||||
buffer.putVertex((float) a4.x, (float) a4.y, (float) a4.z, 0, 0, -1, minU, minV);
|
|
||||||
// outside quad
|
|
||||||
buffer.putVertex((float) b4.x, (float) b4.y, (float) b4.z, 0, 0, 1f, minU, minV);
|
|
||||||
buffer.putVertex((float) b3.x, (float) b3.y, (float) b3.z, 0, 0, 1f, minU, maxV);
|
|
||||||
buffer.putVertex((float) b2.x, (float) b2.y, (float) b2.z, 0, 0, 1f, maxU, maxV);
|
|
||||||
buffer.putVertex((float) b1.x, (float) b1.y, (float) b1.z, 0, 0, 1f, maxU, minV);
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,42 +0,0 @@
|
||||||
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
|
||||||
|
|
||||||
import java.util.Collection;
|
|
||||||
import java.util.LinkedList;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
import com.simibubi.create.AllSoundEvents;
|
|
||||||
import com.simibubi.create.foundation.networking.AllPackets;
|
|
||||||
import com.simibubi.create.foundation.utility.BlockFace;
|
|
||||||
import com.simibubi.create.foundation.utility.WorldAttached;
|
|
||||||
|
|
||||||
import net.minecraft.util.Mth;
|
|
||||||
import net.minecraft.world.level.Level;
|
|
||||||
import net.minecraftforge.network.PacketDistributor;
|
|
||||||
|
|
||||||
public class GlueQueue {
|
|
||||||
|
|
||||||
private WorldAttached<List<BlockFace>> QUEUED_GLUE = new WorldAttached<>(level -> new LinkedList<>());
|
|
||||||
|
|
||||||
public void tick(Level level) {
|
|
||||||
List<BlockFace> list = QUEUED_GLUE.get(level);
|
|
||||||
if (list.isEmpty())
|
|
||||||
return;
|
|
||||||
BlockFace next = list.remove(0);
|
|
||||||
if (!level.isLoaded(next.getPos()))
|
|
||||||
return;
|
|
||||||
|
|
||||||
SuperGlueEntity entity = new SuperGlueEntity(level, next.getPos(), next.getFace());
|
|
||||||
level.addFreshEntity(entity);
|
|
||||||
AllSoundEvents.SLIME_ADDED.playFrom(entity, 0.125F, Mth.clamp(8f / (list.size() + 1), 0.75f, 1f));
|
|
||||||
|
|
||||||
AllPackets.channel.send(PacketDistributor.ALL.noArg(),
|
|
||||||
new GlueEffectPacket(entity.getHangingPosition(), entity.getFacingDirection()
|
|
||||||
.getOpposite(), false));
|
|
||||||
}
|
|
||||||
|
|
||||||
public void add(Level level, Collection<BlockFace> entries) {
|
|
||||||
QUEUED_GLUE.get(level)
|
|
||||||
.addAll(entries);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,8 +1,6 @@
|
||||||
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
||||||
|
|
||||||
import javax.annotation.Nullable;
|
import java.util.Set;
|
||||||
|
|
||||||
import org.apache.commons.lang3.Validate;
|
|
||||||
|
|
||||||
import com.simibubi.create.AllBlocks;
|
import com.simibubi.create.AllBlocks;
|
||||||
import com.simibubi.create.AllEntityTypes;
|
import com.simibubi.create.AllEntityTypes;
|
||||||
|
@ -15,23 +13,19 @@ import com.simibubi.create.content.contraptions.components.structureMovement.cha
|
||||||
import com.simibubi.create.content.schematics.ISpecialEntityItemRequirement;
|
import com.simibubi.create.content.schematics.ISpecialEntityItemRequirement;
|
||||||
import com.simibubi.create.content.schematics.ItemRequirement;
|
import com.simibubi.create.content.schematics.ItemRequirement;
|
||||||
import com.simibubi.create.content.schematics.ItemRequirement.ItemUseType;
|
import com.simibubi.create.content.schematics.ItemRequirement.ItemUseType;
|
||||||
import com.simibubi.create.foundation.networking.AllPackets;
|
import com.simibubi.create.foundation.utility.Iterate;
|
||||||
import com.simibubi.create.foundation.utility.AnimationTickHolder;
|
import com.simibubi.create.foundation.utility.VecHelper;
|
||||||
import com.simibubi.create.foundation.utility.BlockFace;
|
|
||||||
import com.simibubi.create.foundation.utility.worldWrappers.WrappedWorld;
|
|
||||||
|
|
||||||
import net.minecraft.client.Minecraft;
|
|
||||||
import net.minecraft.client.multiplayer.ClientLevel;
|
|
||||||
import net.minecraft.client.player.LocalPlayer;
|
|
||||||
import net.minecraft.core.BlockPos;
|
import net.minecraft.core.BlockPos;
|
||||||
import net.minecraft.core.Direction;
|
import net.minecraft.core.Direction;
|
||||||
import net.minecraft.core.Direction.Axis;
|
import net.minecraft.core.Direction.Axis;
|
||||||
|
import net.minecraft.core.Direction.AxisDirection;
|
||||||
|
import net.minecraft.core.particles.ParticleTypes;
|
||||||
import net.minecraft.nbt.CompoundTag;
|
import net.minecraft.nbt.CompoundTag;
|
||||||
|
import net.minecraft.nbt.Tag;
|
||||||
import net.minecraft.network.FriendlyByteBuf;
|
import net.minecraft.network.FriendlyByteBuf;
|
||||||
import net.minecraft.network.protocol.Packet;
|
import net.minecraft.network.protocol.Packet;
|
||||||
import net.minecraft.server.level.ServerLevel;
|
import net.minecraft.server.level.ServerLevel;
|
||||||
import net.minecraft.sounds.SoundEvents;
|
|
||||||
import net.minecraft.util.Mth;
|
|
||||||
import net.minecraft.world.InteractionHand;
|
import net.minecraft.world.InteractionHand;
|
||||||
import net.minecraft.world.InteractionResult;
|
import net.minecraft.world.InteractionResult;
|
||||||
import net.minecraft.world.damagesource.DamageSource;
|
import net.minecraft.world.damagesource.DamageSource;
|
||||||
|
@ -39,14 +33,11 @@ import net.minecraft.world.entity.Entity;
|
||||||
import net.minecraft.world.entity.EntityDimensions;
|
import net.minecraft.world.entity.EntityDimensions;
|
||||||
import net.minecraft.world.entity.EntityType;
|
import net.minecraft.world.entity.EntityType;
|
||||||
import net.minecraft.world.entity.LightningBolt;
|
import net.minecraft.world.entity.LightningBolt;
|
||||||
import net.minecraft.world.entity.Mob;
|
|
||||||
import net.minecraft.world.entity.MoverType;
|
import net.minecraft.world.entity.MoverType;
|
||||||
import net.minecraft.world.entity.Pose;
|
import net.minecraft.world.entity.Pose;
|
||||||
import net.minecraft.world.entity.item.ItemEntity;
|
|
||||||
import net.minecraft.world.entity.player.Player;
|
import net.minecraft.world.entity.player.Player;
|
||||||
import net.minecraft.world.item.ItemStack;
|
|
||||||
import net.minecraft.world.level.GameRules;
|
|
||||||
import net.minecraft.world.level.Level;
|
import net.minecraft.world.level.Level;
|
||||||
|
import net.minecraft.world.level.LevelAccessor;
|
||||||
import net.minecraft.world.level.block.Blocks;
|
import net.minecraft.world.level.block.Blocks;
|
||||||
import net.minecraft.world.level.block.DirectionalBlock;
|
import net.minecraft.world.level.block.DirectionalBlock;
|
||||||
import net.minecraft.world.level.block.Mirror;
|
import net.minecraft.world.level.block.Mirror;
|
||||||
|
@ -54,145 +45,51 @@ import net.minecraft.world.level.block.Rotation;
|
||||||
import net.minecraft.world.level.block.state.BlockState;
|
import net.minecraft.world.level.block.state.BlockState;
|
||||||
import net.minecraft.world.level.block.state.properties.BooleanProperty;
|
import net.minecraft.world.level.block.state.properties.BooleanProperty;
|
||||||
import net.minecraft.world.phys.AABB;
|
import net.minecraft.world.phys.AABB;
|
||||||
import net.minecraft.world.phys.BlockHitResult;
|
|
||||||
import net.minecraft.world.phys.HitResult;
|
|
||||||
import net.minecraft.world.phys.HitResult.Type;
|
|
||||||
import net.minecraft.world.phys.Vec3;
|
import net.minecraft.world.phys.Vec3;
|
||||||
import net.minecraftforge.api.distmarker.Dist;
|
|
||||||
import net.minecraftforge.api.distmarker.OnlyIn;
|
|
||||||
import net.minecraftforge.common.util.FakePlayer;
|
|
||||||
import net.minecraftforge.entity.IEntityAdditionalSpawnData;
|
import net.minecraftforge.entity.IEntityAdditionalSpawnData;
|
||||||
import net.minecraftforge.fml.DistExecutor;
|
|
||||||
import net.minecraftforge.network.NetworkHooks;
|
import net.minecraftforge.network.NetworkHooks;
|
||||||
import net.minecraftforge.network.PacketDistributor;
|
|
||||||
|
|
||||||
public class SuperGlueEntity extends Entity
|
public class SuperGlueEntity extends Entity implements IEntityAdditionalSpawnData, ISpecialEntityItemRequirement {
|
||||||
implements IEntityAdditionalSpawnData, ISpecialEntityItemRequirement {
|
|
||||||
|
|
||||||
private int validationTimer;
|
public static AABB span(BlockPos startPos, BlockPos endPos) {
|
||||||
protected BlockPos hangingPosition;
|
return new AABB(startPos, endPos).expandTowards(1, 1, 1);
|
||||||
protected Direction facingDirection = Direction.SOUTH;
|
}
|
||||||
|
|
||||||
|
public static boolean isGlued(LevelAccessor level, BlockPos blockPos, Direction direction,
|
||||||
|
Set<SuperGlueEntity> cached) {
|
||||||
|
BlockPos targetPos = blockPos.relative(direction);
|
||||||
|
if (cached != null)
|
||||||
|
for (SuperGlueEntity glueEntity : cached)
|
||||||
|
if (glueEntity.contains(blockPos) && glueEntity.contains(targetPos))
|
||||||
|
return true;
|
||||||
|
for (SuperGlueEntity glueEntity : level.getEntitiesOfClass(SuperGlueEntity.class, span(blockPos, targetPos))) {
|
||||||
|
if (!glueEntity.contains(blockPos) || !glueEntity.contains(targetPos))
|
||||||
|
continue;
|
||||||
|
if (cached != null)
|
||||||
|
cached.add(glueEntity);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
public SuperGlueEntity(EntityType<?> type, Level world) {
|
public SuperGlueEntity(EntityType<?> type, Level world) {
|
||||||
super(type, world);
|
super(type, world);
|
||||||
}
|
}
|
||||||
|
|
||||||
public SuperGlueEntity(Level world, BlockPos pos, Direction direction) {
|
public SuperGlueEntity(Level world, AABB boundingBox) {
|
||||||
this(AllEntityTypes.SUPER_GLUE.get(), world);
|
this(AllEntityTypes.SUPER_GLUE.get(), world);
|
||||||
hangingPosition = pos;
|
setBoundingBox(boundingBox);
|
||||||
facingDirection = direction;
|
resetPositionToBB();
|
||||||
updateFacingWithBoundingBox();
|
}
|
||||||
|
|
||||||
|
public void resetPositionToBB() {
|
||||||
|
AABB bb = getBoundingBox();
|
||||||
|
setPosRaw(bb.getCenter().x, bb.minY, bb.getCenter().z);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void defineSynchedData() {}
|
protected void defineSynchedData() {}
|
||||||
|
|
||||||
public int getWidthPixels() {
|
|
||||||
return 12;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getHeightPixels() {
|
|
||||||
return 12;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void onBroken(@Nullable Entity breaker) {
|
|
||||||
playSound(SoundEvents.SLIME_SQUISH_SMALL, 1.0F, 1.0F);
|
|
||||||
if (onValidSurface()) {
|
|
||||||
AllPackets.channel.send(PacketDistributor.TRACKING_ENTITY.with(() -> this),
|
|
||||||
new GlueEffectPacket(getHangingPosition(), getFacingDirection().getOpposite(), false));
|
|
||||||
AllSoundEvents.SLIME_ADDED.playFrom(this, 0.5F, 0.5F);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void playPlaceSound() {
|
|
||||||
AllSoundEvents.SLIME_ADDED.playFrom(this, 0.5F, 0.75F);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected void updateFacingWithBoundingBox() {
|
|
||||||
Validate.notNull(getFacingDirection());
|
|
||||||
if (getFacingDirection().getAxis()
|
|
||||||
.isHorizontal()) {
|
|
||||||
setXRot(0);
|
|
||||||
setYRot(getFacingDirection().get2DDataValue() * 90);
|
|
||||||
} else {
|
|
||||||
setXRot(-90 * getFacingDirection().getAxisDirection()
|
|
||||||
.getStep());
|
|
||||||
setYRot(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.xRotO = this.getXRot();
|
|
||||||
this.yRotO = this.getYRot();
|
|
||||||
this.updateBoundingBox();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected void updateBoundingBox() {
|
|
||||||
if (this.getFacingDirection() != null) {
|
|
||||||
double offset = 0.5 - 1 / 256d;
|
|
||||||
double x = hangingPosition.getX() + 0.5 - facingDirection.getStepX() * offset;
|
|
||||||
double y = hangingPosition.getY() + 0.5 - facingDirection.getStepY() * offset;
|
|
||||||
double z = hangingPosition.getZ() + 0.5 - facingDirection.getStepZ() * offset;
|
|
||||||
this.setPosRaw(x, y, z);
|
|
||||||
double w = getWidthPixels();
|
|
||||||
double h = getHeightPixels();
|
|
||||||
double l = getWidthPixels();
|
|
||||||
Axis axis = this.getFacingDirection()
|
|
||||||
.getAxis();
|
|
||||||
double depth = 2 - 1 / 128f;
|
|
||||||
|
|
||||||
switch (axis) {
|
|
||||||
case X -> w = depth;
|
|
||||||
case Y -> h = depth;
|
|
||||||
case Z -> l = depth;
|
|
||||||
}
|
|
||||||
|
|
||||||
w = w / 32.0D;
|
|
||||||
h = h / 32.0D;
|
|
||||||
l = l / 32.0D;
|
|
||||||
this.setBoundingBox(new AABB(x - w, y - h, z - l, x + w, y + h, z + l));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void tick() {
|
|
||||||
if (this.validationTimer++ == 10 && !this.level.isClientSide) {
|
|
||||||
this.validationTimer = 0;
|
|
||||||
if (isAlive() && !this.onValidSurface()) {
|
|
||||||
kill();
|
|
||||||
onBroken(null);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isVisible() {
|
|
||||||
if (!isAlive())
|
|
||||||
return false;
|
|
||||||
if (level instanceof WrappedWorld)
|
|
||||||
return true;
|
|
||||||
|
|
||||||
BlockPos pos = hangingPosition;
|
|
||||||
BlockPos pos2 = pos.relative(getFacingDirection().getOpposite());
|
|
||||||
return isValidFace(level, pos2, getFacingDirection()) != isValidFace(level, pos,
|
|
||||||
getFacingDirection().getOpposite());
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean onValidSurface() {
|
|
||||||
BlockPos pos = hangingPosition;
|
|
||||||
BlockPos pos2 = hangingPosition.relative(getFacingDirection().getOpposite());
|
|
||||||
if (level.isOutsideBuildHeight(pos2))
|
|
||||||
return false;
|
|
||||||
if (!level.isLoaded(pos) || !level.isLoaded(pos2))
|
|
||||||
return true;
|
|
||||||
if (!isValidFace(level, pos2, getFacingDirection())
|
|
||||||
&& !isValidFace(level, pos, getFacingDirection().getOpposite()))
|
|
||||||
return false;
|
|
||||||
if (isSideSticky(level, pos2, getFacingDirection())
|
|
||||||
|| isSideSticky(level, pos, getFacingDirection().getOpposite()))
|
|
||||||
return false;
|
|
||||||
return level.getEntities(this, getBoundingBox(), e -> e instanceof SuperGlueEntity)
|
|
||||||
.isEmpty();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static boolean isValidFace(Level world, BlockPos pos, Direction direction) {
|
public static boolean isValidFace(Level world, BlockPos pos, Direction direction) {
|
||||||
BlockState state = world.getBlockState(pos);
|
BlockState state = world.getBlockState(pos);
|
||||||
if (BlockMovementChecks.isBlockAttachedTowards(state, world, pos, direction))
|
if (BlockMovementChecks.isBlockAttachedTowards(state, world, pos, direction))
|
||||||
|
@ -237,61 +134,36 @@ public class SuperGlueEntity extends Entity
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean isPickable() {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean skipAttackInteraction(Entity entity) {
|
|
||||||
return entity instanceof Player ? hurt(DamageSource.playerAttack((Player) entity), 0) : false;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Direction getDirection() {
|
|
||||||
return this.getFacingDirection();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean hurt(DamageSource source, float amount) {
|
public boolean hurt(DamageSource source, float amount) {
|
||||||
if (this.isInvulnerableTo(source))
|
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
boolean mobGriefing = level.getGameRules()
|
|
||||||
.getBoolean(GameRules.RULE_MOBGRIEFING);
|
|
||||||
Entity trueSource = source.getEntity();
|
|
||||||
if (!mobGriefing && trueSource instanceof Mob)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
Entity immediateSource = source.getDirectEntity();
|
|
||||||
if (!isVisible() && immediateSource instanceof Player) {
|
|
||||||
if (!AllItems.SUPER_GLUE.isIn(((Player) immediateSource).getMainHandItem()))
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isAlive() && !level.isClientSide) {
|
@Override
|
||||||
onBroken(source.getEntity());
|
public void tick() {
|
||||||
kill();
|
super.tick();
|
||||||
markHurt();
|
if (getBoundingBox().getXsize() == 0)
|
||||||
|
discard();
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
@Override
|
||||||
|
public void setPos(double x, double y, double z) {
|
||||||
|
setPosRaw(x, y, z);
|
||||||
|
getBoundingBox().move(getBoundingBox().getCenter()
|
||||||
|
.scale(-1)
|
||||||
|
.add(x, y, z));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void move(MoverType typeIn, Vec3 pos) {
|
public void move(MoverType typeIn, Vec3 pos) {
|
||||||
if (!level.isClientSide && isAlive() && pos.lengthSqr() > 0.0D) {
|
if (!level.isClientSide && isAlive() && pos.lengthSqr() > 0.0D)
|
||||||
discard();
|
discard();
|
||||||
onBroken(null);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void push(double x, double y, double z) {
|
public void push(double x, double y, double z) {
|
||||||
if (!level.isClientSide && isAlive() && x * x + y * y + z * z > 0.0D) {
|
if (!level.isClientSide && isAlive() && x * x + y * y + z * z > 0.0D)
|
||||||
discard();
|
discard();
|
||||||
onBroken(null);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -299,9 +171,8 @@ public class SuperGlueEntity extends Entity
|
||||||
return 0.0F;
|
return 0.0F;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
public void playPlaceSound() {
|
||||||
public ItemStack getPickedResult(HitResult target) {
|
AllSoundEvents.SLIME_ADDED.playFrom(this, 0.5F, 0.75F);
|
||||||
return AllItems.SUPER_GLUE.asStack();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -311,79 +182,30 @@ public class SuperGlueEntity extends Entity
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public InteractionResult interact(Player player, InteractionHand hand) {
|
public InteractionResult interact(Player player, InteractionHand hand) {
|
||||||
if (player instanceof FakePlayer)
|
|
||||||
return InteractionResult.PASS;
|
return InteractionResult.PASS;
|
||||||
DistExecutor.unsafeRunWhenOn(Dist.CLIENT, () -> () -> {
|
|
||||||
triggerPlaceBlock(player, hand);
|
|
||||||
});
|
|
||||||
return InteractionResult.CONSUME;
|
|
||||||
}
|
|
||||||
|
|
||||||
@OnlyIn(Dist.CLIENT)
|
|
||||||
private void triggerPlaceBlock(Player player, InteractionHand hand) {
|
|
||||||
if (!(player instanceof LocalPlayer))
|
|
||||||
return;
|
|
||||||
if (!(player.level instanceof ClientLevel))
|
|
||||||
return;
|
|
||||||
|
|
||||||
LocalPlayer cPlayer = (LocalPlayer) player;
|
|
||||||
Minecraft mc = Minecraft.getInstance();
|
|
||||||
HitResult ray = cPlayer.pick(mc.gameMode.getPickRange(), AnimationTickHolder.getPartialTicks(), false);
|
|
||||||
|
|
||||||
if (!(ray instanceof BlockHitResult))
|
|
||||||
return;
|
|
||||||
if (ray.getType() == Type.MISS)
|
|
||||||
return;
|
|
||||||
BlockHitResult blockRay = (BlockHitResult) ray;
|
|
||||||
BlockFace rayFace = new BlockFace(blockRay.getBlockPos(), blockRay.getDirection());
|
|
||||||
BlockFace hangingFace = new BlockFace(getHangingPosition(), getFacingDirection().getOpposite());
|
|
||||||
if (!rayFace.isEquivalent(hangingFace))
|
|
||||||
return;
|
|
||||||
|
|
||||||
for (InteractionHand handIn : InteractionHand.values()) {
|
|
||||||
ItemStack itemstack = cPlayer.getItemInHand(handIn);
|
|
||||||
int countBefore = itemstack.getCount();
|
|
||||||
InteractionResult actionResultType =
|
|
||||||
mc.gameMode.useItemOn(cPlayer, (ClientLevel) cPlayer.level, handIn, blockRay);
|
|
||||||
if (actionResultType != InteractionResult.SUCCESS)
|
|
||||||
return;
|
|
||||||
|
|
||||||
cPlayer.swing(handIn);
|
|
||||||
if (!itemstack.isEmpty() && (itemstack.getCount() != countBefore || mc.gameMode.hasInfiniteItems()))
|
|
||||||
mc.gameRenderer.itemInHandRenderer.itemUsed(handIn);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void addAdditionalSaveData(CompoundTag compound) {
|
public void addAdditionalSaveData(CompoundTag compound) {
|
||||||
compound.putByte("Facing", (byte) this.getFacingDirection()
|
Vec3 position = position();
|
||||||
.get3DDataValue());
|
writeBoundingBox(compound, getBoundingBox().move(position.scale(-1)));
|
||||||
BlockPos blockpos = this.getHangingPosition();
|
|
||||||
compound.putInt("TileX", blockpos.getX());
|
|
||||||
compound.putInt("TileY", blockpos.getY());
|
|
||||||
compound.putInt("TileZ", blockpos.getZ());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void readAdditionalSaveData(CompoundTag compound) {
|
public void readAdditionalSaveData(CompoundTag compound) {
|
||||||
this.hangingPosition =
|
Vec3 position = position();
|
||||||
new BlockPos(compound.getInt("TileX"), compound.getInt("TileY"), compound.getInt("TileZ"));
|
setBoundingBox(readBoundingBox(compound).move(position));
|
||||||
this.facingDirection = Direction.from3DDataValue(compound.getByte("Facing"));
|
|
||||||
updateFacingWithBoundingBox();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
public static void writeBoundingBox(CompoundTag compound, AABB bb) {
|
||||||
public ItemEntity spawnAtLocation(ItemStack stack, float yOffset) {
|
compound.put("From", VecHelper.writeNBT(new Vec3(bb.minX, bb.minY, bb.minZ)));
|
||||||
float xOffset = (float) this.getFacingDirection()
|
compound.put("To", VecHelper.writeNBT(new Vec3(bb.maxX, bb.maxY, bb.maxZ)));
|
||||||
.getStepX() * 0.15F;
|
}
|
||||||
float zOffset = (float) this.getFacingDirection()
|
|
||||||
.getStepZ() * 0.15F;
|
public static AABB readBoundingBox(CompoundTag compound) {
|
||||||
ItemEntity itementity =
|
Vec3 from = VecHelper.readNBT(compound.getList("From", Tag.TAG_DOUBLE));
|
||||||
new ItemEntity(this.level, this.getX() + xOffset, this.getY() + yOffset, this.getZ() + zOffset, stack);
|
Vec3 to = VecHelper.readNBT(compound.getList("To", Tag.TAG_DOUBLE));
|
||||||
itementity.setDefaultPickUpDelay();
|
return new AABB(from, to);
|
||||||
this.level.addFreshEntity(itementity);
|
|
||||||
return itementity;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -391,51 +213,17 @@ public class SuperGlueEntity extends Entity
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setPos(double x, double y, double z) {
|
|
||||||
hangingPosition = new BlockPos(x, y, z);
|
|
||||||
updateBoundingBox();
|
|
||||||
hasImpulse = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public float rotate(Rotation transformRotation) {
|
public float rotate(Rotation transformRotation) {
|
||||||
if (this.getFacingDirection()
|
AABB bb = getBoundingBox().move(position().scale(-1));
|
||||||
.getAxis() != Direction.Axis.Y) {
|
if (transformRotation == Rotation.CLOCKWISE_90 || transformRotation == Rotation.COUNTERCLOCKWISE_90)
|
||||||
switch (transformRotation) {
|
setBoundingBox(new AABB(bb.minZ, bb.minY, bb.minX, bb.maxZ, bb.maxY, bb.maxX).move(position()));
|
||||||
case CLOCKWISE_180:
|
return super.rotate(transformRotation);
|
||||||
facingDirection = facingDirection.getOpposite();
|
|
||||||
break;
|
|
||||||
case COUNTERCLOCKWISE_90:
|
|
||||||
facingDirection = facingDirection.getCounterClockWise();
|
|
||||||
break;
|
|
||||||
case CLOCKWISE_90:
|
|
||||||
facingDirection = facingDirection.getClockWise();
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
float f = Mth.wrapDegrees(this.getYRot());
|
|
||||||
return switch (transformRotation) {
|
|
||||||
case CLOCKWISE_180 -> f + 180.0F;
|
|
||||||
case COUNTERCLOCKWISE_90 -> f + 90.0F;
|
|
||||||
case CLOCKWISE_90 -> f + 270.0F;
|
|
||||||
default -> f;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public BlockPos getHangingPosition() {
|
|
||||||
return this.hangingPosition;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public float mirror(Mirror transformMirror) {
|
public float mirror(Mirror transformMirror) {
|
||||||
return this.rotate(transformMirror.getRotation(this.getFacingDirection()));
|
return super.mirror(transformMirror);
|
||||||
}
|
|
||||||
|
|
||||||
public Direction getAttachedDirection(BlockPos pos) {
|
|
||||||
return !pos.equals(hangingPosition) ? getFacingDirection() : getFacingDirection().getOpposite();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -467,10 +255,6 @@ public class SuperGlueEntity extends Entity
|
||||||
readAdditionalSaveData(additionalData.readNbt());
|
readAdditionalSaveData(additionalData.readNbt());
|
||||||
}
|
}
|
||||||
|
|
||||||
public Direction getFacingDirection() {
|
|
||||||
return facingDirection;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ItemRequirement getRequiredItems() {
|
public ItemRequirement getRequiredItems() {
|
||||||
return new ItemRequirement(ItemUseType.DAMAGE, AllItems.SUPER_GLUE.get());
|
return new ItemRequirement(ItemUseType.DAMAGE, AllItems.SUPER_GLUE.get());
|
||||||
|
@ -480,4 +264,53 @@ public class SuperGlueEntity extends Entity
|
||||||
public boolean isIgnoringBlockTriggers() {
|
public boolean isIgnoringBlockTriggers() {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean contains(BlockPos pos) {
|
||||||
|
return getBoundingBox().contains(Vec3.atCenterOf(pos));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void spawnParticles() {
|
||||||
|
AABB bb = getBoundingBox();
|
||||||
|
Vec3 origin = new Vec3(bb.minX, bb.minY, bb.minZ);
|
||||||
|
Vec3 extents = new Vec3(bb.getXsize(), bb.getYsize(), bb.getZsize());
|
||||||
|
|
||||||
|
if (!(level instanceof ServerLevel slevel))
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (Axis axis : Iterate.axes) {
|
||||||
|
AxisDirection positive = AxisDirection.POSITIVE;
|
||||||
|
double max = axis.choose(extents.x, extents.y, extents.z);
|
||||||
|
Vec3 normal = Vec3.atLowerCornerOf(Direction.fromAxisAndDirection(axis, positive)
|
||||||
|
.getNormal());
|
||||||
|
for (Axis axis2 : Iterate.axes) {
|
||||||
|
if (axis2 == axis)
|
||||||
|
continue;
|
||||||
|
double max2 = axis2.choose(extents.x, extents.y, extents.z);
|
||||||
|
Vec3 normal2 = Vec3.atLowerCornerOf(Direction.fromAxisAndDirection(axis2, positive)
|
||||||
|
.getNormal());
|
||||||
|
for (Axis axis3 : Iterate.axes) {
|
||||||
|
if (axis3 == axis2 || axis3 == axis)
|
||||||
|
continue;
|
||||||
|
double max3 = axis3.choose(extents.x, extents.y, extents.z);
|
||||||
|
Vec3 normal3 = Vec3.atLowerCornerOf(Direction.fromAxisAndDirection(axis3, positive)
|
||||||
|
.getNormal());
|
||||||
|
|
||||||
|
for (int i = 0; i <= max * 2; i++) {
|
||||||
|
for (int o1 : Iterate.zeroAndOne) {
|
||||||
|
for (int o2 : Iterate.zeroAndOne) {
|
||||||
|
Vec3 v = origin.add(normal.scale(i / 2f))
|
||||||
|
.add(normal2.scale(max2 * o1))
|
||||||
|
.add(normal3.scale(max3 * o2));
|
||||||
|
|
||||||
|
slevel.sendParticles(ParticleTypes.ITEM_SLIME, v.x, v.y, v.z, 1, 0, 0, 0, 0);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,11 +1,12 @@
|
||||||
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
||||||
|
|
||||||
import java.util.HashMap;
|
import java.util.HashSet;
|
||||||
import java.util.List;
|
import java.util.Set;
|
||||||
import java.util.Map;
|
|
||||||
|
|
||||||
import com.simibubi.create.AllItems;
|
import com.simibubi.create.AllItems;
|
||||||
|
import com.simibubi.create.content.contraptions.components.structureMovement.BlockMovementChecks;
|
||||||
import com.simibubi.create.foundation.networking.AllPackets;
|
import com.simibubi.create.foundation.networking.AllPackets;
|
||||||
|
import com.simibubi.create.foundation.utility.Iterate;
|
||||||
import com.simibubi.create.foundation.utility.placement.IPlacementHelper;
|
import com.simibubi.create.foundation.utility.placement.IPlacementHelper;
|
||||||
import com.simibubi.create.foundation.utility.worldWrappers.RayTraceWorld;
|
import com.simibubi.create.foundation.utility.worldWrappers.RayTraceWorld;
|
||||||
|
|
||||||
|
@ -21,7 +22,6 @@ import net.minecraft.world.level.ClipContext;
|
||||||
import net.minecraft.world.level.Level;
|
import net.minecraft.world.level.Level;
|
||||||
import net.minecraft.world.level.LevelAccessor;
|
import net.minecraft.world.level.LevelAccessor;
|
||||||
import net.minecraft.world.level.block.Blocks;
|
import net.minecraft.world.level.block.Blocks;
|
||||||
import net.minecraft.world.phys.AABB;
|
|
||||||
import net.minecraft.world.phys.BlockHitResult;
|
import net.minecraft.world.phys.BlockHitResult;
|
||||||
import net.minecraft.world.phys.HitResult.Type;
|
import net.minecraft.world.phys.HitResult.Type;
|
||||||
import net.minecraft.world.phys.Vec3;
|
import net.minecraft.world.phys.Vec3;
|
||||||
|
@ -34,14 +34,6 @@ import net.minecraftforge.network.PacketDistributor;
|
||||||
@EventBusSubscriber
|
@EventBusSubscriber
|
||||||
public class SuperGlueHandler {
|
public class SuperGlueHandler {
|
||||||
|
|
||||||
public static Map<Direction, SuperGlueEntity> gatherGlue(LevelAccessor world, BlockPos pos) {
|
|
||||||
List<SuperGlueEntity> entities = world.getEntitiesOfClass(SuperGlueEntity.class, new AABB(pos));
|
|
||||||
Map<Direction, SuperGlueEntity> map = new HashMap<>();
|
|
||||||
for (SuperGlueEntity entity : entities)
|
|
||||||
map.put(entity.getAttachedDirection(pos), entity);
|
|
||||||
return map;
|
|
||||||
}
|
|
||||||
|
|
||||||
@SubscribeEvent
|
@SubscribeEvent
|
||||||
public static void glueListensForBlockPlacement(EntityPlaceEvent event) {
|
public static void glueListensForBlockPlacement(EntityPlaceEvent event) {
|
||||||
LevelAccessor world = event.getWorld();
|
LevelAccessor world = event.getWorld();
|
||||||
|
@ -53,10 +45,14 @@ public class SuperGlueHandler {
|
||||||
if (world.isClientSide())
|
if (world.isClientSide())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
Map<Direction, SuperGlueEntity> gatheredGlue = gatherGlue(world, pos);
|
Set<SuperGlueEntity> cached = new HashSet<>();
|
||||||
for (Direction direction : gatheredGlue.keySet())
|
for (Direction direction : Iterate.directions) {
|
||||||
|
BlockPos relative = pos.relative(direction);
|
||||||
|
if (SuperGlueEntity.isGlued(world, pos, direction, cached)
|
||||||
|
&& BlockMovementChecks.isMovementNecessary(world.getBlockState(relative), entity.level, relative))
|
||||||
AllPackets.channel.send(PacketDistributor.TRACKING_ENTITY_AND_SELF.with(() -> entity),
|
AllPackets.channel.send(PacketDistributor.TRACKING_ENTITY_AND_SELF.with(() -> entity),
|
||||||
new GlueEffectPacket(pos, direction, true));
|
new GlueEffectPacket(pos, direction, true));
|
||||||
|
}
|
||||||
|
|
||||||
if (entity instanceof Player)
|
if (entity instanceof Player)
|
||||||
glueInOffHandAppliesOnBlockPlace(event, pos, (Player) entity);
|
glueInOffHandAppliesOnBlockPlace(event, pos, (Player) entity);
|
||||||
|
@ -80,31 +76,33 @@ public class SuperGlueHandler {
|
||||||
|
|
||||||
RayTraceWorld rayTraceWorld =
|
RayTraceWorld rayTraceWorld =
|
||||||
new RayTraceWorld(world, (p, state) -> p.equals(pos) ? Blocks.AIR.defaultBlockState() : state);
|
new RayTraceWorld(world, (p, state) -> p.equals(pos) ? Blocks.AIR.defaultBlockState() : state);
|
||||||
BlockHitResult ray = rayTraceWorld.clip(
|
BlockHitResult ray =
|
||||||
new ClipContext(start, end, ClipContext.Block.OUTLINE, ClipContext.Fluid.NONE, placer));
|
rayTraceWorld.clip(new ClipContext(start, end, ClipContext.Block.OUTLINE, ClipContext.Fluid.NONE, placer));
|
||||||
|
|
||||||
Direction face = ray.getDirection();
|
Direction face = ray.getDirection();
|
||||||
if (face == null || ray.getType() == Type.MISS)
|
if (face == null || ray.getType() == Type.MISS)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (!ray.getBlockPos()
|
BlockPos gluePos = ray.getBlockPos();
|
||||||
.relative(face)
|
if (!gluePos.relative(face)
|
||||||
.equals(pos)) {
|
.equals(pos)) {
|
||||||
event.setCanceled(true);
|
event.setCanceled(true);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
SuperGlueEntity entity = new SuperGlueEntity(world, ray.getBlockPos(), face.getOpposite());
|
if (SuperGlueEntity.isGlued(world, gluePos, face, null))
|
||||||
|
return;
|
||||||
|
|
||||||
|
SuperGlueEntity entity = new SuperGlueEntity(world, SuperGlueEntity.span(gluePos, gluePos.relative(face)));
|
||||||
CompoundTag compoundnbt = itemstack.getTag();
|
CompoundTag compoundnbt = itemstack.getTag();
|
||||||
if (compoundnbt != null)
|
if (compoundnbt != null)
|
||||||
EntityType.updateCustomEntityTag(world, placer, entity, compoundnbt);
|
EntityType.updateCustomEntityTag(world, placer, entity, compoundnbt);
|
||||||
|
|
||||||
if (entity.onValidSurface()) {
|
if (SuperGlueEntity.isValidFace(world, gluePos, face)) {
|
||||||
if (!world.isClientSide) {
|
if (!world.isClientSide) {
|
||||||
entity.playPlaceSound();
|
|
||||||
world.addFreshEntity(entity);
|
world.addFreshEntity(entity);
|
||||||
AllPackets.channel.send(PacketDistributor.TRACKING_ENTITY.with(() -> entity),
|
AllPackets.channel.send(PacketDistributor.TRACKING_ENTITY.with(() -> entity),
|
||||||
new GlueEffectPacket(ray.getBlockPos(), face, true));
|
new GlueEffectPacket(gluePos, face, true));
|
||||||
}
|
}
|
||||||
itemstack.hurtAndBreak(1, placer, SuperGlueItem::onBroken);
|
itemstack.hurtAndBreak(1, placer, SuperGlueItem::onBroken);
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,15 +7,11 @@ import net.minecraft.core.BlockPos;
|
||||||
import net.minecraft.core.Direction;
|
import net.minecraft.core.Direction;
|
||||||
import net.minecraft.core.particles.ItemParticleOption;
|
import net.minecraft.core.particles.ItemParticleOption;
|
||||||
import net.minecraft.core.particles.ParticleTypes;
|
import net.minecraft.core.particles.ParticleTypes;
|
||||||
import net.minecraft.nbt.CompoundTag;
|
|
||||||
import net.minecraft.util.Mth;
|
import net.minecraft.util.Mth;
|
||||||
import net.minecraft.world.InteractionResult;
|
|
||||||
import net.minecraft.world.entity.EntityType;
|
|
||||||
import net.minecraft.world.entity.player.Player;
|
import net.minecraft.world.entity.player.Player;
|
||||||
import net.minecraft.world.item.Item;
|
import net.minecraft.world.item.Item;
|
||||||
import net.minecraft.world.item.ItemStack;
|
import net.minecraft.world.item.ItemStack;
|
||||||
import net.minecraft.world.item.Items;
|
import net.minecraft.world.item.Items;
|
||||||
import net.minecraft.world.item.context.UseOnContext;
|
|
||||||
import net.minecraft.world.level.Level;
|
import net.minecraft.world.level.Level;
|
||||||
import net.minecraft.world.level.block.state.BlockState;
|
import net.minecraft.world.level.block.state.BlockState;
|
||||||
import net.minecraft.world.phys.Vec3;
|
import net.minecraft.world.phys.Vec3;
|
||||||
|
@ -49,47 +45,17 @@ public class SuperGlueItem extends Item {
|
||||||
super(properties);
|
super(properties);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean canAttackBlock(BlockState pState, Level pLevel, BlockPos pPos, Player pPlayer) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean canBeDepleted() {
|
public boolean canBeDepleted() {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
public static void onBroken(Player player) {}
|
||||||
public InteractionResult useOn(UseOnContext context) {
|
|
||||||
BlockPos blockpos = context.getClickedPos();
|
|
||||||
Direction direction = context.getClickedFace();
|
|
||||||
BlockPos blockpos1 = blockpos.relative(direction);
|
|
||||||
Player playerentity = context.getPlayer();
|
|
||||||
ItemStack itemstack = context.getItemInHand();
|
|
||||||
|
|
||||||
if (playerentity == null || !this.canPlace(playerentity, direction, itemstack, blockpos1))
|
|
||||||
return InteractionResult.FAIL;
|
|
||||||
|
|
||||||
Level world = context.getLevel();
|
|
||||||
SuperGlueEntity entity = new SuperGlueEntity(world, blockpos1, direction);
|
|
||||||
CompoundTag compoundnbt = itemstack.getTag();
|
|
||||||
if (compoundnbt != null)
|
|
||||||
EntityType.updateCustomEntityTag(world, playerentity, entity, compoundnbt);
|
|
||||||
|
|
||||||
if (!entity.onValidSurface())
|
|
||||||
return InteractionResult.FAIL;
|
|
||||||
|
|
||||||
if (!world.isClientSide) {
|
|
||||||
entity.playPlaceSound();
|
|
||||||
world.addFreshEntity(entity);
|
|
||||||
}
|
|
||||||
itemstack.hurtAndBreak(1, playerentity, SuperGlueItem::onBroken);
|
|
||||||
|
|
||||||
return InteractionResult.SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void onBroken(Player player) {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
protected boolean canPlace(Player entity, Direction facing, ItemStack stack, BlockPos pos) {
|
|
||||||
return !entity.level.isOutsideBuildHeight(pos) && entity.mayUseItemAt(pos, facing, stack);
|
|
||||||
}
|
|
||||||
|
|
||||||
@OnlyIn(Dist.CLIENT)
|
@OnlyIn(Dist.CLIENT)
|
||||||
public static void spawnParticles(Level world, BlockPos pos, Direction direction, boolean fullBlock) {
|
public static void spawnParticles(Level world, BlockPos pos, Direction direction, boolean fullBlock) {
|
||||||
|
|
|
@ -0,0 +1,53 @@
|
||||||
|
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
||||||
|
|
||||||
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
|
import com.simibubi.create.AllSoundEvents;
|
||||||
|
import com.simibubi.create.foundation.networking.SimplePacketBase;
|
||||||
|
|
||||||
|
import net.minecraft.core.BlockPos;
|
||||||
|
import net.minecraft.network.FriendlyByteBuf;
|
||||||
|
import net.minecraft.server.level.ServerPlayer;
|
||||||
|
import net.minecraft.world.entity.Entity;
|
||||||
|
import net.minecraftforge.network.NetworkEvent.Context;
|
||||||
|
|
||||||
|
public class SuperGlueRemovalPacket extends SimplePacketBase {
|
||||||
|
|
||||||
|
private int entityId;
|
||||||
|
private BlockPos soundSource;
|
||||||
|
|
||||||
|
public SuperGlueRemovalPacket(int id, BlockPos soundSource) {
|
||||||
|
entityId = id;
|
||||||
|
this.soundSource = soundSource;
|
||||||
|
}
|
||||||
|
|
||||||
|
public SuperGlueRemovalPacket(FriendlyByteBuf buffer) {
|
||||||
|
entityId = buffer.readInt();
|
||||||
|
soundSource = buffer.readBlockPos();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(FriendlyByteBuf buffer) {
|
||||||
|
buffer.writeInt(entityId);
|
||||||
|
buffer.writeBlockPos(soundSource);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void handle(Supplier<Context> context) {
|
||||||
|
Context ctx = context.get();
|
||||||
|
ctx.enqueueWork(() -> {
|
||||||
|
ServerPlayer player = ctx.getSender();
|
||||||
|
Entity entity = player.level.getEntity(entityId);
|
||||||
|
if (!(entity instanceof SuperGlueEntity superGlue))
|
||||||
|
return;
|
||||||
|
double range = 32;
|
||||||
|
if (player.distanceToSqr(superGlue.position()) > range * range)
|
||||||
|
return;
|
||||||
|
AllSoundEvents.SLIME_ADDED.play(player.level, null, soundSource, 0.5F, 0.5F);
|
||||||
|
superGlue.spawnParticles();
|
||||||
|
entity.discard();
|
||||||
|
});
|
||||||
|
ctx.setPacketHandled(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -1,146 +1,27 @@
|
||||||
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.util.transform.TransformStack;
|
|
||||||
import com.mojang.blaze3d.vertex.PoseStack;
|
|
||||||
import com.mojang.blaze3d.vertex.PoseStack.Pose;
|
|
||||||
import com.mojang.blaze3d.vertex.VertexConsumer;
|
|
||||||
import com.simibubi.create.AllItems;
|
|
||||||
import com.simibubi.create.Create;
|
|
||||||
import com.simibubi.create.foundation.utility.AngleHelper;
|
|
||||||
import com.simibubi.create.foundation.utility.VecHelper;
|
|
||||||
|
|
||||||
import net.minecraft.client.Minecraft;
|
|
||||||
import net.minecraft.client.renderer.LevelRenderer;
|
|
||||||
import net.minecraft.client.renderer.MultiBufferSource;
|
|
||||||
import net.minecraft.client.renderer.RenderType;
|
|
||||||
import net.minecraft.client.renderer.culling.Frustum;
|
import net.minecraft.client.renderer.culling.Frustum;
|
||||||
import net.minecraft.client.renderer.entity.EntityRenderer;
|
import net.minecraft.client.renderer.entity.EntityRenderer;
|
||||||
import net.minecraft.client.renderer.entity.EntityRendererProvider;
|
import net.minecraft.client.renderer.entity.EntityRendererProvider;
|
||||||
import net.minecraft.client.renderer.texture.OverlayTexture;
|
|
||||||
import net.minecraft.core.BlockPos;
|
|
||||||
import net.minecraft.core.Direction;
|
|
||||||
import net.minecraft.core.Direction.Axis;
|
|
||||||
import net.minecraft.resources.ResourceLocation;
|
import net.minecraft.resources.ResourceLocation;
|
||||||
import net.minecraft.world.entity.player.Player;
|
|
||||||
import net.minecraft.world.level.Level;
|
|
||||||
import net.minecraft.world.phys.Vec3;
|
|
||||||
import net.minecraftforge.api.distmarker.Dist;
|
import net.minecraftforge.api.distmarker.Dist;
|
||||||
import net.minecraftforge.api.distmarker.OnlyIn;
|
import net.minecraftforge.api.distmarker.OnlyIn;
|
||||||
|
|
||||||
@OnlyIn(Dist.CLIENT)
|
@OnlyIn(Dist.CLIENT)
|
||||||
public class SuperGlueRenderer extends EntityRenderer<SuperGlueEntity> {
|
public class SuperGlueRenderer extends EntityRenderer<SuperGlueEntity> {
|
||||||
|
|
||||||
private ResourceLocation regular = Create.asResource("textures/entity/super_glue/slime.png");
|
|
||||||
|
|
||||||
private float[] insideQuad;
|
|
||||||
private float[] outsideQuad;
|
|
||||||
|
|
||||||
public SuperGlueRenderer(EntityRendererProvider.Context context) {
|
public SuperGlueRenderer(EntityRendererProvider.Context context) {
|
||||||
super(context);
|
super(context);
|
||||||
initQuads();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ResourceLocation getTextureLocation(SuperGlueEntity entity) {
|
public ResourceLocation getTextureLocation(SuperGlueEntity entity) {
|
||||||
return regular;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean shouldRender(SuperGlueEntity entity, Frustum frustum, double x, double y, double z) {
|
public boolean shouldRender(SuperGlueEntity entity, Frustum frustum, double x, double y, double z) {
|
||||||
if (super.shouldRender(entity, frustum, x, y, z)) {
|
|
||||||
Player player = Minecraft.getInstance().player;
|
|
||||||
boolean visible = entity.isVisible();
|
|
||||||
boolean holdingGlue = AllItems.SUPER_GLUE.isIn(player.getMainHandItem())
|
|
||||||
|| AllItems.SUPER_GLUE.isIn(player.getOffhandItem());
|
|
||||||
|
|
||||||
if (visible || holdingGlue)
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void render(SuperGlueEntity entity, float yaw, float partialTicks, PoseStack ms,
|
|
||||||
MultiBufferSource buffer, int light) {
|
|
||||||
super.render(entity, yaw, partialTicks, ms, buffer, light);
|
|
||||||
|
|
||||||
VertexConsumer builder = buffer.getBuffer(RenderType.entityCutout(getTextureLocation(entity)));
|
|
||||||
light = getBrightnessForRender(entity);
|
|
||||||
Direction face = entity.getFacingDirection();
|
|
||||||
|
|
||||||
ms.pushPose();
|
|
||||||
TransformStack.cast(ms)
|
|
||||||
.rotateY(AngleHelper.horizontalAngle(face))
|
|
||||||
.rotateX(AngleHelper.verticalAngle(face));
|
|
||||||
Pose peek = ms.last();
|
|
||||||
|
|
||||||
renderQuad(builder, peek, insideQuad, light, -1);
|
|
||||||
renderQuad(builder, peek, outsideQuad, light, 1);
|
|
||||||
|
|
||||||
ms.popPose();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void initQuads() {
|
|
||||||
Vec3 diff = Vec3.atLowerCornerOf(Direction.SOUTH.getNormal());
|
|
||||||
Vec3 extension = diff.normalize()
|
|
||||||
.scale(1 / 32f - 1 / 128f);
|
|
||||||
|
|
||||||
Vec3 plane = VecHelper.axisAlingedPlaneOf(diff);
|
|
||||||
Axis axis = Direction.getNearest(diff.x, diff.y, diff.z)
|
|
||||||
.getAxis();
|
|
||||||
|
|
||||||
Vec3 start = Vec3.ZERO.subtract(extension);
|
|
||||||
Vec3 end = Vec3.ZERO.add(extension);
|
|
||||||
|
|
||||||
plane = plane.scale(1 / 2f);
|
|
||||||
Vec3 a1 = plane.add(start);
|
|
||||||
Vec3 b1 = plane.add(end);
|
|
||||||
plane = VecHelper.rotate(plane, -90, axis);
|
|
||||||
Vec3 a2 = plane.add(start);
|
|
||||||
Vec3 b2 = plane.add(end);
|
|
||||||
plane = VecHelper.rotate(plane, -90, axis);
|
|
||||||
Vec3 a3 = plane.add(start);
|
|
||||||
Vec3 b3 = plane.add(end);
|
|
||||||
plane = VecHelper.rotate(plane, -90, axis);
|
|
||||||
Vec3 a4 = plane.add(start);
|
|
||||||
Vec3 b4 = plane.add(end);
|
|
||||||
|
|
||||||
insideQuad = new float[] {
|
|
||||||
(float) a1.x, (float) a1.y, (float) a1.z, 1, 0,
|
|
||||||
(float) a2.x, (float) a2.y, (float) a2.z, 1, 1,
|
|
||||||
(float) a3.x, (float) a3.y, (float) a3.z, 0, 1,
|
|
||||||
(float) a4.x, (float) a4.y, (float) a4.z, 0, 0,
|
|
||||||
};
|
|
||||||
outsideQuad = new float[] {
|
|
||||||
(float) b4.x, (float) b4.y, (float) b4.z, 0, 0,
|
|
||||||
(float) b3.x, (float) b3.y, (float) b3.z, 0, 1,
|
|
||||||
(float) b2.x, (float) b2.y, (float) b2.z, 1, 1,
|
|
||||||
(float) b1.x, (float) b1.y, (float) b1.z, 1, 0,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
private int getBrightnessForRender(SuperGlueEntity entity) {
|
|
||||||
BlockPos blockpos = entity.getHangingPosition();
|
|
||||||
BlockPos blockpos2 = blockpos.relative(entity.getFacingDirection()
|
|
||||||
.getOpposite());
|
|
||||||
|
|
||||||
Level world = entity.getCommandSenderWorld();
|
|
||||||
int light = world.isLoaded(blockpos) ? LevelRenderer.getLightColor(world, blockpos) : 15;
|
|
||||||
int light2 = world.isLoaded(blockpos2) ? LevelRenderer.getLightColor(world, blockpos2) : 15;
|
|
||||||
return Math.max(light, light2);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Vertex format: pos x, pos y, pos z, u, v
|
|
||||||
private void renderQuad(VertexConsumer builder, Pose matrix, float[] data, int light, float normalZ) {
|
|
||||||
for (int i = 0; i < 4; i++) {
|
|
||||||
builder.vertex(matrix.pose(), data[5 * i], data[5 * i + 1], data[5 * i + 2])
|
|
||||||
.color(255, 255, 255, 255)
|
|
||||||
.uv(data[5 * i + 3], data[5 * i + 4])
|
|
||||||
.overlayCoords(OverlayTexture.NO_OVERLAY)
|
|
||||||
.uv2(light)
|
|
||||||
.normal(matrix.normal(), 0.0f, 0.0f, normalZ)
|
|
||||||
.endVertex();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,42 +1,53 @@
|
||||||
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Optional;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
import com.google.common.base.Objects;
|
import com.google.common.base.Objects;
|
||||||
|
import com.simibubi.create.AllSoundEvents;
|
||||||
import com.simibubi.create.AllSpecialTextures;
|
import com.simibubi.create.AllSpecialTextures;
|
||||||
import com.simibubi.create.CreateClient;
|
import com.simibubi.create.CreateClient;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.chassis.AbstractChassisBlock;
|
import com.simibubi.create.content.contraptions.components.structureMovement.chassis.AbstractChassisBlock;
|
||||||
import com.simibubi.create.foundation.networking.AllPackets;
|
import com.simibubi.create.foundation.networking.AllPackets;
|
||||||
import com.simibubi.create.foundation.utility.BlockFace;
|
|
||||||
import com.simibubi.create.foundation.utility.Lang;
|
import com.simibubi.create.foundation.utility.Lang;
|
||||||
import com.simibubi.create.foundation.utility.Pair;
|
import com.simibubi.create.foundation.utility.RaycastHelper;
|
||||||
import com.simibubi.create.foundation.utility.outliner.Outline.OutlineParams;
|
|
||||||
|
|
||||||
import net.minecraft.client.Minecraft;
|
import net.minecraft.client.Minecraft;
|
||||||
import net.minecraft.client.multiplayer.ClientLevel;
|
import net.minecraft.client.multiplayer.ClientLevel;
|
||||||
import net.minecraft.client.player.LocalPlayer;
|
import net.minecraft.client.player.LocalPlayer;
|
||||||
import net.minecraft.core.BlockPos;
|
import net.minecraft.core.BlockPos;
|
||||||
|
import net.minecraft.core.Direction;
|
||||||
|
import net.minecraft.network.chat.TextComponent;
|
||||||
|
import net.minecraft.sounds.SoundEvents;
|
||||||
|
import net.minecraft.sounds.SoundSource;
|
||||||
import net.minecraft.world.item.ItemStack;
|
import net.minecraft.world.item.ItemStack;
|
||||||
import net.minecraft.world.level.block.state.BlockState;
|
import net.minecraft.world.level.block.state.BlockState;
|
||||||
import net.minecraft.world.phys.AABB;
|
import net.minecraft.world.phys.AABB;
|
||||||
import net.minecraft.world.phys.BlockHitResult;
|
import net.minecraft.world.phys.BlockHitResult;
|
||||||
import net.minecraft.world.phys.HitResult;
|
import net.minecraft.world.phys.HitResult;
|
||||||
import net.minecraft.world.phys.HitResult.Type;
|
import net.minecraft.world.phys.HitResult.Type;
|
||||||
|
import net.minecraft.world.phys.Vec3;
|
||||||
|
import net.minecraftforge.common.ForgeMod;
|
||||||
|
|
||||||
public class SuperGlueSelectionHandler {
|
public class SuperGlueSelectionHandler {
|
||||||
|
|
||||||
private static final int SUCCESS = 0x68c586;
|
private static final int PASSIVE = 0x4D9162;
|
||||||
|
private static final int HIGHLIGHT = 0x68c586;
|
||||||
private static final int FAIL = 0xc5b548;
|
private static final int FAIL = 0xc5b548;
|
||||||
|
|
||||||
private Object clusterOutlineSlot = new Object();
|
private Object clusterOutlineSlot = new Object();
|
||||||
private Object bbOutlineSlot = new Object();
|
private Object bbOutlineSlot = new Object();
|
||||||
|
private int clusterCooldown;
|
||||||
|
|
||||||
private BlockPos firstPos;
|
private BlockPos firstPos;
|
||||||
private BlockPos hoveredPos;
|
private BlockPos hoveredPos;
|
||||||
private Set<BlockPos> currentCluster;
|
private Set<BlockPos> currentCluster;
|
||||||
private int glueRequired;
|
private int glueRequired;
|
||||||
|
|
||||||
|
private SuperGlueEntity selected;
|
||||||
|
private BlockPos soundSourceForRemoval;
|
||||||
|
|
||||||
public void tick() {
|
public void tick() {
|
||||||
Minecraft mc = Minecraft.getInstance();
|
Minecraft mc = Minecraft.getInstance();
|
||||||
LocalPlayer player = mc.player;
|
LocalPlayer player = mc.player;
|
||||||
|
@ -49,6 +60,51 @@ public class SuperGlueSelectionHandler {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (clusterCooldown > 0) {
|
||||||
|
if (clusterCooldown == 25)
|
||||||
|
player.displayClientMessage(TextComponent.EMPTY, true);
|
||||||
|
CreateClient.OUTLINER.keep(clusterOutlineSlot);
|
||||||
|
clusterCooldown--;
|
||||||
|
}
|
||||||
|
|
||||||
|
AABB scanArea = player.getBoundingBox()
|
||||||
|
.inflate(32, 16, 32);
|
||||||
|
|
||||||
|
List<SuperGlueEntity> glueNearby = mc.level.getEntitiesOfClass(SuperGlueEntity.class, scanArea);
|
||||||
|
|
||||||
|
selected = null;
|
||||||
|
if (firstPos == null) {
|
||||||
|
double range = player.getAttribute(ForgeMod.REACH_DISTANCE.get())
|
||||||
|
.getValue() + 1;
|
||||||
|
Vec3 traceOrigin = RaycastHelper.getTraceOrigin(player);
|
||||||
|
Vec3 traceTarget = RaycastHelper.getTraceTarget(player, range, traceOrigin);
|
||||||
|
|
||||||
|
double bestDistance = Double.MAX_VALUE;
|
||||||
|
for (SuperGlueEntity glueEntity : glueNearby) {
|
||||||
|
Optional<Vec3> clip = glueEntity.getBoundingBox()
|
||||||
|
.clip(traceOrigin, traceTarget);
|
||||||
|
if (clip.isEmpty())
|
||||||
|
continue;
|
||||||
|
Vec3 vec3 = clip.get();
|
||||||
|
double distanceToSqr = vec3.distanceToSqr(traceOrigin);
|
||||||
|
if (distanceToSqr > bestDistance)
|
||||||
|
continue;
|
||||||
|
selected = glueEntity;
|
||||||
|
soundSourceForRemoval = new BlockPos(vec3);
|
||||||
|
bestDistance = distanceToSqr;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (SuperGlueEntity glueEntity : glueNearby) {
|
||||||
|
boolean h = clusterCooldown == 0 && glueEntity == selected;
|
||||||
|
AllSpecialTextures faceTex = h ? AllSpecialTextures.GLUE : null;
|
||||||
|
CreateClient.OUTLINER.showAABB(glueEntity, glueEntity.getBoundingBox())
|
||||||
|
.colored(h ? HIGHLIGHT : PASSIVE)
|
||||||
|
.withFaceTextures(faceTex, faceTex)
|
||||||
|
.disableNormals()
|
||||||
|
.lineWidth(h ? 1 / 16f : 1 / 64f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
HitResult hitResult = mc.hitResult;
|
HitResult hitResult = mc.hitResult;
|
||||||
if (hitResult != null && hitResult.getType() == Type.BLOCK)
|
if (hitResult != null && hitResult.getType() == Type.BLOCK)
|
||||||
hovered = ((BlockHitResult) hitResult).getBlockPos();
|
hovered = ((BlockHitResult) hitResult).getBlockPos();
|
||||||
|
@ -83,33 +139,29 @@ public class SuperGlueSelectionHandler {
|
||||||
else if (cancel)
|
else if (cancel)
|
||||||
Lang.sendStatus(player, FAIL, "super_glue.click_to_discard");
|
Lang.sendStatus(player, FAIL, "super_glue.click_to_discard");
|
||||||
else
|
else
|
||||||
Lang.sendStatus(player, SUCCESS, "super_glue.click_to_confirm");
|
Lang.sendStatus(player, HIGHLIGHT, "super_glue.click_to_confirm");
|
||||||
|
|
||||||
|
if (currentSelectionBox != null)
|
||||||
|
CreateClient.OUTLINER.showAABB(bbOutlineSlot, currentSelectionBox)
|
||||||
|
.colored(canReach && canAfford && !cancel ? HIGHLIGHT : FAIL)
|
||||||
|
.withFaceTextures(AllSpecialTextures.GLUE, AllSpecialTextures.GLUE)
|
||||||
|
.disableNormals()
|
||||||
|
.lineWidth(1 / 16f);
|
||||||
|
|
||||||
CreateClient.OUTLINER.showCluster(clusterOutlineSlot, currentCluster)
|
CreateClient.OUTLINER.showCluster(clusterOutlineSlot, currentCluster)
|
||||||
.colored(canReach && canAfford && !cancel ? SUCCESS : FAIL)
|
.colored(0x4D9162)
|
||||||
.withFaceTextures(AllSpecialTextures.CHECKERED, AllSpecialTextures.HIGHLIGHT_CHECKERED)
|
|
||||||
.lineWidth(1 / 16f);
|
|
||||||
}
|
|
||||||
if (currentSelectionBox != null) {
|
|
||||||
OutlineParams params =
|
|
||||||
firstPos == null ? CreateClient.OUTLINER.showAABB(bbOutlineSlot, currentSelectionBox)
|
|
||||||
: CreateClient.OUTLINER.chaseAABB(bbOutlineSlot, currentSelectionBox);
|
|
||||||
params.colored(0x111111)
|
|
||||||
.disableNormals()
|
.disableNormals()
|
||||||
.lineWidth(1 / 128f);
|
.lineWidth(1 / 64f);
|
||||||
}
|
}
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
hoveredPos = hovered;
|
hoveredPos = hovered;
|
||||||
|
|
||||||
Pair<Set<BlockPos>, List<BlockFace>> pair =
|
Set<BlockPos> cluster = SuperGlueSelectionHelper.searchGlueGroup(mc.level, firstPos, hoveredPos, true);
|
||||||
SuperGlueSelectionHelper.searchGlueGroup(mc.level, firstPos, hoveredPos);
|
currentCluster = cluster;
|
||||||
|
glueRequired = 1;
|
||||||
currentCluster = pair == null ? null : pair.getFirst();
|
|
||||||
glueRequired = pair == null ? 0
|
|
||||||
: pair.getSecond()
|
|
||||||
.size();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean isGlue(ItemStack stack) {
|
private boolean isGlue(ItemStack stack) {
|
||||||
|
@ -120,7 +172,7 @@ public class SuperGlueSelectionHandler {
|
||||||
return firstPos == null || hoveredPos == null ? null : new AABB(firstPos, hoveredPos).expandTowards(1, 1, 1);
|
return firstPos == null || hoveredPos == null ? null : new AABB(firstPos, hoveredPos).expandTowards(1, 1, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean onMouseInput() {
|
public boolean onMouseInput(boolean attack) {
|
||||||
Minecraft mc = Minecraft.getInstance();
|
Minecraft mc = Minecraft.getInstance();
|
||||||
LocalPlayer player = mc.player;
|
LocalPlayer player = mc.player;
|
||||||
ClientLevel level = mc.level;
|
ClientLevel level = mc.level;
|
||||||
|
@ -128,6 +180,15 @@ public class SuperGlueSelectionHandler {
|
||||||
if (!isGlue(player.getMainHandItem()))
|
if (!isGlue(player.getMainHandItem()))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
|
if (attack) {
|
||||||
|
if (selected == null)
|
||||||
|
return false;
|
||||||
|
AllPackets.channel.sendToServer(new SuperGlueRemovalPacket(selected.getId(), soundSourceForRemoval));
|
||||||
|
selected = null;
|
||||||
|
clusterCooldown = 0;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if (player.isSteppingCarefully()) {
|
if (player.isSteppingCarefully()) {
|
||||||
if (firstPos != null) {
|
if (firstPos != null) {
|
||||||
discard();
|
discard();
|
||||||
|
@ -139,7 +200,9 @@ public class SuperGlueSelectionHandler {
|
||||||
if (hoveredPos == null)
|
if (hoveredPos == null)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
|
Direction face = null;
|
||||||
if (mc.hitResult instanceof BlockHitResult bhr) {
|
if (mc.hitResult instanceof BlockHitResult bhr) {
|
||||||
|
face = bhr.getDirection();
|
||||||
BlockState blockState = level.getBlockState(hoveredPos);
|
BlockState blockState = level.getBlockState(hoveredPos);
|
||||||
if (blockState.getBlock()instanceof AbstractChassisBlock cb)
|
if (blockState.getBlock()instanceof AbstractChassisBlock cb)
|
||||||
if (cb.getGlueableSide(blockState, bhr.getDirection()) != null)
|
if (cb.getGlueableSide(blockState, bhr.getDirection()) != null)
|
||||||
|
@ -158,7 +221,11 @@ public class SuperGlueSelectionHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
firstPos = hoveredPos;
|
firstPos = hoveredPos;
|
||||||
|
if (face != null)
|
||||||
|
SuperGlueItem.spawnParticles(level, firstPos, face, true);
|
||||||
Lang.sendStatus(player, "super_glue.first_pos");
|
Lang.sendStatus(player, "super_glue.first_pos");
|
||||||
|
AllSoundEvents.SLIME_ADDED.playAt(level, firstPos, 0.5F, 0.85F, false);
|
||||||
|
level.playSound(player, firstPos, SoundEvents.ITEM_FRAME_ADD_ITEM, SoundSource.BLOCKS, 0.75f, 1);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -167,13 +234,25 @@ public class SuperGlueSelectionHandler {
|
||||||
currentCluster = null;
|
currentCluster = null;
|
||||||
firstPos = null;
|
firstPos = null;
|
||||||
Lang.sendStatus(player, "super_glue.abort");
|
Lang.sendStatus(player, "super_glue.abort");
|
||||||
|
clusterCooldown = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void confirm() {
|
public void confirm() {
|
||||||
LocalPlayer player = Minecraft.getInstance().player;
|
LocalPlayer player = Minecraft.getInstance().player;
|
||||||
AllPackets.channel.sendToServer(new SuperGlueSelectionPacket(firstPos, hoveredPos));
|
AllPackets.channel.sendToServer(new SuperGlueSelectionPacket(firstPos, hoveredPos));
|
||||||
|
AllSoundEvents.SLIME_ADDED.playAt(player.level, hoveredPos, 0.5F, 0.95F, false);
|
||||||
|
player.level.playSound(player, hoveredPos, SoundEvents.ITEM_FRAME_ADD_ITEM, SoundSource.BLOCKS, 0.75f, 1);
|
||||||
|
|
||||||
|
if (currentCluster != null)
|
||||||
|
CreateClient.OUTLINER.showCluster(clusterOutlineSlot, currentCluster)
|
||||||
|
.colored(0xB5F2C6)
|
||||||
|
.withFaceTextures(AllSpecialTextures.GLUE, AllSpecialTextures.HIGHLIGHT_CHECKERED)
|
||||||
|
.disableNormals()
|
||||||
|
.lineWidth(1 / 24f);
|
||||||
|
|
||||||
discard();
|
discard();
|
||||||
Lang.sendStatus(player, "super_glue.sucess");
|
Lang.sendStatus(player, "super_glue.sucess");
|
||||||
|
clusterCooldown = 40;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,13 +3,10 @@ package com.simibubi.create.content.contraptions.components.structureMovement.gl
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.BlockMovementChecks;
|
import com.simibubi.create.content.contraptions.components.structureMovement.BlockMovementChecks;
|
||||||
import com.simibubi.create.foundation.utility.BlockFace;
|
|
||||||
import com.simibubi.create.foundation.utility.Iterate;
|
import com.simibubi.create.foundation.utility.Iterate;
|
||||||
import com.simibubi.create.foundation.utility.Pair;
|
|
||||||
|
|
||||||
import net.minecraft.core.BlockPos;
|
import net.minecraft.core.BlockPos;
|
||||||
import net.minecraft.core.Direction;
|
import net.minecraft.core.Direction;
|
||||||
|
@ -22,17 +19,16 @@ import net.minecraft.world.phys.Vec3;
|
||||||
|
|
||||||
public class SuperGlueSelectionHelper {
|
public class SuperGlueSelectionHelper {
|
||||||
|
|
||||||
public static Pair<Set<BlockPos>, List<BlockFace>> searchGlueGroup(Level level, BlockPos startPos,
|
public static Set<BlockPos> searchGlueGroup(Level level, BlockPos startPos, BlockPos endPos, boolean includeOther) {
|
||||||
BlockPos endPos) {
|
|
||||||
if (endPos == null || startPos == null)
|
if (endPos == null || startPos == null)
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
AABB bb = new AABB(startPos, endPos).expandTowards(1, 1, 1);
|
AABB bb = SuperGlueEntity.span(startPos, endPos);
|
||||||
|
|
||||||
List<BlockPos> frontier = new ArrayList<>();
|
List<BlockPos> frontier = new ArrayList<>();
|
||||||
Set<BlockPos> visited = new HashSet<>();
|
Set<BlockPos> visited = new HashSet<>();
|
||||||
Set<BlockPos> attached = new HashSet<>();
|
Set<BlockPos> attached = new HashSet<>();
|
||||||
List<BlockFace> glue = new ArrayList<>();
|
Set<SuperGlueEntity> cachedOther = new HashSet<>();
|
||||||
|
|
||||||
visited.add(startPos);
|
visited.add(startPos);
|
||||||
frontier.add(startPos);
|
frontier.add(startPos);
|
||||||
|
@ -41,11 +37,10 @@ public class SuperGlueSelectionHelper {
|
||||||
BlockPos currentPos = frontier.remove(0);
|
BlockPos currentPos = frontier.remove(0);
|
||||||
attached.add(currentPos);
|
attached.add(currentPos);
|
||||||
|
|
||||||
Map<Direction, SuperGlueEntity> gatheredGlue = SuperGlueHandler.gatherGlue(level, currentPos);
|
|
||||||
for (Direction d : Iterate.directions) {
|
for (Direction d : Iterate.directions) {
|
||||||
BlockPos offset = currentPos.relative(d);
|
BlockPos offset = currentPos.relative(d);
|
||||||
boolean gluePresent = gatheredGlue.containsKey(d);
|
boolean gluePresent = includeOther && SuperGlueEntity.isGlued(level, currentPos, d, cachedOther);
|
||||||
boolean alreadySticky = SuperGlueEntity.isSideSticky(level, currentPos, d)
|
boolean alreadySticky = includeOther && SuperGlueEntity.isSideSticky(level, currentPos, d)
|
||||||
|| SuperGlueEntity.isSideSticky(level, offset, d.getOpposite());
|
|| SuperGlueEntity.isSideSticky(level, offset, d.getOpposite());
|
||||||
|
|
||||||
if (!alreadySticky && !gluePresent && !bb.contains(Vec3.atCenterOf(offset)))
|
if (!alreadySticky && !gluePresent && !bb.contains(Vec3.atCenterOf(offset)))
|
||||||
|
@ -56,20 +51,15 @@ public class SuperGlueSelectionHelper {
|
||||||
|| !SuperGlueEntity.isValidFace(level, offset, d.getOpposite()))
|
|| !SuperGlueEntity.isValidFace(level, offset, d.getOpposite()))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
boolean glueNecessary = !gluePresent && !alreadySticky;
|
if (visited.add(offset))
|
||||||
|
|
||||||
if (visited.add(offset)) {
|
|
||||||
frontier.add(offset);
|
frontier.add(offset);
|
||||||
if (glueNecessary)
|
|
||||||
glue.add(new BlockFace(offset, d));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (attached.size() < 2 && attached.contains(endPos))
|
if (attached.size() < 2 && attached.contains(endPos))
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
return Pair.of(attached, glue);
|
return attached;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static boolean collectGlueFromInventory(Player player, int requiredAmount, boolean simulate) {
|
public static boolean collectGlueFromInventory(Player player, int requiredAmount, boolean simulate) {
|
||||||
|
|
|
@ -1,17 +1,14 @@
|
||||||
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
||||||
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.function.Supplier;
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
import com.simibubi.create.Create;
|
|
||||||
import com.simibubi.create.foundation.networking.SimplePacketBase;
|
import com.simibubi.create.foundation.networking.SimplePacketBase;
|
||||||
import com.simibubi.create.foundation.utility.BlockFace;
|
|
||||||
import com.simibubi.create.foundation.utility.Pair;
|
|
||||||
|
|
||||||
import net.minecraft.core.BlockPos;
|
import net.minecraft.core.BlockPos;
|
||||||
import net.minecraft.network.FriendlyByteBuf;
|
import net.minecraft.network.FriendlyByteBuf;
|
||||||
import net.minecraft.server.level.ServerPlayer;
|
import net.minecraft.server.level.ServerPlayer;
|
||||||
|
import net.minecraft.world.phys.AABB;
|
||||||
import net.minecraft.world.phys.Vec3;
|
import net.minecraft.world.phys.Vec3;
|
||||||
import net.minecraftforge.common.ForgeMod;
|
import net.minecraftforge.common.ForgeMod;
|
||||||
import net.minecraftforge.network.NetworkEvent.Context;
|
import net.minecraftforge.network.NetworkEvent.Context;
|
||||||
|
@ -50,19 +47,19 @@ public class SuperGlueSelectionPacket extends SimplePacketBase {
|
||||||
if (!to.closerThan(from, 25))
|
if (!to.closerThan(from, 25))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
Pair<Set<BlockPos>, List<BlockFace>> group =
|
Set<BlockPos> group = SuperGlueSelectionHelper.searchGlueGroup(player.level, from, to, false);
|
||||||
SuperGlueSelectionHelper.searchGlueGroup(player.level, from, to);
|
|
||||||
if (group == null)
|
if (group == null)
|
||||||
return;
|
return;
|
||||||
if (!group.getFirst()
|
if (!group.contains(to))
|
||||||
.contains(to))
|
|
||||||
return;
|
return;
|
||||||
List<BlockFace> glue = group.getSecond();
|
if (!SuperGlueSelectionHelper.collectGlueFromInventory(player, 1, true))
|
||||||
if (!SuperGlueSelectionHelper.collectGlueFromInventory(player, glue.size(), true))
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
SuperGlueSelectionHelper.collectGlueFromInventory(player, glue.size(), false);
|
AABB bb = SuperGlueEntity.span(from, to);
|
||||||
Create.GLUE_QUEUE.add(player.level, glue);
|
SuperGlueSelectionHelper.collectGlueFromInventory(player, 1, false);
|
||||||
|
SuperGlueEntity entity = new SuperGlueEntity(player.level, bb);
|
||||||
|
player.level.addFreshEntity(entity);
|
||||||
|
entity.spawnParticles();
|
||||||
});
|
});
|
||||||
ctx.setPacketHandled(true);
|
ctx.setPacketHandled(true);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +0,0 @@
|
||||||
@ParametersAreNonnullByDefault @MethodsReturnNonnullByDefault
|
|
||||||
package com.simibubi.create.content.contraptions.components.structureMovement.glue;
|
|
||||||
|
|
||||||
import javax.annotation.ParametersAreNonnullByDefault;
|
|
||||||
|
|
||||||
import net.minecraft.MethodsReturnNonnullByDefault;
|
|
|
@ -122,7 +122,6 @@ public class CommonEvents {
|
||||||
LinkedControllerServerHandler.tick(world);
|
LinkedControllerServerHandler.tick(world);
|
||||||
ControlsServerHandler.tick(world);
|
ControlsServerHandler.tick(world);
|
||||||
Create.RAILWAYS.tick(world);
|
Create.RAILWAYS.tick(world);
|
||||||
Create.GLUE_QUEUE.tick(world);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@SubscribeEvent
|
@SubscribeEvent
|
||||||
|
|
|
@ -8,6 +8,7 @@ import com.simibubi.create.content.logistics.trains.track.CurvedTrackInteraction
|
||||||
import com.simibubi.create.foundation.tileEntity.behaviour.filtering.FilteringHandler;
|
import com.simibubi.create.foundation.tileEntity.behaviour.filtering.FilteringHandler;
|
||||||
import com.simibubi.create.foundation.tileEntity.behaviour.scrollvalue.ScrollValueHandler;
|
import com.simibubi.create.foundation.tileEntity.behaviour.scrollvalue.ScrollValueHandler;
|
||||||
|
|
||||||
|
import net.minecraft.client.KeyMapping;
|
||||||
import net.minecraft.client.Minecraft;
|
import net.minecraft.client.Minecraft;
|
||||||
import net.minecraftforge.api.distmarker.Dist;
|
import net.minecraftforge.api.distmarker.Dist;
|
||||||
import net.minecraftforge.client.event.InputEvent.ClickInputEvent;
|
import net.minecraftforge.client.event.InputEvent.ClickInputEvent;
|
||||||
|
@ -68,12 +69,14 @@ public class InputEvents {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (event.getKeyMapping() == mc.options.keyUse) {
|
KeyMapping key = event.getKeyMapping();
|
||||||
if (CreateClient.GLUE_HANDLER.onMouseInput())
|
|
||||||
|
if (key == mc.options.keyUse || key == mc.options.keyAttack) {
|
||||||
|
if (CreateClient.GLUE_HANDLER.onMouseInput(key == mc.options.keyAttack))
|
||||||
event.setCanceled(true);
|
event.setCanceled(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (event.getKeyMapping() == mc.options.keyPickItem) {
|
if (key == mc.options.keyPickItem) {
|
||||||
if (ToolboxHandlerClient.onPickItem())
|
if (ToolboxHandlerClient.onPickItem())
|
||||||
event.setCanceled(true);
|
event.setCanceled(true);
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -92,19 +92,19 @@ public class CloneCommand {
|
||||||
List<SuperGlueEntity> glue = world.getEntitiesOfClass(SuperGlueEntity.class, AABB.of(sourceArea));
|
List<SuperGlueEntity> glue = world.getEntitiesOfClass(SuperGlueEntity.class, AABB.of(sourceArea));
|
||||||
List<Pair<BlockPos, Direction>> newGlue = Lists.newArrayList();
|
List<Pair<BlockPos, Direction>> newGlue = Lists.newArrayList();
|
||||||
|
|
||||||
for (SuperGlueEntity g : glue) {
|
// for (SuperGlueEntity g : glue) {TODO
|
||||||
BlockPos pos = g.getHangingPosition();
|
// BlockPos pos = g.getHangingPosition();
|
||||||
Direction direction = g.getFacingDirection();
|
// Direction direction = g.getFacingDirection();
|
||||||
newGlue.add(Pair.of(pos.offset(diffToTarget), direction));
|
// newGlue.add(Pair.of(pos.offset(diffToTarget), direction));
|
||||||
}
|
// }
|
||||||
|
//
|
||||||
for (Pair<BlockPos, Direction> p : newGlue) {
|
// for (Pair<BlockPos, Direction> p : newGlue) {
|
||||||
SuperGlueEntity g = new SuperGlueEntity(world, p.getFirst(), p.getSecond());
|
// SuperGlueEntity g = new SuperGlueEntity(world, p.getFirst(), p.getSecond());
|
||||||
if (g.onValidSurface()) {
|
// if (g.onValidSurface()) {
|
||||||
world.addFreshEntity(g);
|
// world.addFreshEntity(g);
|
||||||
gluePastes++;
|
// gluePastes++;
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
return gluePastes;
|
return gluePastes;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -7,26 +7,26 @@ import net.minecraft.commands.CommandSourceStack;
|
||||||
import net.minecraft.commands.Commands;
|
import net.minecraft.commands.Commands;
|
||||||
import net.minecraft.commands.arguments.coordinates.BlockPosArgument;
|
import net.minecraft.commands.arguments.coordinates.BlockPosArgument;
|
||||||
import net.minecraft.core.BlockPos;
|
import net.minecraft.core.BlockPos;
|
||||||
import net.minecraft.core.Direction;
|
|
||||||
import net.minecraft.server.level.ServerLevel;
|
import net.minecraft.server.level.ServerLevel;
|
||||||
|
|
||||||
public class GlueCommand {
|
public class GlueCommand {
|
||||||
public static ArgumentBuilder<CommandSourceStack, ?> register() {
|
public static ArgumentBuilder<CommandSourceStack, ?> register() {
|
||||||
return Commands.literal("glue")
|
return Commands.literal("glue")
|
||||||
.requires(cs -> cs.hasPermission(2))
|
.requires(cs -> cs.hasPermission(2))
|
||||||
.then(Commands.argument("pos", BlockPosArgument.blockPos())
|
.then(Commands.argument("from", BlockPosArgument.blockPos())
|
||||||
//.then(Commands.argument("direction", EnumArgument.enumArgument(Direction.class))
|
.then(Commands.argument("to", BlockPosArgument.blockPos())
|
||||||
.executes(ctx -> {
|
.executes(ctx -> {
|
||||||
BlockPos pos = BlockPosArgument.getLoadedBlockPos(ctx, "pos");
|
BlockPos from = BlockPosArgument.getLoadedBlockPos(ctx, "from");
|
||||||
|
BlockPos to = BlockPosArgument.getLoadedBlockPos(ctx, "to");
|
||||||
|
|
||||||
ServerLevel world = ctx.getSource().getLevel();
|
ServerLevel world = ctx.getSource()
|
||||||
SuperGlueEntity entity = new SuperGlueEntity(world, pos, Direction.UP);
|
.getLevel();
|
||||||
|
|
||||||
|
SuperGlueEntity entity = new SuperGlueEntity(world, SuperGlueEntity.span(from, to));
|
||||||
entity.playPlaceSound();
|
entity.playPlaceSound();
|
||||||
world.addFreshEntity(entity);
|
world.addFreshEntity(entity);
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}));
|
})));
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,6 +12,7 @@ import com.simibubi.create.content.contraptions.components.structureMovement.Con
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.ContraptionStallPacket;
|
import com.simibubi.create.content.contraptions.components.structureMovement.ContraptionStallPacket;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.gantry.GantryContraptionUpdatePacket;
|
import com.simibubi.create.content.contraptions.components.structureMovement.gantry.GantryContraptionUpdatePacket;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.glue.GlueEffectPacket;
|
import com.simibubi.create.content.contraptions.components.structureMovement.glue.GlueEffectPacket;
|
||||||
|
import com.simibubi.create.content.contraptions.components.structureMovement.glue.SuperGlueRemovalPacket;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.glue.SuperGlueSelectionPacket;
|
import com.simibubi.create.content.contraptions.components.structureMovement.glue.SuperGlueSelectionPacket;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.interaction.controls.ControlsInputPacket;
|
import com.simibubi.create.content.contraptions.components.structureMovement.interaction.controls.ControlsInputPacket;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.interaction.controls.ControlsStopControllingPacket;
|
import com.simibubi.create.content.contraptions.components.structureMovement.interaction.controls.ControlsStopControllingPacket;
|
||||||
|
@ -130,6 +131,7 @@ public enum AllPackets {
|
||||||
SELECT_CURVED_TRACK(CurvedTrackSelectionPacket.class, CurvedTrackSelectionPacket::new, PLAY_TO_SERVER),
|
SELECT_CURVED_TRACK(CurvedTrackSelectionPacket.class, CurvedTrackSelectionPacket::new, PLAY_TO_SERVER),
|
||||||
PLACE_CURVED_TRACK(PlaceExtendedCurvePacket.class, PlaceExtendedCurvePacket::new, PLAY_TO_SERVER),
|
PLACE_CURVED_TRACK(PlaceExtendedCurvePacket.class, PlaceExtendedCurvePacket::new, PLAY_TO_SERVER),
|
||||||
GLUE_IN_AREA(SuperGlueSelectionPacket.class, SuperGlueSelectionPacket::new, PLAY_TO_SERVER),
|
GLUE_IN_AREA(SuperGlueSelectionPacket.class, SuperGlueSelectionPacket::new, PLAY_TO_SERVER),
|
||||||
|
GLUE_REMOVED(SuperGlueRemovalPacket.class, SuperGlueRemovalPacket::new, PLAY_TO_SERVER),
|
||||||
|
|
||||||
// Server to Client
|
// Server to Client
|
||||||
SYMMETRY_EFFECT(SymmetryEffectPacket.class, SymmetryEffectPacket::new, PLAY_TO_CLIENT),
|
SYMMETRY_EFFECT(SymmetryEffectPacket.class, SymmetryEffectPacket::new, PLAY_TO_CLIENT),
|
||||||
|
|
|
@ -13,7 +13,6 @@ import com.simibubi.create.content.contraptions.base.KineticBlock;
|
||||||
import com.simibubi.create.content.contraptions.base.KineticTileEntity;
|
import com.simibubi.create.content.contraptions.base.KineticTileEntity;
|
||||||
import com.simibubi.create.content.contraptions.components.crafter.ConnectedInputHandler;
|
import com.simibubi.create.content.contraptions.components.crafter.ConnectedInputHandler;
|
||||||
import com.simibubi.create.content.contraptions.components.crafter.MechanicalCrafterTileEntity;
|
import com.simibubi.create.content.contraptions.components.crafter.MechanicalCrafterTileEntity;
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.glue.SuperGlueEntity;
|
|
||||||
import com.simibubi.create.content.contraptions.components.structureMovement.glue.SuperGlueItem;
|
import com.simibubi.create.content.contraptions.components.structureMovement.glue.SuperGlueItem;
|
||||||
import com.simibubi.create.content.contraptions.fluids.PumpTileEntity;
|
import com.simibubi.create.content.contraptions.fluids.PumpTileEntity;
|
||||||
import com.simibubi.create.content.contraptions.particle.RotationIndicatorParticleData;
|
import com.simibubi.create.content.contraptions.particle.RotationIndicatorParticleData;
|
||||||
|
@ -661,11 +660,6 @@ public class SceneBuilder {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public ElementLink<EntityElement> createGlueEntity(BlockPos pos, Direction face) {
|
|
||||||
effects.superGlue(pos, face, false);
|
|
||||||
return createEntity(world -> new SuperGlueEntity(world, pos, face.getOpposite()));
|
|
||||||
}
|
|
||||||
|
|
||||||
public void createItemOnBeltLike(BlockPos location, Direction insertionSide, ItemStack stack) {
|
public void createItemOnBeltLike(BlockPos location, Direction insertionSide, ItemStack stack) {
|
||||||
addInstruction(scene -> {
|
addInstruction(scene -> {
|
||||||
PonderWorld world = scene.getWorld();
|
PonderWorld world = scene.getWorld();
|
||||||
|
|
|
@ -253,7 +253,7 @@ public class ChassisScenes {
|
||||||
scene.overlay.showControls(new InputWindowElement(glueSurface, Pointing.DOWN).rightClick()
|
scene.overlay.showControls(new InputWindowElement(glueSurface, Pointing.DOWN).rightClick()
|
||||||
.withItem(AllItems.SUPER_GLUE.asStack()), 30);
|
.withItem(AllItems.SUPER_GLUE.asStack()), 30);
|
||||||
scene.idle(7);
|
scene.idle(7);
|
||||||
ElementLink<EntityElement> glueEntity = scene.world.createGlueEntity(chassisPos.west(), Direction.NORTH);
|
// ElementLink<EntityElement> glueEntity = scene.world.createGlueEntity(chassisPos.west(), Direction.NORTH);TODO
|
||||||
scene.idle(20);
|
scene.idle(20);
|
||||||
ElementLink<WorldSectionElement> gluedPlank =
|
ElementLink<WorldSectionElement> gluedPlank =
|
||||||
scene.world.showIndependentSection(util.select.position(3, 3, 1), Direction.SOUTH);
|
scene.world.showIndependentSection(util.select.position(3, 3, 1), Direction.SOUTH);
|
||||||
|
@ -262,7 +262,7 @@ public class ChassisScenes {
|
||||||
scene.effects.superGlue(chassisPos.west(), Direction.NORTH, true);
|
scene.effects.superGlue(chassisPos.west(), Direction.NORTH, true);
|
||||||
scene.idle(20);
|
scene.idle(20);
|
||||||
|
|
||||||
scene.world.modifyEntity(glueEntity, Entity::discard);
|
// scene.world.modifyEntity(glueEntity, Entity::discard);
|
||||||
scene.world.hideIndependentSection(glassSection, Direction.UP);
|
scene.world.hideIndependentSection(glassSection, Direction.UP);
|
||||||
scene.world.hideIndependentSection(gluedPlank, Direction.UP);
|
scene.world.hideIndependentSection(gluedPlank, Direction.UP);
|
||||||
scene.world.hideIndependentSection(topGlassSection, Direction.UP);
|
scene.world.hideIndependentSection(topGlassSection, Direction.UP);
|
||||||
|
@ -499,7 +499,7 @@ public class ChassisScenes {
|
||||||
scene.overlay.showControls(new InputWindowElement(blockSurface, Pointing.DOWN).rightClick()
|
scene.overlay.showControls(new InputWindowElement(blockSurface, Pointing.DOWN).rightClick()
|
||||||
.withItem(AllItems.SUPER_GLUE.asStack()), 40);
|
.withItem(AllItems.SUPER_GLUE.asStack()), 40);
|
||||||
scene.idle(7);
|
scene.idle(7);
|
||||||
ElementLink<EntityElement> glueEntity = scene.world.createGlueEntity(central, Direction.NORTH);
|
// ElementLink<EntityElement> glueEntity = scene.world.createGlueEntity(central, Direction.NORTH);TODO
|
||||||
scene.idle(10);
|
scene.idle(10);
|
||||||
scene.overlay.showText(60)
|
scene.overlay.showText(60)
|
||||||
.pointAt(blockSurface)
|
.pointAt(blockSurface)
|
||||||
|
@ -510,7 +510,7 @@ public class ChassisScenes {
|
||||||
|
|
||||||
scene.world.glueBlockOnto(central.north(), Direction.SOUTH, plank);
|
scene.world.glueBlockOnto(central.north(), Direction.SOUTH, plank);
|
||||||
scene.idle(20);
|
scene.idle(20);
|
||||||
scene.world.modifyEntity(glueEntity, Entity::discard);
|
// scene.world.modifyEntity(glueEntity, Entity::discard);
|
||||||
|
|
||||||
BlockPos bearingPos = util.grid.at(2, 1, 2);
|
BlockPos bearingPos = util.grid.at(2, 1, 2);
|
||||||
scene.world.configureCenterOfRotation(plank, util.vector.centerOf(bearingPos));
|
scene.world.configureCenterOfRotation(plank, util.vector.centerOf(bearingPos));
|
||||||
|
@ -553,13 +553,13 @@ public class ChassisScenes {
|
||||||
scene.world.rotateSection(plank, 0, 360, 0, 80);
|
scene.world.rotateSection(plank, 0, 360, 0, 80);
|
||||||
scene.idle(90);
|
scene.idle(90);
|
||||||
|
|
||||||
glueEntity = scene.world.createGlueEntity(central, Direction.UP);
|
// glueEntity = scene.world.createGlueEntity(central, Direction.UP);TODO
|
||||||
scene.world.destroyBlock(central.above());
|
scene.world.destroyBlock(central.above());
|
||||||
scene.idle(20);
|
scene.idle(20);
|
||||||
scene.addKeyframe();
|
scene.addKeyframe();
|
||||||
scene.overlay.showControls(new InputWindowElement(util.vector.topOf(central), Pointing.DOWN).leftClick(), 40);
|
scene.overlay.showControls(new InputWindowElement(util.vector.topOf(central), Pointing.DOWN).leftClick(), 40);
|
||||||
scene.idle(7);
|
scene.idle(7);
|
||||||
scene.world.modifyEntity(glueEntity, Entity::discard);
|
// scene.world.modifyEntity(glueEntity, Entity::discard);
|
||||||
scene.effects.superGlue(central, Direction.UP, false);
|
scene.effects.superGlue(central, Direction.UP, false);
|
||||||
scene.idle(10);
|
scene.idle(10);
|
||||||
scene.overlay.showText(60)
|
scene.overlay.showText(60)
|
||||||
|
|
|
@ -55,27 +55,47 @@ public class BlockClusterOutline extends Outline {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static Vec3 xyz = new Vec3(-.5, -.5, -.5);
|
||||||
|
static Vec3 Xyz = new Vec3(.5, -.5, -.5);
|
||||||
|
static Vec3 xYz = new Vec3(-.5, .5, -.5);
|
||||||
|
static Vec3 XYz = new Vec3(.5, .5, -.5);
|
||||||
|
static Vec3 xyZ = new Vec3(-.5, -.5, .5);
|
||||||
|
static Vec3 XyZ = new Vec3(.5, -.5, .5);
|
||||||
|
static Vec3 xYZ = new Vec3(-.5, .5, .5);
|
||||||
|
static Vec3 XYZ = new Vec3(.5, .5, .5);
|
||||||
|
|
||||||
protected void renderBlockFace(PoseStack ms, VertexConsumer builder, BlockPos pos, Direction face) {
|
protected void renderBlockFace(PoseStack ms, VertexConsumer builder, BlockPos pos, Direction face) {
|
||||||
Vec3 center = VecHelper.getCenterOf(pos);
|
Vec3 center = VecHelper.getCenterOf(pos);
|
||||||
Vec3 offset = Vec3.atLowerCornerOf(face.getNormal());
|
Vec3 offset = Vec3.atLowerCornerOf(face.getNormal());
|
||||||
Vec3 plane = VecHelper.axisAlingedPlaneOf(offset);
|
offset = offset.scale(1 / 128d);
|
||||||
Axis axis = face.getAxis();
|
center = center.add(offset);
|
||||||
|
|
||||||
offset = offset.scale(1 / 2f + 1 / 128d);
|
ms.pushPose();
|
||||||
plane = plane.scale(1 / 2f)
|
ms.translate(center.x, center.y, center.z);
|
||||||
.add(offset);
|
|
||||||
|
|
||||||
int deg = face.getAxisDirection()
|
switch (face) {
|
||||||
.getStep() * 90;
|
case DOWN:
|
||||||
Vec3 a1 = plane.add(center);
|
putQuad(ms, builder, xyz, Xyz, XyZ, xyZ, face);
|
||||||
plane = VecHelper.rotate(plane, deg, axis);
|
break;
|
||||||
Vec3 a2 = plane.add(center);
|
case EAST:
|
||||||
plane = VecHelper.rotate(plane, deg, axis);
|
putQuad(ms, builder, XYz, XYZ, XyZ, Xyz, face);
|
||||||
Vec3 a3 = plane.add(center);
|
break;
|
||||||
plane = VecHelper.rotate(plane, deg, axis);
|
case NORTH:
|
||||||
Vec3 a4 = plane.add(center);
|
putQuad(ms, builder, xYz, XYz, Xyz, xyz, face);
|
||||||
|
break;
|
||||||
|
case SOUTH:
|
||||||
|
putQuad(ms, builder, XYZ, xYZ, xyZ, XyZ, face);
|
||||||
|
break;
|
||||||
|
case UP:
|
||||||
|
putQuad(ms, builder, xYZ, XYZ, XYz, xYz, face);
|
||||||
|
break;
|
||||||
|
case WEST:
|
||||||
|
putQuad(ms, builder, xYZ, xYz, xyz, xyZ, face);
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
putQuad(ms, builder, a1, a2, a3, a4, face);
|
ms.popPose();
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class Cluster {
|
private static class Cluster {
|
||||||
|
|
BIN
src/main/resources/assets/create/textures/special/glue.png
Normal file
BIN
src/main/resources/assets/create/textures/special/glue.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 275 B |
Loading…
Reference in a new issue