Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add RayTraceConfigurationBuilder #11907

Open
wants to merge 10 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
notTamion marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,180 @@
package io.papermc.paper.raytracing;

import org.bukkit.FluidCollisionMode;
import org.bukkit.Location;
import org.bukkit.block.Block;
import org.bukkit.entity.Entity;
import org.bukkit.util.Vector;
import org.jetbrains.annotations.Contract;
import org.jspecify.annotations.NullMarked;
import org.jspecify.annotations.Nullable;
import java.util.Set;
import java.util.function.Predicate;

/**
* Holds information about how to cast a raytrace.
*/
@NullMarked
public interface PositionedRayTraceConfigurationBuilder {

/**
* Gets the starting location.
*
* @return the starting location
*/
@Nullable
Location start();

/**
* Sets the starting location.
*
* @param start the new starting location
* @return a reference to this object
*/
@Contract("_ -> this")
PositionedRayTraceConfigurationBuilder start(Location start);

/**
* Gets the direction.
*
* @return the direction
*/
@Nullable
Vector direction();

/**
* Sets the direction.
*
* @param direction the new direction
* @return a reference to this object
*/
@Contract("_ -> this")
PositionedRayTraceConfigurationBuilder direction(Vector direction);

/**
* Gets the maximum distance.
*
* @return the maximum distance
*/
double maxDistance();

/**
* Sets the maximum distance.
*
* @param maxDistance the new maxDistance
* @return a reference to this object
*/
@Contract("_ -> this")
PositionedRayTraceConfigurationBuilder maxDistance(double maxDistance);

/**
* Gets the FluidCollisionMode when looking for block collisions.
*
* @return the FluidCollisionMode
*/
FluidCollisionMode fluidCollisionMode();

/**
* Sets the FluidCollisionMode when looking for block collisions.
* <p>
* If collisions with passable blocks are ignored, fluid collisions are
* ignored as well regardless of the fluid collision mode.
*
* @param fluidCollisionMode the new FluidCollisionMode
* @return a reference to this object
*/
@Contract("_ -> this")
PositionedRayTraceConfigurationBuilder fluidCollisionMode(FluidCollisionMode fluidCollisionMode);

/**
* Gets if the raytrace will ignore passable blocks when looking for block collisions.
*
* @return if the raytrace will ignore passable blocks
*/
boolean ignorePassableBlocks();

/**
* Sets whether the raytrace should ignore passable blocks when looking for
* block collisions.
* <p>
* If collisions with passable blocks are ignored, fluid collisions are
* ignored as well regardless of the fluid collision mode.
* <p>
* Portal blocks are only considered passable if the ray starts within them.
* Apart from that collisions with portal blocks will be considered even if
* collisions with passable blocks are otherwise ignored.
*
* @param ignorePassableBlocks if the raytrace should ignore passable blocks
* @return a reference to this object
*/
@Contract("_ -> this")
PositionedRayTraceConfigurationBuilder ignorePassableBlocks(boolean ignorePassableBlocks);

/**
* Gets the size of the raytrace when looking for entity collisions.
* <p>
* Block collisions use the blocks' precise collision shapes.
*
* @return the raytrace size
*/
double raySize();

/**
* Sets the size of the raytrace when looking for entity collisions.
*
* @param raySize the new raytrace size
* @return a reference to this object
*/
@Contract("_ -> this")
PositionedRayTraceConfigurationBuilder raySize(double raySize);

/**
* Gets the current entity filter when looking for entity collisions.
*
* @return predicate for entities the ray can potentially collide with, or null to consider all entities
*/
@Nullable
Predicate<? super Entity> entityFilter();

/**
* Sets the current entity filter when looking for entity collisions.
*
* @param entityFilter predicate for entities the ray can potentially collide with, or null to consider all entities
* @return a reference to this object
*/
@Contract("_ -> this")
PositionedRayTraceConfigurationBuilder entityFilter(@Nullable Predicate<? super Entity> entityFilter);

/**
* Gets the current block filter when looking for block collisions.
*
* @return predicate for blocks the ray can potentially collide with, or null to consider all blocks
*/
@Nullable
Predicate<? super Block> blockFilter();

/**
* Sets the current block filter when looking for block collisions.
*
* @param blockFilter predicate for blocks the ray can potentially collide with, or null to consider all blocks
* @return a reference to this object
*/
@Contract("_ -> this")
PositionedRayTraceConfigurationBuilder blockFilter(@Nullable Predicate<? super Block> blockFilter);

/**
* Gets the current set targets.
*
* @return the targets
*/
@Nullable
Set<RayTraceTarget> targets();

/**
* Sets the targets for the rayTrace.
*
* @return a reference to this object
*/
@Contract("_, _ -> this")
PositionedRayTraceConfigurationBuilder targets(RayTraceTarget first, RayTraceTarget... others);
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
package io.papermc.paper.raytracing;

/**
* List of Targets a builder can target.
*/
public enum RayTraceTarget {
ENTITY,
BLOCK
}
15 changes: 15 additions & 0 deletions paper-api/src/main/java/org/bukkit/World.java
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package org.bukkit;

import java.io.File;
import io.papermc.paper.raytracing.PositionedRayTraceConfigurationBuilder;
import org.bukkit.generator.ChunkGenerator;

import java.util.ArrayList;
Expand Down Expand Up @@ -1938,6 +1939,20 @@ default Iterable<? extends net.kyori.adventure.audience.Audience> audiences() {
@Nullable RayTraceResult rayTrace(io.papermc.paper.math.@NotNull Position start, @NotNull Vector direction, double maxDistance, @NotNull FluidCollisionMode fluidCollisionMode, boolean ignorePassableBlocks, double raySize, @Nullable Predicate<? super Entity> filter, @Nullable Predicate<? super Block> canCollide);
// Paper end

/**
* Performs a ray trace that checks for collisions with the specified
* targets.
* <p>
* This may cause loading of chunks! Some implementations may impose
* artificial restrictions on the maximum distance.
*
* @param builderConsumer a consumer to configure the ray trace configuration.
* The received builder is not valid for use outside the consumer
* @return the closest ray trace hit result with either a block or an
* entity, or <code>null</code> if there is no hit
*/
@Nullable RayTraceResult rayTrace(@NotNull Consumer<PositionedRayTraceConfigurationBuilder> builderConsumer);

/**
* Gets the default spawn {@link Location} of this world
*
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,130 @@
package io.papermc.paper.raytracing;

import org.bukkit.FluidCollisionMode;
import org.bukkit.Location;
import org.bukkit.block.Block;
import org.bukkit.entity.Entity;
import org.bukkit.util.Vector;
import org.jetbrains.annotations.UnmodifiableView;
import org.jspecify.annotations.NullMarked;
import org.jspecify.annotations.Nullable;
import java.util.Collections;
import java.util.EnumSet;
import java.util.Set;
import java.util.function.Predicate;

@NullMarked
public class PositionedRayTraceConfigurationBuilderImpl implements PositionedRayTraceConfigurationBuilder {
notTamion marked this conversation as resolved.
Show resolved Hide resolved

@Nullable
private Location start;
@Nullable
private Vector direction;
private double maxDistance;
private FluidCollisionMode fluidCollisionMode = FluidCollisionMode.NEVER;
private boolean ignorePassableBlocks;
private double raySize = 0.0D;
private Predicate<? super Entity> entityFilter;
private Predicate<? super Block> blockFilter;
@Nullable
private EnumSet<RayTraceTarget> targets;

notTamion marked this conversation as resolved.
Show resolved Hide resolved
@Override
public Location start() {
return this.start == null ? null : this.start.clone();
}
notTamion marked this conversation as resolved.
Show resolved Hide resolved

@Override
public PositionedRayTraceConfigurationBuilder start(final Location start) {
this.start = start.clone();
return this;
}

@Override
public Vector direction() {
return this.direction == null ? null : this.direction.clone();
}

@Override
public PositionedRayTraceConfigurationBuilder direction(final Vector direction) {
this.direction = direction.clone();
return this;
}

@Override
public double maxDistance() {
return maxDistance;
}

@Override
public PositionedRayTraceConfigurationBuilder maxDistance(double maxDistance) {
this.maxDistance = maxDistance;
return this;
}

@Override
public FluidCollisionMode fluidCollisionMode() {
return fluidCollisionMode;
}

@Override
public PositionedRayTraceConfigurationBuilder fluidCollisionMode(FluidCollisionMode fluidCollisionMode) {
this.fluidCollisionMode = fluidCollisionMode;
return this;
}

@Override
public boolean ignorePassableBlocks() {
return ignorePassableBlocks;
}

@Override
public PositionedRayTraceConfigurationBuilder ignorePassableBlocks(boolean ignorePassableBlocks) {
this.ignorePassableBlocks = ignorePassableBlocks;
return this;
}

@Override
public double raySize() {
return raySize;
}

@Override
public PositionedRayTraceConfigurationBuilder raySize(double raySize) {
this.raySize = raySize;
return this;
}

@Override
public Predicate<? super Entity> entityFilter() {
return entityFilter;
}

@Override
public PositionedRayTraceConfigurationBuilder entityFilter(Predicate<? super Entity> entityFilter) {
this.entityFilter = entityFilter;
return this;
}

@Override
public Predicate<? super Block> blockFilter() {
return blockFilter;
}

@Override
public PositionedRayTraceConfigurationBuilder blockFilter(Predicate<? super Block> blockFilter) {
this.blockFilter = blockFilter;
return this;
}

@Override
public @UnmodifiableView Set<RayTraceTarget> targets() {
return this.targets == null ? Set.of() : Collections.unmodifiableSet(this.targets);
}

@Override
public PositionedRayTraceConfigurationBuilder targets(final RayTraceTarget first, final RayTraceTarget ... others) {
this.targets = EnumSet.of(first, others);
return this;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,12 @@
import com.google.common.collect.ImmutableMap;
import com.mojang.datafixers.util.Pair;
import io.papermc.paper.FeatureHooks;
import io.papermc.paper.raytracing.RayTraceTarget;
import io.papermc.paper.registry.RegistryAccess;
import io.papermc.paper.registry.RegistryKey;
import it.unimi.dsi.fastutil.longs.Long2ObjectMap;
import io.papermc.paper.raytracing.PositionedRayTraceConfigurationBuilder;
import io.papermc.paper.raytracing.PositionedRayTraceConfigurationBuilderImpl;
import it.unimi.dsi.fastutil.objects.Object2IntOpenHashMap;
import java.io.File;
import java.util.ArrayList;
Expand Down Expand Up @@ -1248,6 +1251,20 @@ public RayTraceResult rayTrace(io.papermc.paper.math.Position start, Vector dire
return blockHit;
}

@Override
public RayTraceResult rayTrace(Consumer<PositionedRayTraceConfigurationBuilder> builderConsumer) {
PositionedRayTraceConfigurationBuilderImpl builder = new PositionedRayTraceConfigurationBuilderImpl();

builderConsumer.accept(builder);

if (builder.targets().contains(RayTraceTarget.ENTITY)) {
if (builder.targets().contains(RayTraceTarget.BLOCK))
return this.rayTrace(builder.start(), builder.direction(), builder.maxDistance(), builder.fluidCollisionMode(), builder.ignorePassableBlocks(), builder.raySize(), builder.entityFilter(), builder.blockFilter());
return this.rayTraceEntities(builder.start(), builder.direction(), builder.maxDistance(), builder.raySize(), builder.entityFilter());
}
return this.rayTraceBlocks(builder.start(), builder.direction(), builder.maxDistance(), builder.fluidCollisionMode(), builder.ignorePassableBlocks(), builder.blockFilter());
}

@Override
public List<Player> getPlayers() {
List<Player> list = new ArrayList<Player>(this.world.players().size());
Expand Down
Loading