Skip to content

Commit

Permalink
Add closure parameter to spawn_entity() to apply entity attributes be…
Browse files Browse the repository at this point in the history
…fore being added to the world
  • Loading branch information
PseudoKnight committed Aug 8, 2018
1 parent 03f9d18 commit 86d1e75
Show file tree
Hide file tree
Showing 3 changed files with 62 additions and 17 deletions.
3 changes: 3 additions & 0 deletions src/main/java/com/laytonsmith/abstraction/MCWorld.java
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
import com.laytonsmith.abstraction.enums.MCWorldEnvironment;
import com.laytonsmith.abstraction.enums.MCWorldType;
import com.laytonsmith.core.constructs.CArray;
import com.laytonsmith.core.constructs.CClosure;
import com.laytonsmith.core.constructs.Target;

import java.util.List;
Expand Down Expand Up @@ -77,6 +78,8 @@ public interface MCWorld extends MCMetadatable {

MCEntity spawn(MCLocation l, MCEntityType entType);

MCEntity spawn(MCLocation l, MCEntityType entType, final CClosure closure);

MCEntity spawn(MCLocation l, MCEntityType.MCVanillaEntityType entityType);

boolean generateTree(MCLocation l, MCTreeType treeType);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -56,10 +56,13 @@
import com.laytonsmith.abstraction.enums.bukkit.BukkitMCTreeType;
import com.laytonsmith.abstraction.enums.bukkit.BukkitMCWorldEnvironment;
import com.laytonsmith.abstraction.enums.bukkit.BukkitMCWorldType;
import com.laytonsmith.core.Static;
import com.laytonsmith.core.constructs.CArray;
import com.laytonsmith.core.constructs.CClosure;
import com.laytonsmith.core.constructs.CString;
import com.laytonsmith.core.constructs.Target;
import com.laytonsmith.core.exceptions.CRE.CREFormatException;
import com.laytonsmith.core.exceptions.FunctionReturnException;
import org.bukkit.Chunk;
import org.bukkit.Effect;
import org.bukkit.FireworkEffect;
Expand Down Expand Up @@ -125,6 +128,7 @@
import org.bukkit.entity.ZombieHorse;
import org.bukkit.entity.ZombieVillager;
import org.bukkit.inventory.meta.FireworkMeta;
import org.bukkit.util.Consumer;

import java.util.ArrayList;
import java.util.List;
Expand Down Expand Up @@ -294,6 +298,29 @@ public MCEntity spawn(MCLocation l, MCEntityType entType) {
((BukkitMCEntityType) entType).getConcrete()));
}

@Override
public MCEntity spawn(MCLocation l, MCEntityType entType, final CClosure closure) {
EntityType type = (EntityType) entType.getConcrete();
Consumer<? extends Entity> consumer = (Consumer<Entity>) entity -> {
MCEntity temp = BukkitConvertor.BukkitGetCorrectEntity(entity);
Static.InjectEntity(temp);
try {
closure.execute(new CString(entity.getUniqueId().toString(), Target.UNKNOWN));
} catch (FunctionReturnException ex) {
// do nothing
} finally {
Static.UninjectEntity(temp);
}
};
Entity ent = this.spawn((Location) l.getHandle(), type.getEntityClass(), consumer);
return BukkitConvertor.BukkitGetCorrectEntity(ent);
}

@SuppressWarnings("unchecked")
private <T extends Entity> Entity spawn(Location location, Class<T> clazz, Consumer<? extends Entity> consumer) {
return w.spawn(location, clazz, (Consumer<T>) consumer);
}

