Refactor TakeOff

- Make it self-contained by moving actual take-off
  from 'Fly' to this
- Make 'moveToRallyPoint' a simple boolean
- Make AttackMove to rally point a child activity
- Make TakeOff uninterruptible
This commit is contained in:
reaperrr
2019-05-23 18:45:37 +02:00
committed by Paul Chote
parent 4f8f8cfb9d
commit ce3d7c98ad
4 changed files with 74 additions and 48 deletions

View File

@@ -26,7 +26,6 @@ namespace OpenRA.Mods.Common.Activities
Target target; Target target;
Target lastVisibleTarget; Target lastVisibleTarget;
bool useLastVisibleTarget; bool useLastVisibleTarget;
bool soundPlayed;
public Fly(Actor self, Target t, WPos? initialTargetPosition = null, Color? targetLineColor = null) public Fly(Actor self, Target t, WPos? initialTargetPosition = null, Color? targetLineColor = null)
{ {
@@ -103,6 +102,13 @@ namespace OpenRA.Mods.Common.Activities
public override Activity Tick(Actor self) public override Activity Tick(Actor self)
{ {
if (ChildActivity != null)
{
ChildActivity = ActivityUtils.RunActivity(self, ChildActivity);
if (ChildActivity != null)
return this;
}
// Refuse to take off if it would land immediately again. // Refuse to take off if it would land immediately again.
if (aircraft.ForceLanding) if (aircraft.ForceLanding)
Cancel(self); Cancel(self);
@@ -110,6 +116,13 @@ namespace OpenRA.Mods.Common.Activities
if (IsCanceling) if (IsCanceling)
return NextActivity; return NextActivity;
var dat = self.World.Map.DistanceAboveTerrain(aircraft.CenterPosition);
if (dat <= aircraft.LandAltitude)
{
QueueChild(self, new TakeOff(self, target), true);
return this;
}
bool targetIsHiddenActor; bool targetIsHiddenActor;
target = target.Recalculate(self.Owner, out targetIsHiddenActor); target = target.Recalculate(self.Owner, out targetIsHiddenActor);
if (!targetIsHiddenActor && target.Type == TargetType.Actor) if (!targetIsHiddenActor && target.Type == TargetType.Actor)
@@ -126,25 +139,6 @@ namespace OpenRA.Mods.Common.Activities
if (useLastVisibleTarget && !lastVisibleTarget.IsValidFor(self)) if (useLastVisibleTarget && !lastVisibleTarget.IsValidFor(self))
return NextActivity; return NextActivity;
var dat = self.World.Map.DistanceAboveTerrain(aircraft.CenterPosition);
// We are taking off, so remove influence in ground cells.
if (dat <= aircraft.LandAltitude)
{
if (!soundPlayed && aircraft.Info.TakeoffSounds.Length > 0)
{
Game.Sound.Play(SoundType.World, aircraft.Info.TakeoffSounds, self.World, aircraft.CenterPosition);
soundPlayed = true;
}
aircraft.RemoveInfluence();
}
// If we're a VTOL, rise before flying forward
if (aircraft.Info.VTOL)
if (VerticalTakeOffOrLandTick(self, aircraft, aircraft.Facing, aircraft.Info.CruiseAltitude))
return this;
var checkTarget = useLastVisibleTarget ? lastVisibleTarget : target; var checkTarget = useLastVisibleTarget ? lastVisibleTarget : target;
var delta = checkTarget.CenterPosition - self.CenterPosition; var delta = checkTarget.CenterPosition - self.CenterPosition;
var desiredFacing = delta.HorizontalLengthSquared != 0 ? delta.Yaw.Facing : aircraft.Facing; var desiredFacing = delta.HorizontalLengthSquared != 0 ? delta.Yaw.Facing : aircraft.Facing;
@@ -188,10 +182,7 @@ namespace OpenRA.Mods.Common.Activities
return NextActivity; return NextActivity;
} }
// Don't turn until we've reached the cruise altitude if (!aircraft.Info.CanHover)
if (dat < aircraft.Info.CruiseAltitude)
desiredFacing = aircraft.Facing;
else if (!aircraft.Info.CanHover)
{ {
// Using the turn rate, compute a hypothetical circle traced by a continuous turn. // Using the turn rate, compute a hypothetical circle traced by a continuous turn.
// If it contains the destination point, it's unreachable without more complex manuvering. // If it contains the destination point, it's unreachable without more complex manuvering.

View File

@@ -12,6 +12,7 @@
using System; using System;
using OpenRA.Activities; using OpenRA.Activities;
using OpenRA.Mods.Common.Traits; using OpenRA.Mods.Common.Traits;
using OpenRA.Traits;
namespace OpenRA.Mods.Common.Activities namespace OpenRA.Mods.Common.Activities
{ {
@@ -19,23 +20,51 @@ namespace OpenRA.Mods.Common.Activities
{ {
readonly Aircraft aircraft; readonly Aircraft aircraft;
readonly IMove move; readonly IMove move;
Func<Actor, Activity, CPos, bool> moveToRallyPoint; readonly Target target;
bool moveToRallyPoint;
public TakeOff(Actor self) public TakeOff(Actor self, Target target)
{ {
aircraft = self.Trait<Aircraft>(); aircraft = self.Trait<Aircraft>();
move = self.Trait<IMove>(); move = self.Trait<IMove>();
moveToRallyPoint = (actor, activity, pos) => NextActivity == null; this.target = target;
} }
public TakeOff(Actor self, Func<Actor, Activity, CPos, bool> moveToRallyPoint) public TakeOff(Actor self)
: this(self) : this(self, Target.FromCell(self.World, self.Location))
{ {
this.moveToRallyPoint = moveToRallyPoint; var host = aircraft.GetActorBelow();
var rp = host != null ? host.TraitOrDefault<RallyPoint>() : null;
var rallyPointDestination = rp != null ? rp.Location :
(host != null ? self.World.Map.CellContaining(host.CenterPosition) : self.Location);
this.target = Target.FromCell(self.World, rallyPointDestination);
moveToRallyPoint = NextActivity == null && rallyPointDestination != self.Location;
}
protected override void OnFirstRun(Actor self)
{
if (aircraft.ForceLanding)
return;
// We are taking off, so remove reservation and influence in ground cells.
aircraft.UnReserve();
aircraft.RemoveInfluence();
if (aircraft.Info.TakeoffSounds.Length > 0)
Game.Sound.Play(SoundType.World, aircraft.Info.TakeoffSounds, self.World, aircraft.CenterPosition);
} }
public override Activity Tick(Actor self) public override Activity Tick(Actor self)
{ {
if (ChildActivity != null)
{
ChildActivity = ActivityUtils.RunActivity(self, ChildActivity);
if (ChildActivity != null)
return this;
}
// Refuse to take off if it would land immediately again. // Refuse to take off if it would land immediately again.
if (aircraft.ForceLanding) if (aircraft.ForceLanding)
{ {
@@ -43,17 +72,29 @@ namespace OpenRA.Mods.Common.Activities
return NextActivity; return NextActivity;
} }
aircraft.UnReserve(); var dat = self.World.Map.DistanceAboveTerrain(aircraft.CenterPosition);
if (dat < aircraft.Info.CruiseAltitude)
{
// If we're a VTOL, rise before flying forward
if (aircraft.Info.VTOL)
{
Fly.VerticalTakeOffOrLandTick(self, aircraft, aircraft.Facing, aircraft.Info.CruiseAltitude);
return this;
}
else
{
Fly.FlyTick(self, aircraft, aircraft.Facing, aircraft.Info.CruiseAltitude);
return this;
}
}
var host = aircraft.GetActorBelow(); // Checking for NextActivity == null again in case another activity was queued while taking off
var hasHost = host != null; if (moveToRallyPoint && NextActivity == null)
var rp = hasHost ? host.TraitOrDefault<RallyPoint>() : null; {
QueueChild(self, new AttackMoveActivity(self, () => move.MoveToTarget(self, target)), true);
var destination = rp != null ? rp.Location : moveToRallyPoint = false;
(hasHost ? self.World.Map.CellContaining(host.CenterPosition) : self.Location); return this;
}
if (moveToRallyPoint(self, this, destination))
return new AttackMoveActivity(self, () => move.MoveTo(destination, 1));
return NextActivity; return NextActivity;
} }

View File

@@ -120,7 +120,7 @@ namespace OpenRA.Mods.Common.Activities
{ {
aircraft.AllowYieldingReservation(); aircraft.AllowYieldingReservation();
if (aircraft.Info.TakeOffOnResupply) if (aircraft.Info.TakeOffOnResupply)
Queue(self, new TakeOff(self, (a, b, c) => NextActivity == null && b.NextActivity == null)); Queue(self, new TakeOff(self));
} }
return NextActivity; return NextActivity;

View File

@@ -795,13 +795,7 @@ namespace OpenRA.Mods.Common.Traits
public Activity MoveToTarget(Actor self, Target target, public Activity MoveToTarget(Actor self, Target target,
WPos? initialTargetPosition = null, Color? targetLineColor = null) WPos? initialTargetPosition = null, Color? targetLineColor = null)
{ {
if (!Info.CanHover) return new Fly(self, target, initialTargetPosition, targetLineColor);
return new Fly(self, target, WDist.FromCells(3), WDist.FromCells(5),
initialTargetPosition, targetLineColor);
return ActivityUtils.SequenceActivities(self,
new Fly(self, target, initialTargetPosition, targetLineColor),
new Turn(self, Info.InitialFacing));
} }
public Activity MoveIntoTarget(Actor self, Target target) public Activity MoveIntoTarget(Actor self, Target target)