blob: a096ed44e822229b201a37a1790ffbf15fe0e41c [file] [log] [blame] [raw]
/*
* Copyright (c) 2010-2011 Graham Edgecombe.
*
* This file is part of Lightstone.
*
* Lightstone 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 3 of the License, or
* (at your option) any later version.
*
* Lightstone is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Lightstone. If not, see <http://www.gnu.org/licenses/>.
*/
package net.lightstone.model;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
/**
* A class which manages all of the entities within a world.
* @author Graham Edgecombe
*/
public final class EntityManager implements Iterable<Entity> {
/**
* A map of all the entity ids to the corresponding entities.
*/
private final Map<Integer, Entity> entities = new HashMap<Integer, Entity>();
/**
* A map of entity types to a set containing all entities of that type.
*/
private final Map<Class<? extends Entity>, Set<? extends Entity>> groupedEntities = new HashMap<Class<? extends Entity>, Set<? extends Entity>>();
/**
* The next id to check.
*/
private int nextId = 1;
/**
* Gets all entities with the specified type.
* @param type The {@link Class} for the type.
* @param <T> The type of entity.
* @return A collection of entities with the specified type.
*/
@SuppressWarnings("unchecked")
public <T extends Entity> Collection<T> getAll(Class<T> type) {
Set<T> set = (Set<T>) groupedEntities.get(type);
if (set == null) {
set = new HashSet<T>();
groupedEntities.put(type, set);
}
return set;
}
/**
* Gets an entity by its id.
* @param id The id.
* @return The entity, or {@code null} if it could not be found.
*/
public Entity getEntity(int id) {
return entities.get(id);
}
/**
* Allocates the id for an entity.
* @param entity The entity.
* @return The id.
*/
@SuppressWarnings("unchecked")
int allocate(Entity entity) {
for (int id = nextId; id < Integer.MAX_VALUE; id++) {
if (!entities.containsKey(id)) {
entities.put(id, entity);
entity.id = id;
((Collection<Entity>) getAll(entity.getClass())).add(entity);
nextId = id + 1;
return id;
}
}
for (int id = Integer.MIN_VALUE; id < -1; id++) { // as -1 is used as a special value
if (!entities.containsKey(id)) {
entities.put(id, entity);
((Collection<Entity>) getAll(entity.getClass())).add(entity);
nextId = id + 1;
return id;
}
}
throw new IllegalStateException("No free entity ids");
}
/**
* Deallocates the id for an entity.
* @param entity The entity.
*/
void deallocate(Entity entity) {
entities.remove(entity.getId());
getAll(entity.getClass()).remove(entity);
}
@Override
public Iterator<Entity> iterator() {
return entities.values().iterator();
}
}