@Override
public MCEntity spawn(MCLocation l, MCEntityType.MCVanillaEntityType entityType) {
return BukkitConvertor.BukkitGetCorrectEntity(w.spawnEntity(
Expand Down
49 changes: 32 additions & 17 deletions src/main/java/com/laytonsmith/core/functions/EntityManagement.java
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,7 @@
import com.laytonsmith.core.Static;
import com.laytonsmith.core.constructs.CArray;
import com.laytonsmith.core.constructs.CBoolean;
import com.laytonsmith.core.constructs.CClosure;
import com.laytonsmith.core.constructs.CDouble;
import com.laytonsmith.core.constructs.CInt;
import com.laytonsmith.core.constructs.CNull;
Expand All @@ -107,10 +108,10 @@
import com.laytonsmith.core.exceptions.CRE.CREBadEntityTypeException;
import com.laytonsmith.core.exceptions.CRE.CRECastException;
import com.laytonsmith.core.exceptions.CRE.CREFormatException;
import com.laytonsmith.core.exceptions.CRE.CREIllegalArgumentException;
import com.laytonsmith.core.exceptions.CRE.CREIndexOverflowException;
import com.laytonsmith.core.exceptions.CRE.CREInvalidWorldException;
import com.laytonsmith.core.exceptions.CRE.CRELengthException;
import com.laytonsmith.core.exceptions.CRE.CRENotFoundException;
import com.laytonsmith.core.exceptions.CRE.CREPlayerOfflineException;
import com.laytonsmith.core.exceptions.CRE.CRERangeException;
import com.laytonsmith.core.exceptions.CRE.CREThrowable;
Expand Down Expand Up @@ -1128,7 +1129,7 @@ public static class spawn_entity extends EntityFunction {
@Override
public Class<? extends CREThrowable>[] thrown() {
return new Class[]{CRECastException.class, CREFormatException.class, CREBadEntityException.class,
CREInvalidWorldException.class, CREPlayerOfflineException.class, CRENotFoundException.class};
CREInvalidWorldException.class, CREPlayerOfflineException.class, CREIllegalArgumentException.class};
}

@Override
Expand All @@ -1139,8 +1140,16 @@ public Construct exec(Target t, Environment environment, Construct... args) thro
MCEntityType entType;
MCLocation l;
MCEntity ent;
if(args.length == 3) {
CClosure consumer = null;
if(args.length >= 3) {
l = ObjectGenerator.GetGenerator().location(args[2], null, t);
if(args.length == 4) {
if(args[3] instanceof CClosure) {
consumer = (CClosure) args[3];
} else {
throw new CREIllegalArgumentException("Expected a closure as last argument for spawn_entity().", t);
}
}
} else {
if(cs instanceof MCPlayer) {
l = ((MCPlayer) cs).getLocation();
Expand All @@ -1157,15 +1166,11 @@ public Construct exec(Target t, Environment environment, Construct... args) thro
}
try {
entType = MCEntityType.valueOf(args[0].val().toUpperCase());
if(entType == null) {
throw new CRENotFoundException(
"Could not find the entity type internal object (are you running in cmdline mode?)", t);
}
if(!entType.isSpawnable()) {
throw new CREFormatException("Unspawnable entitytype: " + args[0].val(), t);
throw new CREFormatException("Unspawnable entity type: " + args[0].val(), t);
}
} catch (IllegalArgumentException iae) {
throw new CREFormatException("Unknown entitytype: " + args[0].val(), t);
throw new CREFormatException("Unknown entity type: " + args[0].val(), t);
}
for(int i = 0; i < qty; i++) {
switch(entType.getAbstracted()) {
Expand All @@ -1180,13 +1185,21 @@ public Construct exec(Target t, Environment environment, Construct... args) thro
case LEASH_HITCH:
case PAINTING:
try {
ent = l.getWorld().spawn(l.getBlock().getLocation(), entType);
if(consumer != null) {
ent = l.getWorld().spawn(l.getBlock().getLocation(), entType, consumer);
} else {
ent = l.getWorld().spawn(l.getBlock().getLocation(), entType);
}
} catch (NullPointerException | IllegalArgumentException ex) {
throw new CREFormatException("Unspawnable location for " + entType.getAbstracted().name(), t);
}
break;
default:
ent = l.getWorld().spawn(l, entType);
if(consumer != null) {
ent = l.getWorld().spawn(l, entType, consumer);
} else {
ent = l.getWorld().spawn(l, entType);
}
}
ret.push(new CString(ent.getUniqueId().toString(), t), t);
}
Expand All @@ -1211,13 +1224,15 @@ public String docs() {
spawnable.add(type.name());
}
}
return "array {entityType, [qty], [location]} Spawns the specified number of entities of the given type"
+ " at the given location. Returns an array of entityIDs of what is spawned. Qty defaults to 1"
+ " and location defaults to the location of the commandsender, if it is a block or player."
+ " If the commandsender is console, location must be supplied. ---- Entitytype can be one of "
+ StringUtils.Join(spawnable, ", ", " or ", ", or ")
return "array {entityType, [qty], [location], [closure]} Spawns the specified number of entities of the"
+ " given type at the given location. Returns an array of entityIDs of what is spawned."
+ " Qty defaults to 1 and location defaults to the location of the commandsender,"
+ " if it is a block or player. If the commandsender is console, location must be supplied."
+ " ---- Entitytype can be one of " + StringUtils.Join(spawnable, ", ", " or ", ", or ")
+ ". Falling_blocks will be sand by default, and dropped_items will be stone,"
+ " as these entities already have their own functions for spawning.";
+ " as these entities already have their own functions for spawning."
+ " A closure can be used as the last argument to modify the entity before adding it to the world."
+ " The entity's UUID is passed to the closure as the first parameter.";
}

@Override
Expand Down

0 comments on commit 86d1e75

Please sign in to comment.