mirror of
https://github.com/Creators-of-Create/Create.git
synced 2025-01-14 08:16:42 +01:00
Toadstool power tool
- Fixed Saw leaving behind floating bee nests - Saws can now be used to cut down big mushrooms - Fixed getFluidState() accessing chunks of a wrapped server world (-> Fixes tree fertilizer on 1.19)
This commit is contained in:
parent
04f4adcf46
commit
7579187dd0
6 changed files with 85 additions and 49 deletions
|
@ -5633,6 +5633,7 @@ d99d5c67bdffff60789a19bd51a5c5267c75e0a4 data/create/tags/blocks/casing.json
|
|||
74f4ba5f6f61c30e27947c6fb4557e888d018285 data/create/tags/blocks/safe_nbt.json
|
||||
6cdeeac1689f7b5bfd9bc40b462143d8eaf3ad0b data/create/tags/blocks/seats.json
|
||||
d063e12c9ef75f39518c6d129ea35d833464d547 data/create/tags/blocks/toolboxes.json
|
||||
8861f982c335a1f17796df4af53a7d5dc91fc85a data/create/tags/blocks/tree_attachments.json
|
||||
50936b211d94167a35ec78c89954082a336b6269 data/create/tags/blocks/valve_handles.json
|
||||
eac71740fb12bdb38b5dfaa2268613d7ba82b809 data/create/tags/blocks/windmill_sails.json
|
||||
74700d556ca80c7a1db5fd4efb09c3ddb26cad66 data/create/tags/blocks/windowable.json
|
||||
|
|
|
@ -0,0 +1,10 @@
|
|||
{
|
||||
"replace": false,
|
||||
"values": [
|
||||
"minecraft:bee_nest",
|
||||
"minecraft:vine",
|
||||
"minecraft:moss_carpet",
|
||||
"minecraft:shroomlight",
|
||||
"minecraft:cocoa"
|
||||
]
|
||||
}
|
|
@ -124,6 +124,7 @@ public class AllTags {
|
|||
WINDMILL_SAILS,
|
||||
WINDOWABLE,
|
||||
WRENCH_PICKUP,
|
||||
TREE_ATTACHMENTS,
|
||||
|
||||
RELOCATION_NOT_SUPPORTED(FORGE),
|
||||
WG_STONE(FORGE),
|
||||
|
@ -414,6 +415,9 @@ public class AllTags {
|
|||
Blocks.COMPARATOR, Blocks.OBSERVER, Blocks.REDSTONE_WALL_TORCH, Blocks.PISTON, Blocks.STICKY_PISTON,
|
||||
Blocks.TRIPWIRE, Blocks.TRIPWIRE_HOOK, Blocks.DAYLIGHT_DETECTOR, Blocks.TARGET, Blocks.HOPPER);
|
||||
|
||||
AllBlockTags.TREE_ATTACHMENTS.add(Blocks.BEE_NEST, Blocks.VINE, Blocks.MOSS_CARPET, Blocks.SHROOMLIGHT,
|
||||
Blocks.COCOA);
|
||||
|
||||
AllBlockTags.ORE_OVERRIDE_STONE.includeAll(BlockTags.STONE_ORE_REPLACEABLES);
|
||||
|
||||
registerCompat();
|
||||
|
|
|
@ -478,8 +478,7 @@ public class SawTileEntity extends BlockBreakingKineticTileEntity {
|
|||
public static boolean isSawable(BlockState stateToBreak) {
|
||||
if (stateToBreak.is(BlockTags.SAPLINGS))
|
||||
return false;
|
||||
if (stateToBreak.is(BlockTags.LOGS) || AllTags.AllBlockTags.SLIMY_LOGS.matches(stateToBreak)
|
||||
|| (stateToBreak.is(BlockTags.LEAVES)))
|
||||
if (TreeCutter.isLog(stateToBreak) || (stateToBreak.is(BlockTags.LEAVES)))
|
||||
return true;
|
||||
Block block = stateToBreak.getBlock();
|
||||
if (block instanceof BambooBlock)
|
||||
|
|
|
@ -8,12 +8,14 @@ import java.util.List;
|
|||
import java.util.Optional;
|
||||
import java.util.Set;
|
||||
import java.util.function.BiConsumer;
|
||||
import java.util.function.Consumer;
|
||||
import java.util.function.Predicate;
|
||||
|
||||
import javax.annotation.Nonnull;
|
||||
import javax.annotation.Nullable;
|
||||
|
||||
import com.simibubi.create.AllTags;
|
||||
import com.simibubi.create.AllTags.AllBlockTags;
|
||||
import com.simibubi.create.compat.Mods;
|
||||
import com.simibubi.create.compat.dynamictrees.DynamicTree;
|
||||
|
||||
|
@ -42,7 +44,8 @@ public class TreeCutter {
|
|||
public static final Tree NO_TREE = new Tree(Collections.emptyList(), Collections.emptyList());
|
||||
|
||||
public static boolean canDynamicTreeCutFrom(Block startBlock) {
|
||||
return Mods.DYNAMICTREES.runIfInstalled(() -> () -> DynamicTree.isDynamicBranch(startBlock)).orElse(false);
|
||||
return Mods.DYNAMICTREES.runIfInstalled(() -> () -> DynamicTree.isDynamicBranch(startBlock))
|
||||
.orElse(false);
|
||||
}
|
||||
|
||||
@Nonnull
|
||||
|
@ -121,57 +124,61 @@ public class TreeCutter {
|
|||
if (!isLog(reader.getBlockState(currentPos)))
|
||||
continue;
|
||||
logs.add(currentPos);
|
||||
addNeighbours(currentPos, frontier, visited);
|
||||
forNeighbours(currentPos, visited, true, p -> frontier.add(new BlockPos(p)));
|
||||
}
|
||||
|
||||
// Find all leaves
|
||||
visited.clear();
|
||||
visited.addAll(logs);
|
||||
frontier.addAll(logs);
|
||||
|
||||
while (!frontier.isEmpty()) {
|
||||
BlockPos currentPos = frontier.remove(0);
|
||||
if (!logs.contains(currentPos) && visited.contains(currentPos))
|
||||
BlockPos prevPos = frontier.remove(0);
|
||||
if (!logs.contains(prevPos) && visited.contains(prevPos))
|
||||
continue;
|
||||
visited.add(currentPos);
|
||||
|
||||
BlockState blockState = reader.getBlockState(currentPos);
|
||||
boolean isLog = isLog(blockState);
|
||||
boolean isLeaf = isLeaf(blockState);
|
||||
boolean isGenericLeaf = isLeaf || isNonDecayingLeaf(blockState);
|
||||
visited.add(prevPos);
|
||||
BlockState prevState = reader.getBlockState(prevPos);
|
||||
int prevLeafDistance = isLeaf(prevState) ? getLeafDistance(prevState) : 0;
|
||||
|
||||
if (!isLog && !isGenericLeaf)
|
||||
continue;
|
||||
if (isGenericLeaf)
|
||||
leaves.add(currentPos);
|
||||
forNeighbours(prevPos, visited, false, currentPos -> {
|
||||
BlockState state = reader.getBlockState(currentPos);
|
||||
BlockPos subtract = currentPos.subtract(pos);
|
||||
BlockPos currentPosImmutable = currentPos.immutable();
|
||||
|
||||
if (AllBlockTags.TREE_ATTACHMENTS.matches(state)) {
|
||||
leaves.add(currentPosImmutable);
|
||||
visited.add(currentPosImmutable);
|
||||
return;
|
||||
}
|
||||
|
||||
int horizontalDistance = Math.max(Math.abs(subtract.getX()), Math.abs(subtract.getZ()));
|
||||
if (horizontalDistance <= nonDecayingLeafDistance(state)) {
|
||||
leaves.add(currentPosImmutable);
|
||||
frontier.add(currentPosImmutable);
|
||||
return;
|
||||
}
|
||||
|
||||
if (isLeaf(state) && getLeafDistance(state) > prevLeafDistance) {
|
||||
leaves.add(currentPosImmutable);
|
||||
frontier.add(currentPosImmutable);
|
||||
return;
|
||||
}
|
||||
|
||||
});
|
||||
}
|
||||
|
||||
return new Tree(logs, leaves);
|
||||
}
|
||||
|
||||
private static int getLeafDistance(BlockState state) {
|
||||
IntegerProperty distanceProperty = LeavesBlock.DISTANCE;
|
||||
for (Property<?> property : blockState.getValues()
|
||||
for (Property<?> property : state.getValues()
|
||||
.keySet())
|
||||
if (property instanceof IntegerProperty ip && property.getName()
|
||||
.equals("distance"))
|
||||
distanceProperty = ip;
|
||||
|
||||
int distance = !isLeaf ? 0 : blockState.getValue(distanceProperty);
|
||||
for (Direction direction : Iterate.directions) {
|
||||
BlockPos offset = currentPos.relative(direction);
|
||||
if (visited.contains(offset))
|
||||
continue;
|
||||
BlockState state = reader.getBlockState(offset);
|
||||
BlockPos subtract = offset.subtract(pos);
|
||||
|
||||
for (Property<?> property : state.getValues().keySet())
|
||||
if (property instanceof IntegerProperty ip && property.getName().equals("distance"))
|
||||
distanceProperty = ip;
|
||||
|
||||
int horizontalDistance = Math.max(Math.abs(subtract.getX()), Math.abs(subtract.getZ()));
|
||||
if (isLeaf(state) && state.getValue(distanceProperty) > distance
|
||||
|| isNonDecayingLeaf(state) && horizontalDistance < 4)
|
||||
frontier.add(offset);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
return new Tree(logs, leaves);
|
||||
return state.getValue(distanceProperty);
|
||||
}
|
||||
|
||||
public static boolean isChorus(BlockState stateAbove) {
|
||||
|
@ -232,23 +239,32 @@ public class TreeCutter {
|
|||
return true;
|
||||
}
|
||||
|
||||
private static void addNeighbours(BlockPos pos, List<BlockPos> frontier, Set<BlockPos> visited) {
|
||||
BlockPos.betweenClosedStream(pos.offset(-1, -1, -1), pos.offset(1, 1, 1))
|
||||
private static void forNeighbours(BlockPos pos, Set<BlockPos> visited, boolean up, Consumer<BlockPos> acceptor) {
|
||||
BlockPos.betweenClosedStream(pos.offset(-1, up ? 0 : -1, -1), pos.offset(1, 1, 1))
|
||||
.filter(((Predicate<BlockPos>) visited::contains).negate())
|
||||
.forEach(p -> frontier.add(new BlockPos(p)));
|
||||
.forEach(acceptor);
|
||||
}
|
||||
|
||||
private static boolean isLog(BlockState state) {
|
||||
return state.is(BlockTags.LOGS) || AllTags.AllBlockTags.SLIMY_LOGS.matches(state);
|
||||
public static boolean isLog(BlockState state) {
|
||||
return state.is(BlockTags.LOGS) || AllTags.AllBlockTags.SLIMY_LOGS.matches(state)
|
||||
|| state.is(Blocks.MUSHROOM_STEM);
|
||||
}
|
||||
|
||||
private static boolean isNonDecayingLeaf(BlockState state) {
|
||||
return state.is(BlockTags.WART_BLOCKS) || state.getBlock() == Blocks.SHROOMLIGHT;
|
||||
private static int nonDecayingLeafDistance(BlockState state) {
|
||||
if (state.is(Blocks.RED_MUSHROOM_BLOCK))
|
||||
return 2;
|
||||
if (state.is(Blocks.BROWN_MUSHROOM_BLOCK))
|
||||
return 3;
|
||||
if (state.is(BlockTags.WART_BLOCKS) || state.is(Blocks.WEEPING_VINES) || state.is(Blocks.WEEPING_VINES_PLANT))
|
||||
return 3;
|
||||
return -1;
|
||||
}
|
||||
|
||||
private static boolean isLeaf(BlockState state) {
|
||||
for (Property<?> property : state.getValues().keySet())
|
||||
if (property instanceof IntegerProperty && property.getName().equals("distance"))
|
||||
for (Property<?> property : state.getValues()
|
||||
.keySet())
|
||||
if (property instanceof IntegerProperty && property.getName()
|
||||
.equals("distance"))
|
||||
return true;
|
||||
return false;
|
||||
}
|
||||
|
|
|
@ -7,6 +7,7 @@ import net.minecraft.core.BlockPos;
|
|||
import net.minecraft.server.level.ServerLevel;
|
||||
import net.minecraft.world.level.block.Blocks;
|
||||
import net.minecraft.world.level.block.state.BlockState;
|
||||
import net.minecraft.world.level.material.FluidState;
|
||||
|
||||
public class PlacementSimulationServerWorld extends WrappedServerWorld {
|
||||
public HashMap<BlockPos, BlockState> blocksAdded;
|
||||
|
@ -53,4 +54,9 @@ public class PlacementSimulationServerWorld extends WrappedServerWorld {
|
|||
return Blocks.AIR.defaultBlockState();
|
||||
}
|
||||
|
||||
@Override
|
||||
public FluidState getFluidState(BlockPos pos) {
|
||||
return getBlockState(pos).getFluidState();
|
||||
}
|
||||
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue