Merge pull request #3627 from pchote/muzzleflash
Improved Muzzleflash support
This commit is contained in:
@@ -74,7 +74,8 @@ namespace OpenRA.Mods.RA.Activities
|
||||
void LayMine(Actor self)
|
||||
{
|
||||
var limitedAmmo = self.TraitOrDefault<LimitedAmmo>();
|
||||
if (limitedAmmo != null) limitedAmmo.Attacking(self, Target.FromCell(self.Location));
|
||||
if (limitedAmmo != null)
|
||||
limitedAmmo.TakeAmmo();
|
||||
|
||||
self.World.AddFrameEndTask(
|
||||
w => w.CreateActor(self.Info.Traits.Get<MinelayerInfo>().Mine, new TypeDictionary
|
||||
|
||||
@@ -151,7 +151,7 @@ namespace OpenRA.Mods.RA
|
||||
});
|
||||
|
||||
foreach (var na in self.TraitsImplementing<INotifyAttack>())
|
||||
na.Attacking(self, target);
|
||||
na.Attacking(self, target, this, barrel);
|
||||
|
||||
Recoil = Info.Recoil;
|
||||
|
||||
|
||||
@@ -40,7 +40,7 @@ namespace OpenRA.Mods.RA
|
||||
base.Tick( self );
|
||||
}
|
||||
|
||||
public void Attacking(Actor self, Target target)
|
||||
public void Attacking(Actor self, Target target, Armament a, Barrel barrel)
|
||||
{
|
||||
--charges;
|
||||
timeToRecharge = self.Info.Traits.Get<AttackTeslaInfo>().ReloadTime;
|
||||
|
||||
@@ -56,7 +56,7 @@ namespace OpenRA.Mods.RA
|
||||
remainingTime = Math.Max(remainingTime, time);
|
||||
}
|
||||
|
||||
public void Attacking(Actor self, Target target) { Uncloak(); }
|
||||
public void Attacking(Actor self, Target target, Armament a, Barrel barrel) { Uncloak(); }
|
||||
|
||||
public bool Cloaked { get { return remainingTime <= 0; } }
|
||||
|
||||
|
||||
@@ -44,6 +44,7 @@ namespace OpenRA.Mods.RA
|
||||
++ammo;
|
||||
return true;
|
||||
}
|
||||
|
||||
public bool TakeAmmo()
|
||||
{
|
||||
if (ammo <= 0) return false;
|
||||
@@ -53,7 +54,7 @@ namespace OpenRA.Mods.RA
|
||||
|
||||
public int ReloadTimePerAmmo() { return Info.ReloadTicks; }
|
||||
|
||||
public void Attacking(Actor self, Target target) { TakeAmmo(); }
|
||||
public void Attacking(Actor self, Target target, Armament a, Barrel barrel) { TakeAmmo(); }
|
||||
|
||||
public IEnumerable<PipType> GetPips(Actor self)
|
||||
{
|
||||
|
||||
@@ -74,6 +74,11 @@ namespace OpenRA.Mods.RA.Render
|
||||
anim.PlayThen(NormalizeInfantrySequence(self, "heal"), () => State = AnimationState.Idle);
|
||||
}
|
||||
|
||||
public void Attacking(Actor self, Target target, Armament a, Barrel barrel)
|
||||
{
|
||||
Attacking(self, target);
|
||||
}
|
||||
|
||||
public override void Tick(Actor self)
|
||||
{
|
||||
base.Tick(self);
|
||||
|
||||
@@ -11,69 +11,84 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using OpenRA.FileFormats;
|
||||
using OpenRA.Graphics;
|
||||
using OpenRA.Traits;
|
||||
using OpenRA.Mods.RA;
|
||||
|
||||
namespace OpenRA.Mods.RA.Render
|
||||
{
|
||||
class WithMuzzleFlashInfo : ITraitInfo, Requires<RenderSpritesInfo>, Requires<AttackBaseInfo>
|
||||
class WithMuzzleFlashInfo : ITraitInfo, Requires<RenderSpritesInfo>, Requires<AttackBaseInfo>, Requires<ArmamentInfo>
|
||||
{
|
||||
public object Create(ActorInitializer init) { return new WithMuzzleFlash(init.self); }
|
||||
[Desc("Sequence name to use")]
|
||||
public readonly string Sequence = "muzzle";
|
||||
|
||||
[Desc("Armament name")]
|
||||
public readonly string Armament = "primary";
|
||||
|
||||
public object Create(ActorInitializer init) { return new WithMuzzleFlash(init.self, this); }
|
||||
}
|
||||
|
||||
class WithMuzzleFlash : INotifyAttack, IRender, ITick
|
||||
{
|
||||
Dictionary<string, AnimationWithOffset> muzzleFlashes = new Dictionary<string, AnimationWithOffset>();
|
||||
bool isShowing;
|
||||
readonly WithMuzzleFlashInfo info;
|
||||
Dictionary<Barrel, bool> visible = new Dictionary<Barrel, bool>();
|
||||
Dictionary<Barrel, AnimationWithOffset> anims = new Dictionary<Barrel, AnimationWithOffset>();
|
||||
|
||||
public WithMuzzleFlash(Actor self)
|
||||
public WithMuzzleFlash(Actor self, WithMuzzleFlashInfo info)
|
||||
{
|
||||
this.info = info;
|
||||
var render = self.Trait<RenderSprites>();
|
||||
var facing = self.TraitOrDefault<IFacing>();
|
||||
|
||||
var arms = self.TraitsImplementing<Armament>();
|
||||
foreach (var a in arms)
|
||||
foreach(var b in a.Barrels)
|
||||
{
|
||||
var barrel = b;
|
||||
var turreted = self.TraitsImplementing<Turreted>()
|
||||
.FirstOrDefault(t => t.Name == a.Info.Turret);
|
||||
var getFacing = turreted != null ? () => turreted.turretFacing :
|
||||
facing != null ? (Func<int>)(() => facing.Facing) : () => 0;
|
||||
var arm = self.TraitsImplementing<Armament>()
|
||||
.Single(a => a.Info.Name == info.Armament);
|
||||
|
||||
var muzzleFlash = new Animation(render.GetImage(self), getFacing);
|
||||
muzzleFlash.Play("muzzle");
|
||||
foreach (var b in arm.Barrels)
|
||||
{
|
||||
var barrel = b;
|
||||
var turreted = self.TraitsImplementing<Turreted>()
|
||||
.FirstOrDefault(t => t.Name == arm.Info.Turret);
|
||||
|
||||
muzzleFlashes.Add("muzzle{0}".F(muzzleFlashes.Count),
|
||||
new AnimationWithOffset(muzzleFlash,
|
||||
() => a.MuzzleOffset(self, barrel),
|
||||
() => !isShowing));
|
||||
}
|
||||
var getFacing = turreted != null ? () => turreted.turretFacing :
|
||||
facing != null ? (Func<int>)(() => facing.Facing) : () => 0;
|
||||
|
||||
var muzzleFlash = new Animation(render.GetImage(self), getFacing);
|
||||
visible.Add(barrel, false);
|
||||
anims.Add(barrel,
|
||||
new AnimationWithOffset(muzzleFlash,
|
||||
() => arm.MuzzleOffset(self, barrel),
|
||||
() => !visible[barrel]));
|
||||
}
|
||||
}
|
||||
|
||||
public void Attacking(Actor self, Target target)
|
||||
public void Attacking(Actor self, Target target, Armament a, Barrel barrel)
|
||||
{
|
||||
isShowing = true;
|
||||
foreach( var mf in muzzleFlashes.Values )
|
||||
mf.Animation.PlayThen("muzzle", () => isShowing = false);
|
||||
if (a.Info.Name != info.Armament)
|
||||
return;
|
||||
|
||||
visible[barrel] = true;
|
||||
anims[barrel].Animation.PlayThen(info.Sequence, () => visible[barrel] = false);
|
||||
}
|
||||
|
||||
public IEnumerable<IRenderable> Render(Actor self, WorldRenderer wr)
|
||||
{
|
||||
foreach (var a in muzzleFlashes.Values)
|
||||
foreach (var kv in anims)
|
||||
{
|
||||
if (a.DisableFunc != null && a.DisableFunc())
|
||||
if (!visible[kv.Key])
|
||||
continue;
|
||||
|
||||
foreach (var r in a.Render(self, wr, wr.Palette("effect"), 1f))
|
||||
if (kv.Value.DisableFunc != null && kv.Value.DisableFunc())
|
||||
continue;
|
||||
|
||||
foreach (var r in kv.Value.Render(self, wr, wr.Palette("effect"), 1f))
|
||||
yield return r;
|
||||
}
|
||||
}
|
||||
|
||||
public void Tick(Actor self)
|
||||
{
|
||||
foreach (var a in muzzleFlashes.Values)
|
||||
foreach (var a in anims.Values)
|
||||
a.Animation.Tick();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -67,7 +67,7 @@ namespace OpenRA.Mods.RA
|
||||
Panic();
|
||||
}
|
||||
|
||||
public void Attacking(Actor self, Target target)
|
||||
public void Attacking(Actor self, Target target, Armament a, Barrel barrel)
|
||||
{
|
||||
if (self.World.SharedRandom.Next(100 / Info.AttackPanicChance) == 0)
|
||||
Panic();
|
||||
|
||||
@@ -137,7 +137,7 @@ namespace OpenRA.Mods.RA
|
||||
}
|
||||
|
||||
/* lose our disguise if we attack anything */
|
||||
public void Attacking(Actor self, Target target) { DropDisguise(); }
|
||||
public void Attacking(Actor self, Target target, Armament a, Barrel barrel) { DropDisguise(); }
|
||||
}
|
||||
|
||||
class IgnoresDisguiseInfo : TraitInfo<IgnoresDisguise> {}
|
||||
|
||||
@@ -10,6 +10,7 @@
|
||||
|
||||
using System.Collections.Generic;
|
||||
using OpenRA.Mods.RA.Activities;
|
||||
using OpenRA.Traits;
|
||||
|
||||
namespace OpenRA.Mods.RA
|
||||
{
|
||||
@@ -46,4 +47,5 @@ namespace OpenRA.Mods.RA
|
||||
public interface INotifyParachuteLanded { void OnLanded(); }
|
||||
public interface INotifyTransform { void OnTransform(Actor self); }
|
||||
public interface INotifyTransformed { void OnTransformed(Actor toActor); }
|
||||
public interface INotifyAttack { void Attacking(Actor self, Target target, Armament a, Barrel barrel); }
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user