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

com.badlogic.gdx.ai.btree.SingleRunningChildBranch Maven / Gradle / Ivy

/*******************************************************************************
 * Copyright 2014 See AUTHORS file.
 * 
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 *   http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 ******************************************************************************/

package com.badlogic.gdx.ai.btree;

import com.badlogic.gdx.math.MathUtils;
import com.badlogic.gdx.utils.Array;

/** A {@code SingleRunningChildBranch} task is a branch task that supports only one running child at a time.
 * 
 * @param  type of the blackboard object that tasks use to read or modify game state
 * 
 * @author implicit-invocation
 * @author davebaol */
public abstract class SingleRunningChildBranch extends BranchTask {

	/** The child in the running status or {@code null} if no child is running. */
	protected Task runningChild;

	/** The index of the child currently processed. */
	protected int currentChildIndex;

	/** Array of random children. If it's {@code null} this task is deterministic. */
	protected Task[] randomChildren;

	/** Creates a {@code SingleRunningChildBranch} task with no children */
	public SingleRunningChildBranch () {
		super();
	}

	/** Creates a {@code SingleRunningChildBranch} task with a list of children
	 * 
	 * @param tasks list of this task's children, can be empty */
	public SingleRunningChildBranch (Array> tasks) {
		super(tasks);
	}

	@Override
	public void childRunning (Task task, Task reporter) {
		runningChild = task;
		running(); // Return a running status when a child says it's running
	}

	@Override
	public void childSuccess (Task task) {
		this.runningChild = null;
	}

	@Override
	public void childFail (Task task) {
		this.runningChild = null;
	}

	@Override
	public void run () {
		if (runningChild != null) {
			runningChild.run();
		} else {
			if (currentChildIndex < children.size) {
				if (randomChildren != null) {
					int last = children.size - 1;
					if (currentChildIndex < last) {
						// Random swap
						int otherChildIndex = MathUtils.random(currentChildIndex, last);
						Task tmp = randomChildren[currentChildIndex];
						randomChildren[currentChildIndex] = randomChildren[otherChildIndex];
						randomChildren[otherChildIndex] = tmp;
					}
					runningChild = randomChildren[currentChildIndex];
				} else {
					runningChild = children.get(currentChildIndex);
				}
				runningChild.setControl(this);
				runningChild.start();
				if (!runningChild.checkGuard(this))
					runningChild.fail();
				run();
			} else {
				// Should never happen; this case must be handled by subclasses in childXXX methods
			}
		}
	}

	@Override
	public void start () {
		this.currentChildIndex = 0;
		runningChild = null;
	}

	@Override
	protected void cancelRunningChildren (int startIndex) {
		super.cancelRunningChildren(startIndex);
		runningChild = null;
	}

	@Override
	public void reset () {
		super.reset();
		this.currentChildIndex = 0;
		this.runningChild = null;
		this.randomChildren = null;
	}

	@Override
	protected Task copyTo (Task task) {
		SingleRunningChildBranch branch = (SingleRunningChildBranch)task;
		branch.randomChildren = null;

		return super.copyTo(task);
	}

	@SuppressWarnings("unchecked")
	protected Task[] createRandomChildren () {
		Task[] rndChildren = new Task[children.size];
		System.arraycopy(children.items, 0, rndChildren, 0, children.size);
		return rndChildren;
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy