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

com.badlogic.gdx.ai.steer.Proximity 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.steer;

import com.badlogic.gdx.ai.Timepiece;
import com.badlogic.gdx.ai.steer.behaviors.Alignment;
import com.badlogic.gdx.ai.steer.behaviors.BlendedSteering;
import com.badlogic.gdx.ai.steer.behaviors.Cohesion;
import com.badlogic.gdx.ai.steer.behaviors.Separation;
import com.badlogic.gdx.ai.steer.behaviors.Wander;
import com.badlogic.gdx.math.Vector;

/** A {@code Proximity} defines an area that is used by group behaviors to find and process the owner's neighbors.
 * 

* Typically (but not necessarily) different group behaviors share the same {@code Proximity} for a given owner. This allows you to * combine group behaviors so as to get a more complex behavior also known as emergent behavior. Emergent behavior is behavior * that looks complex and/or purposeful to the observer but is actually derived spontaneously from fairly simple rules. The * lower-level agents following the rules have no idea of the bigger picture; they are only aware of themselves and maybe a few of * their neighbors. A typical example of emergence is flocking behavior which is a combination of three group behaviors: * {@link Separation separation}, {@link Alignment alignment}, and {@link Cohesion cohesion}. The three behaviors are typically * combined through a {@link BlendedSteering blended steering}. This works okay but, because of the limited view distance of a * character, it's possible for an agent to become isolated from its flock. If this happens, it will just sit still and do * nothing. To prevent this from happening, you usually add in the {@link Wander wander} behavior too. This way, all the agents * keep moving all the time. Tweaking the magnitudes of each of the contributing behaviors will give you different effects such as * shoals of fish, loose swirling flocks of birds, or bustling close-knit herds of sheep. *

* Before a steering acceleration can be calculated for a combination of group behaviors, the neighbors must be determined and * processed. This is done by the {@link #findNeighbors} method and its callback argument. *

* Notes: *

    *
  • Sharing a {@code Proximity} instance among group behaviors having the same owner can save a little time determining the * neighbors only once from inside the {@code findNeighbors} method. Especially, {@code Proximity} implementation classes can use * {@link Timepiece#getTime() GdxAI.getTimepiece().getTime()} to calculate neighbors only once per frame (assuming delta time is * always greater than 0, if time has changed the frame has changed too). This means that *
      *
    • if you forget to {@link Timepiece#update(float) update the timepiece} on each frame the proximity instance will be * calculated only the very first time, which is not what you want of course.
    • *
    • ideally the timepiece should be updated before the proximity is updated by the {@link #findNeighbors(ProximityCallback)} * method.
    • *
    *
  • *
  • If you want to make sure a Proximity doesn't use as a neighbor a given agent from the list, for example the evader or the * owner itself, you have to implement a callback that prevents it from being considered by returning {@code false} from the method * {@link ProximityCallback#reportNeighbor(Steerable) reportNeighbor}.
  • *
  • If there is some efficient way of pruning potential neighbors before they are processed, the overall performance in time * will improve. Spatial data structures such as multi-resolution maps, quad-trees, oct-trees, and binary space partition (BSP) * trees can be used to get potential neighbors more efficiently. Spatial partitioning techniques are crucial when you have to * deal with lots of agents. Especially, if you're using Bullet or Box2d in your game, it's recommended to implement proximities * that exploit their methods to query the world. Both Bullet and Box2d internally use some kind of spatial partitioning.
  • *
* * @param Type of vector, either 2D or 3D, implementing the {@link Vector} interface * * @author davebaol */ public interface Proximity> { /** Returns the owner of this proximity. */ public Steerable getOwner (); /** Sets the owner of this proximity. */ public void setOwner (Steerable owner); /** Finds the agents that are within the immediate area of the owner. Each of those agents is passed to the * {@link ProximityCallback#reportNeighbor(Steerable) reportNeighbor} method of the specified callback. * @return the number of neighbors found. */ public int findNeighbors (ProximityCallback callback); /** The callback object used by a proximity to report the owner's neighbor. * * @param Type of vector, either 2D or 3D, implementing the {@link Vector} interface * * @author davebaol */ public interface ProximityCallback> { /** The callback method used to report a neighbor. * @param neighbor the reported neighbor. * @return {@code true} if the given neighbor is valid; {@code false} otherwise. */ public boolean reportNeighbor (Steerable neighbor); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy