Abstract docking logic from Harvester and Refinery
This commit is contained in:
committed by
Matthias Mailänder
parent
da16e4ed99
commit
d0974cfdd2
@@ -24,8 +24,6 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
readonly HarvesterInfo harvInfo;
|
readonly HarvesterInfo harvInfo;
|
||||||
readonly Mobile mobile;
|
readonly Mobile mobile;
|
||||||
readonly ResourceClaimLayer claimLayer;
|
readonly ResourceClaimLayer claimLayer;
|
||||||
|
|
||||||
Actor deliverActor;
|
|
||||||
CPos? orderLocation;
|
CPos? orderLocation;
|
||||||
CPos? lastHarvestedCell;
|
CPos? lastHarvestedCell;
|
||||||
bool hasDeliveredLoad;
|
bool hasDeliveredLoad;
|
||||||
@@ -34,19 +32,14 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
|
|
||||||
public bool LastSearchFailed { get; private set; }
|
public bool LastSearchFailed { get; private set; }
|
||||||
|
|
||||||
public FindAndDeliverResources(Actor self, Actor deliverActor = null)
|
public FindAndDeliverResources(Actor self, CPos? orderLocation = null)
|
||||||
{
|
{
|
||||||
harv = self.Trait<Harvester>();
|
harv = self.Trait<Harvester>();
|
||||||
harvInfo = self.Info.TraitInfo<HarvesterInfo>();
|
harvInfo = self.Info.TraitInfo<HarvesterInfo>();
|
||||||
mobile = self.Trait<Mobile>();
|
mobile = self.Trait<Mobile>();
|
||||||
claimLayer = self.World.WorldActor.Trait<ResourceClaimLayer>();
|
claimLayer = self.World.WorldActor.Trait<ResourceClaimLayer>();
|
||||||
this.deliverActor = deliverActor;
|
if (orderLocation.HasValue)
|
||||||
}
|
this.orderLocation = orderLocation.Value;
|
||||||
|
|
||||||
public FindAndDeliverResources(Actor self, CPos orderLocation)
|
|
||||||
: this(self, null)
|
|
||||||
{
|
|
||||||
this.orderLocation = orderLocation;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void OnFirstRun(Actor self)
|
protected override void OnFirstRun(Actor self)
|
||||||
@@ -63,14 +56,6 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
if (harv.IsFull)
|
if (harv.IsFull)
|
||||||
QueueChild(new MoveToDock(self));
|
QueueChild(new MoveToDock(self));
|
||||||
}
|
}
|
||||||
|
|
||||||
// If an explicit "deliver" order is given, the harvester goes immediately to the refinery.
|
|
||||||
if (deliverActor != null)
|
|
||||||
{
|
|
||||||
QueueChild(new MoveToDock(self, deliverActor));
|
|
||||||
hasDeliveredLoad = true;
|
|
||||||
deliverActor = null;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public override bool Tick(Actor self)
|
public override bool Tick(Actor self)
|
||||||
@@ -92,9 +77,12 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
// Are we full or have nothing more to gather? Deliver resources.
|
// Are we full or have nothing more to gather? Deliver resources.
|
||||||
if (harv.IsFull || (!harv.IsEmpty && LastSearchFailed))
|
if (harv.IsFull || (!harv.IsEmpty && LastSearchFailed))
|
||||||
{
|
{
|
||||||
|
// If we are reserved it means docking was already initiated and we should wait.
|
||||||
|
if (harv.DockClientManager.ReservedHost != null)
|
||||||
|
return false;
|
||||||
|
|
||||||
QueueChild(new MoveToDock(self));
|
QueueChild(new MoveToDock(self));
|
||||||
hasDeliveredLoad = true;
|
hasDeliveredLoad = true;
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// After a failed search, wait and sit still for a bit before searching again.
|
// After a failed search, wait and sit still for a bit before searching again.
|
||||||
@@ -128,13 +116,13 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
// of the refinery entrance.
|
// of the refinery entrance.
|
||||||
if (LastSearchFailed)
|
if (LastSearchFailed)
|
||||||
{
|
{
|
||||||
var lastproc = harv.LastLinkedProc ?? harv.LinkedProc;
|
var lastproc = harv.DockClientManager.LastReservedHost;
|
||||||
if (lastproc != null && !lastproc.Disposed)
|
if (lastproc != null)
|
||||||
{
|
{
|
||||||
var deliveryLoc = lastproc.Trait<IAcceptResources>().DeliveryPosition;
|
var deliveryLoc = self.World.Map.CellContaining(lastproc.DockPosition);
|
||||||
if (self.CenterPosition == deliveryLoc && harv.IsEmpty)
|
if (self.Location == deliveryLoc && harv.IsEmpty)
|
||||||
{
|
{
|
||||||
var unblockCell = self.World.Map.CellContaining(deliveryLoc) + harv.Info.UnblockCell;
|
var unblockCell = deliveryLoc + harv.Info.UnblockCell;
|
||||||
var moveTo = mobile.NearestMoveableCell(unblockCell, 1, 5);
|
var moveTo = mobile.NearestMoveableCell(unblockCell, 1, 5);
|
||||||
QueueChild(mobile.MoveTo(moveTo, 1));
|
QueueChild(mobile.MoveTo(moveTo, 1));
|
||||||
}
|
}
|
||||||
@@ -171,14 +159,31 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Determine where to search from and how far to search:
|
// Determine where to search from and how far to search:
|
||||||
var procLoc = GetSearchFromProcLocation();
|
// Prioritise search by these locations in this order: lastHarvestedCell -> lastLinkedDock -> self.
|
||||||
var searchFromLoc = lastHarvestedCell ?? procLoc ?? self.Location;
|
CPos searchFromLoc;
|
||||||
var searchRadius = lastHarvestedCell.HasValue ? harvInfo.SearchFromHarvesterRadius : harvInfo.SearchFromProcRadius;
|
int searchRadius;
|
||||||
|
WPos? dockPos = null;
|
||||||
|
if (lastHarvestedCell.HasValue)
|
||||||
|
{
|
||||||
|
searchRadius = harvInfo.SearchFromHarvesterRadius;
|
||||||
|
searchFromLoc = lastHarvestedCell.Value;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
searchRadius = harvInfo.SearchFromProcRadius;
|
||||||
|
var dock = harv.DockClientManager.LastReservedHost;
|
||||||
|
if (dock != null)
|
||||||
|
{
|
||||||
|
dockPos = dock.DockPosition;
|
||||||
|
searchFromLoc = self.World.Map.CellContaining(dockPos.Value);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
searchFromLoc = self.Location;
|
||||||
|
}
|
||||||
|
|
||||||
var searchRadiusSquared = searchRadius * searchRadius;
|
var searchRadiusSquared = searchRadius * searchRadius;
|
||||||
|
|
||||||
var map = self.World.Map;
|
var map = self.World.Map;
|
||||||
var procPos = procLoc.HasValue ? (WPos?)map.CenterOfCell(procLoc.Value) : null;
|
|
||||||
var harvPos = self.CenterPosition;
|
var harvPos = self.CenterPosition;
|
||||||
|
|
||||||
// Find any harvestable resources:
|
// Find any harvestable resources:
|
||||||
@@ -196,19 +201,19 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
|
|
||||||
// Add a cost modifier to harvestable cells to prefer resources that are closer to the refinery.
|
// Add a cost modifier to harvestable cells to prefer resources that are closer to the refinery.
|
||||||
// This reduces the tendency for harvesters to move in straight lines
|
// This reduces the tendency for harvesters to move in straight lines
|
||||||
if (procPos.HasValue && harvInfo.ResourceRefineryDirectionPenalty > 0 && harv.CanHarvestCell(loc))
|
if (dockPos.HasValue && harvInfo.ResourceRefineryDirectionPenalty > 0 && harv.CanHarvestCell(loc))
|
||||||
{
|
{
|
||||||
var pos = map.CenterOfCell(loc);
|
var pos = map.CenterOfCell(loc);
|
||||||
|
|
||||||
// Calculate harv-cell-refinery angle (cosine rule)
|
// Calculate harv-cell-refinery angle (cosine rule)
|
||||||
var b = pos - procPos.Value;
|
var b = pos - dockPos.Value;
|
||||||
|
|
||||||
if (b != WVec.Zero)
|
if (b != WVec.Zero)
|
||||||
{
|
{
|
||||||
var c = pos - harvPos;
|
var c = pos - harvPos;
|
||||||
if (c != WVec.Zero)
|
if (c != WVec.Zero)
|
||||||
{
|
{
|
||||||
var a = harvPos - procPos.Value;
|
var a = harvPos - dockPos.Value;
|
||||||
var cosA = (int)(512 * (b.LengthSquared + c.LengthSquared - a.LengthSquared) / b.Length / c.Length);
|
var cosA = (int)(512 * (b.LengthSquared + c.LengthSquared - a.LengthSquared) / b.Length / c.Length);
|
||||||
|
|
||||||
// Cost modifier varies between 0 and ResourceRefineryDirectionPenalty
|
// Cost modifier varies between 0 and ResourceRefineryDirectionPenalty
|
||||||
@@ -239,19 +244,12 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
|
|
||||||
if (orderLocation != null)
|
if (orderLocation != null)
|
||||||
yield return new TargetLineNode(Target.FromCell(self.World, orderLocation.Value), harvInfo.HarvestLineColor);
|
yield return new TargetLineNode(Target.FromCell(self.World, orderLocation.Value), harvInfo.HarvestLineColor);
|
||||||
else if (deliverActor != null)
|
else
|
||||||
yield return new TargetLineNode(Target.FromActor(deliverActor), harvInfo.DeliverLineColor);
|
|
||||||
}
|
|
||||||
|
|
||||||
CPos? GetSearchFromProcLocation()
|
|
||||||
{
|
{
|
||||||
if (harv.LastLinkedProc != null && !harv.LastLinkedProc.IsDead && harv.LastLinkedProc.IsInWorld)
|
var manager = harv.DockClientManager;
|
||||||
return harv.LastLinkedProc.World.Map.CellContaining(harv.LastLinkedProc.Trait<IAcceptResources>().DeliveryPosition);
|
if (manager.ReservedHostActor != null)
|
||||||
|
yield return new TargetLineNode(Target.FromActor(manager.ReservedHostActor), manager.DockLineColor);
|
||||||
if (harv.LinkedProc != null && !harv.LinkedProc.IsDead && harv.LinkedProc.IsInWorld)
|
}
|
||||||
return harv.LinkedProc.World.Map.CellContaining(harv.LinkedProc.Trait<IAcceptResources>().DeliveryPosition);
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -24,12 +24,12 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
{
|
{
|
||||||
protected enum DockingState { Wait, Drag, Dock, Loop, Undock, Complete }
|
protected enum DockingState { Wait, Drag, Dock, Loop, Undock, Complete }
|
||||||
|
|
||||||
protected readonly Actor RefineryActor;
|
protected readonly Actor DockHostActor;
|
||||||
|
protected readonly IDockHost DockHost;
|
||||||
protected readonly WithDockingOverlay DockHostSpriteOverlay;
|
protected readonly WithDockingOverlay DockHostSpriteOverlay;
|
||||||
protected readonly Harvester Harv;
|
protected readonly DockClientManager DockClient;
|
||||||
protected readonly IDockClientBody DockClientBody;
|
protected readonly IDockClientBody DockClientBody;
|
||||||
protected readonly bool IsDragRequired;
|
protected readonly bool IsDragRequired;
|
||||||
protected readonly WVec DragOffset;
|
|
||||||
protected readonly int DragLength;
|
protected readonly int DragLength;
|
||||||
protected readonly WPos StartDrag;
|
protected readonly WPos StartDrag;
|
||||||
protected readonly WPos EndDrag;
|
protected readonly WPos EndDrag;
|
||||||
@@ -41,20 +41,30 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
|
|
||||||
bool dockInitiated = false;
|
bool dockInitiated = false;
|
||||||
|
|
||||||
public GenericDockSequence(Actor self, Actor refineryActor, Refinery refinery)
|
public GenericDockSequence(Actor self, DockClientManager client, Actor hostActor, IDockHost host)
|
||||||
{
|
{
|
||||||
dockingState = DockingState.Drag;
|
dockingState = DockingState.Drag;
|
||||||
RefineryActor = refineryActor;
|
|
||||||
DockHostSpriteOverlay = refineryActor.TraitOrDefault<WithDockingOverlay>();
|
DockClient = client;
|
||||||
IsDragRequired = refinery.IsDragRequired;
|
|
||||||
DragOffset = refinery.DragOffset;
|
|
||||||
DragLength = refinery.DragLength;
|
|
||||||
Harv = self.Trait<Harvester>();
|
|
||||||
DockClientBody = self.TraitOrDefault<IDockClientBody>();
|
DockClientBody = self.TraitOrDefault<IDockClientBody>();
|
||||||
StartDrag = self.CenterPosition;
|
|
||||||
EndDrag = refineryActor.CenterPosition + DragOffset;
|
|
||||||
notifyDockClients = self.TraitsImplementing<INotifyDockClient>().ToArray();
|
notifyDockClients = self.TraitsImplementing<INotifyDockClient>().ToArray();
|
||||||
notifyDockHosts = refineryActor.TraitsImplementing<INotifyDockHost>().ToArray();
|
|
||||||
|
DockHost = host;
|
||||||
|
DockHostActor = hostActor;
|
||||||
|
DockHostSpriteOverlay = hostActor.TraitOrDefault<WithDockingOverlay>();
|
||||||
|
notifyDockHosts = hostActor.TraitsImplementing<INotifyDockHost>().ToArray();
|
||||||
|
|
||||||
|
if (host is IDockHostDrag sequence)
|
||||||
|
{
|
||||||
|
IsDragRequired = sequence.IsDragRequired;
|
||||||
|
DragLength = sequence.DragLength;
|
||||||
|
StartDrag = self.CenterPosition;
|
||||||
|
EndDrag = hostActor.CenterPosition + sequence.DragOffset;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
IsDragRequired = false;
|
||||||
|
|
||||||
|
QueueChild(new Wait(host.DockWait));
|
||||||
}
|
}
|
||||||
|
|
||||||
public override bool Tick(Actor self)
|
public override bool Tick(Actor self)
|
||||||
@@ -65,8 +75,11 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
return false;
|
return false;
|
||||||
|
|
||||||
case DockingState.Drag:
|
case DockingState.Drag:
|
||||||
if (IsCanceling || !RefineryActor.IsInWorld || RefineryActor.IsDead || Harv.IsTraitDisabled)
|
if (IsCanceling || DockHostActor.IsDead || !DockHostActor.IsInWorld || !DockClient.CanDockAt(DockHostActor, DockHost, false, true))
|
||||||
|
{
|
||||||
|
DockClient.UnreserveHost();
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
dockingState = DockingState.Dock;
|
dockingState = DockingState.Dock;
|
||||||
if (IsDragRequired)
|
if (IsDragRequired)
|
||||||
@@ -75,10 +88,12 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
return false;
|
return false;
|
||||||
|
|
||||||
case DockingState.Dock:
|
case DockingState.Dock:
|
||||||
if (!IsCanceling && RefineryActor.IsInWorld && !RefineryActor.IsDead && !Harv.IsTraitDisabled)
|
if (!IsCanceling && !DockHostActor.IsDead && DockHostActor.IsInWorld && DockClient.CanDockAt(DockHostActor, DockHost, false, true))
|
||||||
{
|
{
|
||||||
dockInitiated = true;
|
dockInitiated = true;
|
||||||
PlayDockAnimations(self);
|
PlayDockAnimations(self);
|
||||||
|
DockHost.OnDockStarted(DockHostActor, self, DockClient);
|
||||||
|
DockClient.OnDockStarted(self, DockHostActor, DockHost);
|
||||||
NotifyDocked(self);
|
NotifyDocked(self);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -87,7 +102,7 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
return false;
|
return false;
|
||||||
|
|
||||||
case DockingState.Loop:
|
case DockingState.Loop:
|
||||||
if (IsCanceling || !RefineryActor.IsInWorld || RefineryActor.IsDead || Harv.IsTraitDisabled || Harv.TickUnload(self, RefineryActor))
|
if (IsCanceling || DockHostActor.IsDead || !DockHostActor.IsInWorld || DockClient.OnDockTick(self, DockHostActor, DockHost))
|
||||||
dockingState = DockingState.Undock;
|
dockingState = DockingState.Undock;
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
@@ -101,8 +116,8 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
return false;
|
return false;
|
||||||
|
|
||||||
case DockingState.Complete:
|
case DockingState.Complete:
|
||||||
Harv.LastLinkedProc = Harv.LinkedProc;
|
DockHost.OnDockCompleted(DockHostActor, self, DockClient);
|
||||||
Harv.LinkProc(null);
|
DockClient.OnDockCompleted(self, DockHostActor, DockHost);
|
||||||
NotifyUndocked(self);
|
NotifyUndocked(self);
|
||||||
if (IsDragRequired)
|
if (IsDragRequired)
|
||||||
QueueChild(new Drag(self, EndDrag, StartDrag, DragLength));
|
QueueChild(new Drag(self, EndDrag, StartDrag, DragLength));
|
||||||
@@ -145,7 +160,7 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
|
|
||||||
public virtual void PlayUndockAnimations(Actor self)
|
public virtual void PlayUndockAnimations(Actor self)
|
||||||
{
|
{
|
||||||
if (RefineryActor.IsInWorld && !RefineryActor.IsDead && DockHostSpriteOverlay != null && !DockHostSpriteOverlay.Visible)
|
if (DockHostActor.IsInWorld && !DockHostActor.IsDead && DockHostSpriteOverlay != null && !DockHostSpriteOverlay.Visible)
|
||||||
{
|
{
|
||||||
dockingState = DockingState.Wait;
|
dockingState = DockingState.Wait;
|
||||||
DockHostSpriteOverlay.Visible = true;
|
DockHostSpriteOverlay.Visible = true;
|
||||||
@@ -176,30 +191,30 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
void NotifyDocked(Actor self)
|
void NotifyDocked(Actor self)
|
||||||
{
|
{
|
||||||
foreach (var nd in notifyDockClients)
|
foreach (var nd in notifyDockClients)
|
||||||
nd.Docked(self, RefineryActor);
|
nd.Docked(self, DockHostActor);
|
||||||
|
|
||||||
foreach (var nd in notifyDockHosts)
|
foreach (var nd in notifyDockHosts)
|
||||||
nd.Docked(RefineryActor, self);
|
nd.Docked(DockHostActor, self);
|
||||||
}
|
}
|
||||||
|
|
||||||
void NotifyUndocked(Actor self)
|
void NotifyUndocked(Actor self)
|
||||||
{
|
{
|
||||||
foreach (var nd in notifyDockClients)
|
foreach (var nd in notifyDockClients)
|
||||||
nd.Undocked(self, RefineryActor);
|
nd.Undocked(self, DockHostActor);
|
||||||
|
|
||||||
if (RefineryActor.IsInWorld && !RefineryActor.IsDead)
|
if (DockHostActor.IsInWorld && !DockHostActor.IsDead)
|
||||||
foreach (var nd in notifyDockHosts)
|
foreach (var nd in notifyDockHosts)
|
||||||
nd.Undocked(RefineryActor, self);
|
nd.Undocked(DockHostActor, self);
|
||||||
}
|
}
|
||||||
|
|
||||||
public override IEnumerable<Target> GetTargets(Actor self)
|
public override IEnumerable<Target> GetTargets(Actor self)
|
||||||
{
|
{
|
||||||
yield return Target.FromActor(RefineryActor);
|
yield return Target.FromActor(DockHostActor);
|
||||||
}
|
}
|
||||||
|
|
||||||
public override IEnumerable<TargetLineNode> TargetLineNodes(Actor self)
|
public override IEnumerable<TargetLineNode> TargetLineNodes(Actor self)
|
||||||
{
|
{
|
||||||
yield return new TargetLineNode(Target.FromActor(RefineryActor), Color.Green);
|
yield return new TargetLineNode(Target.FromActor(DockHostActor), Color.Green);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,66 +19,71 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
{
|
{
|
||||||
public class MoveToDock : Activity
|
public class MoveToDock : Activity
|
||||||
{
|
{
|
||||||
readonly IMove movement;
|
readonly DockClientManager dockClient;
|
||||||
readonly Harvester harv;
|
Actor dockHostActor;
|
||||||
readonly Actor targetActor;
|
IDockHost dockHost;
|
||||||
readonly INotifyHarvesterAction[] notifyHarvesterActions;
|
readonly INotifyHarvesterAction[] notifyHarvesterActions;
|
||||||
|
|
||||||
Actor proc;
|
public MoveToDock(Actor self, Actor dockHostActor = null, IDockHost dockHost = null)
|
||||||
|
|
||||||
public MoveToDock(Actor self, Actor targetActor = null)
|
|
||||||
{
|
{
|
||||||
movement = self.Trait<IMove>();
|
dockClient = self.Trait<DockClientManager>();
|
||||||
harv = self.Trait<Harvester>();
|
this.dockHostActor = dockHostActor;
|
||||||
this.targetActor = targetActor;
|
this.dockHost = dockHost;
|
||||||
notifyHarvesterActions = self.TraitsImplementing<INotifyHarvesterAction>().ToArray();
|
notifyHarvesterActions = self.TraitsImplementing<INotifyHarvesterAction>().ToArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void OnFirstRun(Actor self)
|
|
||||||
{
|
|
||||||
if (targetActor != null && targetActor.IsInWorld)
|
|
||||||
harv.LinkProc(targetActor);
|
|
||||||
}
|
|
||||||
|
|
||||||
public override bool Tick(Actor self)
|
public override bool Tick(Actor self)
|
||||||
{
|
{
|
||||||
if (harv.IsTraitDisabled)
|
|
||||||
Cancel(self, true);
|
|
||||||
|
|
||||||
if (IsCanceling)
|
if (IsCanceling)
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
// Find the nearest best refinery if not explicitly ordered to a specific refinery:
|
if (dockClient.IsTraitDisabled)
|
||||||
if (harv.LinkedProc == null || !harv.LinkedProc.IsInWorld)
|
|
||||||
harv.ChooseNewProc(self, null);
|
|
||||||
|
|
||||||
// No refineries exist; check again after delay defined in Harvester.
|
|
||||||
if (harv.LinkedProc == null)
|
|
||||||
{
|
{
|
||||||
QueueChild(new Wait(harv.Info.SearchForDeliveryBuildingDelay));
|
Cancel(self, true);
|
||||||
return false;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
proc = harv.LinkedProc;
|
// Find the nearest DockHost if not explicitly ordered to a specific dock.
|
||||||
var iao = proc.Trait<IAcceptResources>();
|
if (dockHost == null || !dockHost.IsEnabledAndInWorld)
|
||||||
|
{
|
||||||
|
var host = dockClient.ClosestDock(null);
|
||||||
|
if (host.HasValue)
|
||||||
|
{
|
||||||
|
dockHost = host.Value.Trait;
|
||||||
|
dockHostActor = host.Value.Actor;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// No docks exist; check again after delay defined in dockClient.
|
||||||
|
QueueChild(new Wait(dockClient.Info.SearchForDockDelay));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (self.CenterPosition != iao.DeliveryPosition)
|
if (dockClient.ReserveHost(dockHostActor, dockHost))
|
||||||
|
{
|
||||||
|
if (dockHost.QueueMoveActivity(this, dockHostActor, self, dockClient))
|
||||||
{
|
{
|
||||||
foreach (var n in notifyHarvesterActions)
|
foreach (var n in notifyHarvesterActions)
|
||||||
n.MovingToRefinery(self, proc);
|
n.MovingToRefinery(self, dockHostActor);
|
||||||
|
|
||||||
var target = Target.FromActor(proc);
|
|
||||||
QueueChild(movement.MoveOntoTarget(self, target, iao.DeliveryPosition - proc.CenterPosition, iao.DeliveryAngle));
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
QueueChild(new Wait(10));
|
dockHost.QueueDockActivity(this, dockHostActor, self, dockClient);
|
||||||
iao.OnDock(self, this);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// The dock explicitely chosen by the user is currently occupied. Wait and check again.
|
||||||
|
QueueChild(new Wait(dockClient.Info.SearchForDockDelay));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public override void Cancel(Actor self, bool keepQueue = false)
|
public override void Cancel(Actor self, bool keepQueue = false)
|
||||||
{
|
{
|
||||||
|
dockClient.UnreserveHost();
|
||||||
foreach (var n in notifyHarvesterActions)
|
foreach (var n in notifyHarvesterActions)
|
||||||
n.MovementCancelled(self);
|
n.MovementCancelled(self);
|
||||||
|
|
||||||
@@ -87,10 +92,13 @@ namespace OpenRA.Mods.Common.Activities
|
|||||||
|
|
||||||
public override IEnumerable<TargetLineNode> TargetLineNodes(Actor self)
|
public override IEnumerable<TargetLineNode> TargetLineNodes(Actor self)
|
||||||
{
|
{
|
||||||
if (proc != null)
|
if (dockHostActor != null)
|
||||||
yield return new TargetLineNode(Target.FromActor(proc), harv.Info.DeliverLineColor);
|
yield return new TargetLineNode(Target.FromActor(dockHostActor), dockClient.DockLineColor);
|
||||||
else
|
else
|
||||||
yield return new TargetLineNode(Target.FromActor(harv.LinkedProc), harv.Info.DeliverLineColor);
|
{
|
||||||
|
if (dockClient.ReservedHostActor != null)
|
||||||
|
yield return new TargetLineNode(Target.FromActor(dockClient.ReservedHostActor), dockClient.DockLineColor);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,32 +12,14 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using OpenRA.Activities;
|
|
||||||
using OpenRA.Mods.Common.Activities;
|
|
||||||
using OpenRA.Mods.Common.Effects;
|
using OpenRA.Mods.Common.Effects;
|
||||||
using OpenRA.Mods.Common.Traits.Render;
|
using OpenRA.Mods.Common.Traits.Render;
|
||||||
using OpenRA.Primitives;
|
|
||||||
using OpenRA.Traits;
|
using OpenRA.Traits;
|
||||||
|
|
||||||
namespace OpenRA.Mods.Common.Traits
|
namespace OpenRA.Mods.Common.Traits
|
||||||
{
|
{
|
||||||
public class RefineryInfo : TraitInfo, Requires<WithSpriteBodyInfo>, IAcceptResourcesInfo
|
public class RefineryInfo : TraitInfo, Requires<WithSpriteBodyInfo>, Requires<IDockHostInfo>
|
||||||
{
|
{
|
||||||
[Desc("Actual harvester facing when docking.")]
|
|
||||||
public readonly WAngle DockAngle = WAngle.Zero;
|
|
||||||
|
|
||||||
[Desc("Docking cell relative to top-left cell.")]
|
|
||||||
public readonly CVec DockOffset = CVec.Zero;
|
|
||||||
|
|
||||||
[Desc("Does the refinery require the harvester to be dragged in?")]
|
|
||||||
public readonly bool IsDragRequired = false;
|
|
||||||
|
|
||||||
[Desc("Vector by which the harvester will be dragged when docking.")]
|
|
||||||
public readonly WVec DragOffset = WVec.Zero;
|
|
||||||
|
|
||||||
[Desc("In how many steps to perform the dragging?")]
|
|
||||||
public readonly int DragLength = 0;
|
|
||||||
|
|
||||||
[Desc("Store resources in silos. Adds cash directly without storing if set to false.")]
|
[Desc("Store resources in silos. Adds cash directly without storing if set to false.")]
|
||||||
public readonly bool UseStorage = true;
|
public readonly bool UseStorage = true;
|
||||||
|
|
||||||
@@ -50,33 +32,16 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
public override object Create(ActorInitializer init) { return new Refinery(init.Self, this); }
|
public override object Create(ActorInitializer init) { return new Refinery(init.Self, this); }
|
||||||
}
|
}
|
||||||
|
|
||||||
public class Refinery : INotifyCreated, ITick, IAcceptResources, INotifySold, INotifyCapture,
|
public class Refinery : IAcceptResources, INotifyCreated, ITick, INotifyOwnerChanged
|
||||||
INotifyOwnerChanged, ISync, INotifyActorDisposing
|
|
||||||
{
|
{
|
||||||
readonly Actor self;
|
|
||||||
readonly RefineryInfo info;
|
readonly RefineryInfo info;
|
||||||
PlayerResources playerResources;
|
PlayerResources playerResources;
|
||||||
IEnumerable<int> resourceValueModifiers;
|
IEnumerable<int> resourceValueModifiers;
|
||||||
|
|
||||||
int currentDisplayTick = 0;
|
int currentDisplayTick = 0;
|
||||||
int currentDisplayValue = 0;
|
int currentDisplayValue = 0;
|
||||||
|
|
||||||
[Sync]
|
|
||||||
Actor dockedHarv = null;
|
|
||||||
|
|
||||||
[Sync]
|
|
||||||
bool preventDock = false;
|
|
||||||
|
|
||||||
public bool AllowDocking => !preventDock;
|
|
||||||
public WPos DeliveryPosition => self.World.Map.CenterOfCell(self.Location + info.DockOffset);
|
|
||||||
public WAngle DeliveryAngle => info.DockAngle;
|
|
||||||
public bool IsDragRequired => info.IsDragRequired;
|
|
||||||
public WVec DragOffset => info.DragOffset;
|
|
||||||
public int DragLength => info.DragLength;
|
|
||||||
|
|
||||||
public Refinery(Actor self, RefineryInfo info)
|
public Refinery(Actor self, RefineryInfo info)
|
||||||
{
|
{
|
||||||
this.self = self;
|
|
||||||
this.info = info;
|
this.info = info;
|
||||||
playerResources = self.Owner.PlayerActor.Trait<PlayerResources>();
|
playerResources = self.Owner.PlayerActor.Trait<PlayerResources>();
|
||||||
currentDisplayTick = info.TickRate;
|
currentDisplayTick = info.TickRate;
|
||||||
@@ -87,13 +52,7 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
resourceValueModifiers = self.TraitsImplementing<IResourceValueModifier>().ToArray().Select(m => m.GetResourceValueModifier());
|
resourceValueModifiers = self.TraitsImplementing<IResourceValueModifier>().ToArray().Select(m => m.GetResourceValueModifier());
|
||||||
}
|
}
|
||||||
|
|
||||||
public IEnumerable<TraitPair<Harvester>> GetLinkedHarvesters()
|
int IAcceptResources.AcceptResources(Actor self, string resourceType, int count)
|
||||||
{
|
|
||||||
return self.World.ActorsWithTrait<Harvester>()
|
|
||||||
.Where(a => a.Trait.LinkedProc == self);
|
|
||||||
}
|
|
||||||
|
|
||||||
int IAcceptResources.AcceptResources(string resourceType, int count)
|
|
||||||
{
|
{
|
||||||
if (!playerResources.Info.ResourceValues.TryGetValue(resourceType, out var resourceValue))
|
if (!playerResources.Info.ResourceValues.TryGetValue(resourceType, out var resourceValue))
|
||||||
return 0;
|
return 0;
|
||||||
@@ -131,17 +90,8 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
void CancelDock()
|
|
||||||
{
|
|
||||||
preventDock = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void ITick.Tick(Actor self)
|
void ITick.Tick(Actor self)
|
||||||
{
|
{
|
||||||
// Harvester was killed while unloading
|
|
||||||
if (dockedHarv != null && dockedHarv.IsDead)
|
|
||||||
dockedHarv = null;
|
|
||||||
|
|
||||||
if (info.ShowTicks && currentDisplayValue > 0 && --currentDisplayTick <= 0)
|
if (info.ShowTicks && currentDisplayValue > 0 && --currentDisplayTick <= 0)
|
||||||
{
|
{
|
||||||
var temp = currentDisplayValue;
|
var temp = currentDisplayValue;
|
||||||
@@ -152,49 +102,9 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void INotifyActorDisposing.Disposing(Actor self)
|
|
||||||
{
|
|
||||||
CancelDock();
|
|
||||||
foreach (var harv in GetLinkedHarvesters())
|
|
||||||
harv.Trait.UnlinkProc(harv.Actor, self);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnDock(Actor harv, MoveToDock dockOrder)
|
|
||||||
{
|
|
||||||
if (!preventDock)
|
|
||||||
{
|
|
||||||
dockOrder.QueueChild(new CallFunc(() => dockedHarv = harv, false));
|
|
||||||
dockOrder.QueueChild(new HarvesterDockSequence(harv, self, this));
|
|
||||||
dockOrder.QueueChild(new CallFunc(() => dockedHarv = null, false));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void INotifyOwnerChanged.OnOwnerChanged(Actor self, Player oldOwner, Player newOwner)
|
void INotifyOwnerChanged.OnOwnerChanged(Actor self, Player oldOwner, Player newOwner)
|
||||||
{
|
{
|
||||||
// Unlink any harvesters
|
|
||||||
foreach (var harv in GetLinkedHarvesters())
|
|
||||||
harv.Trait.UnlinkProc(harv.Actor, self);
|
|
||||||
|
|
||||||
playerResources = newOwner.PlayerActor.Trait<PlayerResources>();
|
playerResources = newOwner.PlayerActor.Trait<PlayerResources>();
|
||||||
}
|
}
|
||||||
|
|
||||||
void INotifyCapture.OnCapture(Actor self, Actor captor, Player oldOwner, Player newOwner, BitSet<CaptureType> captureTypes)
|
|
||||||
{
|
|
||||||
// Steal any docked harv too
|
|
||||||
if (dockedHarv != null)
|
|
||||||
{
|
|
||||||
dockedHarv.ChangeOwner(newOwner);
|
|
||||||
|
|
||||||
// Relink to this refinery
|
|
||||||
dockedHarv.Trait<Harvester>().LinkProc(self);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void INotifySold.Selling(Actor self) { CancelDock(); }
|
|
||||||
void INotifySold.Sold(Actor self)
|
|
||||||
{
|
|
||||||
foreach (var harv in GetLinkedHarvesters())
|
|
||||||
harv.Trait.UnlinkProc(harv.Actor, self);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -36,10 +36,9 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
|
|
||||||
void INotifyHarvesterAction.MovingToRefinery(Actor self, Actor refineryActor)
|
void INotifyHarvesterAction.MovingToRefinery(Actor self, Actor refineryActor)
|
||||||
{
|
{
|
||||||
var iao = refineryActor.Trait<IAcceptResources>();
|
var dock = refineryActor.Trait<IDockHost>();
|
||||||
var location = self.World.Map.CellContaining(iao.DeliveryPosition);
|
|
||||||
foreach (var t in transports)
|
foreach (var t in transports)
|
||||||
t.RequestTransport(self, location);
|
t.RequestTransport(self, self.World.Map.CellContaining(dock.DockPosition));
|
||||||
}
|
}
|
||||||
|
|
||||||
void INotifyHarvesterAction.MovementCancelled(Actor self)
|
void INotifyHarvesterAction.MovementCancelled(Actor self)
|
||||||
|
|||||||
54
OpenRA.Mods.Common/Traits/DockClientBase.cs
Normal file
54
OpenRA.Mods.Common/Traits/DockClientBase.cs
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
#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 OpenRA.Primitives;
|
||||||
|
using OpenRA.Traits;
|
||||||
|
|
||||||
|
namespace OpenRA.Mods.Common.Traits
|
||||||
|
{
|
||||||
|
public abstract class DockClientBaseInfo : ConditionalTraitInfo, IDockClientInfo, Requires<DockClientManagerInfo> { }
|
||||||
|
|
||||||
|
public abstract class DockClientBase<InfoType> : ConditionalTrait<InfoType>, IDockClient, INotifyCreated where InfoType : DockClientBaseInfo
|
||||||
|
{
|
||||||
|
readonly Actor self;
|
||||||
|
|
||||||
|
public abstract BitSet<DockType> GetDockType { get; }
|
||||||
|
public DockClientManager DockClientManager { get; }
|
||||||
|
|
||||||
|
protected DockClientBase(Actor self, InfoType info)
|
||||||
|
: base(info)
|
||||||
|
{
|
||||||
|
this.self = self;
|
||||||
|
DockClientManager = self.Trait<DockClientManager>();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected virtual bool CanDock()
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual bool IsDockingPossible(BitSet<DockType> type, bool forceEnter = false)
|
||||||
|
{
|
||||||
|
return !IsTraitDisabled && GetDockType.Overlaps(type) && (forceEnter || CanDock());
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual bool CanDockAt(Actor hostActor, IDockHost host, bool forceEnter = false, bool ignoreOccupancy = false)
|
||||||
|
{
|
||||||
|
return (forceEnter || self.Owner.IsAlliedWith(hostActor.Owner)) && IsDockingPossible(host.GetDockType, forceEnter) && host.IsDockingPossible(self, this, ignoreOccupancy);
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual void OnDockStarted(Actor self, Actor hostActor, IDockHost host) { }
|
||||||
|
|
||||||
|
public virtual bool OnDockTick(Actor self, Actor hostActor, IDockHost host) { return false; }
|
||||||
|
|
||||||
|
public virtual void OnDockCompleted(Actor self, Actor hostActor, IDockHost host) { }
|
||||||
|
}
|
||||||
|
}
|
||||||
334
OpenRA.Mods.Common/Traits/DockClientManager.cs
Normal file
334
OpenRA.Mods.Common/Traits/DockClientManager.cs
Normal file
@@ -0,0 +1,334 @@
|
|||||||
|
#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.Mods.Common.Activities;
|
||||||
|
using OpenRA.Mods.Common.Orders;
|
||||||
|
using OpenRA.Primitives;
|
||||||
|
using OpenRA.Traits;
|
||||||
|
|
||||||
|
namespace OpenRA.Mods.Common.Traits
|
||||||
|
{
|
||||||
|
[Desc("Manages DockClients on the actor.")]
|
||||||
|
public class DockClientManagerInfo : ConditionalTraitInfo
|
||||||
|
{
|
||||||
|
[Desc("How long (in ticks) to wait until (re-)checking for a nearby available DockHost.")]
|
||||||
|
public readonly int SearchForDockDelay = 125;
|
||||||
|
|
||||||
|
[Desc("The pathfinding cost penalty applied for each dock client waiting to unload at a DockHost.")]
|
||||||
|
public readonly int OccupancyCostModifier = 12;
|
||||||
|
|
||||||
|
[CursorReference]
|
||||||
|
[Desc("Cursor to display when able to dock at target actor.")]
|
||||||
|
public readonly string EnterCursor = "enter";
|
||||||
|
|
||||||
|
[CursorReference]
|
||||||
|
[Desc("Cursor to display when unable to dock at target actor.")]
|
||||||
|
public readonly string EnterBlockedCursor = "enter-blocked";
|
||||||
|
|
||||||
|
[VoiceReference]
|
||||||
|
[Desc("Voice to be played when ordered to dock.")]
|
||||||
|
public readonly string Voice = "Action";
|
||||||
|
|
||||||
|
[Desc("Color to use for the target line of docking orders.")]
|
||||||
|
public readonly Color DockLineColor = Color.Green;
|
||||||
|
|
||||||
|
public override object Create(ActorInitializer init) { return new DockClientManager(init.Self, this); }
|
||||||
|
}
|
||||||
|
|
||||||
|
public class DockClientManager : ConditionalTrait<DockClientManagerInfo>, IResolveOrder, IOrderVoice, IIssueOrder, INotifyKilled, INotifyActorDisposing
|
||||||
|
{
|
||||||
|
readonly Actor self;
|
||||||
|
protected IDockClient[] dockClients;
|
||||||
|
public Color DockLineColor => Info.DockLineColor;
|
||||||
|
public int OccupancyCostModifier => Info.OccupancyCostModifier;
|
||||||
|
|
||||||
|
public DockClientManager(Actor self, DockClientManagerInfo info)
|
||||||
|
: base(info)
|
||||||
|
{
|
||||||
|
this.self = self;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Created(Actor self)
|
||||||
|
{
|
||||||
|
base.Created(self);
|
||||||
|
dockClients = self.TraitsImplementing<IDockClient>().ToArray();
|
||||||
|
}
|
||||||
|
|
||||||
|
public Actor ReservedHostActor { get; protected set; }
|
||||||
|
public IDockHost ReservedHost { get; protected set; }
|
||||||
|
|
||||||
|
IDockHost lastReservedDockHost = null;
|
||||||
|
public IDockHost LastReservedHost
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
if (lastReservedDockHost != null)
|
||||||
|
{
|
||||||
|
if (!lastReservedDockHost.IsEnabledAndInWorld)
|
||||||
|
lastReservedDockHost = null;
|
||||||
|
else
|
||||||
|
return lastReservedDockHost;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ReservedHost;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UnreserveHost()
|
||||||
|
{
|
||||||
|
if (ReservedHost != null)
|
||||||
|
{
|
||||||
|
lastReservedDockHost = ReservedHost;
|
||||||
|
ReservedHost = null;
|
||||||
|
ReservedHostActor = null;
|
||||||
|
lastReservedDockHost.Unreserve(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>In addition returns true if reservation was succesful or we have already been reserved at <paramref name="host"/>.</summary>
|
||||||
|
public bool ReserveHost(Actor hostActor, IDockHost host)
|
||||||
|
{
|
||||||
|
if (host == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (ReservedHost == host)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
UnreserveHost();
|
||||||
|
if (host.Reserve(hostActor, this))
|
||||||
|
{
|
||||||
|
ReservedHost = host;
|
||||||
|
ReservedHostActor = hostActor;
|
||||||
|
|
||||||
|
// After we have reserved a new Host we want to forget our old host.
|
||||||
|
lastReservedDockHost = null;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnDockStarted(Actor self, Actor hostActor, IDockHost host)
|
||||||
|
{
|
||||||
|
foreach (var client in dockClients)
|
||||||
|
client.OnDockStarted(self, hostActor, host);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool OnDockTick(Actor self, Actor hostActor, IDockHost host)
|
||||||
|
{
|
||||||
|
if (IsTraitDisabled)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
var cancel = true;
|
||||||
|
foreach (var client in dockClients)
|
||||||
|
if (!client.OnDockTick(self, hostActor, host))
|
||||||
|
cancel = false;
|
||||||
|
|
||||||
|
return cancel;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnDockCompleted(Actor self, Actor hostActor, IDockHost host)
|
||||||
|
{
|
||||||
|
foreach (var client in dockClients)
|
||||||
|
client.OnDockCompleted(self, hostActor, host);
|
||||||
|
|
||||||
|
UnreserveHost();
|
||||||
|
}
|
||||||
|
|
||||||
|
IEnumerable<IOrderTargeter> IIssueOrder.Orders
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
yield return new EnterAlliedActorTargeter<IDockHostInfo>(
|
||||||
|
"ForceDock",
|
||||||
|
6,
|
||||||
|
Info.EnterCursor,
|
||||||
|
Info.EnterBlockedCursor,
|
||||||
|
DockingPossible,
|
||||||
|
target => CanDockAt(target, true, true));
|
||||||
|
yield return new EnterAlliedActorTargeter<IDockHostInfo>(
|
||||||
|
"Dock",
|
||||||
|
5,
|
||||||
|
Info.EnterCursor,
|
||||||
|
Info.EnterBlockedCursor,
|
||||||
|
(actor, modifiers) => DockingPossible(actor),
|
||||||
|
target => CanDockAt(target, false, true));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void IResolveOrder.ResolveOrder(Actor self, Order order)
|
||||||
|
{
|
||||||
|
if (order.OrderString == "Dock")
|
||||||
|
{
|
||||||
|
var target = order.Target;
|
||||||
|
|
||||||
|
// Deliver orders are only valid for own/allied actors,
|
||||||
|
// which are guaranteed to never be frozen.
|
||||||
|
// TODO: support frozen actors
|
||||||
|
if (target.Type != TargetType.Actor)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (IsTraitDisabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var dock = AvailableDockHosts(target.Actor, false, true).ClosestDock(self, this);
|
||||||
|
if (!dock.HasValue)
|
||||||
|
return;
|
||||||
|
|
||||||
|
self.QueueActivity(order.Queued, new MoveToDock(self, dock.Value.Actor, dock.Value.Trait));
|
||||||
|
self.ShowTargetLines();
|
||||||
|
}
|
||||||
|
else if (order.OrderString == "ForceDock")
|
||||||
|
{
|
||||||
|
var target = order.Target;
|
||||||
|
|
||||||
|
// Deliver orders are only valid for own/allied actors,
|
||||||
|
// which are guaranteed to never be frozen.
|
||||||
|
// TODO: support frozen actors
|
||||||
|
if (target.Type != TargetType.Actor)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (IsTraitDisabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var dock = AvailableDockHosts(target.Actor, true, true).ClosestDock(self, this);
|
||||||
|
if (!dock.HasValue)
|
||||||
|
return;
|
||||||
|
|
||||||
|
self.QueueActivity(order.Queued, new MoveToDock(self, dock.Value.Actor, dock.Value.Trait));
|
||||||
|
self.ShowTargetLines();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
string IOrderVoice.VoicePhraseForOrder(Actor self, Order order)
|
||||||
|
{
|
||||||
|
if (order.Target.Type != TargetType.Actor || IsTraitDisabled)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
if (order.OrderString == "Dock" && CanDockAt(order.Target.Actor, false, true))
|
||||||
|
return Info.Voice;
|
||||||
|
else if (order.OrderString == "ForceDock" && CanDockAt(order.Target.Actor, true, true))
|
||||||
|
return Info.Voice;
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
Order IIssueOrder.IssueOrder(Actor self, IOrderTargeter order, in Target target, bool queued)
|
||||||
|
{
|
||||||
|
if (order.OrderID == "Dock" || order.OrderID == "ForceDock")
|
||||||
|
return new Order(order.OrderID, self, target, queued);
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>Do we have an enabled client with matching <paramref name="type"/>.</summary>
|
||||||
|
public bool DockingPossible(BitSet<DockType> type, bool forceEnter = false)
|
||||||
|
{
|
||||||
|
return !IsTraitDisabled && dockClients.Any(client => client.IsDockingPossible(type, forceEnter));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>Does this <paramref name="target"/> contain at least one enabled <see cref="IDockHost"/> with maching <see cref="DockType"/>.</summary>
|
||||||
|
public bool DockingPossible(Actor target)
|
||||||
|
{
|
||||||
|
return !IsTraitDisabled && target.TraitsImplementing<IDockHost>().Any(host => dockClients.Any(client => client.IsDockingPossible(host.GetDockType)));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>Does this <paramref name="target"/> contain at least one enabled <see cref="IDockHost"/> with maching <see cref="DockType"/>.</summary>
|
||||||
|
public bool DockingPossible(Actor target, TargetModifiers modifiers)
|
||||||
|
{
|
||||||
|
var forceEnter = modifiers.HasModifier(TargetModifiers.ForceMove);
|
||||||
|
return !IsTraitDisabled && target.TraitsImplementing<IDockHost>().Any(host => dockClients.Any(client => client.IsDockingPossible(host.GetDockType, forceEnter)));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>Can we dock to this <paramref name="host"/>.</summary>
|
||||||
|
public bool CanDockAt(Actor hostActor, IDockHost host, bool forceEnter = false, bool ignoreOccupancy = false)
|
||||||
|
{
|
||||||
|
return !IsTraitDisabled && dockClients.Any(client => client.CanDockAt(hostActor, host, forceEnter, ignoreOccupancy));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>Can we dock to this <paramref name="target"/>.</summary>
|
||||||
|
public bool CanDockAt(Actor target, bool forceEnter = false, bool ignoreOccupancy = false)
|
||||||
|
{
|
||||||
|
return !IsTraitDisabled && target.TraitsImplementing<IDockHost>().Any(host => dockClients.Any(client => client.CanDockAt(target, host, forceEnter, ignoreOccupancy)));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>Find the closest viable <see cref="IDockHost"/>.</summary>
|
||||||
|
/// <remarks>If <paramref name="type"/> is not set, scans all clients. Does not check if <see cref="DockClientManager"/> is enabled.</remarks>
|
||||||
|
public TraitPair<IDockHost>? ClosestDock(IDockHost ignore, BitSet<DockType> type = default, bool forceEnter = false, bool ignoreOccupancy = false)
|
||||||
|
{
|
||||||
|
var clients = type.IsEmpty ? dockClients : AvailableDockClients(type);
|
||||||
|
return self.World.ActorsWithTrait<IDockHost>()
|
||||||
|
.Where(host => host.Trait != ignore && clients.Any(client => client.CanDockAt(host.Actor, host.Trait, forceEnter, ignoreOccupancy)))
|
||||||
|
.ClosestDock(self, this);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>Get viable <see cref="IDockHost"/>'s on the <paramref name="target"/>.</summary>
|
||||||
|
/// <remarks>Does not check if <see cref="DockClientManager"/> is enabled.</remarks>
|
||||||
|
public IEnumerable<TraitPair<IDockHost>> AvailableDockHosts(Actor target, bool forceEnter = false, bool ignoreOccupancy = false)
|
||||||
|
{
|
||||||
|
return target.TraitsImplementing<IDockHost>()
|
||||||
|
.Where(host => dockClients.Any(client => client.CanDockAt(target, host, forceEnter, ignoreOccupancy)))
|
||||||
|
.Select(host => new TraitPair<IDockHost>(target, host));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>Get clients of matching <paramref name="type"/>.</summary>
|
||||||
|
/// <remarks>Does not check if <see cref="DockClientManager"/> is enabled.</remarks>
|
||||||
|
public IEnumerable<IDockClient> AvailableDockClients(BitSet<DockType> type)
|
||||||
|
{
|
||||||
|
return dockClients.Where(client => client.IsDockingPossible(type));
|
||||||
|
}
|
||||||
|
|
||||||
|
void INotifyKilled.Killed(Actor self, AttackInfo e) { UnreserveHost(); }
|
||||||
|
|
||||||
|
void INotifyActorDisposing.Disposing(Actor self) { UnreserveHost(); }
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class DockExts
|
||||||
|
{
|
||||||
|
public static TraitPair<IDockHost>? ClosestDock(this IEnumerable<TraitPair<IDockHost>> docks, Actor clientActor, DockClientManager client)
|
||||||
|
{
|
||||||
|
var mobile = clientActor.TraitOrDefault<Mobile>();
|
||||||
|
if (mobile != null)
|
||||||
|
{
|
||||||
|
// Overlapping docks can become hidden.
|
||||||
|
var lookup = docks.ToDictionary(dock => clientActor.World.Map.CellContaining(dock.Trait.DockPosition));
|
||||||
|
|
||||||
|
// Start a search from each docks position:
|
||||||
|
var path = mobile.PathFinder.FindPathToTargetCell(
|
||||||
|
clientActor, lookup.Keys, clientActor.Location, BlockedByActor.None,
|
||||||
|
location =>
|
||||||
|
{
|
||||||
|
if (!lookup.ContainsKey(location))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
var dock = lookup[location];
|
||||||
|
|
||||||
|
// Prefer docks with less occupancy (multiplier is to offset distance cost):
|
||||||
|
// TODO: add custom wieghts. E.g. owner vs allied.
|
||||||
|
return dock.Trait.ReservationCount * client.OccupancyCostModifier;
|
||||||
|
});
|
||||||
|
|
||||||
|
if (path.Count > 0)
|
||||||
|
return lookup[path.Last()];
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return docks
|
||||||
|
.OrderBy(dock => (clientActor.Location - clientActor.World.Map.CellContaining(dock.Trait.DockPosition)).LengthSquared + dock.Trait.ReservationCount * client.OccupancyCostModifier)
|
||||||
|
.FirstOrDefault();
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
183
OpenRA.Mods.Common/Traits/DockHost.cs
Normal file
183
OpenRA.Mods.Common/Traits/DockHost.cs
Normal file
@@ -0,0 +1,183 @@
|
|||||||
|
#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 OpenRA.Activities;
|
||||||
|
using OpenRA.Mods.Common.Activities;
|
||||||
|
using OpenRA.Primitives;
|
||||||
|
using OpenRA.Traits;
|
||||||
|
|
||||||
|
namespace OpenRA.Mods.Common.Traits
|
||||||
|
{
|
||||||
|
public sealed class DockType { DockType() { } }
|
||||||
|
|
||||||
|
[Desc("A generic dock that services DockClients.")]
|
||||||
|
public class DockHostInfo : ConditionalTraitInfo, IDockHostInfo
|
||||||
|
{
|
||||||
|
[Desc("Docking type.")]
|
||||||
|
public readonly BitSet<DockType> Type;
|
||||||
|
|
||||||
|
[Desc("How many clients can this dock be reserved for?")]
|
||||||
|
public readonly int MaxQueueLength = 3;
|
||||||
|
|
||||||
|
[Desc("How long should the client wait before starting the docking sequence.")]
|
||||||
|
public readonly int DockWait = 10;
|
||||||
|
|
||||||
|
[Desc("Actual client facing when docking.")]
|
||||||
|
public readonly WAngle DockAngle = WAngle.Zero;
|
||||||
|
|
||||||
|
[Desc("Docking cell relative to the centre of the actor.")]
|
||||||
|
public readonly WVec DockOffset = WVec.Zero;
|
||||||
|
|
||||||
|
[Desc("Does client need to be dragged in?")]
|
||||||
|
public readonly bool IsDragRequired = false;
|
||||||
|
|
||||||
|
[Desc("Vector by which the client will be dragged when docking.")]
|
||||||
|
public readonly WVec DragOffset = WVec.Zero;
|
||||||
|
|
||||||
|
[Desc("In how many steps to perform the dragging?")]
|
||||||
|
public readonly int DragLength = 0;
|
||||||
|
|
||||||
|
public override object Create(ActorInitializer init) { return new DockHost(init.Self, this); }
|
||||||
|
}
|
||||||
|
|
||||||
|
public class DockHost : ConditionalTrait<DockHostInfo>, IDockHost, IDockHostDrag, ITick, INotifySold, INotifyCapture, INotifyOwnerChanged, ISync, INotifyKilled, INotifyActorDisposing
|
||||||
|
{
|
||||||
|
readonly Actor self;
|
||||||
|
|
||||||
|
public BitSet<DockType> GetDockType => Info.Type;
|
||||||
|
public bool IsEnabledAndInWorld => !preventDock && !IsTraitDisabled && !self.IsDead && self.IsInWorld;
|
||||||
|
public int ReservationCount => ReservedDockClients.Count;
|
||||||
|
public bool CanBeReserved => ReservationCount < Info.MaxQueueLength;
|
||||||
|
protected readonly List<DockClientManager> ReservedDockClients = new();
|
||||||
|
|
||||||
|
public WPos DockPosition => self.CenterPosition + Info.DockOffset;
|
||||||
|
public int DockWait => Info.DockWait;
|
||||||
|
public WAngle DockAngle => Info.DockAngle;
|
||||||
|
|
||||||
|
bool IDockHostDrag.IsDragRequired => Info.IsDragRequired;
|
||||||
|
WVec IDockHostDrag.DragOffset => Info.DragOffset;
|
||||||
|
int IDockHostDrag.DragLength => Info.DragLength;
|
||||||
|
|
||||||
|
[Sync]
|
||||||
|
bool preventDock = false;
|
||||||
|
|
||||||
|
[Sync]
|
||||||
|
protected Actor dockedClientActor = null;
|
||||||
|
protected DockClientManager dockedClient = null;
|
||||||
|
|
||||||
|
public DockHost(Actor self, DockHostInfo info)
|
||||||
|
: base(info)
|
||||||
|
{
|
||||||
|
this.self = self;
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual bool IsDockingPossible(Actor clientActor, IDockClient client, bool ignoreReservations = false)
|
||||||
|
{
|
||||||
|
return !IsTraitDisabled && (ignoreReservations || CanBeReserved || ReservedDockClients.Contains(client.DockClientManager));
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual bool Reserve(Actor self, DockClientManager client)
|
||||||
|
{
|
||||||
|
if (CanBeReserved && !ReservedDockClients.Contains(client))
|
||||||
|
{
|
||||||
|
ReservedDockClients.Add(client);
|
||||||
|
client.ReserveHost(self, this);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual void UnreserveAll()
|
||||||
|
{
|
||||||
|
while (ReservedDockClients.Count > 0)
|
||||||
|
Unreserve(ReservedDockClients[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual void Unreserve(DockClientManager client)
|
||||||
|
{
|
||||||
|
if (ReservedDockClients.Contains(client))
|
||||||
|
{
|
||||||
|
ReservedDockClients.Remove(client);
|
||||||
|
client.UnreserveHost();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual void OnDockStarted(Actor self, Actor clientActor, DockClientManager client)
|
||||||
|
{
|
||||||
|
dockedClientActor = clientActor;
|
||||||
|
dockedClient = client;
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual void OnDockCompleted(Actor self, Actor clientActor, DockClientManager client)
|
||||||
|
{
|
||||||
|
dockedClientActor = null;
|
||||||
|
dockedClient = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ITick.Tick(Actor self)
|
||||||
|
{
|
||||||
|
Tick(self);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected virtual void Tick(Actor self)
|
||||||
|
{
|
||||||
|
// Client was killed during docking.
|
||||||
|
if (dockedClientActor != null && (dockedClientActor.IsDead || !dockedClientActor.IsInWorld))
|
||||||
|
OnDockCompleted(self, dockedClientActor, dockedClient);
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual bool QueueMoveActivity(Activity moveToDockActivity, Actor self, Actor clientActor, DockClientManager client)
|
||||||
|
{
|
||||||
|
var move = clientActor.Trait<IMove>();
|
||||||
|
|
||||||
|
// Make sure the actor is at dock, at correct facing, and aircraft are landed.
|
||||||
|
// Mobile cannot freely move in WPos, so when we calculate close enough we convert to CPos.
|
||||||
|
if ((move is Mobile ? clientActor.Location != clientActor.World.Map.CellContaining(DockPosition) : clientActor.CenterPosition != DockPosition)
|
||||||
|
|| move is not IFacing facing || facing.Facing != DockAngle)
|
||||||
|
{
|
||||||
|
moveToDockActivity.QueueChild(move.MoveOntoTarget(clientActor, Target.FromActor(self), DockPosition - self.CenterPosition, DockAngle));
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual void QueueDockActivity(Activity moveToDockActivity, Actor self, Actor clientActor, DockClientManager client)
|
||||||
|
{
|
||||||
|
moveToDockActivity.QueueChild(new GenericDockSequence(clientActor, client, self, this));
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void TraitDisabled(Actor self) { UnreserveAll(); }
|
||||||
|
|
||||||
|
void INotifyOwnerChanged.OnOwnerChanged(Actor self, Player oldOwner, Player newOwner) { UnreserveAll(); }
|
||||||
|
|
||||||
|
void INotifyCapture.OnCapture(Actor self, Actor captor, Player oldOwner, Player newOwner, BitSet<CaptureType> captureTypes)
|
||||||
|
{
|
||||||
|
// Steal any docked unit too.
|
||||||
|
if (dockedClientActor != null && !dockedClientActor.IsDead && dockedClientActor.IsInWorld)
|
||||||
|
{
|
||||||
|
dockedClientActor.ChangeOwner(newOwner);
|
||||||
|
|
||||||
|
// On capture OnOwnerChanged event is called first, so we need to re-reserve.
|
||||||
|
dockedClient.ReserveHost(self, this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void INotifySold.Selling(Actor self) { preventDock = true; }
|
||||||
|
|
||||||
|
void INotifySold.Sold(Actor self) { UnreserveAll(); }
|
||||||
|
|
||||||
|
void INotifyKilled.Killed(Actor self, AttackInfo e) { UnreserveAll(); }
|
||||||
|
|
||||||
|
void INotifyActorDisposing.Disposing(Actor self) { preventDock = true; UnreserveAll(); }
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -14,18 +14,15 @@ using System.Collections.Generic;
|
|||||||
using System.Collections.ObjectModel;
|
using System.Collections.ObjectModel;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using OpenRA.Mods.Common.Activities;
|
using OpenRA.Mods.Common.Activities;
|
||||||
using OpenRA.Mods.Common.Orders;
|
|
||||||
using OpenRA.Primitives;
|
using OpenRA.Primitives;
|
||||||
using OpenRA.Traits;
|
using OpenRA.Traits;
|
||||||
|
|
||||||
namespace OpenRA.Mods.Common.Traits
|
namespace OpenRA.Mods.Common.Traits
|
||||||
{
|
{
|
||||||
public class HarvesterInfo : ConditionalTraitInfo, Requires<MobileInfo>
|
public class HarvesterInfo : DockClientBaseInfo, Requires<MobileInfo>
|
||||||
{
|
{
|
||||||
public readonly HashSet<string> DeliveryBuildings = new();
|
[Desc("Docking type")]
|
||||||
|
public readonly BitSet<DockType> Type = new("Unload");
|
||||||
[Desc("How long (in ticks) to wait until (re-)checking for a nearby available DeliveryBuilding if not yet linked to one.")]
|
|
||||||
public readonly int SearchForDeliveryBuildingDelay = 125;
|
|
||||||
|
|
||||||
[Desc("Cell to move to when automatically unblocking DeliveryBuilding.")]
|
[Desc("Cell to move to when automatically unblocking DeliveryBuilding.")]
|
||||||
public readonly CVec UnblockCell = new(0, 4);
|
public readonly CVec UnblockCell = new(0, 4);
|
||||||
@@ -61,12 +58,6 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
[Desc("Interval to wait between searches when there are no resources nearby.")]
|
[Desc("Interval to wait between searches when there are no resources nearby.")]
|
||||||
public readonly int WaitDuration = 25;
|
public readonly int WaitDuration = 25;
|
||||||
|
|
||||||
[Desc("Find a new refinery to unload at if more than this many harvesters are already waiting.")]
|
|
||||||
public readonly int MaxUnloadQueue = 3;
|
|
||||||
|
|
||||||
[Desc("The pathfinding cost penalty applied for each harvester waiting to unload at a refinery.")]
|
|
||||||
public readonly int UnloadQueueCostModifier = 12;
|
|
||||||
|
|
||||||
[Desc("The pathfinding cost penalty applied for cells directly away from the refinery.")]
|
[Desc("The pathfinding cost penalty applied for cells directly away from the refinery.")]
|
||||||
public readonly int ResourceRefineryDirectionPenalty = 200;
|
public readonly int ResourceRefineryDirectionPenalty = 200;
|
||||||
|
|
||||||
@@ -80,23 +71,9 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
[VoiceReference]
|
[VoiceReference]
|
||||||
public readonly string HarvestVoice = "Action";
|
public readonly string HarvestVoice = "Action";
|
||||||
|
|
||||||
[VoiceReference]
|
|
||||||
public readonly string DeliverVoice = "Action";
|
|
||||||
|
|
||||||
[Desc("Color to use for the target line of harvest orders.")]
|
[Desc("Color to use for the target line of harvest orders.")]
|
||||||
public readonly Color HarvestLineColor = Color.Crimson;
|
public readonly Color HarvestLineColor = Color.Crimson;
|
||||||
|
|
||||||
[Desc("Color to use for the target line of harvest orders.")]
|
|
||||||
public readonly Color DeliverLineColor = Color.Green;
|
|
||||||
|
|
||||||
[CursorReference]
|
|
||||||
[Desc("Cursor to display when able to unload at target actor.")]
|
|
||||||
public readonly string EnterCursor = "enter";
|
|
||||||
|
|
||||||
[CursorReference]
|
|
||||||
[Desc("Cursor to display when unable to unload at target actor.")]
|
|
||||||
public readonly string EnterBlockedCursor = "enter-blocked";
|
|
||||||
|
|
||||||
[CursorReference]
|
[CursorReference]
|
||||||
[Desc("Cursor to display when ordering to harvest resources.")]
|
[Desc("Cursor to display when ordering to harvest resources.")]
|
||||||
public readonly string HarvestCursor = "harvest";
|
public readonly string HarvestCursor = "harvest";
|
||||||
@@ -104,7 +81,7 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
public override object Create(ActorInitializer init) { return new Harvester(init.Self, this); }
|
public override object Create(ActorInitializer init) { return new Harvester(init.Self, this); }
|
||||||
}
|
}
|
||||||
|
|
||||||
public class Harvester : ConditionalTrait<HarvesterInfo>, IIssueOrder, IResolveOrder, IOrderVoice,
|
public class Harvester : DockClientBase<HarvesterInfo>, IIssueOrder, IResolveOrder, IOrderVoice,
|
||||||
ISpeedModifier, ISync, INotifyCreated
|
ISpeedModifier, ISync, INotifyCreated
|
||||||
{
|
{
|
||||||
public readonly IReadOnlyDictionary<string, int> Contents;
|
public readonly IReadOnlyDictionary<string, int> Contents;
|
||||||
@@ -115,11 +92,7 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
readonly Dictionary<string, int> contents = new();
|
readonly Dictionary<string, int> contents = new();
|
||||||
int conditionToken = Actor.InvalidConditionToken;
|
int conditionToken = Actor.InvalidConditionToken;
|
||||||
|
|
||||||
[Sync]
|
public override BitSet<DockType> GetDockType => Info.Type;
|
||||||
public Actor LastLinkedProc = null;
|
|
||||||
|
|
||||||
[Sync]
|
|
||||||
public Actor LinkedProc = null;
|
|
||||||
|
|
||||||
[Sync]
|
[Sync]
|
||||||
int currentUnloadTicks;
|
int currentUnloadTicks;
|
||||||
@@ -137,7 +110,7 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
}
|
}
|
||||||
|
|
||||||
public Harvester(Actor self, HarvesterInfo info)
|
public Harvester(Actor self, HarvesterInfo info)
|
||||||
: base(info)
|
: base(self, info)
|
||||||
{
|
{
|
||||||
Contents = new ReadOnlyDictionary<string, int>(contents);
|
Contents = new ReadOnlyDictionary<string, int>(contents);
|
||||||
mobile = self.Trait<Mobile>();
|
mobile = self.Trait<Mobile>();
|
||||||
@@ -156,70 +129,15 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
base.Created(self);
|
base.Created(self);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void LinkProc(Actor proc)
|
|
||||||
{
|
|
||||||
LinkedProc = proc;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void UnlinkProc(Actor self, Actor proc)
|
|
||||||
{
|
|
||||||
if (LinkedProc == proc)
|
|
||||||
ChooseNewProc(self, proc);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void ChooseNewProc(Actor self, Actor ignore)
|
|
||||||
{
|
|
||||||
LastLinkedProc = null;
|
|
||||||
LinkProc(ClosestProc(self, ignore));
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IsAcceptableProcType(Actor proc)
|
|
||||||
{
|
|
||||||
return Info.DeliveryBuildings.Count == 0 ||
|
|
||||||
Info.DeliveryBuildings.Contains(proc.Info.Name);
|
|
||||||
}
|
|
||||||
|
|
||||||
public Actor ClosestProc(Actor self, Actor ignore)
|
|
||||||
{
|
|
||||||
// Find all refineries and their occupancy count:
|
|
||||||
// Exclude refineries with too many harvesters clogging the delivery location.
|
|
||||||
var refineries = self.World.ActorsWithTrait<IAcceptResources>()
|
|
||||||
.Where(r => r.Actor != ignore && r.Actor.Owner == self.Owner && IsAcceptableProcType(r.Actor))
|
|
||||||
.Select(r => new
|
|
||||||
{
|
|
||||||
Location = r.Actor.World.Map.CellContaining(r.Trait.DeliveryPosition),
|
|
||||||
Actor = r.Actor,
|
|
||||||
Occupancy = self.World.ActorsHavingTrait<Harvester>(h => h.LinkedProc == r.Actor).Count()
|
|
||||||
})
|
|
||||||
.Where(r => r.Occupancy < Info.MaxUnloadQueue)
|
|
||||||
.ToDictionary(r => r.Location);
|
|
||||||
|
|
||||||
if (refineries.Count == 0)
|
|
||||||
return null;
|
|
||||||
|
|
||||||
// Start a search from each refinery's delivery location:
|
|
||||||
var path = mobile.PathFinder.FindPathToTargetCells(
|
|
||||||
self, self.Location, refineries.Select(r => r.Key), BlockedByActor.None,
|
|
||||||
location =>
|
|
||||||
{
|
|
||||||
if (!refineries.ContainsKey(location))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
// Prefer refineries with less occupancy (multiplier is to offset distance cost):
|
|
||||||
var occupancy = refineries[location].Occupancy;
|
|
||||||
return occupancy * Info.UnloadQueueCostModifier;
|
|
||||||
});
|
|
||||||
|
|
||||||
if (path.Count > 0)
|
|
||||||
return refineries[path[0]].Actor;
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool IsFull => contents.Values.Sum() == Info.Capacity;
|
public bool IsFull => contents.Values.Sum() == Info.Capacity;
|
||||||
public bool IsEmpty => contents.Values.Sum() == 0;
|
public bool IsEmpty => contents.Values.Sum() == 0;
|
||||||
public int Fullness => contents.Values.Sum() * 100 / Info.Capacity;
|
public int Fullness => contents.Values.Sum() * 100 / Info.Capacity;
|
||||||
|
|
||||||
|
protected override bool CanDock()
|
||||||
|
{
|
||||||
|
return !IsEmpty;
|
||||||
|
}
|
||||||
|
|
||||||
void UpdateCondition(Actor self)
|
void UpdateCondition(Actor self)
|
||||||
{
|
{
|
||||||
if (string.IsNullOrEmpty(Info.EmptyCondition))
|
if (string.IsNullOrEmpty(Info.EmptyCondition))
|
||||||
@@ -243,21 +161,29 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
UpdateCondition(self);
|
UpdateCondition(self);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns true when unloading is complete
|
IAcceptResources acceptResources;
|
||||||
public virtual bool TickUnload(Actor self, Actor proc)
|
public override void OnDockStarted(Actor self, Actor hostActor, IDockHost host)
|
||||||
{
|
{
|
||||||
|
if (IsDockingPossible(host.GetDockType))
|
||||||
|
acceptResources = hostActor.TraitOrDefault<IAcceptResources>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public override bool OnDockTick(Actor self, Actor hostActor, IDockHost host)
|
||||||
|
{
|
||||||
|
if (acceptResources == null || IsTraitDisabled)
|
||||||
|
return true;
|
||||||
|
|
||||||
// Wait until the next bale is ready
|
// Wait until the next bale is ready
|
||||||
if (--currentUnloadTicks > 0)
|
if (--currentUnloadTicks > 0)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (contents.Keys.Count > 0)
|
if (contents.Keys.Count > 0)
|
||||||
{
|
{
|
||||||
var acceptResources = proc.Trait<IAcceptResources>();
|
|
||||||
foreach (var c in contents)
|
foreach (var c in contents)
|
||||||
{
|
{
|
||||||
var resourceType = c.Key;
|
var resourceType = c.Key;
|
||||||
var count = Math.Min(c.Value, Info.BaleUnloadAmount);
|
var count = Math.Min(c.Value, Info.BaleUnloadAmount);
|
||||||
var accepted = acceptResources.AcceptResources(resourceType, count);
|
var accepted = acceptResources.AcceptResources(hostActor, resourceType, count);
|
||||||
if (accepted == 0)
|
if (accepted == 0)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
@@ -274,6 +200,19 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
return contents.Count == 0;
|
return contents.Count == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public override void OnDockCompleted(Actor self, Actor hostActor, IDockHost dock)
|
||||||
|
{
|
||||||
|
acceptResources = null;
|
||||||
|
|
||||||
|
// After having docked at a refinery make sure we are running FindAndDeliverResources activity.
|
||||||
|
if (GetDockType.Overlaps(dock.GetDockType))
|
||||||
|
{
|
||||||
|
var currentActivity = self.CurrentActivity;
|
||||||
|
if (currentActivity == null || (currentActivity is not FindAndDeliverResources && currentActivity.NextActivity == null))
|
||||||
|
self.QueueActivity(true, new FindAndDeliverResources(self));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public bool CanHarvestCell(CPos cell)
|
public bool CanHarvestCell(CPos cell)
|
||||||
{
|
{
|
||||||
// Resources only exist in the ground layer
|
// Resources only exist in the ground layer
|
||||||
@@ -295,20 +234,13 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
if (IsTraitDisabled)
|
if (IsTraitDisabled)
|
||||||
yield break;
|
yield break;
|
||||||
|
|
||||||
yield return new EnterAlliedActorTargeter<IAcceptResourcesInfo>(
|
|
||||||
"Deliver",
|
|
||||||
5,
|
|
||||||
Info.EnterCursor,
|
|
||||||
Info.EnterBlockedCursor,
|
|
||||||
(proc, _) => IsAcceptableProcType(proc),
|
|
||||||
proc => proc.Trait<IAcceptResources>().AllowDocking);
|
|
||||||
yield return new HarvestOrderTargeter();
|
yield return new HarvestOrderTargeter();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Order IIssueOrder.IssueOrder(Actor self, IOrderTargeter order, in Target target, bool queued)
|
Order IIssueOrder.IssueOrder(Actor self, IOrderTargeter order, in Target target, bool queued)
|
||||||
{
|
{
|
||||||
if (order.OrderID == "Deliver" || order.OrderID == "Harvest")
|
if (order.OrderID == "Harvest")
|
||||||
return new Order(order.OrderID, self, target, queued);
|
return new Order(order.OrderID, self, target, queued);
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
@@ -319,9 +251,6 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
if (order.OrderString == "Harvest")
|
if (order.OrderString == "Harvest")
|
||||||
return Info.HarvestVoice;
|
return Info.HarvestVoice;
|
||||||
|
|
||||||
if (order.OrderString == "Deliver" && !IsEmpty)
|
|
||||||
return Info.DeliverVoice;
|
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -329,9 +258,6 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
{
|
{
|
||||||
if (order.OrderString == "Harvest")
|
if (order.OrderString == "Harvest")
|
||||||
{
|
{
|
||||||
// NOTE: An explicit harvest order allows the harvester to decide which refinery to deliver to.
|
|
||||||
LinkProc(null);
|
|
||||||
|
|
||||||
CPos loc;
|
CPos loc;
|
||||||
if (order.Target.Type != TargetType.Invalid)
|
if (order.Target.Type != TargetType.Invalid)
|
||||||
{
|
{
|
||||||
@@ -349,21 +275,6 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
self.QueueActivity(order.Queued, new FindAndDeliverResources(self, loc));
|
self.QueueActivity(order.Queued, new FindAndDeliverResources(self, loc));
|
||||||
self.ShowTargetLines();
|
self.ShowTargetLines();
|
||||||
}
|
}
|
||||||
else if (order.OrderString == "Deliver")
|
|
||||||
{
|
|
||||||
// Deliver orders are only valid for own/allied actors,
|
|
||||||
// which are guaranteed to never be frozen.
|
|
||||||
if (order.Target.Type != TargetType.Actor)
|
|
||||||
return;
|
|
||||||
|
|
||||||
var targetActor = order.Target.Actor;
|
|
||||||
var iao = targetActor.TraitOrDefault<IAcceptResources>();
|
|
||||||
if (iao == null || !iao.AllowDocking || !IsAcceptableProcType(targetActor))
|
|
||||||
return;
|
|
||||||
|
|
||||||
self.QueueActivity(order.Queued, new FindAndDeliverResources(self, targetActor));
|
|
||||||
self.ShowTargetLines();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int ISpeedModifier.GetSpeedModifier()
|
int ISpeedModifier.GetSpeedModifier()
|
||||||
@@ -373,8 +284,7 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
|
|
||||||
protected override void TraitDisabled(Actor self)
|
protected override void TraitDisabled(Actor self)
|
||||||
{
|
{
|
||||||
LastLinkedProc = null;
|
base.TraitDisabled(self);
|
||||||
LinkedProc = null;
|
|
||||||
contents.Clear();
|
contents.Clear();
|
||||||
|
|
||||||
if (conditionToken != Actor.InvalidConditionToken)
|
if (conditionToken != Actor.InvalidConditionToken)
|
||||||
|
|||||||
@@ -13,7 +13,6 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using OpenRA.Activities;
|
using OpenRA.Activities;
|
||||||
using OpenRA.Graphics;
|
using OpenRA.Graphics;
|
||||||
using OpenRA.Mods.Common.Activities;
|
|
||||||
using OpenRA.Mods.Common.Graphics;
|
using OpenRA.Mods.Common.Graphics;
|
||||||
using OpenRA.Mods.Common.Terrain;
|
using OpenRA.Mods.Common.Terrain;
|
||||||
using OpenRA.Mods.Common.Widgets;
|
using OpenRA.Mods.Common.Widgets;
|
||||||
@@ -211,6 +210,72 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
void Harvested(Actor self, string resourceType);
|
void Harvested(Actor self, string resourceType);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public interface IDockClientInfo : ITraitInfoInterface { }
|
||||||
|
|
||||||
|
public interface IDockClient
|
||||||
|
{
|
||||||
|
BitSet<DockType> GetDockType { get; }
|
||||||
|
DockClientManager DockClientManager { get; }
|
||||||
|
void OnDockStarted(Actor self, Actor hostActor, IDockHost host);
|
||||||
|
bool OnDockTick(Actor self, Actor hostActor, IDockHost dock);
|
||||||
|
void OnDockCompleted(Actor self, Actor hostActor, IDockHost host);
|
||||||
|
|
||||||
|
/// <summary>Is this client allowed to dock.</summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Does not check if <see cref="Traits.DockClientManager"/> is enabled.
|
||||||
|
/// Function should only be called from within <see cref="IDockClient"/> or <see cref="Traits.DockClientManager"/>.
|
||||||
|
/// </remarks>
|
||||||
|
bool IsDockingPossible(BitSet<DockType> type, bool forceEnter = false);
|
||||||
|
|
||||||
|
/// <summary>Is this client allowed to dock to <paramref name="host"/>.</summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Does not check if <see cref="Traits.DockClientManager"/> is enabled.
|
||||||
|
/// Function should only be called from within <see cref="IDockClient"/> or <see cref="Traits.DockClientManager"/>.
|
||||||
|
/// </remarks>
|
||||||
|
bool CanDockAt(Actor hostActor, IDockHost host, bool forceEnter = false, bool ignoreOccupancy = false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public interface IDockHostInfo : ITraitInfoInterface { }
|
||||||
|
|
||||||
|
public interface IDockHost
|
||||||
|
{
|
||||||
|
BitSet<DockType> GetDockType { get; }
|
||||||
|
|
||||||
|
/// <summary>Use this function instead of ConditionalTrait.IsTraitDisabled.</summary>
|
||||||
|
bool IsEnabledAndInWorld { get; }
|
||||||
|
int ReservationCount { get; }
|
||||||
|
bool CanBeReserved { get; }
|
||||||
|
WPos DockPosition { get; }
|
||||||
|
int DockWait { get; }
|
||||||
|
WAngle DockAngle { get; }
|
||||||
|
|
||||||
|
/// <summary>Can this <paramref name="client"/> dock at this <see cref="IDockHost"/>.</summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Does not check <see cref="DockType"/>.
|
||||||
|
/// Does not check if <see cref="IDockClient"/> is enabled.
|
||||||
|
/// Does not check if <see cref="DockClientManager"/> is enabled.
|
||||||
|
/// </remarks>
|
||||||
|
bool IsDockingPossible(Actor clientActor, IDockClient client, bool ignoreReservations = false);
|
||||||
|
bool Reserve(Actor self, DockClientManager client);
|
||||||
|
void UnreserveAll();
|
||||||
|
void Unreserve(DockClientManager client);
|
||||||
|
void OnDockStarted(Actor self, Actor clientActor, DockClientManager client);
|
||||||
|
void OnDockCompleted(Actor self, Actor clientActor, DockClientManager client);
|
||||||
|
|
||||||
|
/// <summary>If <paramref name="client"/> is not in range of <see cref="IDockHost"/> queues a child move activity and returns true. If in range returns false.</summary>
|
||||||
|
bool QueueMoveActivity(Activity moveToDockActivity, Actor self, Actor clientActor, DockClientManager client);
|
||||||
|
|
||||||
|
/// <summary>Should be called when in range of <see cref="IDockHost"/>.</summary>
|
||||||
|
void QueueDockActivity(Activity moveToDockActivity, Actor self, Actor clientActor, DockClientManager client);
|
||||||
|
}
|
||||||
|
|
||||||
|
public interface IDockHostDrag
|
||||||
|
{
|
||||||
|
bool IsDragRequired { get; }
|
||||||
|
WVec DragOffset { get; }
|
||||||
|
int DragLength { get; }
|
||||||
|
}
|
||||||
|
|
||||||
[RequireExplicitImplementation]
|
[RequireExplicitImplementation]
|
||||||
public interface INotifyLoadCargo
|
public interface INotifyLoadCargo
|
||||||
{
|
{
|
||||||
@@ -280,14 +345,9 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
void Undeploy(Actor self, bool skipMakeAnim);
|
void Undeploy(Actor self, bool skipMakeAnim);
|
||||||
}
|
}
|
||||||
|
|
||||||
public interface IAcceptResourcesInfo : ITraitInfoInterface { }
|
|
||||||
public interface IAcceptResources
|
public interface IAcceptResources
|
||||||
{
|
{
|
||||||
void OnDock(Actor harv, MoveToDock dockOrder);
|
int AcceptResources(Actor self, string resourceType, int count = 1);
|
||||||
int AcceptResources(string resourceType, int count = 1);
|
|
||||||
WPos DeliveryPosition { get; }
|
|
||||||
WAngle DeliveryAngle { get; }
|
|
||||||
bool AllowDocking { get; }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public interface IDockClientBody
|
public interface IDockClientBody
|
||||||
|
|||||||
166
OpenRA.Mods.Common/UpdateRules/Rules/20230801/AbstractDocking.cs
Normal file
166
OpenRA.Mods.Common/UpdateRules/Rules/20230801/AbstractDocking.cs
Normal file
@@ -0,0 +1,166 @@
|
|||||||
|
#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;
|
||||||
|
|
||||||
|
namespace OpenRA.Mods.Common.UpdateRules.Rules
|
||||||
|
{
|
||||||
|
public class AbstractDocking : UpdateRule, IBeforeUpdateActors
|
||||||
|
{
|
||||||
|
readonly string[] moveRefineyValues = { "DockAngle", "IsDragRequired", "DragOffset", "DragLength" };
|
||||||
|
readonly string[] moveHarvesterValues = { "EnterCursor", "EnterBlockedCursor" };
|
||||||
|
readonly string[] buildings = { "Building", "D2kBuilding" };
|
||||||
|
readonly string[,] moveAndRenameHarvesterValues = new string[4, 2]
|
||||||
|
{
|
||||||
|
{ "DeliverVoice", "Voice" },
|
||||||
|
{ "DeliverLineColor", "DockLineColor" },
|
||||||
|
{ "UnloadQueueCostModifier", "OccupancyCostModifier" },
|
||||||
|
{ "SearchForDeliveryBuildingDelay", "SearchForDockDelay" }
|
||||||
|
};
|
||||||
|
|
||||||
|
readonly Dictionary<string, List<MiniYamlNodeBuilder>> refineryNodes = new();
|
||||||
|
public override string Name => "Docking was abstracted from Refinery & Harvester.";
|
||||||
|
|
||||||
|
public override string Description =>
|
||||||
|
"Fields moved from Refinery to new trait DockHost, fields moved from Harvester to new trait DockClientManager and to DockHost";
|
||||||
|
|
||||||
|
public IEnumerable<string> BeforeUpdateActors(ModData modData, List<MiniYamlNodeBuilder> resolvedActors)
|
||||||
|
{
|
||||||
|
grid = modData.Manifest.Get<MapGrid>();
|
||||||
|
var harvesters = new Dictionary<string, HashSet<string>>();
|
||||||
|
var refineries = new List<string>();
|
||||||
|
foreach (var actorNode in resolvedActors)
|
||||||
|
{
|
||||||
|
var harvesterNode = actorNode.ChildrenMatching("Harvester", includeRemovals: false).FirstOrDefault();
|
||||||
|
if (harvesterNode != null)
|
||||||
|
harvesters[actorNode.Key] = harvesterNode.ChildrenMatching("DeliveryBuildings", includeRemovals: false)
|
||||||
|
.FirstOrDefault()?.NodeValue<HashSet<string>>() ?? new HashSet<string>();
|
||||||
|
|
||||||
|
if (actorNode.ChildrenMatching("Refinery", includeRemovals: false).FirstOrDefault() != null)
|
||||||
|
refineries.Add(actorNode.Key.ToLowerInvariant());
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var harvester in harvesters)
|
||||||
|
{
|
||||||
|
foreach (var deliveryBuildingHigh in harvester.Value)
|
||||||
|
{
|
||||||
|
var deliveryBuilding = deliveryBuildingHigh.ToLowerInvariant();
|
||||||
|
foreach (var refinery in refineries)
|
||||||
|
{
|
||||||
|
if (refinery == deliveryBuilding)
|
||||||
|
{
|
||||||
|
if (!refineryNodes.ContainsKey(refinery))
|
||||||
|
refineryNodes[refinery] = new List<MiniYamlNodeBuilder>();
|
||||||
|
|
||||||
|
var node = new MiniYamlNodeBuilder("Type", deliveryBuilding.ToString());
|
||||||
|
if (!refineryNodes[refinery].Any(n => n.Key == node.Key))
|
||||||
|
refineryNodes[refinery].Add(node);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
yield break;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override IEnumerable<string> UpdateActorNode(ModData modData, MiniYamlNodeBuilder actorNode)
|
||||||
|
{
|
||||||
|
var refineryNode = actorNode.ChildrenMatching("Refinery", includeRemovals: false).FirstOrDefault();
|
||||||
|
if (refineryNode != null)
|
||||||
|
{
|
||||||
|
var dockNode = new MiniYamlNodeBuilder("DockHost", "");
|
||||||
|
|
||||||
|
var lowActorName = actorNode.Key.ToLowerInvariant();
|
||||||
|
if (!refineryNodes.ContainsKey(lowActorName) || !refineryNodes[lowActorName].Any(n => n.Key == "Type"))
|
||||||
|
dockNode.AddNode("Type", "Unload");
|
||||||
|
else
|
||||||
|
dockNode.AddNode(refineryNodes[lowActorName].First(n => n.Key == "Type"));
|
||||||
|
|
||||||
|
foreach (var value in moveRefineyValues)
|
||||||
|
{
|
||||||
|
foreach (var node in refineryNode.ChildrenMatching(value).ToList())
|
||||||
|
{
|
||||||
|
dockNode.AddNode(node);
|
||||||
|
refineryNode.RemoveNode(node);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var oldOffset = CVec.Zero;
|
||||||
|
var dockOffsetNode = refineryNode.ChildrenMatching("DockOffset", includeRemovals: false).FirstOrDefault();
|
||||||
|
if (dockOffsetNode != null)
|
||||||
|
{
|
||||||
|
oldOffset = dockOffsetNode.NodeValue<CVec>();
|
||||||
|
refineryNode.RemoveNode(dockOffsetNode);
|
||||||
|
}
|
||||||
|
|
||||||
|
var buildingNode = actorNode.Value.Nodes.FirstOrDefault(n => buildings.Any(b => n.KeyMatches(b, includeRemovals: false)));
|
||||||
|
if (buildingNode != null)
|
||||||
|
{
|
||||||
|
var dimensions = buildingNode.ChildrenMatching("Dimensions", includeRemovals: false).FirstOrDefault()?.NodeValue<CVec>() ?? new CVec(1, 1);
|
||||||
|
var localCenterOffset = buildingNode.ChildrenMatching("LocalCenterOffset", includeRemovals: false).FirstOrDefault()?.NodeValue<WVec>() ?? WVec.Zero;
|
||||||
|
|
||||||
|
var offset = CenterOfCell(oldOffset) - CenterOfCell(CVec.Zero) - BuildingCenter(dimensions, localCenterOffset);
|
||||||
|
if (offset != WVec.Zero)
|
||||||
|
dockNode.AddNode("DockOffset", offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
actorNode.AddNode(dockNode);
|
||||||
|
}
|
||||||
|
|
||||||
|
var harvesterNode = actorNode.ChildrenMatching("Harvester", includeRemovals: false).FirstOrDefault();
|
||||||
|
if (harvesterNode != null)
|
||||||
|
{
|
||||||
|
var dockClientNode = new MiniYamlNodeBuilder("DockClientManager", "");
|
||||||
|
|
||||||
|
foreach (var value in moveHarvesterValues)
|
||||||
|
{
|
||||||
|
foreach (var node in harvesterNode.ChildrenMatching(value).ToList())
|
||||||
|
{
|
||||||
|
dockClientNode.AddNode(node);
|
||||||
|
harvesterNode.RemoveNode(node);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (var i = 0; i < moveAndRenameHarvesterValues.GetLength(0); i++)
|
||||||
|
{
|
||||||
|
foreach (var node in harvesterNode.ChildrenMatching(moveAndRenameHarvesterValues[i, 0]).ToList())
|
||||||
|
{
|
||||||
|
harvesterNode.RemoveNode(node);
|
||||||
|
node.RenameKey(moveAndRenameHarvesterValues[i, 1]);
|
||||||
|
dockClientNode.AddNode(node);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
harvesterNode.RenameChildrenMatching("DeliveryBuildings", "DockType");
|
||||||
|
harvesterNode.RemoveNodes("MaxUnloadQueue");
|
||||||
|
|
||||||
|
actorNode.AddNode(dockClientNode);
|
||||||
|
}
|
||||||
|
|
||||||
|
yield break;
|
||||||
|
}
|
||||||
|
|
||||||
|
MapGrid grid;
|
||||||
|
public WVec CenterOfCell(CVec cell)
|
||||||
|
{
|
||||||
|
if (grid.Type == MapGridType.Rectangular)
|
||||||
|
return new WVec(1024 * cell.X + 512, 1024 * cell.Y + 512, 0);
|
||||||
|
|
||||||
|
return new WVec(724 * (cell.X - cell.Y + 1), 724 * (cell.X + cell.Y + 1), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
public WVec BuildingCenter(CVec dimensions, WVec localCenterOffset)
|
||||||
|
{
|
||||||
|
return (CenterOfCell(dimensions) - CenterOfCell(new CVec(1, 1))) / 2 + localCenterOffset;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -88,9 +88,8 @@ namespace OpenRA.Mods.Common.UpdateRules
|
|||||||
new UnhardcodeBaseBuilderBotModule(),
|
new UnhardcodeBaseBuilderBotModule(),
|
||||||
}),
|
}),
|
||||||
|
|
||||||
new UpdatePath("release-20230225", new UpdateRule[]
|
new UpdatePath("release-20230225", "playtest-20230801", new UpdateRule[]
|
||||||
{
|
{
|
||||||
// bleed only changes here
|
|
||||||
new TextNotificationsDisplayWidgetRemoveTime(),
|
new TextNotificationsDisplayWidgetRemoveTime(),
|
||||||
new RenameEngineerRepair(),
|
new RenameEngineerRepair(),
|
||||||
new ProductionTabsWidgetAddTabButtonCollection(),
|
new ProductionTabsWidgetAddTabButtonCollection(),
|
||||||
@@ -104,7 +103,15 @@ namespace OpenRA.Mods.Common.UpdateRules
|
|||||||
new ExplicitSequenceFilenames(),
|
new ExplicitSequenceFilenames(),
|
||||||
new RemoveSequenceHasEmbeddedPalette(),
|
new RemoveSequenceHasEmbeddedPalette(),
|
||||||
new RemoveNegativeSequenceLength(),
|
new RemoveNegativeSequenceLength(),
|
||||||
})
|
}),
|
||||||
|
|
||||||
|
new UpdatePath("playtest-20230801", new UpdateRule[]
|
||||||
|
{
|
||||||
|
// bleed only changes here.
|
||||||
|
|
||||||
|
// Execute these rules last to avoid premature yaml merge crashes.
|
||||||
|
new AbstractDocking(),
|
||||||
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static IEnumerable<UpdateRule> FromSource(ObjectCreator objectCreator, string source, bool chain = true)
|
public static IEnumerable<UpdateRule> FromSource(ObjectCreator objectCreator, string source, bool chain = true)
|
||||||
|
|||||||
@@ -239,12 +239,14 @@ PROC:
|
|||||||
RevealsShroud:
|
RevealsShroud:
|
||||||
Range: 6c0
|
Range: 6c0
|
||||||
Refinery:
|
Refinery:
|
||||||
|
TickRate: 15
|
||||||
|
DockHost:
|
||||||
|
Type: Unload
|
||||||
DockAngle: 448
|
DockAngle: 448
|
||||||
DockOffset: 0,2
|
DockOffset: -1c0, 1c0, 0
|
||||||
IsDragRequired: True
|
IsDragRequired: True
|
||||||
DragOffset: -554,512,0
|
DragOffset: -554,512,0
|
||||||
DragLength: 12
|
DragLength: 12
|
||||||
TickRate: 15
|
|
||||||
StoresResources:
|
StoresResources:
|
||||||
Capacity: 1000
|
Capacity: 1000
|
||||||
Selectable:
|
Selectable:
|
||||||
|
|||||||
@@ -63,6 +63,7 @@ HARV:
|
|||||||
SearchFromHarvesterRadius: 8
|
SearchFromHarvesterRadius: 8
|
||||||
HarvestFacings: 8
|
HarvestFacings: 8
|
||||||
EmptyCondition: no-tiberium
|
EmptyCondition: no-tiberium
|
||||||
|
DockClientManager:
|
||||||
Mobile:
|
Mobile:
|
||||||
Speed: 72
|
Speed: 72
|
||||||
Health:
|
Health:
|
||||||
|
|||||||
@@ -282,9 +282,11 @@ refinery:
|
|||||||
RevealsShroud:
|
RevealsShroud:
|
||||||
Range: 3c768
|
Range: 3c768
|
||||||
Refinery:
|
Refinery:
|
||||||
DockAngle: 640
|
|
||||||
DockOffset: 2,1
|
|
||||||
TickRate: 20
|
TickRate: 20
|
||||||
|
DockHost:
|
||||||
|
Type: Unload
|
||||||
|
DockAngle: 640
|
||||||
|
DockOffset: 1c0,512,0
|
||||||
StoresResources:
|
StoresResources:
|
||||||
Capacity: 2000
|
Capacity: 2000
|
||||||
CustomSellValue:
|
CustomSellValue:
|
||||||
|
|||||||
@@ -76,6 +76,7 @@ harvester:
|
|||||||
BaleUnloadDelay: 5
|
BaleUnloadDelay: 5
|
||||||
SearchFromProcRadius: 30
|
SearchFromProcRadius: 30
|
||||||
SearchFromHarvesterRadius: 15
|
SearchFromHarvesterRadius: 15
|
||||||
|
DockClientManager:
|
||||||
CarryableHarvester:
|
CarryableHarvester:
|
||||||
Health:
|
Health:
|
||||||
HP: 45000
|
HP: 45000
|
||||||
|
|||||||
@@ -1287,8 +1287,10 @@ PROC:
|
|||||||
RevealsShroud@GAPGEN:
|
RevealsShroud@GAPGEN:
|
||||||
Range: 4c0
|
Range: 4c0
|
||||||
Refinery:
|
Refinery:
|
||||||
|
DockHost:
|
||||||
|
Type: Unload
|
||||||
DockAngle: 256
|
DockAngle: 256
|
||||||
DockOffset: 1,2
|
DockOffset: 0, 1c0, 0
|
||||||
StoresResources:
|
StoresResources:
|
||||||
Capacity: 2000
|
Capacity: 2000
|
||||||
CustomSellValue:
|
CustomSellValue:
|
||||||
|
|||||||
@@ -327,6 +327,7 @@ HARV:
|
|||||||
SearchFromHarvesterRadius: 8
|
SearchFromHarvesterRadius: 8
|
||||||
HarvestFacings: 8
|
HarvestFacings: 8
|
||||||
EmptyCondition: no-ore
|
EmptyCondition: no-ore
|
||||||
|
DockClientManager:
|
||||||
Health:
|
Health:
|
||||||
HP: 60000
|
HP: 60000
|
||||||
Armor:
|
Armor:
|
||||||
|
|||||||
@@ -589,8 +589,10 @@ NAWAST:
|
|||||||
Range: 6c0
|
Range: 6c0
|
||||||
MaxHeightDelta: 3
|
MaxHeightDelta: 3
|
||||||
Refinery:
|
Refinery:
|
||||||
|
DockHost:
|
||||||
|
Type: UnloadWeed
|
||||||
DockAngle: 640
|
DockAngle: 640
|
||||||
DockOffset: 2,1
|
DockOffset: 724,724,0
|
||||||
StoresResources:
|
StoresResources:
|
||||||
Capacity: 56
|
Capacity: 56
|
||||||
Power:
|
Power:
|
||||||
|
|||||||
@@ -351,7 +351,7 @@ WEED:
|
|||||||
Prerequisites: ~naweap, nawast, ~techlevel.superweapons
|
Prerequisites: ~naweap, nawast, ~techlevel.superweapons
|
||||||
Description: Collects veins for processing.\n Unarmed
|
Description: Collects veins for processing.\n Unarmed
|
||||||
Harvester:
|
Harvester:
|
||||||
DeliveryBuildings: nawast
|
Type: UnloadWeed
|
||||||
Capacity: 7
|
Capacity: 7
|
||||||
Resources: Veins
|
Resources: Veins
|
||||||
BaleUnloadDelay: 20
|
BaleUnloadDelay: 20
|
||||||
@@ -359,7 +359,8 @@ WEED:
|
|||||||
SearchFromProcRadius: 72
|
SearchFromProcRadius: 72
|
||||||
SearchFromHarvesterRadius: 36
|
SearchFromHarvesterRadius: 36
|
||||||
HarvestVoice: Attack
|
HarvestVoice: Attack
|
||||||
DeliverVoice: Move
|
DockClientManager:
|
||||||
|
Voice: Move
|
||||||
Mobile:
|
Mobile:
|
||||||
Speed: 71
|
Speed: 71
|
||||||
TurnSpeed: 20
|
TurnSpeed: 20
|
||||||
|
|||||||
@@ -120,9 +120,11 @@ PROC:
|
|||||||
Range: 6c0
|
Range: 6c0
|
||||||
MaxHeightDelta: 3
|
MaxHeightDelta: 3
|
||||||
Refinery:
|
Refinery:
|
||||||
DockAngle: 640
|
|
||||||
DockOffset: 2,1
|
|
||||||
DiscardExcessResources: true
|
DiscardExcessResources: true
|
||||||
|
DockHost:
|
||||||
|
Type: Unload
|
||||||
|
DockAngle: 640
|
||||||
|
DockOffset: 362,362,0
|
||||||
StoresResources:
|
StoresResources:
|
||||||
Capacity: 2000
|
Capacity: 2000
|
||||||
CustomSellValue:
|
CustomSellValue:
|
||||||
|
|||||||
@@ -59,7 +59,6 @@ HARV:
|
|||||||
Bounds: 1086, 2172
|
Bounds: 1086, 2172
|
||||||
DecorationBounds: 1086, 2172
|
DecorationBounds: 1086, 2172
|
||||||
Harvester:
|
Harvester:
|
||||||
DeliveryBuildings: proc
|
|
||||||
Capacity: 28
|
Capacity: 28
|
||||||
Resources: Tiberium, BlueTiberium
|
Resources: Tiberium, BlueTiberium
|
||||||
BaleLoadDelay: 15
|
BaleLoadDelay: 15
|
||||||
@@ -68,8 +67,9 @@ HARV:
|
|||||||
SearchFromProcRadius: 36
|
SearchFromProcRadius: 36
|
||||||
SearchFromHarvesterRadius: 18
|
SearchFromHarvesterRadius: 18
|
||||||
HarvestVoice: Attack
|
HarvestVoice: Attack
|
||||||
DeliverVoice: Move
|
|
||||||
EmptyCondition: no-tiberium
|
EmptyCondition: no-tiberium
|
||||||
|
DockClientManager:
|
||||||
|
Voice: Move
|
||||||
Mobile:
|
Mobile:
|
||||||
Speed: 71
|
Speed: 71
|
||||||
Health:
|
Health:
|
||||||
|
|||||||
Reference in New Issue
Block a user