WIP entity selector argument

This commit is contained in:
themode 2021-01-09 00:59:03 +01:00
parent 2c2a691af7
commit 8f99753235
7 changed files with 193 additions and 43 deletions

View File

@ -2,13 +2,13 @@ package net.minestom.server.command.builder;
import net.minestom.server.chat.ChatColor; import net.minestom.server.chat.ChatColor;
import net.minestom.server.command.builder.arguments.Argument; import net.minestom.server.command.builder.arguments.Argument;
import net.minestom.server.entity.Entity;
import net.minestom.server.entity.EntityType; import net.minestom.server.entity.EntityType;
import net.minestom.server.instance.block.Block; import net.minestom.server.instance.block.Block;
import net.minestom.server.item.Enchantment; import net.minestom.server.item.Enchantment;
import net.minestom.server.item.ItemStack; import net.minestom.server.item.ItemStack;
import net.minestom.server.particle.Particle; import net.minestom.server.particle.Particle;
import net.minestom.server.potion.PotionEffect; import net.minestom.server.potion.PotionEffect;
import net.minestom.server.utils.entity.EntityFinder;
import net.minestom.server.utils.location.RelativeBlockPosition; import net.minestom.server.utils.location.RelativeBlockPosition;
import net.minestom.server.utils.location.RelativeVec; import net.minestom.server.utils.location.RelativeVec;
import net.minestom.server.utils.math.FloatRange; import net.minestom.server.utils.math.FloatRange;
@ -20,7 +20,6 @@ import org.jglrxavpok.hephaistos.nbt.NBT;
import org.jglrxavpok.hephaistos.nbt.NBTCompound; import org.jglrxavpok.hephaistos.nbt.NBTCompound;
import java.util.HashMap; import java.util.HashMap;
import java.util.List;
import java.util.Map; import java.util.Map;
/** /**
@ -122,8 +121,8 @@ public final class Arguments {
} }
@NotNull @NotNull
public List<Entity> getEntities(@NotNull String id) { public EntityFinder getEntities(@NotNull String id) {
return (List<Entity>) getObject(id); return (EntityFinder) getObject(id);
} }
@NotNull @NotNull

View File

@ -100,7 +100,6 @@ public class ArgumentType {
return new ArgumentFloatRange(id); return new ArgumentFloatRange(id);
} }
@Deprecated
public static ArgumentEntities Entities(@NotNull String id) { public static ArgumentEntities Entities(@NotNull String id) {
return new ArgumentEntities(id); return new ArgumentEntities(id);
} }

View File

@ -2,8 +2,13 @@ package net.minestom.server.command.builder.arguments.minecraft;
import net.minestom.server.MinecraftServer; import net.minestom.server.MinecraftServer;
import net.minestom.server.command.builder.arguments.Argument; import net.minestom.server.command.builder.arguments.Argument;
import net.minestom.server.entity.Entity; import net.minestom.server.command.builder.exception.ArgumentSyntaxException;
import net.minestom.server.entity.EntityType;
import net.minestom.server.entity.GameMode;
import net.minestom.server.network.ConnectionManager; import net.minestom.server.network.ConnectionManager;
import net.minestom.server.registry.Registries;
import net.minestom.server.utils.entity.EntityFinder;
import net.minestom.server.utils.math.IntRange;
import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.NotNull;
import java.util.Arrays; import java.util.Arrays;
@ -15,21 +20,30 @@ import java.util.List;
* Represents the target selector argument. * Represents the target selector argument.
* https://minecraft.gamepedia.com/Commands#Target_selectors * https://minecraft.gamepedia.com/Commands#Target_selectors
*/ */
public class ArgumentEntities extends Argument<List<Entity>> { public class ArgumentEntities extends Argument<EntityFinder> {
private static final int SUCCESS = 0; private static final int SUCCESS = 0;
public static final int INVALID_SYNTAX = -2; public static final int INVALID_SYNTAX = -2;
public static final int ONLY_SINGLE_ENTITY_ERROR = -3; public static final int ONLY_SINGLE_ENTITY_ERROR = -3;
public static final int ONLY_PLAYERS_ERROR = -4; public static final int ONLY_PLAYERS_ERROR = -4;
public static final int INVALID_ARGUMENT_NAME = -5;
public static final int INVALID_ARGUMENT_VALUE = -6;
private static final ConnectionManager CONNECTION_MANAGER = MinecraftServer.getConnectionManager(); private static final ConnectionManager CONNECTION_MANAGER = MinecraftServer.getConnectionManager();
private static final List<String> selectorVariables = Arrays.asList("@p", "@r", "@a", "@e", "@s"); private static final List<String> selectorVariables = Arrays.asList("@p", "@r", "@a", "@e", "@s");
private static final List<String> playersOnlySelector = Arrays.asList("@p", "@r", "@a", "@s"); private static final List<String> playersOnlySelector = Arrays.asList("@p", "@r", "@a", "@s");
private static final List<String> singleOnlySelector = Arrays.asList("@p", "@r", "@s"); private static final List<String> singleOnlySelector = Arrays.asList("@p", "@r", "@s");
// List with all the valid arguments
private static final List<String> validArguments = Arrays.asList("x", "y", "z", private static final List<String> validArguments = Arrays.asList("x", "y", "z",
"distance", "dx", "dy", "dz", "distance", "dx", "dy", "dz",
"scores", "tag", "team", "limit", "sort", "level", "gamemode", "name", "scores", "tag", "team", "limit", "sort", "level", "gamemode",
"x_rotation", "y_rotation", "type", "nbt", "advancements", "predicate"); "x_rotation", "y_rotation", "type", "nbt", "advancements", "predicate");
// List with all the easily parsable arguments which only require reading until a specific character (comma)
private static final List<String> simpleArguments = Arrays.asList("x", "y", "z",
"distance", "dx", "dy", "dz",
"scores", "tag", "team", "limit", "sort", "level", "gamemode", "name",
"x_rotation", "y_rotation", "type", "advancements", "predicate");
private boolean onlySingleEntity; private boolean onlySingleEntity;
private boolean onlyPlayers; private boolean onlyPlayers;
@ -47,78 +61,164 @@ public class ArgumentEntities extends Argument<List<Entity>> {
return this; return this;
} }
public int getCorrectionResult(@NotNull String value) { @NotNull
System.out.println("check: " + value); @Override
public EntityFinder parse(@NotNull String input) throws ArgumentSyntaxException {
System.out.println("check: " + input);
// Check for raw player name // Check for raw player name
if (value.length() <= 16) { if (input.length() <= 16) {
if (CONNECTION_MANAGER.getPlayer(value) != null) if (CONNECTION_MANAGER.getPlayer(input) != null) {
return SUCCESS; // TODO success
}
} }
// The minimum size is always 0 (for the selector variable, ex: @p) // The minimum size is always 0 (for the selector variable, ex: @p)
if (value.length() < 2) if (input.length() < 2)
return INVALID_SYNTAX; throw new ArgumentSyntaxException("Length needs to be > 1", input, INVALID_SYNTAX);
// The target selector variable always start by '@' // The target selector variable always start by '@'
if (!value.startsWith("@")) if (!input.startsWith("@"))
return INVALID_SYNTAX; throw new ArgumentSyntaxException("Target selector needs to start with @", input, INVALID_SYNTAX);
final String selectorVariable = value.substring(0, 2); final String selectorVariable = input.substring(0, 2);
// Check if the selector variable used exists // Check if the selector variable used exists
if (!selectorVariables.contains(selectorVariable)) if (!selectorVariables.contains(selectorVariable))
return INVALID_SYNTAX; throw new ArgumentSyntaxException("Invalid selector variable", input, INVALID_SYNTAX);
// Check if it should only select single entity and if the selector variable valid the condition // Check if it should only select single entity and if the selector variable valid the condition
if (onlySingleEntity && !singleOnlySelector.contains(selectorVariable)) if (onlySingleEntity && !singleOnlySelector.contains(selectorVariable))
return ONLY_SINGLE_ENTITY_ERROR; throw new ArgumentSyntaxException("Argument requires only a single entity", input, ONLY_SINGLE_ENTITY_ERROR);
// Check if it should only select players and if the selector variable valid the condition // Check if it should only select players and if the selector variable valid the condition
if (onlyPlayers && !playersOnlySelector.contains(selectorVariable)) if (onlyPlayers && !playersOnlySelector.contains(selectorVariable))
return ONLY_PLAYERS_ERROR; throw new ArgumentSyntaxException("Argument requires only players", input, ONLY_PLAYERS_ERROR);
// Create the EntityFinder which will be used for the rest of the parsing
final EntityFinder entityFinder = new EntityFinder()
.setTargetSelector(toTargetSelector(selectorVariable));
// The selector is a single selector variable which verify all the conditions // The selector is a single selector variable which verify all the conditions
if (value.length() == 2) if (input.length() == 2)
return SUCCESS; return entityFinder;
// START PARSING THE STRUCTURE // START PARSING THE STRUCTURE
final String structure = value.substring(2); final String structure = input.substring(2);
return parseStructure(input, entityFinder, structure);
}
@NotNull
private EntityFinder parseStructure(@NotNull String input,
@NotNull EntityFinder entityFinder,
@NotNull String structure) throws ArgumentSyntaxException {
// The structure isn't opened or closed properly // The structure isn't opened or closed properly
if (!structure.startsWith("[") || !structure.endsWith("]")) if (!structure.startsWith("[") || !structure.endsWith("]"))
return INVALID_SYNTAX; throw new ArgumentSyntaxException("Target selector needs to start and end with brackets", input, INVALID_SYNTAX);
// Remove brackets
final String structureData = structure.substring(1, structure.length() - 1); final String structureData = structure.substring(1, structure.length() - 1);
System.out.println("structure data: " + structureData);
String currentArgument = ""; String currentArgument = "";
for (int i = 0; i < structureData.length(); i++) { for (int i = 0; i < structureData.length(); i++) {
final char c = structureData.charAt(i); final char c = structureData.charAt(i);
if (c == '=') { if (c == '=') {
i = retrieveArgument(structureData, currentArgument, i); System.out.println("type: " + currentArgument);
if (!validArguments.contains(currentArgument))
throw new ArgumentSyntaxException("Argument name '" + currentArgument + "' does not exist", input, INVALID_ARGUMENT_NAME);
i = parseArgument(entityFinder, currentArgument, input, structureData, i);
currentArgument = ""; // Reset current argument
} else { } else {
currentArgument += c; currentArgument += c;
} }
} }
return 0; return entityFinder;
} }
private int retrieveArgument(String structureData, String argument, int index) { private int parseArgument(@NotNull EntityFinder entityFinder,
int finalIndex = index; @NotNull String argumentName,
for (int i = index + 1; i < structureData.length(); i++) { @NotNull String input,
System.out.println("char: " + structureData.charAt(i)); @NotNull String structureData, int beginIndex) throws ArgumentSyntaxException {
System.out.println("retrieve: " + argument); final char comma = ',';
final boolean isSimple = simpleArguments.contains(argumentName);
int finalIndex = beginIndex + 1;
StringBuilder valueBuilder = new StringBuilder();
for (; finalIndex < structureData.length(); finalIndex++) {
final char c = structureData.charAt(finalIndex);
// Command is parsed
if (isSimple && c == comma)
break;
valueBuilder.append(c);
}
final String value = valueBuilder.toString().trim();
//System.out.println("value: " + value);
switch (argumentName) {
case "type": {
final boolean include = !value.startsWith("!");
final String entityName = include ? value : value.substring(1);
final EntityType entityType = Registries.getEntityType(entityName);
if (entityType == null)
throw new ArgumentSyntaxException("Invalid entity name", input, INVALID_ARGUMENT_VALUE);
entityFinder.setEntity(entityType, include ? EntityFinder.ToggleableType.INCLUDE : EntityFinder.ToggleableType.EXCLUDE);
break;
}
case "gamemode": {
final boolean include = !value.startsWith("!");
final String gameModeName = include ? value : value.substring(1);
try {
final GameMode gameMode = GameMode.valueOf(gameModeName);
entityFinder.setGameMode(gameMode, include ? EntityFinder.ToggleableType.INCLUDE : EntityFinder.ToggleableType.EXCLUDE);
} catch (IllegalArgumentException e) {
throw new ArgumentSyntaxException("Invalid entity game mode", input, INVALID_ARGUMENT_VALUE);
}
break;
}
case "limit":
try {
final int limit = Integer.parseInt(value);
entityFinder.setLimit(limit);
} catch (NumberFormatException e) {
throw new ArgumentSyntaxException("Invalid limit number", input, INVALID_ARGUMENT_VALUE);
}
break;
case "sort":
try {
EntityFinder.EntitySort entitySort = EntityFinder.EntitySort.valueOf(value.toUpperCase());
entityFinder.setEntitySort(entitySort);
} catch (IllegalArgumentException e) {
throw new ArgumentSyntaxException("Invalid entity sort", input, INVALID_ARGUMENT_VALUE);
}
break;
case "level":
try {
final IntRange level = ArgumentIntRange.staticParse(value);
entityFinder.setLevel(level);
} catch (ArgumentSyntaxException e) {
throw new ArgumentSyntaxException("Invalid level number", input, INVALID_ARGUMENT_VALUE);
}
break;
case "distance":
try {
final IntRange distance = ArgumentIntRange.staticParse(value);
entityFinder.setDistance(distance);
} catch (ArgumentSyntaxException e) {
throw new ArgumentSyntaxException("Invalid level number", input, INVALID_ARGUMENT_VALUE);
}
break;
} }
return finalIndex; return finalIndex;
} }
@NotNull
@Override
public List<Entity> parse(@NotNull String value) {
return null;
}
public boolean isOnlySingleEntity() { public boolean isOnlySingleEntity() {
return onlySingleEntity; return onlySingleEntity;
} }
@ -126,4 +226,18 @@ public class ArgumentEntities extends Argument<List<Entity>> {
public boolean isOnlyPlayers() { public boolean isOnlyPlayers() {
return onlyPlayers; return onlyPlayers;
} }
private static EntityFinder.TargetSelector toTargetSelector(@NotNull String selectorVariable) {
if (selectorVariable.equals("@p"))
return EntityFinder.TargetSelector.NEAREST_PLAYER;
if (selectorVariable.equals("@r"))
return EntityFinder.TargetSelector.RANDOM_PLAYER;
if (selectorVariable.equals("@a"))
return EntityFinder.TargetSelector.ALL_PLAYERS;
if (selectorVariable.equals("@e"))
return EntityFinder.TargetSelector.ALL_ENTITIES;
if (selectorVariable.equals("@s"))
return EntityFinder.TargetSelector.SELF;
throw new IllegalStateException("Weird selector variable: " + selectorVariable);
}
} }

