diff --git a/data/conf/zones.xsd b/data/conf/zones.xsd
index 1fd34be1f91..569b5553765 100644
--- a/data/conf/zones.xsd
+++ b/data/conf/zones.xsd
@@ -25,6 +25,7 @@
maxOccurs="unbounded"/>
+
@@ -178,4 +179,27 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/doc/CHANGES.txt b/doc/CHANGES.txt
index bac0f249a45..4bf753c0c62 100644
--- a/doc/CHANGES.txt
+++ b/doc/CHANGES.txt
@@ -8,6 +8,7 @@ Changelog
- renamed chef Patrick to Preston
- new achievements:
- 30 Minutes or Less: Deliver 25 hot pizzas
+- support added for dynamic creature spawning
*web client*
- fixed text extending past edge of notification bubbles
diff --git a/src/games/stendhal/server/core/config/ZonesXMLLoader.java b/src/games/stendhal/server/core/config/ZonesXMLLoader.java
index 0798dfd9d2c..26c742d8ebf 100644
--- a/src/games/stendhal/server/core/config/ZonesXMLLoader.java
+++ b/src/games/stendhal/server/core/config/ZonesXMLLoader.java
@@ -4,7 +4,7 @@
* $Id$
*/
/***************************************************************************
- * (C) Copyright 2003-2023 - Stendhal *
+ * (C) Copyright 2003-2024 - Stendhal *
***************************************************************************
***************************************************************************
* *
@@ -38,6 +38,7 @@
import games.stendhal.common.tiled.StendhalMapStructure;
import games.stendhal.server.core.config.zone.AttributesXMLReader;
import games.stendhal.server.core.config.zone.ConfiguratorXMLReader;
+import games.stendhal.server.core.config.zone.CreatureSpawnsXMLReader;
import games.stendhal.server.core.config.zone.EntitySetupXMLReader;
import games.stendhal.server.core.config.zone.PortalSetupXMLReader;
import games.stendhal.server.core.config.zone.RegionNameSubstitutionHelper;
@@ -60,6 +61,8 @@ public final class ZonesXMLLoader {
/** Zone attributes reader. */
private static final SetupXMLReader attributesReader = new AttributesXMLReader();
+ /** Dynamically spawned creatures. */
+ private static final SetupXMLReader creatureSpawnsReader = new CreatureSpawnsXMLReader();
/**
* The ConfiguratorDescriptor XML reader.
*/
@@ -417,6 +420,13 @@ public ZoneDesc readZone(final Element element) {
continue;
} else if (tag.equals("associated")) {
desc.setAssociatedZones(child.getAttribute("zones"));
+ } else if (tag.equals("spawns")) {
+ for (final Element spawn: XMLUtil.getElements(child)) {
+ final String spawnType = spawn.getTagName();
+ if ("creatures".equals(spawnType)) {
+ setupDesc = creatureSpawnsReader.read(spawn);
+ }
+ }
} else {
logger.warn("Zone [" + name + "] has unknown element: " + tag);
continue;
diff --git a/src/games/stendhal/server/core/config/zone/CreatureSpawnsXMLReader.java b/src/games/stendhal/server/core/config/zone/CreatureSpawnsXMLReader.java
new file mode 100644
index 00000000000..e022d93c166
--- /dev/null
+++ b/src/games/stendhal/server/core/config/zone/CreatureSpawnsXMLReader.java
@@ -0,0 +1,56 @@
+/***************************************************************************
+ * Copyright © 2024 - Faiumoni e. V. *
+ ***************************************************************************
+ ***************************************************************************
+ * *
+ * This program is free software; you can redistribute it and/or modify *
+ * it under the terms of the GNU General Public License as published by *
+ * the Free Software Foundation; either version 2 of the License, or *
+ * (at your option) any later version. *
+ * *
+ ***************************************************************************/
+package games.stendhal.server.core.config.zone;
+
+import java.util.Map;
+
+import org.apache.log4j.Logger;
+import org.w3c.dom.Element;
+
+import games.stendhal.common.MathHelper;
+import games.stendhal.server.core.engine.StendhalRPZone;
+import games.stendhal.server.entity.mapstuff.spawner.DynamicCreatureSpawner;
+
+
+/**
+ * Parses info for dynamically spawned creatures.
+ */
+public class CreatureSpawnsXMLReader extends SetupXMLReader {
+
+ private static Logger logger = Logger.getLogger(CreatureSpawnsXMLReader.class);
+
+
+ @Override
+ public SetupDescriptor read(final Element element) {
+ final CreatureSpawnsDescriptor desc = new CreatureSpawnsDescriptor();
+ readParameters(desc, element);
+ return desc;
+ }
+
+ private static class CreatureSpawnsDescriptor extends SetupDescriptor {
+ @Override
+ public void setup(final StendhalRPZone zone) {
+ final DynamicCreatureSpawner spawner = new DynamicCreatureSpawner(zone);
+ final Map params = getParameters();
+ for (final String name: params.keySet()) {
+ final int max = MathHelper.parseIntDefault(params.get(name), 0);
+ if (max < 0) {
+ logger.warn("Max must be a positive integer, not registering dynamic spawn for creature \"" + name + "\"");
+ continue;
+ }
+ spawner.register(name, max);
+ }
+ // NOTE: does zone need to know about dynamic spawner?
+ zone.setDynamicSpawner(spawner);
+ }
+ }
+}
diff --git a/src/games/stendhal/server/core/engine/StendhalRPZone.java b/src/games/stendhal/server/core/engine/StendhalRPZone.java
index 38af2975b1d..061c1faa96a 100644
--- a/src/games/stendhal/server/core/engine/StendhalRPZone.java
+++ b/src/games/stendhal/server/core/engine/StendhalRPZone.java
@@ -39,6 +39,7 @@
import games.stendhal.common.Direction;
import games.stendhal.common.Line;
import games.stendhal.common.MathHelper;
+import games.stendhal.common.Rand;
import games.stendhal.common.filter.FilterCriteria;
import games.stendhal.common.grammar.Grammar;
import games.stendhal.common.tiled.LayerDefinition;
@@ -63,6 +64,7 @@
import games.stendhal.server.entity.mapstuff.portal.OneWayPortalDestination;
import games.stendhal.server.entity.mapstuff.portal.Portal;
import games.stendhal.server.entity.mapstuff.spawner.CreatureRespawnPoint;
+import games.stendhal.server.entity.mapstuff.spawner.DynamicCreatureSpawner;
import games.stendhal.server.entity.mapstuff.spawner.PassiveEntityRespawnPoint;
import games.stendhal.server.entity.mapstuff.spawner.PassiveEntityRespawnPointFactory;
import games.stendhal.server.entity.mapstuff.spawner.SheepFood;
@@ -72,6 +74,7 @@
import games.stendhal.server.entity.npc.TrainingDummyFactory;
import games.stendhal.server.entity.player.Player;
import games.stendhal.server.util.StringUtils;
+import marauroa.common.Pair;
import marauroa.common.game.IRPZone;
import marauroa.common.game.RPObject;
import marauroa.common.game.RPSlot;
@@ -112,8 +115,12 @@ public class StendhalRPZone extends MarauroaRPZone {
*/
private final List sheepFoods;
+ /** Statically spawned creatures. */
private final List respawnPoints;
+ /** Dynamically spawned creatures. */
+ private DynamicCreatureSpawner dynamicSpawner;
+
private final List plantGrowers;
private final List playersAndFriends;
@@ -310,6 +317,30 @@ public void remove(final CreatureRespawnPoint point) {
respawnPoints.remove(point);
}
+ /**
+ * Retrieves the dynamic creature spawner associated with this zone.
+ *
+ * @return
+ * Creature spawner instance.
+ */
+ public DynamicCreatureSpawner getDynamicSpawner() {
+ return dynamicSpawner;
+ }
+
+ /**
+ * Sets the dynamic creature spawner associated with this zone.
+ *
+ * @param spawner
+ * Creature spawner instance.
+ */
+ public void setDynamicSpawner(final DynamicCreatureSpawner spawner) {
+ if (dynamicSpawner != null) {
+ dynamicSpawner.remove();
+ }
+ dynamicSpawner = spawner;
+ dynamicSpawner.init();
+ }
+
/**
* Retrieves growers in this zone.
*/
@@ -1514,6 +1545,7 @@ public void nextTurn() {
os.append("playersAndFriends: " + playersAndFriends.size() + "\n");
os.append("portals: " + portals.size() + "\n");
os.append("respawnPoints: " + respawnPoints.size() + "\n");
+ os.append("dynamicRespawns: " + dynamicSpawner.maxTotal() + "\n");
os.append("sheepFoods: " + sheepFoods.size() + "\n");
os.append("objects: " + objects.size() + "\n");
logger.info(os);
@@ -1945,4 +1977,24 @@ public List getAssociatedZonesList() {
return Arrays.asList(getAssociatedZones().split(","));
}
+
+ /**
+ * Attempts to find a position where an entity can be spawned.
+ *
+ * @param entity
+ * Entity attempting to spawn.
+ * @return
+ * Appropriate position or `null` if none found.
+ */
+ public Pair getRandomSpawnPosition(final Entity entity) {
+ final short retries = 50;
+ for (short t = 0; t < retries; t++) {
+ final int x = Rand.rand(getWidth());
+ final int y = Rand.rand(getHeight());
+ if (!collides(entity, x, y)) {
+ return new Pair<>(x, y);
+ }
+ }
+ return null;
+ }
}
diff --git a/src/games/stendhal/server/entity/mapstuff/spawner/DynamicCreatureSpawner.java b/src/games/stendhal/server/entity/mapstuff/spawner/DynamicCreatureSpawner.java
new file mode 100644
index 00000000000..564b1f7895e
--- /dev/null
+++ b/src/games/stendhal/server/entity/mapstuff/spawner/DynamicCreatureSpawner.java
@@ -0,0 +1,344 @@
+/***************************************************************************
+ * Copyright © 2024 - Faiumoni e. V. *
+ ***************************************************************************
+ ***************************************************************************
+ * *
+ * This program is free software; you can redistribute it and/or modify *
+ * it under the terms of the GNU General Public License as published by *
+ * the Free Software Foundation; either version 2 of the License, or *
+ * (at your option) any later version. *
+ * *
+ ***************************************************************************/
+package games.stendhal.server.entity.mapstuff.spawner;
+
+import java.util.LinkedHashMap;
+import java.util.LinkedList;
+import java.util.List;
+
+import org.apache.log4j.Logger;
+
+import games.stendhal.common.MathHelper;
+import games.stendhal.common.Rand;
+import games.stendhal.common.Triple;
+import games.stendhal.server.core.engine.SingletonRepository;
+import games.stendhal.server.core.engine.StendhalRPZone;
+import games.stendhal.server.core.rp.StendhalRPAction;
+import games.stendhal.server.core.rule.defaultruleset.DefaultCreature;
+import games.stendhal.server.entity.creature.Creature;
+import games.stendhal.server.util.Observer;
+import marauroa.common.Pair;
+
+
+/**
+ * Class that spawn multiple creatures at random points in a zone.
+ */
+public class DynamicCreatureSpawner extends LinkedHashMap> implements CreatureSpawner {
+
+ private static final Logger logger = Logger.getLogger(DynamicCreatureSpawner.class);
+
+ /** Zone associated with this spawner. */
+ private final StendhalRPZone zone;
+ /** Creatures to be spawned at turn reached. */
+ final List> queued;
+
+ final List observers;
+
+ /** Denotes whether turn notifier is active. */
+ private int spawnsAt = -1;
+
+
+ /**
+ * Creates a new spawner.
+ *
+ * @param zone
+ */
+ public DynamicCreatureSpawner(final StendhalRPZone zone) {
+ this.zone = zone;
+ this.observers = new LinkedList();
+ this.queued = new LinkedList>();
+ }
+
+ /**
+ * Starts the spawning cycle of all registered creatures.
+ */
+ public void init() {
+ for (final String name: keySet()) {
+ queue(name);
+ }
+ // NOTE: Do we want to queue all possible spawns of a creature simultaneously instead of
+ // sequentially? E.g. if max active = 10 queue up all 10 instances instead of only the
+ // first.
+ startNotifier();
+ }
+
+ /**
+ * Disables associated turn notifiers for clean removal.
+ */
+ public void remove() {
+ SingletonRepository.getTurnNotifier().dontNotify(this);
+ // NOTE: should active spawned creatures also be removed?
+ }
+
+ /**
+ * Adds a creature to spawn list.
+ *
+ * Registration is represented in a map indexed by creature name. The value is a triple of
+ * consisting of the following integer representations:
+ * - first: respawn time
+ * - second: max active
+ * - third: actual acitve
+ *
+ * @param name
+ * Creature name.
+ * @param max
+ * Maximum instances allowed at one time from this spawner.
+ */
+ public void register(final String name, final int max) {
+ if (max < 1) {
+ logger.warn("Max must be a positive integer");
+ return;
+ }
+ final DefaultCreature creature = SingletonRepository.getEntityManager().getDefaultCreature(name);
+ if (creature == null) {
+ logger.warn("Creature \"" + name + "\" not found");
+ return;
+ }
+ put(name, new Triple(creature.getRespawnTime(), max, 0));
+ }
+
+ /**
+ * Retrieves respawn time value of a creature.
+ *
+ * @param name
+ * Creature name.
+ * @return
+ * Rate at which creature respawns.
+ */
+ private int getRespawnTime(final String name) {
+ if (containsKey(name)) {
+ return get(name).getFirst();
+ }
+ return 0;
+ }
+
+ /**
+ * Retrieves maximum allowed active spawned instances.
+ *
+ * @param name
+ * Creature name.
+ * @return
+ * Maximum allowed creatures at one time.
+ */
+ private int getActiveMax(final String name) {
+ if (containsKey(name)) {
+ return get(name).getSecond();
+ }
+ return 0;
+ }
+
+ /**
+ * Retrieves the current number of active creatures from this spawner.
+ *
+ * @param name
+ * Creature name.
+ * @return
+ * Active creatures.
+ */
+ private int getActiveCount(final String name) {
+ if (containsKey(name)) {
+ return get(name).getThird();
+ }
+ return 0;
+ }
+
+ /**
+ * Retrieves the total number of active creatures that can be spawned at the same time.
+ *
+ * @return
+ * Max total active creatures.
+ */
+ public int maxTotal() {
+ int total = 0;
+ for (final Triple tr: values()) {
+ total += tr.getSecond();
+ }
+ return total;
+ }
+
+ /**
+ * Adds a new creature instance to zone at a random position.
+ *
+ * @param name
+ * Creature name.
+ */
+ private boolean spawn(final String name) {
+ final DefaultCreature defaultCreature = SingletonRepository.getEntityManager().getDefaultCreature(name);
+ if (defaultCreature == null) {
+ logger.error("Default creature \"" + name + "\" not found");
+ return false;
+ }
+ final Creature creature = defaultCreature.getCreatureRandomizeStats();
+ try {
+ // find a suitable location on map for spawning
+ final Pair pos = zone.getRandomSpawnPosition(creature);
+ if (pos == null) {
+ throw new Exception("No suitable position for dynamic spawning available in zone " + zone.getName());
+ }
+ final int x = pos.first();
+ final int y = pos.second();
+ // NOTE: behavior copied from `CreatureRespawnPoint`, not sure what it does
+ creature.registerObjectsForNotification(observers);
+ if (StendhalRPAction.placeat(zone, creature, x, y)) {
+ onSpawned(creature);
+ return true;
+ } else {
+ // could not place the creature anywhere so treat it like it died
+ onRemoved(creature);
+ logger.warn("Could not spawn " + creature.getName() + " near " + zone.getName() + " " + x + " " + y);
+ }
+ } catch (final Exception e) {
+ logger.error("Error spawning entity " + creature, e);
+ }
+ return false;
+ }
+
+ /**
+ * Adds creature to spawn queue.
+ *
+ * @param name
+ * Creature name.
+ */
+ private void queue(final String name) {
+ if (getActiveCount(name) >= getActiveMax(name)) {
+ // maximum number of creatures is active so don't spawn more
+ return;
+ }
+ final int nextTurn = MathHelper.clamp(Rand.randExponential(getRespawnTime(name)), MIN_RESPAWN_TIME, MAX_RESPAWN_TIME);
+ queued.add(new Pair<>(nextTurn, name));
+ }
+
+ /**
+ * Removes next entry from the queue.
+ */
+ private void unqueue() {
+ final int idx = queued.indexOf(getNextQueued(false));
+ if (idx > -1) {
+ queued.remove(idx);
+ }
+ }
+
+ /**
+ * Gets the next entry in the queue.
+ *
+ * @param copy
+ * If {@code true} make a copy before returning.
+ * @return
+ * Attributes of next entry.
+ */
+ private Pair getNextQueued(final boolean copy) {
+ Pair next = null;
+ // NOTE: might be more efficient to get first value of sorted list
+ for (final Pair p: queued) {
+ final int t = p.first();
+ if (next == null || t < next.first()) {
+ next = p;
+ }
+ }
+ if (copy && next != null) {
+ return new Pair<>(next.first(), next.second());
+ }
+ return next;
+ }
+
+ /**
+ * Increments the known number of active creature instances.
+ *
+ * @param name
+ * Creature name.
+ * @param amount
+ * Number to adjust by (usually 1 or -1).
+ */
+ private void incActive(final String name, final short amount) {
+ if (!containsKey(name)) {
+ logger.warn("Creature \"" + name + "\" not registered");
+ return;
+ }
+ final Triple oldValue = get(name);
+ put(name, new Triple(oldValue.getFirst(), oldValue.getSecond(), oldValue.getThird() + amount));
+ }
+
+ @Override
+ public void onSpawned(final Creature spawned) {
+ spawned.init();
+ spawned.setSpawner(this);
+ final String name = spawned.getName();
+ incActive(name, (short) 1);
+ // spawning succeeded so reset active spawning property
+ spawnsAt = -1;
+ // add another to queue
+ queue(name);
+ // don't restart notifier here as it will be done in {@code TurnListener.onTurnReached}
+ }
+
+ @Override
+ public void onRemoved(final Creature removed) {
+ final String name = removed.getName();
+ incActive(name, (short) -1);
+ // add another to queue
+ queue(name);
+ // restart notifier if needed
+ startNotifier();
+ }
+
+ /**
+ * Starts turn notifier for next queued entry.
+ */
+ private void startNotifier() {
+ final Pair next = getNextQueued(false);
+ if (next == null) {
+ return;
+ }
+ final int turns = next.first();
+ if (spawnsAt > -1 && turns >= spawnsAt) {
+ // don't override earlier spawning
+ return;
+ }
+ spawnsAt = turns;
+ SingletonRepository.getTurnNotifier().notifyInTurns(turns, this);
+ }
+
+ @Override
+ public void onTurnReached(final int currentTurn) {
+ final Pair next = getNextQueued(true);
+ if (next != null) {
+ final String nextName = next.second();
+ /* this check doesn't work because currentTurn is total turns since server started &
+ * next.first() is turns since time of queue
+ if (next.first() != currentTurn) {
+ logger.warn("Creature \"" + nextName + "\" scheduled to spawn at turn " + nextTurn + " but spawning at " + currentTurn);
+ }
+ */
+ // remove from queue
+ unqueue();
+ spawn(nextName);
+ } else {
+ logger.warn("Turn reached but nothing queued for spawn");
+ }
+
+ // restart notifier if another entry is in queue
+ startNotifier();
+ if (spawnsAt > -1 && queued.size() > 0) {
+ logger.warn("Queue not empty but not scheduled for spawn");
+ }
+ }
+
+ @Override
+ public void addObserver(final Observer observer) {
+ observers.add(observer);
+ }
+
+ @Override
+ public void removeObserver(final Observer observer) {
+ observers.remove(observer);
+ }
+}