#region Copyright & License Information /* * Copyright 2007-2013 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. For more information, * see COPYING. */ #endregion using System; using System.Collections; using System.Collections.Generic; using System.Linq; using NLua; using OpenRA.Effects; using OpenRA.FileFormats; using OpenRA.Mods.RA.Activities; using OpenRA.Mods.RA.Air; using OpenRA.Mods.RA.Missions; using OpenRA.Scripting; using OpenRA.Support; using OpenRA.Traits; using WorldRenderer = OpenRA.Graphics.WorldRenderer; namespace OpenRA.Mods.RA.Scripting { public class LuaScriptInterfaceInfo : ITraitInfo, Requires { public readonly string[] LuaScripts = { }; public object Create(ActorInitializer init) { return new LuaScriptInterface(this); } } public class LuaScriptInterface : IWorldLoaded, ITick { World world; Dictionary mapActors; readonly LuaScriptContext context = new LuaScriptContext(); readonly LuaScriptInterfaceInfo info; public LuaScriptInterface(LuaScriptInterfaceInfo info) { this.info = info; } public void WorldLoaded(World w, WorldRenderer wr) { world = w; mapActors = world.WorldActor.Trait().Actors; context.Lua["World"] = w; context.Lua["WorldRenderer"] = wr; context.RegisterObject(this, "Internal", false); context.RegisterType(typeof(WVec), "WVec", true); context.RegisterType(typeof(CVec), "CVec", true); context.RegisterType(typeof(WPos), "WPos", true); context.RegisterType(typeof(CPos), "CPos", true); context.RegisterType(typeof(WRot), "WRot", true); context.RegisterType(typeof(WAngle), "WAngle", true); context.RegisterType(typeof(WRange), "WRange", true); context.RegisterType(typeof(int2), "int2", true); context.RegisterType(typeof(float2), "float2", true); var sharedScripts = Game.modData.Manifest.LuaScripts ?? new string[0]; if (sharedScripts.Any()) context.LoadLuaScripts(f => FileSystem.Open(f).ReadAllText(), sharedScripts); AddMapActorGlobals(); context.LoadLuaScripts(f => w.Map.Container.GetContent(f).ReadAllText(), info.LuaScripts); context.InvokeLuaFunction("WorldLoaded"); } void AddMapActorGlobals() { foreach (var kv in mapActors) { if (context.Lua[kv.Key] != null) context.ShowErrorMessage("{0}: The global name '{1}' is reserved and may not be used by map actor {2}".F(GetType().Name, kv.Key, kv.Value), null); else context.Lua[kv.Key] = kv.Value; } } public void Tick(Actor self) { using (new PerfSample("tick_lua")) context.InvokeLuaFunction("Tick"); } [LuaGlobal] public object New(string typeName, LuaTable args) { var type = Game.modData.ObjectCreator.FindType(typeName); if (type == null) throw new InvalidOperationException("Cannot locate type: {0}".F(typeName)); if (args == null) return Activator.CreateInstance(type); var argsArray = ConvertArgs(args); return Activator.CreateInstance(type, argsArray); } object[] ConvertArgs(LuaTable args) { var argsArray = new object[args.Keys.Count]; for (var i = 1; i <= args.Keys.Count; i++) { var arg = args[i] as LuaTable; if (arg != null && arg[1] != null && arg[2] != null) argsArray[i - 1] = Convert.ChangeType(arg[1], Enum.Parse(arg[2].ToString())); else argsArray[i - 1] = args[i]; } return argsArray; } [LuaGlobal] public void Debug(object obj) { if (obj != null) Game.Debug(obj.ToString()); } [LuaGlobal] public object TraitOrDefault(Actor actor, string className) { var type = Game.modData.ObjectCreator.FindType(className); if (type == null) return null; var method = typeof(Actor).GetMethod("TraitOrDefault"); var genericMethod = method.MakeGenericMethod(type); return genericMethod.Invoke(actor, null); } [LuaGlobal] public object Trait(Actor actor, string className) { var ret = TraitOrDefault(actor, className); if (ret == null) throw new InvalidOperationException("Actor {0} does not have trait of type {1}".F(actor, className)); return ret; } [LuaGlobal] public bool HasTrait(Actor actor, string className) { var ret = TraitOrDefault(actor, className); return ret != null; } [LuaGlobal] public object[] ActorsWithTrait(string className) { var type = Game.modData.ObjectCreator.FindType(className); if (type == null) throw new InvalidOperationException("Cannot locate type: {0}".F(className)); var method = typeof(World).GetMethod("ActorsWithTrait"); var genericMethod = method.MakeGenericMethod(type); var result = ((IEnumerable)genericMethod.Invoke(world, null)).Cast().ToArray(); return result; } [LuaGlobal] public object TraitInfoOrDefault(string actorType, string className) { var type = Game.modData.ObjectCreator.FindType(className); if (type == null || !Rules.Info.ContainsKey(actorType)) return null; return Rules.Info[actorType].Traits.GetOrDefault(type); } [LuaGlobal] public object TraitInfo(string actorType, string className) { var ret = TraitInfoOrDefault(actorType, className); if (ret == null) throw new InvalidOperationException("Actor type {0} does not have trait info of type {1}".F(actorType, className)); return ret; } [LuaGlobal] public bool HasTraitInfo(string actorType, string className) { var ret = TraitInfoOrDefault(actorType, className); return ret != null; } [LuaGlobal] public void RunAfterDelay(double delay, Action func) { world.AddFrameEndTask(w => w.Add(new DelayedAction((int)delay, func))); } [LuaGlobal] public void PlaySpeechNotification(Player player, string notification) { Sound.PlayNotification(player, "Speech", notification, player != null ? player.Country.Race : null); } [LuaGlobal] public void PlaySoundNotification(Player player, string notification) { Sound.PlayNotification(player, "Sounds", notification, player != null ? player.Country.Race : null); } [LuaGlobal] public void WaitFor(Actor actor, Func func) { actor.QueueActivity(new WaitFor(func)); } [LuaGlobal] public void CallFunc(Actor actor, Action func) { actor.QueueActivity(new CallFunc(func)); } [LuaGlobal] public int GetFacing(object vec, double currentFacing) { if (vec is CVec) return Util.GetFacing((CVec)vec, (int)currentFacing); if (vec is WVec) return Util.GetFacing((WVec)vec, (int)currentFacing); throw new ArgumentException("Unsupported vector type: {0}".F(vec.GetType())); } [LuaGlobal] public WRange GetWRangeFromCells(double cells) { return WRange.FromCells((int)cells); } [LuaGlobal] public void SetWinState(Player player, string winState) { player.WinState = Enum.Parse(winState); } [LuaGlobal] public void PlayRandomMusic() { MissionUtils.PlayMissionMusic(); } [LuaGlobal] public bool IsDead(Actor actor) { return actor.IsDead(); } [LuaGlobal] public void PlayMovieFullscreen(string movie, Action onComplete) { Media.PlayFMVFullscreen(world, movie, onComplete); } [LuaGlobal] public void FlyToPos(Actor actor, WPos pos) { actor.QueueActivity(new Fly(actor, Target.FromPos(pos))); } [LuaGlobal] public void FlyAttackActor(Actor actor, Actor targetActor) { actor.QueueActivity(new FlyAttack(Target.FromActor(targetActor))); } [LuaGlobal] public void FlyAttackCell(Actor actor, CPos location) { actor.QueueActivity(new FlyAttack(Target.FromCell(location))); } [LuaGlobal] public void HeliFlyToPos(Actor actor, WPos pos) { actor.QueueActivity(new HeliFly(actor, Target.FromPos(pos))); } [LuaGlobal] public void SetUnitStance(Actor actor, string stance) { var at = actor.TraitOrDefault(); if (at != null) at.Stance = Enum.Parse(stance); } [LuaGlobal] public bool RequiredUnitsAreDestroyed(Player player) { return world.ActorsWithTrait().All(p => p.Actor.Owner != player); } [LuaGlobal] public void AttackMove(Actor actor, CPos location) { if (actor.HasTrait()) actor.QueueActivity(new AttackMove.AttackMoveActivity(actor, new Move.Move(location, 0))); else actor.QueueActivity(new Move.Move(location, 0)); } [LuaGlobal] public int GetRandomInteger(double low, double high) { return world.SharedRandom.Next((int)low, (int)high); } [LuaGlobal] public CPos GetRandomCell() { return world.ChooseRandomCell(world.SharedRandom); } [LuaGlobal] public CPos GetRandomEdgeCell() { return world.ChooseRandomEdgeCell(); } [LuaGlobal] public Actor GetNamedActor(string actorName) { return mapActors[actorName]; } } }