mirror of
https://gitea.wildfiregames.com/0ad/0ad
synced 2026-06-17 22:03:56 -07:00
The ModifiersManager system component provides an interface to add and remove modifiers, and get modified stats. The goal is to merge all the different stat-modifying systems 0 A.D. has implemented over the years. This commit makes technologies and auras use ModifiersManager. Some cheats and AI bonuses also have a similar stat-modifying effect that have not yet been updated. Further, this system component makes it possible for e.g. triggers to easily add modifiers, enabling the writing of Castle Blood Automatic, RPG or Tower Defense maps without the need for mods or hacks. The 'Modifier' name was preferred over 'Modification' as it is shorter and more readable, along with the logic that 'modifiers' store 'modifications' and this stores modifiers. Renaming of other functions and classes has been left for future work for now. Internally, this uses a JS data structure. If performance issues arise with it in the future, this data structure or the whole component could be moved to C++. The performance has been tested to be about as fast as the current implementations (and specifically much faster for global auras with no icons). Testing showed that sending value modification messages was by far the slowest part. Comments by: leper, Stan, elexis Differential Revision: https://code.wildfiregames.com/D274 This was SVN commit r22767.
168 lines
4.2 KiB
JavaScript
168 lines
4.2 KiB
JavaScript
function PlayerManager() {}
|
|
|
|
PlayerManager.prototype.Schema =
|
|
"<a:component type='system'/><empty/>";
|
|
|
|
PlayerManager.prototype.Init = function()
|
|
{
|
|
this.playerEntities = []; // list of player entity IDs
|
|
};
|
|
|
|
PlayerManager.prototype.AddPlayer = function(ent)
|
|
{
|
|
var id = this.playerEntities.length;
|
|
var cmpPlayer = Engine.QueryInterface(ent, IID_Player);
|
|
cmpPlayer.SetPlayerID(id);
|
|
this.playerEntities.push(ent);
|
|
// initialize / update the diplomacy arrays
|
|
var newDiplo = [];
|
|
for (var i = 0; i < id; i++)
|
|
{
|
|
var cmpOtherPlayer = Engine.QueryInterface(this.GetPlayerByID(i), IID_Player);
|
|
cmpOtherPlayer.diplomacy[id] = -1;
|
|
newDiplo[i] = -1;
|
|
}
|
|
newDiplo[id] = 1;
|
|
cmpPlayer.SetDiplomacy(newDiplo);
|
|
|
|
Engine.BroadcastMessage(MT_PlayerEntityChanged, {
|
|
"player": id,
|
|
"from": INVALID_ENTITY,
|
|
"to": ent
|
|
});
|
|
|
|
return id;
|
|
};
|
|
|
|
/**
|
|
* To avoid possible problems,
|
|
* we first remove all entities from this player, and add them back after the replacement.
|
|
* Note: This should only be called during setup/init and not during the game
|
|
*/
|
|
PlayerManager.prototype.ReplacePlayer = function(id, ent)
|
|
{
|
|
let cmpRangeManager = Engine.QueryInterface(SYSTEM_ENTITY, IID_RangeManager);
|
|
let entities = cmpRangeManager.GetEntitiesByPlayer(id);
|
|
for (let e of entities)
|
|
{
|
|
let cmpOwnership = Engine.QueryInterface(e, IID_Ownership);
|
|
if (cmpOwnership)
|
|
cmpOwnership.SetOwner(INVALID_PLAYER);
|
|
}
|
|
|
|
let oldent = this.playerEntities[id];
|
|
let oldCmpPlayer = Engine.QueryInterface(oldent, IID_Player);
|
|
let diplo = oldCmpPlayer.GetDiplomacy();
|
|
let color = oldCmpPlayer.GetColor();
|
|
|
|
let newCmpPlayer = Engine.QueryInterface(ent, IID_Player);
|
|
newCmpPlayer.SetPlayerID(id);
|
|
this.playerEntities[id] = ent;
|
|
newCmpPlayer.SetColor(color);
|
|
newCmpPlayer.SetDiplomacy(diplo);
|
|
|
|
Engine.BroadcastMessage(MT_PlayerEntityChanged, {
|
|
"player": id,
|
|
"from": oldent,
|
|
"to": ent
|
|
});
|
|
|
|
for (let e of entities)
|
|
{
|
|
let cmpOwnership = Engine.QueryInterface(e, IID_Ownership);
|
|
if (cmpOwnership)
|
|
cmpOwnership.SetOwner(id);
|
|
}
|
|
|
|
Engine.DestroyEntity(oldent);
|
|
Engine.FlushDestroyedEntities();
|
|
};
|
|
|
|
/**
|
|
* Returns the player entity ID for the given player ID.
|
|
* The player ID must be valid (else there will be an error message).
|
|
*/
|
|
PlayerManager.prototype.GetPlayerByID = function(id)
|
|
{
|
|
if (id in this.playerEntities)
|
|
return this.playerEntities[id];
|
|
|
|
// All players at or below ID 0 get gaia-level data. (Observers for example)
|
|
if (id <= 0)
|
|
return this.playerEntities[0];
|
|
|
|
var stack = new Error().stack.trimRight().replace(/^/mg, ' '); // indent each line
|
|
warn("GetPlayerByID: no player defined for id '"+id+"'\n"+stack);
|
|
|
|
return INVALID_ENTITY;
|
|
};
|
|
|
|
/**
|
|
* Returns the number of players including gaia.
|
|
*/
|
|
PlayerManager.prototype.GetNumPlayers = function()
|
|
{
|
|
return this.playerEntities.length;
|
|
};
|
|
|
|
/**
|
|
* Returns IDs of all players including gaia.
|
|
*/
|
|
PlayerManager.prototype.GetAllPlayers = function()
|
|
{
|
|
let players = [];
|
|
for (let i = 0; i < this.playerEntities.length; ++i)
|
|
players.push(i);
|
|
return players;
|
|
};
|
|
|
|
/**
|
|
* Returns IDs of all players excluding gaia.
|
|
*/
|
|
PlayerManager.prototype.GetNonGaiaPlayers = function()
|
|
{
|
|
let players = [];
|
|
for (let i = 1; i < this.playerEntities.length; ++i)
|
|
players.push(i);
|
|
return players;
|
|
};
|
|
|
|
/**
|
|
* Returns IDs of all players excluding gaia that are not defeated nor have won.
|
|
*/
|
|
PlayerManager.prototype.GetActivePlayers = function()
|
|
{
|
|
return this.GetNonGaiaPlayers().filter(playerID => QueryPlayerIDInterface(playerID).GetState() == "active");
|
|
};
|
|
|
|
PlayerManager.prototype.RemoveAllPlayers = function()
|
|
{
|
|
// Destroy existing player entities
|
|
for (let player in this.playerEntities)
|
|
{
|
|
Engine.BroadcastMessage(MT_PlayerEntityChanged, {
|
|
"player": player,
|
|
"from": this.playerEntities[player],
|
|
"to": INVALID_ENTITY
|
|
});
|
|
Engine.DestroyEntity(this.playerEntities[player]);
|
|
}
|
|
|
|
this.playerEntities = [];
|
|
};
|
|
|
|
PlayerManager.prototype.RemoveLastPlayer = function()
|
|
{
|
|
if (this.playerEntities.length == 0)
|
|
return;
|
|
|
|
var lastId = this.playerEntities.pop();
|
|
Engine.BroadcastMessage(MT_PlayerEntityChanged, {
|
|
"player": this.playerEntities.length + 1,
|
|
"from": lastId,
|
|
"to": INVALID_ENTITY
|
|
});
|
|
Engine.DestroyEntity(lastId);
|
|
};
|
|
|
|
Engine.RegisterSystemComponentType(IID_PlayerManager, "PlayerManager", PlayerManager);
|