View File

@ -20,6 +20,11 @@ public class ArgumentIntRange extends ArgumentRange<IntRange> {
@NotNull @NotNull
@Override @Override
public IntRange parse(@NotNull String input) throws ArgumentSyntaxException { public IntRange parse(@NotNull String input) throws ArgumentSyntaxException {
return staticParse(input);
}
@NotNull
public static IntRange staticParse(@NotNull String input) throws ArgumentSyntaxException {
try { try {
if (input.contains("..")) { if (input.contains("..")) {
final int index = input.indexOf('.'); final int index = input.indexOf('.');

View File

@ -9,6 +9,9 @@ import org.jetbrains.annotations.NotNull;
* Exception triggered when an {@link Argument} is wrongly parsed. * Exception triggered when an {@link Argument} is wrongly parsed.
* <p> * <p>
* Retrieved in {@link ArgumentCallback} defined in {@link Command#setArgumentCallback(ArgumentCallback, Argument)}. * Retrieved in {@link ArgumentCallback} defined in {@link Command#setArgumentCallback(ArgumentCallback, Argument)}.
* <p>
* Be aware that the message returned by {@link #getMessage()} is only here for debugging purpose,
* you should refer to {@link #getErrorCode()} to identify the exceptions.
*/ */
public class ArgumentSyntaxException extends Exception { public class ArgumentSyntaxException extends Exception {

View File

@ -35,10 +35,10 @@ public class EntityFinder {
// By traits // By traits
private OptionalInt limit; private OptionalInt limit;
private ToggleableMap<EntityType> entityTypes; private final ToggleableMap<EntityType> entityTypes = new ToggleableMap<>();
// Players specific // Players specific
private ToggleableMap<GameMode> gameModes; private final ToggleableMap<GameMode> gameModes = new ToggleableMap<>();
private IntRange level; private IntRange level;
public EntityFinder setTargetSelector(@NotNull TargetSelector targetSelector) { public EntityFinder setTargetSelector(@NotNull TargetSelector targetSelector) {
@ -71,6 +71,16 @@ public class EntityFinder {
return this; return this;
} }
public EntityFinder setEntity(@NotNull EntityType entityType, @NotNull ToggleableType toggleableType) {
this.entityTypes.put(entityType, toggleableType.getValue());
return this;
}
public EntityFinder setGameMode(@NotNull GameMode gameMode, @NotNull ToggleableType toggleableType) {
this.gameModes.put(gameMode, toggleableType.getValue());
return this;
}
public EntityFinder setDifference(double dx, double dy, double dz) { public EntityFinder setDifference(double dx, double dy, double dz) {
this.dx = OptionalDouble.of(dx); this.dx = OptionalDouble.of(dx);
this.dy = OptionalDouble.of(dy); this.dy = OptionalDouble.of(dy);
@ -125,7 +135,7 @@ public class EntityFinder {
} }
// Entity type // Entity type
if (entityTypes != null && !entityTypes.isEmpty()) { if (!entityTypes.isEmpty()) {
final EntityType requirement = entityTypes.requirement; final EntityType requirement = entityTypes.requirement;
result = result.stream().filter(entity -> { result = result.stream().filter(entity -> {
final EntityType entityType = entity.getEntityType(); final EntityType entityType = entity.getEntityType();
@ -137,7 +147,7 @@ public class EntityFinder {
} }
// GameMode // GameMode
if (gameModes != null && !gameModes.isEmpty()) { if (!gameModes.isEmpty()) {
final GameMode requirement = gameModes.requirement; final GameMode requirement = gameModes.requirement;
result = result.stream().filter(entity -> { result = result.stream().filter(entity -> {
if (!(entity instanceof Player)) if (!(entity instanceof Player))
@ -205,6 +215,20 @@ public class EntityFinder {
ARBITRARY, FURTHEST, NEAREST, RANDOM ARBITRARY, FURTHEST, NEAREST, RANDOM
} }
public enum ToggleableType {
INCLUDE(true), EXCLUDE(false);
private final boolean value;
ToggleableType(boolean value) {
this.value = value;
}
public boolean getValue() {
return value;
}
}
private static class ToggleableMap<T> extends Object2BooleanOpenHashMap<T> { private static class ToggleableMap<T> extends Object2BooleanOpenHashMap<T> {
@Nullable @Nullable

View File

@ -6,6 +6,7 @@ import net.minestom.server.command.builder.Command;
import net.minestom.server.command.builder.arguments.ArgumentType; import net.minestom.server.command.builder.arguments.ArgumentType;
import net.minestom.server.command.builder.arguments.minecraft.ArgumentEntities; import net.minestom.server.command.builder.arguments.minecraft.ArgumentEntities;
import net.minestom.server.entity.Entity; import net.minestom.server.entity.Entity;
import net.minestom.server.utils.entity.EntityFinder;
import java.util.List; import java.util.List;
@ -14,13 +15,18 @@ public class EntitySelectorCommand extends Command {
public EntitySelectorCommand() { public EntitySelectorCommand() {
super("ent"); super("ent");
setDefaultExecutor((sender, args) -> System.out.println("DEFAULT"));
ArgumentEntities argumentEntities = ArgumentType.Entities("entities"); ArgumentEntities argumentEntities = ArgumentType.Entities("entities");
setArgumentCallback((sender, exception) -> exception.printStackTrace(), argumentEntities);
addSyntax(this::executor, argumentEntities); addSyntax(this::executor, argumentEntities);
} }
private void executor(CommandSender commandSender, Arguments arguments) { private void executor(CommandSender commandSender, Arguments arguments) {
List<Entity> entities = arguments.getEntities("entities"); EntityFinder query = arguments.getEntities("entities");
System.out.println("SUCCESS COMMAND");
} }
} }