#region Copyright & License Information /* * Copyright (c) The OpenRA Developers and Contributors * 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, either version 3 of * the License, or (at your option) any later version. For more * information, see COPYING. */ #endregion using System.Collections.Generic; using System.Linq; using OpenRA.Activities; using OpenRA.Mods.Common.Traits; using OpenRA.Primitives; using OpenRA.Traits; namespace OpenRA.Mods.Common.Activities { // Assumes you have Minelayer on that unit public class LayMines : Activity { readonly Minelayer minelayer; readonly AmmoPool[] ammoPools; readonly IMove movement; readonly IMoveInfo moveInfo; readonly RearmableInfo rearmableInfo; readonly MoveCooldownHelper moveCooldownHelper; List minefield; bool returnToBase; Actor rearmTarget; bool layingMine; public LayMines(Actor self, List minefield = null) { minelayer = self.Trait(); ammoPools = self.TraitsImplementing().ToArray(); movement = self.Trait(); moveInfo = self.Info.TraitInfo(); rearmableInfo = self.Info.TraitInfoOrDefault(); moveCooldownHelper = new MoveCooldownHelper(self.World, movement as Mobile) { RetryIfDestinationBlocked = true }; this.minefield = minefield; } protected override void OnFirstRun(Actor self) { minefield ??= new List { self.Location }; } CPos? NextValidCell(Actor self) { if (minefield != null) foreach (var c in minefield) if (CanLayMine(self, c)) return c; return null; } public override bool Tick(Actor self) { returnToBase = false; if (IsCanceling) { if (layingMine) foreach (var t in self.TraitsImplementing()) t.MineLayingCanceled(self, self.Location); return true; } if (layingMine) { layingMine = false; if (LayMine(self)) { if (minelayer.Info.AfterLayingDelay > 0) QueueChild(new Wait(minelayer.Info.AfterLayingDelay)); // The tick has to end now, because otherwise a next cell is picked and Move activity is queued (which will change minelayer's location in current tick). return false; } } var result = moveCooldownHelper.Tick(false); if (result != null) return result.Value; if ((minefield == null || minefield.Contains(self.Location)) && CanLayMine(self, self.Location)) { if (rearmableInfo != null && ammoPools.Any(p => p.Info.Name == minelayer.Info.AmmoPoolName && !p.HasAmmo)) { // Rearm (and possibly repair) at rearm building, then back out here to refill the minefield some more rearmTarget = self.World.Actors .Where(a => self.Owner.RelationshipWith(a.Owner) == PlayerRelationship.Ally && rearmableInfo.RearmActors.Contains(a.Info.Name)) .ClosestToWithPathFrom(self); if (rearmTarget == null) return true; // Add a CloseEnough range of 512 to the Rearm/Repair activities in order to ensure that we're at the host actor moveCooldownHelper.NotifyMoveQueued(); QueueChild(new MoveAdjacentTo(self, Target.FromActor(rearmTarget))); QueueChild(movement.MoveTo(self.World.Map.CellContaining(rearmTarget.CenterPosition), ignoreActor: rearmTarget)); QueueChild(new Resupply(self, rearmTarget, new WDist(512))); returnToBase = true; return false; } if (!StartLayingMine(self)) return false; if (minelayer.Info.PreLayDelay == 0) { if (LayMine(self) && minelayer.Info.AfterLayingDelay > 0) QueueChild(new Wait(minelayer.Info.AfterLayingDelay)); } else { layingMine = true; QueueChild(new Wait(minelayer.Info.PreLayDelay)); } return false; } var nextCell = NextValidCell(self); if (nextCell != null) { moveCooldownHelper.NotifyMoveQueued(); QueueChild(movement.MoveTo(nextCell.Value, 0)); return false; } // TODO: Return somewhere likely to be safe (near rearm building) so we're not sitting out in the minefield. return true; } public void CleanMineField(Actor self) { // Remove cells that have already been mined // or that are revealed to be unmineable. if (minefield != null) { var positionable = (IPositionable)movement; var mobile = positionable as Mobile; minefield.RemoveAll(c => self.World.ActorMap.GetActorsAt(c) .Any(a => a.Info.Name == minelayer.Info.Mine.ToLowerInvariant() && a.CanBeViewedByPlayer(self.Owner)) || ((!positionable.CanEnterCell(c, null, BlockedByActor.Immovable) || (mobile != null && !mobile.CanStayInCell(c))) && self.Owner.Shroud.IsVisible(c))); } } public override IEnumerable TargetLineNodes(Actor self) { if (returnToBase) yield return new TargetLineNode(Target.FromActor(rearmTarget), moveInfo.GetTargetLineColor()); if (minefield == null || minefield.Count == 0) yield break; var nextCell = NextValidCell(self); if (nextCell != null) yield return new TargetLineNode(Target.FromCell(self.World, nextCell.Value), minelayer.Info.TargetLineColor); if (minefield.Count > 1) foreach (var c in minefield) yield return new TargetLineNode(Target.FromCell(self.World, c), minelayer.Info.TargetLineColor, tile: minelayer.Tile); } static bool CanLayMine(Actor self, CPos p) { // If there is no unit (other than me) here, we can place a mine here return self.World.ActorMap.GetActorsAt(p).All(a => a == self); } bool StartLayingMine(Actor self) { if (ammoPools != null) { var pool = ammoPools.FirstOrDefault(x => x.Info.Name == minelayer.Info.AmmoPoolName); if (pool == null) return false; if (pool.CurrentAmmoCount < minelayer.Info.AmmoUsage) return false; } foreach (var t in self.TraitsImplementing()) t.MineLaying(self, self.Location); return true; } bool LayMine(Actor self) { if (ammoPools != null) { var pool = ammoPools.FirstOrDefault(x => x.Info.Name == minelayer.Info.AmmoPoolName); if (pool == null) return false; if (!pool.TakeAmmo(self, minelayer.Info.AmmoUsage)) return false; } minefield.Remove(self.Location); self.World.AddFrameEndTask(w => { if (!CanLayMine(self, self.Location)) return; var mine = w.CreateActor(minelayer.Info.Mine, new TypeDictionary { new LocationInit(self.Location), new OwnerInit(self.Owner), }); foreach (var t in self.TraitsImplementing()) t.MineLaid(self, mine); }); return true; } } }