Merge pull request #6437 from pchote/fix-crate-crash

Fixes #5953
Fixes #6427
This commit is contained in:
Matthias Mailänder
2014-09-13 08:25:32 +02:00
16 changed files with 191 additions and 139 deletions

View File

@@ -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)

View File

@@ -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)));
} }
} }
} }

View File

@@ -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);
}
} }
} }

View File

@@ -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);
} }
} }

View File

@@ -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);

View File

@@ -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;
} }
} }
} }

View File

@@ -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);
} }

View File

@@ -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);
} }
} }
} }

View File

@@ -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);
} }
} }
} }

View File

@@ -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);

View File

@@ -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);
} }
} }
} }

View File

@@ -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);
} }
} }
} }

View File

@@ -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 =>
{ {

View File

@@ -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);
} }
} }

View File

@@ -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

View File

@@ -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