#region Copyright & License Information /* * Copyright 2007-2014 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.Generic; using System.Linq; using OpenRA.Graphics; using OpenRA.Mods.RA.Buildings; using OpenRA.Traits; namespace OpenRA.Mods.RA { [Desc("This actor will remain visible (but not updated visually) under fog, once discovered.")] public class FrozenUnderFogInfo : ITraitInfo, Requires { public readonly bool StartsRevealed = false; public object Create(ActorInitializer init) { return new FrozenUnderFog(init, this); } } public class FrozenUnderFog : IRenderModifier, IVisibilityModifier, ITick, ITickRender, ISync { [Sync] public int VisibilityHash; readonly bool startsRevealed; readonly CPos[] footprint; readonly CellRegion footprintRegion; readonly Lazy tooltip; readonly Lazy health; readonly Dictionary visible; readonly Dictionary frozen; bool initialized; public FrozenUnderFog(ActorInitializer init, FrozenUnderFogInfo info) { // Spawned actors (e.g. building husks) shouldn't be revealed startsRevealed = info.StartsRevealed && !init.Contains(); footprint = FootprintUtils.Tiles(init.self).ToArray(); footprintRegion = CellRegion.BoundingRegion(init.world.Map.TileShape, footprint); tooltip = Exts.Lazy(() => init.self.TraitsImplementing().FirstOrDefault()); tooltip = Exts.Lazy(() => init.self.TraitsImplementing().FirstOrDefault()); health = Exts.Lazy(() => init.self.TraitOrDefault()); frozen = new Dictionary(); visible = init.world.Players.ToDictionary(p => p, p => false); } public bool IsVisible(Actor self, Player byPlayer) { return byPlayer == null || visible[byPlayer]; } public void Tick(Actor self) { if (self.Destroyed) return; VisibilityHash = 0; foreach (var p in self.World.Players) { var isVisible = footprint.Any(p.Shroud.IsVisibleTest(footprintRegion)); visible[p] = isVisible; if (isVisible) VisibilityHash += p.ClientIndex; } if (!initialized) { foreach (var p in self.World.Players) { visible[p] |= startsRevealed; p.PlayerActor.Trait().Add(frozen[p] = new FrozenActor(self, footprint, footprintRegion)); } initialized = true; } foreach (var player in self.World.Players) { if (!visible[player]) continue; var actor = frozen[player]; actor.Owner = self.Owner; if (health.Value != null) { actor.HP = health.Value.HP; actor.DamageState = health.Value.DamageState; } if (tooltip.Value != null) { actor.TooltipName = tooltip.Value.Name(); actor.TooltipOwner = tooltip.Value.Owner(); } } } public void TickRender(WorldRenderer wr, Actor self) { if (self.Destroyed || !initialized || !visible.Values.Any(v => v)) return; var renderables = self.Render(wr).ToArray(); foreach (var player in self.World.Players) if (visible[player]) frozen[player].Renderables = renderables; } public IEnumerable ModifyRender(Actor self, WorldRenderer wr, IEnumerable r) { return IsVisible(self, self.World.RenderPlayer) ? r : SpriteRenderable.None; } } }