ExternalCondition: remove unneeded source references
This commit is contained in:
@@ -32,18 +32,27 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
public object Create(ActorInitializer init) { return new ExternalCondition(init.Self, this); }
|
public object Create(ActorInitializer init) { return new ExternalCondition(init.Self, this); }
|
||||||
}
|
}
|
||||||
|
|
||||||
public class ExternalCondition
|
public class ExternalCondition : ITick
|
||||||
{
|
{
|
||||||
class TimedToken
|
class TimedToken
|
||||||
{
|
{
|
||||||
public int Token;
|
public readonly int Token;
|
||||||
public int Expires;
|
public readonly int Expires;
|
||||||
|
public readonly object Source;
|
||||||
|
|
||||||
|
public TimedToken(int token, Actor self, object source, int duration)
|
||||||
|
{
|
||||||
|
Token = token;
|
||||||
|
Expires = self.World.WorldTick + duration;
|
||||||
|
Source = source;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public readonly ExternalConditionInfo Info;
|
public readonly ExternalConditionInfo Info;
|
||||||
readonly ConditionManager conditionManager;
|
readonly ConditionManager conditionManager;
|
||||||
readonly Dictionary<object, HashSet<int>> permanentTokens = new Dictionary<object, HashSet<int>>();
|
readonly Dictionary<object, HashSet<int>> permanentTokens = new Dictionary<object, HashSet<int>>();
|
||||||
readonly Dictionary<object, HashSet<TimedToken>> timedTokens = new Dictionary<object, HashSet<TimedToken>>();
|
readonly Dictionary<object, Dictionary<int, TimedToken>> timedTokensBySource = new Dictionary<object, Dictionary<int, TimedToken>>();
|
||||||
|
readonly List<KeyValuePair<int, TimedToken>> timedTokensByExpiration = new List<KeyValuePair<int, TimedToken>>();
|
||||||
|
|
||||||
public ExternalCondition(Actor self, ExternalConditionInfo info)
|
public ExternalCondition(Actor self, ExternalConditionInfo info)
|
||||||
{
|
{
|
||||||
@@ -58,39 +67,57 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
|
|
||||||
// Timed tokens do not count towards the source cap: the condition with the shortest
|
// Timed tokens do not count towards the source cap: the condition with the shortest
|
||||||
// remaining duration can always be revoked to make room.
|
// remaining duration can always be revoked to make room.
|
||||||
if (Info.SourceCap > 0 && permanentTokens.GetOrAdd(source).Count >= Info.SourceCap)
|
if (Info.SourceCap > 0)
|
||||||
|
{
|
||||||
|
HashSet<int> permanentTokensForSource;
|
||||||
|
if (permanentTokens.TryGetValue(source, out permanentTokensForSource) && permanentTokensForSource.Count >= Info.SourceCap)
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if (Info.TotalCap > 0 && permanentTokens.Values.SelectMany(t => t).Count() >= Info.TotalCap)
|
if (Info.TotalCap > 0 && permanentTokens.Values.Sum(t => t.Count) >= Info.TotalCap)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void RemoveFromExpiration(TimedToken timedToken)
|
||||||
|
{
|
||||||
|
timedTokensByExpiration.RemoveAt(timedTokensByExpiration.FindIndex(p => p.Value == timedToken));
|
||||||
|
}
|
||||||
|
|
||||||
|
void AddForExpiration(TimedToken timedToken)
|
||||||
|
{
|
||||||
|
var index = timedTokensByExpiration.FindIndex(p => p.Key >= timedToken.Expires);
|
||||||
|
if (index >= 0)
|
||||||
|
timedTokensByExpiration.Insert(index, new KeyValuePair<int, TimedToken>(timedToken.Expires, timedToken));
|
||||||
|
else
|
||||||
|
timedTokensByExpiration.Add(new KeyValuePair<int, TimedToken>(timedToken.Expires, timedToken));
|
||||||
|
}
|
||||||
|
|
||||||
public int GrantCondition(Actor self, object source, int duration = 0)
|
public int GrantCondition(Actor self, object source, int duration = 0)
|
||||||
{
|
{
|
||||||
if (conditionManager == null || source == null || !CanGrantCondition(self, source))
|
if (!CanGrantCondition(self, source))
|
||||||
return ConditionManager.InvalidConditionToken;
|
return ConditionManager.InvalidConditionToken;
|
||||||
|
|
||||||
var token = conditionManager.GrantCondition(self, Info.Condition, duration);
|
var token = conditionManager.GrantCondition(self, Info.Condition, duration);
|
||||||
var permanent = permanentTokens.GetOrAdd(source);
|
HashSet<int> permanent;
|
||||||
|
permanentTokens.TryGetValue(source, out permanent);
|
||||||
|
|
||||||
if (duration > 0)
|
if (duration > 0)
|
||||||
{
|
{
|
||||||
var timed = timedTokens.GetOrAdd(source);
|
var timed = timedTokensBySource.GetOrAdd(source);
|
||||||
|
|
||||||
// Remove expired tokens
|
|
||||||
timed.RemoveWhere(t => t.Expires < self.World.WorldTick);
|
|
||||||
|
|
||||||
// Check level caps
|
// Check level caps
|
||||||
if (Info.SourceCap > 0)
|
if (Info.SourceCap > 0)
|
||||||
{
|
{
|
||||||
if (permanent.Count + timed.Count >= Info.SourceCap)
|
if ((permanent != null ? permanent.Count + timed.Count : timed.Count) >= Info.SourceCap)
|
||||||
{
|
{
|
||||||
var expire = timed.MinByOrDefault(t => t.Expires);
|
var expire = timed.MinByOrDefault(t => t.Value.Expires).Value;
|
||||||
if (expire != null)
|
if (expire != null)
|
||||||
{
|
{
|
||||||
timed.Remove(expire);
|
timed.Remove(expire.Token);
|
||||||
|
RemoveFromExpiration(expire);
|
||||||
|
|
||||||
if (conditionManager.TokenValid(self, expire.Token))
|
if (conditionManager.TokenValid(self, expire.Token))
|
||||||
conditionManager.RevokeCondition(self, expire.Token);
|
conditionManager.RevokeCondition(self, expire.Token);
|
||||||
}
|
}
|
||||||
@@ -99,24 +126,28 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
|
|
||||||
if (Info.TotalCap > 0)
|
if (Info.TotalCap > 0)
|
||||||
{
|
{
|
||||||
var totalCount = permanentTokens.Values.SelectMany(t => t).Count() + timedTokens.Values.SelectMany(t => t).Count();
|
var totalCount = permanentTokens.Values.Sum(t => t.Count) + timedTokensByExpiration.Count;
|
||||||
if (totalCount >= Info.TotalCap)
|
if (totalCount >= Info.TotalCap)
|
||||||
{
|
{
|
||||||
// Prefer tokens from the same source
|
// Prefer tokens from the same source
|
||||||
var expire = timedTokens.SelectMany(t => t.Value.Select(tt => new Tuple<object, TimedToken>(t.Key, tt)))
|
var expire = timedTokensByExpiration.FirstOrDefault().Value;
|
||||||
.MinByOrDefault(t => t.Item2.Expires);
|
|
||||||
if (expire != null)
|
if (expire != null)
|
||||||
{
|
{
|
||||||
if (conditionManager.TokenValid(self, expire.Item2.Token))
|
if (conditionManager.TokenValid(self, expire.Token))
|
||||||
conditionManager.RevokeCondition(self, expire.Item2.Token);
|
conditionManager.RevokeCondition(self, expire.Token);
|
||||||
|
|
||||||
timedTokens[expire.Item1].Remove(expire.Item2);
|
timedTokensBySource[expire.Source].Remove(expire.Token);
|
||||||
|
RemoveFromExpiration(expire);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
timed.Add(new TimedToken { Expires = self.World.WorldTick + duration, Token = token });
|
var timedToken = new TimedToken(token, self, source, duration);
|
||||||
|
timed.Add(token, timedToken);
|
||||||
|
AddForExpiration(timedToken);
|
||||||
}
|
}
|
||||||
|
else if (permanent == null)
|
||||||
|
permanentTokens.Add(source, new HashSet<int> { token });
|
||||||
else
|
else
|
||||||
permanent.Add(token);
|
permanent.Add(token);
|
||||||
|
|
||||||
@@ -138,9 +169,16 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
HashSet<TimedToken> timedTokensForSource = null;
|
Dictionary<int, TimedToken> timedTokensForSource;
|
||||||
if (timedTokens.TryGetValue(source, out timedTokensForSource) && timedTokensForSource.RemoveWhere(t => t.Token == token) == 0)
|
if (timedTokensBySource.TryGetValue(source, out timedTokensForSource))
|
||||||
|
{
|
||||||
|
TimedToken timedToken;
|
||||||
|
if (!timedTokensForSource.TryGetValue(token, out timedToken))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
|
timedTokensForSource.Remove(token);
|
||||||
|
RemoveFromExpiration(timedToken);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (conditionManager.TokenValid(self, token))
|
if (conditionManager.TokenValid(self, token))
|
||||||
@@ -148,5 +186,29 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ITick.Tick(Actor self)
|
||||||
|
{
|
||||||
|
if (timedTokensByExpiration.Count == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// Remove expired tokens
|
||||||
|
var worldTick = self.World.WorldTick;
|
||||||
|
var pair = timedTokensByExpiration[0];
|
||||||
|
while (pair.Key < worldTick)
|
||||||
|
{
|
||||||
|
var timed = pair.Value;
|
||||||
|
var timedTokensForSource = timedTokensBySource[timed.Source];
|
||||||
|
timedTokensForSource.Remove(timed.Token);
|
||||||
|
if (timedTokensForSource.Count == 0)
|
||||||
|
timedTokensBySource.Remove(timed.Source);
|
||||||
|
|
||||||
|
timedTokensByExpiration.RemoveAt(0);
|
||||||
|
if (timedTokensByExpiration.Count == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
pair = timedTokensByExpiration[0];
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user