200 lines
6.4 KiB
C#
200 lines
6.4 KiB
C#
#region Copyright & License Information
|
|
/*
|
|
* Copyright 2007-2017 The OpenRA Developers (see AUTHORS)
|
|
* 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;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using Eluant;
|
|
using OpenRA.Activities;
|
|
using OpenRA.Effects;
|
|
using OpenRA.Mods.Common.Activities;
|
|
using OpenRA.Mods.Common.Traits;
|
|
using OpenRA.Primitives;
|
|
using OpenRA.Scripting;
|
|
using OpenRA.Traits;
|
|
|
|
namespace OpenRA.Mods.Common.Scripting
|
|
{
|
|
[ScriptGlobal("Reinforcements")]
|
|
public class ReinforcementsGlobal : ScriptGlobal
|
|
{
|
|
public ReinforcementsGlobal(ScriptContext context) : base(context) { }
|
|
|
|
Actor CreateActor(Player owner, string actorType, bool addToWorld, CPos? entryLocation = null, CPos? nextLocation = null)
|
|
{
|
|
ActorInfo ai;
|
|
if (!Context.World.Map.Rules.Actors.TryGetValue(actorType, out ai))
|
|
throw new LuaException("Unknown actor type '{0}'".F(actorType));
|
|
|
|
var initDict = new TypeDictionary();
|
|
|
|
initDict.Add(new OwnerInit(owner));
|
|
|
|
if (entryLocation.HasValue)
|
|
{
|
|
var pi = ai.TraitInfoOrDefault<AircraftInfo>();
|
|
initDict.Add(new CenterPositionInit(owner.World.Map.CenterOfCell(entryLocation.Value) + new WVec(0, 0, pi != null ? pi.CruiseAltitude.Length : 0)));
|
|
initDict.Add(new LocationInit(entryLocation.Value));
|
|
}
|
|
|
|
if (entryLocation.HasValue && nextLocation.HasValue)
|
|
initDict.Add(new FacingInit(Context.World.Map.FacingBetween(CPos.Zero, CPos.Zero + (nextLocation.Value - entryLocation.Value), 0)));
|
|
|
|
var actor = Context.World.CreateActor(addToWorld, actorType, initDict);
|
|
|
|
return actor;
|
|
}
|
|
|
|
void Move(Actor actor, CPos dest)
|
|
{
|
|
var move = actor.TraitOrDefault<IMove>();
|
|
if (move == null)
|
|
return;
|
|
|
|
actor.QueueActivity(move.MoveTo(dest, 2));
|
|
}
|
|
|
|
[Desc("Send reinforcements consisting of multiple units. Supports ground-based, naval and air units. " +
|
|
"The first member of the entryPath array will be the units' spawnpoint, " +
|
|
"while the last one will be their destination. If actionFunc is given, " +
|
|
"it will be executed once a unit has reached its destination. actionFunc " +
|
|
"will be called as actionFunc(Actor actor)")]
|
|
public Actor[] Reinforce(Player owner, string[] actorTypes, CPos[] entryPath, int interval = 25, LuaFunction actionFunc = null)
|
|
{
|
|
var actors = new List<Actor>();
|
|
for (var i = 0; i < actorTypes.Length; i++)
|
|
{
|
|
var af = actionFunc != null ? (LuaFunction)actionFunc.CopyReference() : null;
|
|
var actor = CreateActor(owner, actorTypes[i], false, entryPath[0], entryPath.Length > 1 ? entryPath[1] : (CPos?)null);
|
|
actors.Add(actor);
|
|
|
|
var actionDelay = i * interval;
|
|
Action actorAction = () =>
|
|
{
|
|
Context.World.Add(actor);
|
|
for (var j = 1; j < entryPath.Length; j++)
|
|
Move(actor, entryPath[j]);
|
|
|
|
if (af != null)
|
|
{
|
|
actor.QueueActivity(new CallFunc(() =>
|
|
{
|
|
using (af)
|
|
using (var a = actor.ToLuaValue(Context))
|
|
af.Call(a);
|
|
}));
|
|
}
|
|
};
|
|
|
|
Context.World.AddFrameEndTask(w => w.Add(new DelayedAction(actionDelay, actorAction)));
|
|
}
|
|
|
|
return actors.ToArray();
|
|
}
|
|
|
|
[Desc("Send reinforcements in a transport. A transport can be a ground unit (APC etc.), ships and aircraft. " +
|
|
"The first member of the entryPath array will be the spawnpoint for the transport, " +
|
|
"while the last one will be its destination. The last member of the exitPath array " +
|
|
"is be the place where the transport will be removed from the game. When the transport " +
|
|
"has reached the destination, it will unload its cargo unless a custom actionFunc has " +
|
|
"been supplied. Afterwards, the transport will follow the exitPath and leave the map, " +
|
|
"unless a custom exitFunc has been supplied. actionFunc will be called as " +
|
|
"actionFunc(Actor transport, Actor[] cargo). exitFunc will be called as exitFunc(Actor transport).")]
|
|
public LuaTable ReinforceWithTransport(Player owner, string actorType, string[] cargoTypes, CPos[] entryPath, CPos[] exitPath = null,
|
|
LuaFunction actionFunc = null, LuaFunction exitFunc = null)
|
|
{
|
|
var transport = CreateActor(owner, actorType, true, entryPath[0], entryPath.Length > 1 ? entryPath[1] : (CPos?)null);
|
|
var cargo = transport.TraitOrDefault<Cargo>();
|
|
|
|
var passengers = new List<Actor>();
|
|
if (cargo != null && cargoTypes != null)
|
|
{
|
|
foreach (var cargoType in cargoTypes)
|
|
{
|
|
var passenger = CreateActor(owner, cargoType, false, entryPath[0]);
|
|
passengers.Add(passenger);
|
|
cargo.Load(transport, passenger);
|
|
}
|
|
}
|
|
|
|
for (var i = 1; i < entryPath.Length; i++)
|
|
Move(transport, entryPath[i]);
|
|
|
|
if (actionFunc != null)
|
|
{
|
|
var af = (LuaFunction)actionFunc.CopyReference();
|
|
transport.QueueActivity(new CallFunc(() =>
|
|
{
|
|
using (af)
|
|
using (LuaValue t = transport.ToLuaValue(Context), p = passengers.ToArray().ToLuaValue(Context))
|
|
af.Call(t, p);
|
|
}));
|
|
}
|
|
else
|
|
{
|
|
var aircraftInfo = transport.TraitOrDefault<Aircraft>();
|
|
if (aircraftInfo != null)
|
|
{
|
|
if (!aircraftInfo.IsPlane)
|
|
{
|
|
transport.QueueActivity(new Turn(transport, aircraftInfo.Info.InitialFacing));
|
|
transport.QueueActivity(new HeliLand(transport, true));
|
|
}
|
|
else
|
|
{
|
|
transport.QueueActivity(new Land(transport, Target.FromCell(transport.World, entryPath.Last())));
|
|
}
|
|
|
|
transport.QueueActivity(new Wait(15));
|
|
}
|
|
|
|
if (cargo != null)
|
|
{
|
|
transport.QueueActivity(new UnloadCargo(transport, true));
|
|
transport.QueueActivity(new WaitFor(() => cargo.IsEmpty(transport)));
|
|
}
|
|
|
|
transport.QueueActivity(new Wait(aircraftInfo != null ? 50 : 25));
|
|
}
|
|
|
|
if (exitFunc != null)
|
|
{
|
|
var ef = (LuaFunction)exitFunc.CopyReference();
|
|
transport.QueueActivity(new CallFunc(() =>
|
|
{
|
|
using (ef)
|
|
using (var t = transport.ToLuaValue(Context))
|
|
ef.Call(t);
|
|
}));
|
|
}
|
|
else if (exitPath != null)
|
|
{
|
|
foreach (var wpt in exitPath)
|
|
Move(transport, wpt);
|
|
|
|
transport.QueueActivity(new RemoveSelf());
|
|
}
|
|
|
|
var ret = Context.CreateTable();
|
|
using (LuaValue
|
|
tKey = 1,
|
|
tValue = transport.ToLuaValue(Context),
|
|
pKey = 2,
|
|
pValue = passengers.ToArray().ToLuaValue(Context))
|
|
{
|
|
ret.Add(tKey, tValue);
|
|
ret.Add(pKey, pValue);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
}
|
|
} |