Moves traits from Common and Power namespaces to Common.Traits namespace
This commit is contained in:
86
OpenRA.Mods.Common/Traits/Upgrades/UpgradableTrait.cs
Normal file
86
OpenRA.Mods.Common/Traits/Upgrades/UpgradableTrait.cs
Normal file
@@ -0,0 +1,86 @@
|
||||
#region Copyright & License Information
|
||||
/*
|
||||
* Copyright 2007-2014 The OpenRA Developers (see AUTHORS)
|
||||
* This file is part of OpenRA, which is free software. It is made
|
||||
* available to you under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation. For more information,
|
||||
* see COPYING.
|
||||
*/
|
||||
#endregion
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using OpenRA.Traits;
|
||||
|
||||
namespace OpenRA.Mods.Common.Traits
|
||||
{
|
||||
/// <summary>Use as base class for *Info to subclass of UpgradableTrait. (See UpgradableTrait.)</summary>
|
||||
public abstract class UpgradableTraitInfo
|
||||
{
|
||||
[Desc("The upgrade types which can enable or disable this trait.")]
|
||||
public readonly string[] UpgradeTypes = { };
|
||||
|
||||
[Desc("The minimum upgrade level at which this trait is enabled.", "Defaults to 0 (enabled by default).")]
|
||||
public readonly int UpgradeMinEnabledLevel = 0;
|
||||
|
||||
[Desc("The maximum upgrade level at which the trait is enabled.",
|
||||
"Defaults to UpgradeMaxAcceptedLevel (enabled for all levels greater than UpgradeMinEnabledLevel).",
|
||||
"Set this to a value smaller than UpgradeMaxAcceptedLevel to disable the trait at higher levels.",
|
||||
"Use UpgradeMaxAcceptedLevel: 2 (1 more) to be able to extend upgrade time.")]
|
||||
public readonly int UpgradeMaxEnabledLevel = int.MaxValue;
|
||||
|
||||
[Desc("The maximum upgrade level that this trait will accept.")]
|
||||
public readonly int UpgradeMaxAcceptedLevel = 1;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Abstract base for enabling and disabling trait using upgrades.
|
||||
/// Requires basing *Info on UpgradableTraitInfo and using base(info) constructor.
|
||||
/// Note that EnabledByUpgrade is not called at creation even if this starts as enabled.
|
||||
/// </summary>,
|
||||
public abstract class UpgradableTrait<InfoType> : IUpgradable, IDisabledTrait, ISync where InfoType : UpgradableTraitInfo
|
||||
{
|
||||
public readonly InfoType Info;
|
||||
public IEnumerable<string> UpgradeTypes { get { return Info.UpgradeTypes; } }
|
||||
[Sync] public bool IsTraitDisabled { get; private set; }
|
||||
|
||||
public UpgradableTrait(InfoType info)
|
||||
{
|
||||
Info = info;
|
||||
IsTraitDisabled = info.UpgradeTypes != null && info.UpgradeTypes.Length > 0 && info.UpgradeMinEnabledLevel > 0;
|
||||
}
|
||||
|
||||
public bool AcceptsUpgradeLevel(Actor self, string type, int level)
|
||||
{
|
||||
return level > 0 && level <= Info.UpgradeMaxAcceptedLevel;
|
||||
}
|
||||
|
||||
public void UpgradeLevelChanged(Actor self, string type, int oldLevel, int newLevel)
|
||||
{
|
||||
if (!Info.UpgradeTypes.Contains(type))
|
||||
return;
|
||||
|
||||
// Restrict the levels to the allowed range
|
||||
oldLevel = oldLevel.Clamp(0, Info.UpgradeMaxAcceptedLevel);
|
||||
newLevel = newLevel.Clamp(0, Info.UpgradeMaxAcceptedLevel);
|
||||
if (oldLevel == newLevel)
|
||||
return;
|
||||
|
||||
var wasDisabled = IsTraitDisabled;
|
||||
IsTraitDisabled = newLevel < Info.UpgradeMinEnabledLevel || newLevel > Info.UpgradeMaxEnabledLevel;
|
||||
UpgradeLevelChanged(self, oldLevel, newLevel);
|
||||
|
||||
if (IsTraitDisabled != wasDisabled)
|
||||
{
|
||||
if (wasDisabled)
|
||||
UpgradeEnabled(self);
|
||||
else
|
||||
UpgradeDisabled(self);
|
||||
}
|
||||
}
|
||||
|
||||
// Subclasses can add upgrade support by querying IsTraitDisabled and/or overriding these methods.
|
||||
protected virtual void UpgradeLevelChanged(Actor self, int oldLevel, int newLevel) { }
|
||||
protected virtual void UpgradeEnabled(Actor self) { }
|
||||
protected virtual void UpgradeDisabled(Actor self) { }
|
||||
}
|
||||
}
|
||||
161
OpenRA.Mods.Common/Traits/Upgrades/UpgradeManager.cs
Normal file
161
OpenRA.Mods.Common/Traits/Upgrades/UpgradeManager.cs
Normal file
@@ -0,0 +1,161 @@
|
||||
#region Copyright & License Information
|
||||
/*
|
||||
* Copyright 2007-2014 The OpenRA Developers (see AUTHORS)
|
||||
* This file is part of OpenRA, which is free software. It is made
|
||||
* available to you under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation. For more information,
|
||||
* see COPYING.
|
||||
*/
|
||||
#endregion
|
||||
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Drawing;
|
||||
using System.Linq;
|
||||
using OpenRA.Graphics;
|
||||
using OpenRA.Primitives;
|
||||
using OpenRA.Traits;
|
||||
|
||||
namespace OpenRA.Mods.Common.Traits
|
||||
{
|
||||
[Desc("Attach this to a unit to enable dynamic upgrades by warheads, experience, crates, support powers, etc.")]
|
||||
public class UpgradeManagerInfo : ITraitInfo
|
||||
{
|
||||
public object Create(ActorInitializer init) { return new UpgradeManager(init); }
|
||||
}
|
||||
|
||||
public class UpgradeManager : ITick
|
||||
{
|
||||
class TimedUpgrade
|
||||
{
|
||||
public readonly string Upgrade;
|
||||
public readonly int Duration;
|
||||
public int Remaining;
|
||||
|
||||
public TimedUpgrade(string upgrade, int duration)
|
||||
{
|
||||
Upgrade = upgrade;
|
||||
Duration = duration;
|
||||
Remaining = duration;
|
||||
}
|
||||
|
||||
public void Tick() { Remaining--; }
|
||||
}
|
||||
|
||||
class UpgradeState
|
||||
{
|
||||
public readonly List<IUpgradable> Traits = new List<IUpgradable>();
|
||||
public readonly List<object> Sources = new List<object>();
|
||||
public readonly List<Action<int, int>> Watchers = new List<Action<int, int>>();
|
||||
}
|
||||
|
||||
readonly List<TimedUpgrade> timedUpgrades = new List<TimedUpgrade>();
|
||||
readonly Lazy<Dictionary<string, UpgradeState>> upgrades;
|
||||
readonly Dictionary<IUpgradable, int> levels = new Dictionary<IUpgradable, int>();
|
||||
|
||||
public UpgradeManager(ActorInitializer init)
|
||||
{
|
||||
upgrades = Exts.Lazy(() =>
|
||||
{
|
||||
var ret = new Dictionary<string, UpgradeState>();
|
||||
foreach (var up in init.self.TraitsImplementing<IUpgradable>())
|
||||
foreach (var t in up.UpgradeTypes)
|
||||
ret.GetOrAdd(t).Traits.Add(up);
|
||||
|
||||
return ret;
|
||||
});
|
||||
}
|
||||
|
||||
public void GrantTimedUpgrade(Actor self, string upgrade, int duration)
|
||||
{
|
||||
var timed = timedUpgrades.FirstOrDefault(u => u.Upgrade == upgrade);
|
||||
if (timed == null)
|
||||
{
|
||||
timed = new TimedUpgrade(upgrade, duration);
|
||||
timedUpgrades.Add(timed);
|
||||
GrantUpgrade(self, upgrade, timed);
|
||||
}
|
||||
else
|
||||
timed.Remaining = Math.Max(duration, timed.Remaining);
|
||||
}
|
||||
|
||||
// Different upgradeable traits may define (a) different level ranges for the same upgrade type,
|
||||
// and (b) multiple upgrade types for the same trait. The unrestricted level for each trait is
|
||||
// tracked independently so that we can can correctly revoke levels without adding the burden of
|
||||
// tracking both the overall (unclamped) and effective (clamped) levels on each individual trait.
|
||||
void NotifyUpgradeLevelChanged(IEnumerable<IUpgradable> traits, Actor self, string upgrade, int levelAdjust)
|
||||
{
|
||||
foreach (var up in traits)
|
||||
{
|
||||
var oldLevel = levels.GetOrAdd(up);
|
||||
var newLevel = levels[up] = oldLevel + levelAdjust;
|
||||
|
||||
// This will internally clamp the levels to its own restricted range
|
||||
up.UpgradeLevelChanged(self, upgrade, oldLevel, newLevel);
|
||||
}
|
||||
}
|
||||
|
||||
int GetOverallLevel(IUpgradable upgradable)
|
||||
{
|
||||
int level;
|
||||
return levels.TryGetValue(upgradable, out level) ? level : 0;
|
||||
}
|
||||
|
||||
public void GrantUpgrade(Actor self, string upgrade, object source)
|
||||
{
|
||||
UpgradeState s;
|
||||
if (!upgrades.Value.TryGetValue(upgrade, out s))
|
||||
return;
|
||||
|
||||
// Track the upgrade source so that the upgrade can be removed without conflicts
|
||||
s.Sources.Add(source);
|
||||
|
||||
NotifyUpgradeLevelChanged(s.Traits, self, upgrade, 1);
|
||||
}
|
||||
|
||||
public void RevokeUpgrade(Actor self, string upgrade, object source)
|
||||
{
|
||||
UpgradeState s;
|
||||
if (!upgrades.Value.TryGetValue(upgrade, out s))
|
||||
return;
|
||||
|
||||
if (!s.Sources.Remove(source))
|
||||
throw new InvalidOperationException("Object <{0}> revoked more levels of upgrade {1} than it granted for {2}.".F(source, upgrade, self));
|
||||
|
||||
NotifyUpgradeLevelChanged(s.Traits, self, upgrade, -1);
|
||||
}
|
||||
|
||||
public bool AcceptsUpgrade(Actor self, string upgrade)
|
||||
{
|
||||
UpgradeState s;
|
||||
if (!upgrades.Value.TryGetValue(upgrade, out s))
|
||||
return false;
|
||||
|
||||
return s.Traits.Any(up => up.AcceptsUpgradeLevel(self, upgrade, GetOverallLevel(up) + 1));
|
||||
}
|
||||
|
||||
public void RegisterWatcher(string upgrade, Action<int, int> action)
|
||||
{
|
||||
UpgradeState s;
|
||||
if (!upgrades.Value.TryGetValue(upgrade, out s))
|
||||
return;
|
||||
|
||||
s.Watchers.Add(action);
|
||||
}
|
||||
|
||||
public void Tick(Actor self)
|
||||
{
|
||||
foreach (var u in timedUpgrades)
|
||||
{
|
||||
u.Tick();
|
||||
if (u.Remaining <= 0)
|
||||
RevokeUpgrade(self, u.Upgrade, u);
|
||||
|
||||
foreach (var a in upgrades.Value[u.Upgrade].Watchers)
|
||||
a(u.Duration, u.Remaining);
|
||||
}
|
||||
|
||||
timedUpgrades.RemoveAll(u => u.Remaining <= 0);
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user