All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.artemis.systems.DelayedEntityProcessingSystem Maven / Gradle / Ivy

package com.artemis.systems;

import com.artemis.Aspect;
import com.artemis.Entity;
import com.artemis.EntitySystem;
import com.artemis.World;
import com.artemis.utils.IntBag;


/**
 * The purpose of this class is to allow systems to execute at varying
 * intervals.
 * 

* An example system would be an ExpirationSystem, that deletes entities after * a certain lifetime. Instead of running a system that decrements a timeLeft * value for each entity, you can simply use this system to execute in a future * at a time of the shortest lived entity, and then reset the system to run at * a time in a future at a time of the shortest lived entity, etc. *

* Another example system would be an AnimationSystem. You know when you have * to animate a certain entity, e.g. in 300 milliseconds. So you can set the * system to run in 300 ms to perform the animation. *

* This will save CPU cycles in some scenarios. *

* Implementation notes:
* Within {@link #processExpired(Entity) processExpired(Entity e)} * you must call {@link #offerDelay(float) offerDelay(float delay)} if the * entity's delay time is renewed. That method is also called by {@link #inserted(int) inserted(int entityId)} * for each newly matched entity. *

* * @author Arni Arent */ public abstract class DelayedEntityProcessingSystem extends EntitySystem { /** The time until an entity should be processed. */ private float delay; /** If the system is running and counting down delays. */ private boolean running; /** The countdown, accumulates world deltas. */ private float acc; private Entity flyweight; /** * Creates a new DelayedEntityProcessingSystem. * * @param aspect * the aspect to match against entities */ public DelayedEntityProcessingSystem(Aspect.Builder aspect) { super(aspect); } @Override protected void setWorld(World world) { super.setWorld(world); flyweight = createFlyweightEntity(); } @Override protected final void processSystem() { IntBag actives = subscription.getEntities(); int processed = actives.size(); if (processed == 0) { stop(); return; } delay = Float.MAX_VALUE; int[] array = actives.getData(); Entity e = flyweight; for (int i = 0; processed > i; i++) { e.id = array[i]; processDelta(e, acc); float remaining = getRemainingDelay(e); if(remaining <= 0) { processExpired(e); } else { offerDelay(remaining); } } acc = 0; } @Override protected void inserted(int entityId) { Entity entity = world.getEntity(entityId); float remainingDelay = getRemainingDelay(entity); processDelta(entity, -acc); if(remainingDelay > 0) { offerDelay(remainingDelay); } } /** * Return the delay until this entity should be processed. * * @param e * entity * * @return delay */ protected abstract float getRemainingDelay(Entity e); @Override protected final boolean checkProcessing() { if (running) { acc += getTimeDelta(); return acc >= delay; } return false; } /** * Overridable method to provide custom time delta. */ protected float getTimeDelta() { return world.getDelta(); } /** * Process an entity this system is interested in. *

* Substract the accumulatedDelta from the entities defined delay. *

* * @param e * the entity to process * @param accumulatedDelta * the delta time since this system was last executed */ protected abstract void processDelta(Entity e, float accumulatedDelta); protected abstract void processExpired(Entity e); /** * Start processing of entities after a certain amount of delta time. *

* Cancels current delayed run and starts a new one. *

* * @param delay * time delay until processing starts * @deprecated bugged and unnecessary. don't use. */ @Deprecated public void restart(float delay) { this.delay = delay; this.acc = 0; running = true; } /** * Restarts the system only if the delay offered is shorter than the time * that the system is currently scheduled to execute at. *

* If the system is already stopped (not running) then the offered delay * will be used to restart the system with no matter its value. *

* If the system is already counting down, and the offered delay is larger * than the time remaining, the system will ignore it. If the offered delay * is shorter than the time remaining, the system will restart itself to * run at the offered delay. *

* * @param offeredDelay * delay to offer */ public void offerDelay(float offeredDelay) { if (!running) { running = true; delay = offeredDelay; } else { delay = Math.min(delay, offeredDelay); } } /** * Get the initial delay that the system was ordered to process entities * after. * * @return the originally set delay */ public float getInitialTimeDelay() { return delay; } /** * Get the time until the system is scheduled to run at. *

* Returns zero (0) if the system is not running. * Use {@link #isRunning() isRunning()} before checking this value. *

* * @return time when system will run at */ public float getRemainingTimeUntilProcessing() { if(running) { return delay-acc; } return 0; } /** * Check if the system is counting down towards processing. * * @return {@code true} if it's counting down, false if it's not running */ public boolean isRunning() { return running; } /** * Stops the system from running, aborts current countdown. *

* Call offerDelay or restart to run it again. *

*/ public void stop() { this.running = false; this.acc = 0; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy