Merge pull request #6437 from pchote/fix-crate-crash
Fixes #5953 Fixes #6427
This commit is contained in:
@@ -11,6 +11,7 @@
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using OpenRA.Mods.RA.Buildings;
|
using OpenRA.Mods.RA.Buildings;
|
||||||
|
using OpenRA.Mods.RA.Move;
|
||||||
using OpenRA.Mods.RA.Render;
|
using OpenRA.Mods.RA.Render;
|
||||||
using OpenRA.Primitives;
|
using OpenRA.Primitives;
|
||||||
using OpenRA.Traits;
|
using OpenRA.Traits;
|
||||||
@@ -53,10 +54,12 @@ namespace OpenRA.Mods.RA
|
|||||||
|
|
||||||
public void OnCrush(Actor crusher)
|
public void OnCrush(Actor crusher)
|
||||||
{
|
{
|
||||||
if (collected) return;
|
if (collected)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var shares = self.TraitsImplementing<CrateAction>()
|
||||||
|
.Select(a => Pair.New(a, a.GetSelectionSharesOuter(crusher)));
|
||||||
|
|
||||||
var shares = self.TraitsImplementing<CrateAction>().Select(
|
|
||||||
a => Pair.New(a, a.GetSelectionSharesOuter(crusher)));
|
|
||||||
var totalShares = shares.Sum(a => a.Second);
|
var totalShares = shares.Sum(a => a.Second);
|
||||||
var n = self.World.SharedRandom.Next(totalShares);
|
var n = self.World.SharedRandom.Next(totalShares);
|
||||||
|
|
||||||
@@ -64,6 +67,7 @@ namespace OpenRA.Mods.RA
|
|||||||
collected = true;
|
collected = true;
|
||||||
|
|
||||||
foreach (var s in shares)
|
foreach (var s in shares)
|
||||||
|
{
|
||||||
if (n < s.Second)
|
if (n < s.Second)
|
||||||
{
|
{
|
||||||
s.First.Activate(crusher);
|
s.First.Activate(crusher);
|
||||||
@@ -71,15 +75,34 @@ namespace OpenRA.Mods.RA
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
n -= s.Second;
|
n -= s.Second;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void OnLanded()
|
public void OnLanded()
|
||||||
{
|
{
|
||||||
|
// Check whether the crate landed on anything
|
||||||
var landedOn = self.World.ActorMap.GetUnitsAt(self.Location)
|
var landedOn = self.World.ActorMap.GetUnitsAt(self.Location)
|
||||||
.FirstOrDefault(a => a != self);
|
.Where(a => a != self);
|
||||||
|
|
||||||
if (landedOn != null)
|
if (!landedOn.Any())
|
||||||
OnCrush(landedOn);
|
return;
|
||||||
|
|
||||||
|
var collector = landedOn.FirstOrDefault(a =>
|
||||||
|
{
|
||||||
|
// Mobile is (currently) the only trait that supports crushing
|
||||||
|
var mi = a.Info.Traits.GetOrDefault<MobileInfo>();
|
||||||
|
if (mi == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Make sure that the actor can collect this crate type
|
||||||
|
return CrushableBy(mi.Crushes, a.Owner);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Destroy the crate if none of the units in the cell are valid collectors
|
||||||
|
if (collector != null)
|
||||||
|
OnCrush(collector);
|
||||||
|
else
|
||||||
|
self.Destroy();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Tick(Actor self)
|
public void Tick(Actor self)
|
||||||
|
|||||||
@@ -42,8 +42,8 @@ namespace OpenRA.Mods.RA
|
|||||||
|
|
||||||
public class CrateAction
|
public class CrateAction
|
||||||
{
|
{
|
||||||
public Actor self;
|
readonly Actor self;
|
||||||
public CrateActionInfo info;
|
readonly CrateActionInfo info;
|
||||||
|
|
||||||
public CrateAction(Actor self, CrateActionInfo info)
|
public CrateAction(Actor self, CrateActionInfo info)
|
||||||
{
|
{
|
||||||
@@ -75,8 +75,7 @@ namespace OpenRA.Mods.RA
|
|||||||
Sound.PlayToPlayer(collector.Owner, info.Notification);
|
Sound.PlayToPlayer(collector.Owner, info.Notification);
|
||||||
|
|
||||||
if (info.Effect != null)
|
if (info.Effect != null)
|
||||||
collector.World.AddFrameEndTask(
|
collector.World.AddFrameEndTask(w => w.Add(new CrateEffect(collector, info.Effect, info.Palette)));
|
||||||
w => w.Add(new CrateEffect(collector, info.Effect, info.Palette)));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,6 +8,7 @@
|
|||||||
*/
|
*/
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using OpenRA.Mods.RA.Move;
|
using OpenRA.Mods.RA.Move;
|
||||||
@@ -22,14 +23,17 @@ namespace OpenRA.Mods.RA.Crates
|
|||||||
[Desc("The maximum number of duplicates to make.")]
|
[Desc("The maximum number of duplicates to make.")]
|
||||||
public readonly int MaxAmount = 2;
|
public readonly int MaxAmount = 2;
|
||||||
|
|
||||||
[Desc("The minimum number of duplicates to make.","Overrules MaxDuplicatesWorth.")]
|
[Desc("The minimum number of duplicates to make. Overrules MaxDuplicatesWorth.")]
|
||||||
public readonly int MinAmount = 1;
|
public readonly int MinAmount = 1;
|
||||||
|
|
||||||
[Desc("The maximum total cost allowed for the duplicates.","Duplication stops if the total worth will exceed this number.","-1 = no limit")]
|
[Desc("The maximum total value allowed for the duplicates.", "Duplication stops if the total worth will exceed this number.", "-1 = no limit")]
|
||||||
public readonly int MaxDuplicatesWorth = -1;
|
public readonly int MaxDuplicateValue = -1;
|
||||||
|
|
||||||
[Desc("The list of unit types we are allowed to duplicate.")]
|
[Desc("The maximum radius (in cells) that duplicates can be spawned.")]
|
||||||
public readonly string[] ValidDuplicateTypes = { "Ground", "Water" };
|
public readonly int MaxRadius = 4;
|
||||||
|
|
||||||
|
[Desc("The list of unit target types we are allowed to duplicate.")]
|
||||||
|
public readonly string[] ValidTargets = { "Ground", "Water" };
|
||||||
|
|
||||||
[Desc("Which races this crate action can occur for.")]
|
[Desc("Which races this crate action can occur for.")]
|
||||||
public readonly string[] ValidRaces = { };
|
public readonly string[] ValidRaces = { };
|
||||||
@@ -42,80 +46,67 @@ namespace OpenRA.Mods.RA.Crates
|
|||||||
|
|
||||||
class DuplicateUnitCrateAction : CrateAction
|
class DuplicateUnitCrateAction : CrateAction
|
||||||
{
|
{
|
||||||
public readonly DuplicateUnitCrateActionInfo Info;
|
readonly DuplicateUnitCrateActionInfo info;
|
||||||
readonly List<CPos> usedCells = new List<CPos>();
|
|
||||||
|
|
||||||
public DuplicateUnitCrateAction(Actor self, DuplicateUnitCrateActionInfo info)
|
public DuplicateUnitCrateAction(Actor self, DuplicateUnitCrateActionInfo info)
|
||||||
: base(self, info) { Info = info; }
|
: base(self, info)
|
||||||
|
{
|
||||||
|
this.info = info;
|
||||||
|
}
|
||||||
|
|
||||||
public bool CanGiveTo(Actor collector)
|
public bool CanGiveTo(Actor collector)
|
||||||
{
|
{
|
||||||
if (Info.ValidRaces.Any() && !Info.ValidRaces.Contains(collector.Owner.Country.Race))
|
if (info.ValidRaces.Any() && !info.ValidRaces.Contains(collector.Owner.Country.Race))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
var targetable = collector.Info.Traits.GetOrDefault<ITargetableInfo>();
|
var targetable = collector.Info.Traits.GetOrDefault<ITargetableInfo>();
|
||||||
if (targetable == null ||
|
if (targetable == null || !info.ValidTargets.Intersect(targetable.GetTargetTypes()).Any())
|
||||||
!Info.ValidDuplicateTypes.Intersect(targetable.GetTargetTypes()).Any())
|
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (!GetSuitableCells(collector.Location, collector.Info.Name).Any()) return false;
|
var positionable = collector.TraitOrDefault<IPositionable>();
|
||||||
|
if (positionable == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
return true;
|
return collector.World.Map.FindTilesInCircle(collector.Location, info.MaxRadius)
|
||||||
|
.Any(c => positionable.CanEnterCell(c));
|
||||||
}
|
}
|
||||||
|
|
||||||
public override int GetSelectionShares(Actor collector)
|
public override int GetSelectionShares(Actor collector)
|
||||||
{
|
{
|
||||||
if (!CanGiveTo(collector)) return 0;
|
if (!CanGiveTo(collector))
|
||||||
|
return 0;
|
||||||
|
|
||||||
return base.GetSelectionShares(collector);
|
return base.GetSelectionShares(collector);
|
||||||
}
|
}
|
||||||
|
|
||||||
public override void Activate(Actor collector)
|
public override void Activate(Actor collector)
|
||||||
{
|
{
|
||||||
int AllowedWorthLeft = Info.MaxDuplicatesWorth;
|
var positionable = collector.Trait<IPositionable>();
|
||||||
int DupesMade = 0;
|
var candidateCells = collector.World.Map.FindTilesInCircle(collector.Location, info.MaxRadius)
|
||||||
|
.Where(c => positionable.CanEnterCell(c)).Shuffle(collector.World.SharedRandom)
|
||||||
|
.ToArray();
|
||||||
|
|
||||||
while ((DupesMade < Info.MaxAmount) && (AllowedWorthLeft > 0) || (DupesMade < Info.MinAmount))
|
var duplicates = Math.Min(candidateCells.Length, info.MaxAmount);
|
||||||
|
|
||||||
|
// Restrict duplicate count to a maximum value
|
||||||
|
if (info.MaxDuplicateValue > 0)
|
||||||
{
|
{
|
||||||
//If the collector has a cost, and we have a max duplicate worth, then update how much dupe worth is left
|
var vi = collector.Info.Traits.GetOrDefault<ValuedInfo>();
|
||||||
var unitCost = collector.Info.Traits.Get<ValuedInfo>().Cost;
|
if (vi != null && vi.Cost > 0)
|
||||||
AllowedWorthLeft -= (Info.MaxDuplicatesWorth > 0) ? unitCost : 0;
|
duplicates = Math.Min(duplicates, info.MaxDuplicateValue / vi.Cost);
|
||||||
if ((AllowedWorthLeft < 0) && (DupesMade >= Info.MinAmount))
|
|
||||||
break;
|
|
||||||
|
|
||||||
DupesMade++;
|
|
||||||
|
|
||||||
var location = ChooseEmptyCellNear(collector, collector.Info.Name);
|
|
||||||
if (location != null)
|
|
||||||
{
|
|
||||||
usedCells.Add(location.Value);
|
|
||||||
collector.World.AddFrameEndTask(
|
|
||||||
w => w.CreateActor(collector.Info.Name, new TypeDictionary
|
|
||||||
{
|
|
||||||
new LocationInit(location.Value),
|
|
||||||
new OwnerInit(Info.Owner ?? collector.Owner.InternalName)
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for (var i = 0; i < duplicates; i++)
|
||||||
|
{
|
||||||
|
var cell = candidateCells[i]; // Avoid modified closure bug
|
||||||
|
collector.World.AddFrameEndTask(w => w.CreateActor(collector.Info.Name, new TypeDictionary
|
||||||
|
{
|
||||||
|
new LocationInit(cell),
|
||||||
|
new OwnerInit(info.Owner ?? collector.Owner.InternalName)
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
base.Activate(collector);
|
base.Activate(collector);
|
||||||
}
|
}
|
||||||
|
|
||||||
IEnumerable<CPos> GetSuitableCells(CPos near, string unitName)
|
|
||||||
{
|
|
||||||
var mi = self.World.Map.Rules.Actors[unitName].Traits.Get<MobileInfo>();
|
|
||||||
|
|
||||||
for (var i = -3; i < 4; i++)
|
|
||||||
for (var j = -3; j < 4; j++)
|
|
||||||
if (mi.CanEnterCell(self.World, self, near + new CVec(i, j)))
|
|
||||||
yield return near + new CVec(i, j);
|
|
||||||
}
|
|
||||||
|
|
||||||
CPos? ChooseEmptyCellNear(Actor a, string unit)
|
|
||||||
{
|
|
||||||
var possibleCells = GetSuitableCells(a.Location, unit).Where(c => !usedCells.Contains(c)).ToArray();
|
|
||||||
if (possibleCells.Length == 0)
|
|
||||||
return null;
|
|
||||||
|
|
||||||
return possibleCells.Random(self.World.SharedRandom);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -25,13 +25,19 @@ namespace OpenRA.Mods.RA
|
|||||||
|
|
||||||
class ExplodeCrateAction : CrateAction
|
class ExplodeCrateAction : CrateAction
|
||||||
{
|
{
|
||||||
|
readonly ExplodeCrateActionInfo info;
|
||||||
|
|
||||||
public ExplodeCrateAction(Actor self, ExplodeCrateActionInfo info)
|
public ExplodeCrateAction(Actor self, ExplodeCrateActionInfo info)
|
||||||
: base(self, info) {}
|
: base(self, info)
|
||||||
|
{
|
||||||
|
this.info = info;
|
||||||
|
}
|
||||||
|
|
||||||
public override void Activate(Actor collector)
|
public override void Activate(Actor collector)
|
||||||
{
|
{
|
||||||
var weapon = self.World.Map.Rules.Weapons[((ExplodeCrateActionInfo)info).Weapon.ToLowerInvariant()];
|
var weapon = collector.World.Map.Rules.Weapons[info.Weapon.ToLowerInvariant()];
|
||||||
weapon.Impact(Target.FromPos(collector.CenterPosition), self, Enumerable.Empty<int>());
|
weapon.Impact(Target.FromPos(collector.CenterPosition), collector, Enumerable.Empty<int>());
|
||||||
|
|
||||||
base.Activate(collector);
|
base.Activate(collector);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -27,19 +27,21 @@ namespace OpenRA.Mods.RA
|
|||||||
|
|
||||||
class GiveCashCrateAction : CrateAction
|
class GiveCashCrateAction : CrateAction
|
||||||
{
|
{
|
||||||
|
readonly GiveCashCrateActionInfo info;
|
||||||
public GiveCashCrateAction(Actor self, GiveCashCrateActionInfo info)
|
public GiveCashCrateAction(Actor self, GiveCashCrateActionInfo info)
|
||||||
: base(self, info) {}
|
: base(self, info)
|
||||||
|
{
|
||||||
|
this.info = info;
|
||||||
|
}
|
||||||
|
|
||||||
public override void Activate(Actor collector)
|
public override void Activate(Actor collector)
|
||||||
{
|
{
|
||||||
collector.World.AddFrameEndTask(w =>
|
collector.World.AddFrameEndTask(w =>
|
||||||
{
|
{
|
||||||
var crateInfo = (GiveCashCrateActionInfo)info;
|
collector.Owner.PlayerActor.Trait<PlayerResources>().GiveCash(info.Amount);
|
||||||
var amount = crateInfo.Amount;
|
|
||||||
collector.Owner.PlayerActor.Trait<PlayerResources>().GiveCash(amount);
|
|
||||||
|
|
||||||
if (crateInfo.UseCashTick)
|
if (info.UseCashTick)
|
||||||
w.Add(new FloatingText(collector.CenterPosition, collector.Owner.Color.RGB, FloatingText.FormatCashTick(amount), 30));
|
w.Add(new FloatingText(collector.CenterPosition, collector.Owner.Color.RGB, FloatingText.FormatCashTick(info.Amount), 30));
|
||||||
});
|
});
|
||||||
|
|
||||||
base.Activate(collector);
|
base.Activate(collector);
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ using System.Linq;
|
|||||||
|
|
||||||
namespace OpenRA.Mods.RA.Crates
|
namespace OpenRA.Mods.RA.Crates
|
||||||
{
|
{
|
||||||
[Desc("Spawns units when collected.","Adjust selection shares when player has no base.")]
|
[Desc("Spawns units when collected.", "Adjust selection shares when player has no base.")]
|
||||||
class GiveMcvCrateActionInfo : GiveUnitCrateActionInfo
|
class GiveMcvCrateActionInfo : GiveUnitCrateActionInfo
|
||||||
{
|
{
|
||||||
[Desc("The selection shares to use if the collector has no base.")]
|
[Desc("The selection shares to use if the collector has no base.")]
|
||||||
@@ -23,19 +23,23 @@ namespace OpenRA.Mods.RA.Crates
|
|||||||
|
|
||||||
class GiveMcvCrateAction : GiveUnitCrateAction
|
class GiveMcvCrateAction : GiveUnitCrateAction
|
||||||
{
|
{
|
||||||
|
readonly GiveMcvCrateActionInfo info;
|
||||||
public GiveMcvCrateAction(Actor self, GiveMcvCrateActionInfo info)
|
public GiveMcvCrateAction(Actor self, GiveMcvCrateActionInfo info)
|
||||||
: base(self, info) { }
|
: base(self, info)
|
||||||
|
{
|
||||||
|
this.info = info;
|
||||||
|
}
|
||||||
|
|
||||||
public override int GetSelectionShares(Actor collector)
|
public override int GetSelectionShares(Actor collector)
|
||||||
{
|
{
|
||||||
|
// There's some other really good reason why we shouldn't give this.
|
||||||
if (!CanGiveTo(collector))
|
if (!CanGiveTo(collector))
|
||||||
return 0; // there's some other really good reason why we shouldn't give this.
|
return 0;
|
||||||
|
|
||||||
var hasBase = self.World.ActorsWithTrait<BaseBuilding>()
|
var hasBase = collector.World.ActorsWithTrait<BaseBuilding>()
|
||||||
.Any(a => a.Actor.Owner == collector.Owner);
|
.Any(a => a.Actor.Owner == collector.Owner);
|
||||||
|
|
||||||
return hasBase ? info.SelectionShares :
|
return hasBase ? info.SelectionShares : info.NoBaseSelectionShares;
|
||||||
((GiveMcvCrateActionInfo)info).NoBaseSelectionShares;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -34,26 +34,29 @@ namespace OpenRA.Mods.RA.Crates
|
|||||||
|
|
||||||
class GiveUnitCrateAction : CrateAction
|
class GiveUnitCrateAction : CrateAction
|
||||||
{
|
{
|
||||||
public readonly GiveUnitCrateActionInfo Info;
|
readonly Actor self;
|
||||||
|
readonly GiveUnitCrateActionInfo info;
|
||||||
readonly List<CPos> usedCells = new List<CPos>();
|
readonly List<CPos> usedCells = new List<CPos>();
|
||||||
|
|
||||||
public GiveUnitCrateAction(Actor self, GiveUnitCrateActionInfo info)
|
public GiveUnitCrateAction(Actor self, GiveUnitCrateActionInfo info)
|
||||||
: base(self, info)
|
: base(self, info)
|
||||||
{
|
{
|
||||||
Info = info;
|
this.self = self;
|
||||||
if (!Info.Units.Any())
|
this.info = info;
|
||||||
|
if (!info.Units.Any())
|
||||||
throw new YamlException("A GiveUnitCrateAction does not specify any units to give. This might be because the yaml is referring to 'Unit' rather than 'Units'.");
|
throw new YamlException("A GiveUnitCrateAction does not specify any units to give. This might be because the yaml is referring to 'Unit' rather than 'Units'.");
|
||||||
}
|
}
|
||||||
|
|
||||||
public bool CanGiveTo(Actor collector)
|
public bool CanGiveTo(Actor collector)
|
||||||
{
|
{
|
||||||
if (Info.ValidRaces.Any() && !Info.ValidRaces.Contains(collector.Owner.Country.Race))
|
if (info.ValidRaces.Any() && !info.ValidRaces.Contains(collector.Owner.Country.Race))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
foreach (string unit in Info.Units)
|
foreach (string unit in info.Units)
|
||||||
{
|
{
|
||||||
// avoid dumping tanks in the sea, and ships on dry land.
|
// avoid dumping tanks in the sea, and ships on dry land.
|
||||||
if (!GetSuitableCells(collector.Location, unit).Any()) return false;
|
if (!GetSuitableCells(collector.Location, unit).Any())
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
@@ -61,13 +64,15 @@ namespace OpenRA.Mods.RA.Crates
|
|||||||
|
|
||||||
public override int GetSelectionShares(Actor collector)
|
public override int GetSelectionShares(Actor collector)
|
||||||
{
|
{
|
||||||
if (!CanGiveTo(collector)) return 0;
|
if (!CanGiveTo(collector))
|
||||||
|
return 0;
|
||||||
|
|
||||||
return base.GetSelectionShares(collector);
|
return base.GetSelectionShares(collector);
|
||||||
}
|
}
|
||||||
|
|
||||||
public override void Activate(Actor collector)
|
public override void Activate(Actor collector)
|
||||||
{
|
{
|
||||||
foreach (var u in Info.Units)
|
foreach (var u in info.Units)
|
||||||
{
|
{
|
||||||
var unit = u; // avoiding access to modified closure
|
var unit = u; // avoiding access to modified closure
|
||||||
|
|
||||||
@@ -79,10 +84,11 @@ namespace OpenRA.Mods.RA.Crates
|
|||||||
w => w.CreateActor(unit, new TypeDictionary
|
w => w.CreateActor(unit, new TypeDictionary
|
||||||
{
|
{
|
||||||
new LocationInit(location.Value),
|
new LocationInit(location.Value),
|
||||||
new OwnerInit(Info.Owner ?? collector.Owner.InternalName)
|
new OwnerInit(info.Owner ?? collector.Owner.InternalName)
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
base.Activate(collector);
|
base.Activate(collector);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -26,15 +26,14 @@ namespace OpenRA.Mods.RA.Crates
|
|||||||
|
|
||||||
public override void Activate(Actor collector)
|
public override void Activate(Actor collector)
|
||||||
{
|
{
|
||||||
base.Activate(collector);
|
|
||||||
foreach (var unit in collector.World.Actors.Where(a => a.Owner == collector.Owner))
|
foreach (var unit in collector.World.Actors.Where(a => a.Owner == collector.Owner))
|
||||||
{
|
{
|
||||||
var health = unit.TraitOrDefault<Health>();
|
var health = unit.TraitOrDefault<Health>();
|
||||||
if (health != null && !health.IsDead)
|
if (health != null && !health.IsDead)
|
||||||
{
|
|
||||||
health.InflictDamage(unit, unit, -(health.MaxHP - health.HP), null, true);
|
health.InflictDamage(unit, unit, -(health.MaxHP - health.HP), null, true);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
base.Activate(collector);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -32,9 +32,10 @@ namespace OpenRA.Mods.RA
|
|||||||
|
|
||||||
public override void Activate(Actor collector)
|
public override void Activate(Actor collector)
|
||||||
{
|
{
|
||||||
base.Activate(collector);
|
|
||||||
if (collector.Owner == collector.World.LocalPlayer)
|
if (collector.Owner == collector.World.LocalPlayer)
|
||||||
collector.Owner.Shroud.ResetExploration();
|
collector.Owner.Shroud.ResetExploration();
|
||||||
|
|
||||||
|
base.Activate(collector);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,12 +29,14 @@ namespace OpenRA.Mods.RA
|
|||||||
|
|
||||||
class LevelUpCrateAction : CrateAction
|
class LevelUpCrateAction : CrateAction
|
||||||
{
|
{
|
||||||
LevelUpCrateActionInfo Info;
|
readonly Actor self;
|
||||||
|
readonly LevelUpCrateActionInfo info;
|
||||||
|
|
||||||
public LevelUpCrateAction(Actor self, LevelUpCrateActionInfo info)
|
public LevelUpCrateAction(Actor self, LevelUpCrateActionInfo info)
|
||||||
: base(self, info)
|
: base(self, info)
|
||||||
{
|
{
|
||||||
Info = info;
|
this.self = self;
|
||||||
|
this.info = info;
|
||||||
}
|
}
|
||||||
|
|
||||||
public override int GetSelectionShares(Actor collector)
|
public override int GetSelectionShares(Actor collector)
|
||||||
@@ -45,34 +47,33 @@ namespace OpenRA.Mods.RA
|
|||||||
|
|
||||||
public override void Activate(Actor collector)
|
public override void Activate(Actor collector)
|
||||||
{
|
{
|
||||||
collector.World.AddFrameEndTask(w =>
|
var inRange = self.World.FindActorsInCircle(self.CenterPosition, info.Range).Where(a =>
|
||||||
{
|
{
|
||||||
var gainsExperience = collector.TraitOrDefault<GainsExperience>();
|
// Don't upgrade the same unit twice
|
||||||
if (gainsExperience != null)
|
if (a == collector)
|
||||||
gainsExperience.GiveLevels(((LevelUpCrateActionInfo)info).Levels);
|
return false;
|
||||||
|
|
||||||
|
// Only upgrade the collecting player's units
|
||||||
|
// TODO: Also apply to allied units?
|
||||||
|
if (a.Owner != collector.Owner)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Ignore units that can't level up
|
||||||
|
var ge = a.TraitOrDefault<GainsExperience>();
|
||||||
|
return ge != null && ge.CanGainLevel;
|
||||||
});
|
});
|
||||||
|
|
||||||
var inRange = self.World.FindActorsInCircle(self.CenterPosition, Info.Range);
|
if (info.MaxExtraCollectors > -1)
|
||||||
inRange = inRange.Where(a =>
|
inRange = inRange.Take(info.MaxExtraCollectors);
|
||||||
(a.Owner == collector.Owner) &&
|
|
||||||
(a != collector) &&
|
|
||||||
(a.TraitOrDefault<GainsExperience>() != null) &&
|
|
||||||
(a.TraitOrDefault<GainsExperience>().CanGainLevel));
|
|
||||||
if (inRange.Any())
|
|
||||||
{
|
|
||||||
if (Info.MaxExtraCollectors > -1)
|
|
||||||
inRange = inRange.Take(Info.MaxExtraCollectors);
|
|
||||||
|
|
||||||
if (inRange.Any())
|
foreach (var actor in inRange.Append(collector))
|
||||||
foreach (Actor actor in inRange)
|
{
|
||||||
{
|
actor.World.AddFrameEndTask(w =>
|
||||||
actor.World.AddFrameEndTask(w =>
|
{
|
||||||
{
|
var gainsExperience = actor.TraitOrDefault<GainsExperience>();
|
||||||
var gainsExperience = actor.TraitOrDefault<GainsExperience>();
|
if (gainsExperience != null)
|
||||||
if (gainsExperience != null)
|
gainsExperience.GiveLevels(((LevelUpCrateActionInfo)info).Levels);
|
||||||
gainsExperience.GiveLevels(((LevelUpCrateActionInfo)info).Levels);
|
});
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
base.Activate(collector);
|
base.Activate(collector);
|
||||||
|
|||||||
@@ -23,12 +23,17 @@ namespace OpenRA.Mods.RA
|
|||||||
|
|
||||||
class RevealMapCrateAction : CrateAction
|
class RevealMapCrateAction : CrateAction
|
||||||
{
|
{
|
||||||
|
readonly RevealMapCrateActionInfo info;
|
||||||
|
|
||||||
public RevealMapCrateAction(Actor self, RevealMapCrateActionInfo info)
|
public RevealMapCrateAction(Actor self, RevealMapCrateActionInfo info)
|
||||||
: base(self, info) {}
|
: base(self, info)
|
||||||
|
{
|
||||||
|
this.info = info;
|
||||||
|
}
|
||||||
|
|
||||||
bool ShouldReveal(Player collectingPlayer)
|
bool ShouldReveal(Player collectingPlayer)
|
||||||
{
|
{
|
||||||
if (((RevealMapCrateActionInfo)info).IncludeAllies)
|
if (info.IncludeAllies)
|
||||||
return collectingPlayer.World.LocalPlayer != null &&
|
return collectingPlayer.World.LocalPlayer != null &&
|
||||||
collectingPlayer.Stances[collectingPlayer.World.LocalPlayer] == Stance.Ally;
|
collectingPlayer.Stances[collectingPlayer.World.LocalPlayer] == Stance.Ally;
|
||||||
|
|
||||||
@@ -37,10 +42,10 @@ namespace OpenRA.Mods.RA
|
|||||||
|
|
||||||
public override void Activate(Actor collector)
|
public override void Activate(Actor collector)
|
||||||
{
|
{
|
||||||
base.Activate(collector);
|
if (ShouldReveal(collector.Owner))
|
||||||
|
|
||||||
if (ShouldReveal( collector.Owner ))
|
|
||||||
collector.Owner.Shroud.ExploreAll(collector.World);
|
collector.Owner.Shroud.ExploreAll(collector.World);
|
||||||
|
|
||||||
|
base.Activate(collector);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -24,20 +24,23 @@ namespace OpenRA.Mods.RA.Crates
|
|||||||
|
|
||||||
class SupportPowerCrateAction : CrateAction
|
class SupportPowerCrateAction : CrateAction
|
||||||
{
|
{
|
||||||
SupportPowerCrateActionInfo Info;
|
SupportPowerCrateActionInfo info;
|
||||||
public SupportPowerCrateAction(Actor self, SupportPowerCrateActionInfo info)
|
public SupportPowerCrateAction(Actor self, SupportPowerCrateActionInfo info)
|
||||||
: base(self, info) { Info = info; }
|
: base(self, info)
|
||||||
|
{
|
||||||
|
this.info = info;
|
||||||
|
}
|
||||||
|
|
||||||
// The free unit crate requires same race and the actor needs to be mobile.
|
// The free unit crate requires same race and the actor needs to be mobile.
|
||||||
// We want neither of these properties for crate power proxies.
|
// We want neither of these properties for crate power proxies.
|
||||||
public override void Activate(Actor collector)
|
public override void Activate(Actor collector)
|
||||||
{
|
{
|
||||||
base.Activate(collector);
|
collector.World.AddFrameEndTask(w => w.CreateActor(info.Proxy, new TypeDictionary
|
||||||
|
|
||||||
collector.World.AddFrameEndTask(w => w.CreateActor(Info.Proxy, new TypeDictionary
|
|
||||||
{
|
{
|
||||||
new OwnerInit(collector.Owner)
|
new OwnerInit(collector.Owner)
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
base.Activate(collector);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ namespace OpenRA.Mods.RA.Crates
|
|||||||
public class UnitUpgradeCrateActionInfo : CrateActionInfo
|
public class UnitUpgradeCrateActionInfo : CrateActionInfo
|
||||||
{
|
{
|
||||||
[Desc("The upgrade to grant.")]
|
[Desc("The upgrade to grant.")]
|
||||||
public readonly string[] Upgrades = {};
|
public readonly string[] Upgrades = { };
|
||||||
|
|
||||||
[Desc("The range to search for extra collectors in.", "Extra collectors will also be granted the crate action.")]
|
[Desc("The range to search for extra collectors in.", "Extra collectors will also be granted the crate action.")]
|
||||||
public readonly WRange Range = new WRange(3);
|
public readonly WRange Range = new WRange(3);
|
||||||
@@ -30,24 +30,26 @@ namespace OpenRA.Mods.RA.Crates
|
|||||||
|
|
||||||
public class UnitUpgradeCrateAction : CrateAction
|
public class UnitUpgradeCrateAction : CrateAction
|
||||||
{
|
{
|
||||||
readonly UnitUpgradeCrateActionInfo Info;
|
readonly Actor self;
|
||||||
|
readonly UnitUpgradeCrateActionInfo info;
|
||||||
|
|
||||||
public UnitUpgradeCrateAction(Actor self, UnitUpgradeCrateActionInfo info)
|
public UnitUpgradeCrateAction(Actor self, UnitUpgradeCrateActionInfo info)
|
||||||
: base(self, info)
|
: base(self, info)
|
||||||
{
|
{
|
||||||
Info = info;
|
this.self = self;
|
||||||
|
this.info = info;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AcceptsUpgrade(Actor a)
|
bool AcceptsUpgrade(Actor a)
|
||||||
{
|
{
|
||||||
return a.TraitsImplementing<IUpgradable>()
|
return a.TraitsImplementing<IUpgradable>()
|
||||||
.Any(up => Info.Upgrades.Any(u => up.AcceptsUpgrade(u)));
|
.Any(up => info.Upgrades.Any(u => up.AcceptsUpgrade(u)));
|
||||||
}
|
}
|
||||||
|
|
||||||
void GrantActorUpgrades(Actor a)
|
void GrantActorUpgrades(Actor a)
|
||||||
{
|
{
|
||||||
foreach (var up in a.TraitsImplementing<IUpgradable>())
|
foreach (var up in a.TraitsImplementing<IUpgradable>())
|
||||||
foreach (var u in Info.Upgrades)
|
foreach (var u in info.Upgrades)
|
||||||
if (up.AcceptsUpgrade(u))
|
if (up.AcceptsUpgrade(u))
|
||||||
up.UpgradeAvailable(a, u, true);
|
up.UpgradeAvailable(a, u, true);
|
||||||
}
|
}
|
||||||
@@ -61,13 +63,13 @@ namespace OpenRA.Mods.RA.Crates
|
|||||||
{
|
{
|
||||||
collector.World.AddFrameEndTask(w => GrantActorUpgrades(collector));
|
collector.World.AddFrameEndTask(w => GrantActorUpgrades(collector));
|
||||||
|
|
||||||
var actorsInRange = self.World.FindActorsInCircle(self.CenterPosition, Info.Range)
|
var actorsInRange = self.World.FindActorsInCircle(self.CenterPosition, info.Range)
|
||||||
.Where(a => a != self && a.Owner == collector.Owner && AcceptsUpgrade(a));
|
.Where(a => a != self && a.Owner == collector.Owner && AcceptsUpgrade(a));
|
||||||
|
|
||||||
if (actorsInRange.Any())
|
if (actorsInRange.Any())
|
||||||
{
|
{
|
||||||
if (Info.MaxExtraCollectors > -1)
|
if (info.MaxExtraCollectors > -1)
|
||||||
actorsInRange = actorsInRange.Take(Info.MaxExtraCollectors);
|
actorsInRange = actorsInRange.Take(info.MaxExtraCollectors);
|
||||||
|
|
||||||
collector.World.AddFrameEndTask(w =>
|
collector.World.AddFrameEndTask(w =>
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -517,6 +517,16 @@ namespace OpenRA.Utility
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// DuplicateUnitCrateAction was tidied up
|
||||||
|
if (engineVersion < 20140912)
|
||||||
|
{
|
||||||
|
if (depth == 2 && node.Key == "MaxDuplicatesWorth" && parentKey == "DuplicateUnitCrateAction")
|
||||||
|
node.Key = "MaxDuplicateValue";
|
||||||
|
|
||||||
|
if (depth == 2 && node.Key == "ValidDuplicateTypes" && parentKey == "DuplicateUnitCrateAction")
|
||||||
|
node.Key = "ValidTargets";
|
||||||
|
}
|
||||||
|
|
||||||
UpgradeActorRules(engineVersion, ref node.Value.Nodes, node, depth + 1);
|
UpgradeActorRules(engineVersion, ref node.Value.Nodes, node, depth + 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,7 +22,7 @@ CRATE:
|
|||||||
SelectionShares: 10
|
SelectionShares: 10
|
||||||
MaxAmount: 5
|
MaxAmount: 5
|
||||||
MinAmount: 1
|
MinAmount: 1
|
||||||
MaxDuplicatesWorth: 1250
|
MaxDuplicateValue: 1250
|
||||||
GiveMcvCrateAction:
|
GiveMcvCrateAction:
|
||||||
SelectionShares: 0
|
SelectionShares: 0
|
||||||
NoBaseSelectionShares: 120
|
NoBaseSelectionShares: 120
|
||||||
|
|||||||
@@ -93,7 +93,7 @@ CRATE:
|
|||||||
SelectionShares: 10
|
SelectionShares: 10
|
||||||
MaxAmount: 5
|
MaxAmount: 5
|
||||||
MinAmount: 1
|
MinAmount: 1
|
||||||
MaxDuplicatesWorth: 1500
|
MaxDuplicateValue: 1500
|
||||||
GiveMcvCrateAction:
|
GiveMcvCrateAction:
|
||||||
SelectionShares: 2
|
SelectionShares: 2
|
||||||
NoBaseSelectionShares: 100
|
NoBaseSelectionShares: 100
|
||||||
|
|||||||
Reference in New Issue
Block a user