Use IRender.ScreenBounds in ScreenMap.
Traits are now required to trigger a ScreenMap update whenever they believe that their ScreenBounds have changed.
This commit is contained in:
@@ -204,6 +204,23 @@ namespace OpenRA
|
|||||||
yield return renderable;
|
yield return renderable;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public IEnumerable<Rectangle> ScreenBounds(WorldRenderer wr)
|
||||||
|
{
|
||||||
|
var bounds = Bounds(wr);
|
||||||
|
foreach (var modifier in renderModifiers)
|
||||||
|
bounds = modifier.ModifyScreenBounds(this, wr, bounds);
|
||||||
|
return bounds;
|
||||||
|
}
|
||||||
|
|
||||||
|
IEnumerable<Rectangle> Bounds(WorldRenderer wr)
|
||||||
|
{
|
||||||
|
// PERF: Avoid LINQ. See comments for Renderables
|
||||||
|
foreach (var render in renders)
|
||||||
|
foreach (var r in render.ScreenBounds(this, wr))
|
||||||
|
if (!r.IsEmpty)
|
||||||
|
yield return r;
|
||||||
|
}
|
||||||
|
|
||||||
public void QueueActivity(bool queued, Activity nextActivity)
|
public void QueueActivity(bool queued, Activity nextActivity)
|
||||||
{
|
{
|
||||||
if (!queued)
|
if (!queued)
|
||||||
|
|||||||
@@ -31,7 +31,6 @@ namespace OpenRA.Traits
|
|||||||
{
|
{
|
||||||
public readonly PPos[] Footprint;
|
public readonly PPos[] Footprint;
|
||||||
public readonly WPos CenterPosition;
|
public readonly WPos CenterPosition;
|
||||||
public readonly Rectangle RenderBounds;
|
|
||||||
public readonly Rectangle SelectableBounds;
|
public readonly Rectangle SelectableBounds;
|
||||||
public readonly HashSet<string> TargetTypes;
|
public readonly HashSet<string> TargetTypes;
|
||||||
readonly Actor actor;
|
readonly Actor actor;
|
||||||
@@ -51,7 +50,10 @@ namespace OpenRA.Traits
|
|||||||
public bool Shrouded { get; private set; }
|
public bool Shrouded { get; private set; }
|
||||||
public bool NeedRenderables { get; set; }
|
public bool NeedRenderables { get; set; }
|
||||||
public IRenderable[] Renderables = NoRenderables;
|
public IRenderable[] Renderables = NoRenderables;
|
||||||
|
public Rectangle[] ScreenBounds = NoBounds;
|
||||||
|
|
||||||
static readonly IRenderable[] NoRenderables = new IRenderable[0];
|
static readonly IRenderable[] NoRenderables = new IRenderable[0];
|
||||||
|
static readonly Rectangle[] NoBounds = new Rectangle[0];
|
||||||
|
|
||||||
int flashTicks;
|
int flashTicks;
|
||||||
|
|
||||||
@@ -78,7 +80,6 @@ namespace OpenRA.Traits
|
|||||||
footprint.Select(p => shroud.Contains(p).ToString()).JoinWith("|")));
|
footprint.Select(p => shroud.Contains(p).ToString()).JoinWith("|")));
|
||||||
|
|
||||||
CenterPosition = self.CenterPosition;
|
CenterPosition = self.CenterPosition;
|
||||||
RenderBounds = self.RenderBounds;
|
|
||||||
SelectableBounds = self.SelectableBounds;
|
SelectableBounds = self.SelectableBounds;
|
||||||
TargetTypes = self.GetEnabledTargetTypes().ToHashSet();
|
TargetTypes = self.GetEnabledTargetTypes().ToHashSet();
|
||||||
|
|
||||||
|
|||||||
@@ -61,22 +61,6 @@ namespace OpenRA.Traits
|
|||||||
|
|
||||||
public void WorldLoaded(World w, WorldRenderer wr) { worldRenderer = wr; }
|
public void WorldLoaded(World w, WorldRenderer wr) { worldRenderer = wr; }
|
||||||
|
|
||||||
Rectangle FrozenActorBounds(FrozenActor fa)
|
|
||||||
{
|
|
||||||
var pos = worldRenderer.ScreenPxPosition(fa.CenterPosition);
|
|
||||||
var bounds = fa.RenderBounds;
|
|
||||||
bounds.Offset(pos.X, pos.Y);
|
|
||||||
return bounds;
|
|
||||||
}
|
|
||||||
|
|
||||||
Rectangle ActorBounds(Actor a)
|
|
||||||
{
|
|
||||||
var pos = worldRenderer.ScreenPxPosition(a.CenterPosition);
|
|
||||||
var bounds = a.RenderBounds;
|
|
||||||
bounds.Offset(pos.X, pos.Y);
|
|
||||||
return bounds;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void AddOrUpdate(Player viewer, FrozenActor fa)
|
public void AddOrUpdate(Player viewer, FrozenActor fa)
|
||||||
{
|
{
|
||||||
if (removeFrozenActors[viewer].Contains(fa))
|
if (removeFrozenActors[viewer].Contains(fa))
|
||||||
@@ -200,11 +184,23 @@ namespace OpenRA.Traits
|
|||||||
return partitionedFrozenActors[p].InBox(r).Where(frozenActorIsValid);
|
return partitionedFrozenActors[p].InBox(r).Where(frozenActorIsValid);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Rectangle AggregateBounds(IEnumerable<Rectangle> screenBounds)
|
||||||
|
{
|
||||||
|
if (!screenBounds.Any())
|
||||||
|
return Rectangle.Empty;
|
||||||
|
|
||||||
|
var bounds = screenBounds.First();
|
||||||
|
foreach (var b in screenBounds.Skip(1))
|
||||||
|
bounds = Rectangle.Union(bounds, b);
|
||||||
|
|
||||||
|
return bounds;
|
||||||
|
}
|
||||||
|
|
||||||
public void Tick()
|
public void Tick()
|
||||||
{
|
{
|
||||||
foreach (var a in addOrUpdateActors)
|
foreach (var a in addOrUpdateActors)
|
||||||
{
|
{
|
||||||
var bounds = ActorBounds(a);
|
var bounds = AggregateBounds(a.ScreenBounds(worldRenderer));
|
||||||
if (!bounds.Size.IsEmpty)
|
if (!bounds.Size.IsEmpty)
|
||||||
{
|
{
|
||||||
if (partitionedActors.Contains(a))
|
if (partitionedActors.Contains(a))
|
||||||
@@ -226,7 +222,7 @@ namespace OpenRA.Traits
|
|||||||
{
|
{
|
||||||
foreach (var fa in kv.Value)
|
foreach (var fa in kv.Value)
|
||||||
{
|
{
|
||||||
var bounds = FrozenActorBounds(fa);
|
var bounds = AggregateBounds(fa.ScreenBounds);
|
||||||
if (!bounds.Size.IsEmpty)
|
if (!bounds.Size.IsEmpty)
|
||||||
{
|
{
|
||||||
if (partitionedFrozenActors[kv.Key].Contains(fa))
|
if (partitionedFrozenActors[kv.Key].Contains(fa))
|
||||||
|
|||||||
@@ -40,6 +40,7 @@ namespace OpenRA.Mods.Cnc.Traits.Render
|
|||||||
readonly Cargo cargo;
|
readonly Cargo cargo;
|
||||||
readonly BodyOrientation body;
|
readonly BodyOrientation body;
|
||||||
readonly IFacing facing;
|
readonly IFacing facing;
|
||||||
|
int cachedFacing;
|
||||||
|
|
||||||
Dictionary<Actor, IActorPreview[]> previews = new Dictionary<Actor, IActorPreview[]>();
|
Dictionary<Actor, IActorPreview[]> previews = new Dictionary<Actor, IActorPreview[]>();
|
||||||
|
|
||||||
@@ -58,6 +59,15 @@ namespace OpenRA.Mods.Cnc.Traits.Render
|
|||||||
if (actorPreviews != null)
|
if (actorPreviews != null)
|
||||||
foreach (var preview in actorPreviews)
|
foreach (var preview in actorPreviews)
|
||||||
preview.Tick();
|
preview.Tick();
|
||||||
|
|
||||||
|
// HACK: We don't have an efficient way to know when the preview
|
||||||
|
// bounds change, so assume that we need to update the screen map
|
||||||
|
// (only) when the facing changes
|
||||||
|
if (facing.Facing != cachedFacing && previews.Any())
|
||||||
|
{
|
||||||
|
self.World.ScreenMap.AddOrUpdate(self);
|
||||||
|
cachedFacing = facing.Facing;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
IEnumerable<IRenderable> IRender.Render(Actor self, WorldRenderer wr)
|
IEnumerable<IRenderable> IRender.Render(Actor self, WorldRenderer wr)
|
||||||
@@ -110,12 +120,16 @@ namespace OpenRA.Mods.Cnc.Traits.Render
|
|||||||
void INotifyPassengerEntered.OnPassengerEntered(Actor self, Actor passenger)
|
void INotifyPassengerEntered.OnPassengerEntered(Actor self, Actor passenger)
|
||||||
{
|
{
|
||||||
if (info.DisplayTypes.Contains(passenger.Trait<Passenger>().Info.CargoType))
|
if (info.DisplayTypes.Contains(passenger.Trait<Passenger>().Info.CargoType))
|
||||||
|
{
|
||||||
previews.Add(passenger, null);
|
previews.Add(passenger, null);
|
||||||
|
self.World.ScreenMap.AddOrUpdate(self);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void INotifyPassengerExited.OnPassengerExited(Actor self, Actor passenger)
|
void INotifyPassengerExited.OnPassengerExited(Actor self, Actor passenger)
|
||||||
{
|
{
|
||||||
previews.Remove(passenger);
|
previews.Remove(passenger);
|
||||||
|
self.World.ScreenMap.AddOrUpdate(self);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -61,6 +61,7 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
[Sync] public Actor Carryable { get; private set; }
|
[Sync] public Actor Carryable { get; private set; }
|
||||||
public CarryallState State { get; private set; }
|
public CarryallState State { get; private set; }
|
||||||
|
|
||||||
|
int cachedFacing;
|
||||||
IActorPreview[] carryablePreview = null;
|
IActorPreview[] carryablePreview = null;
|
||||||
|
|
||||||
/// <summary>Offset between the carryall's and the carried actor's CenterPositions</summary>
|
/// <summary>Offset between the carryall's and the carried actor's CenterPositions</summary>
|
||||||
@@ -84,6 +85,15 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
// Cargo may be killed in the same tick as, but after they are attached
|
// Cargo may be killed in the same tick as, but after they are attached
|
||||||
if (Carryable != null && Carryable.IsDead)
|
if (Carryable != null && Carryable.IsDead)
|
||||||
DetachCarryable(self);
|
DetachCarryable(self);
|
||||||
|
|
||||||
|
// HACK: We don't have an efficient way to know when the preview
|
||||||
|
// bounds change, so assume that we need to update the screen map
|
||||||
|
// (only) when the facing changes
|
||||||
|
if (facing.Facing != cachedFacing && carryablePreview != null)
|
||||||
|
{
|
||||||
|
self.World.ScreenMap.AddOrUpdate(self);
|
||||||
|
cachedFacing = facing.Facing;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void INotifyActorDisposing.Disposing(Actor self)
|
void INotifyActorDisposing.Disposing(Actor self)
|
||||||
@@ -126,6 +136,7 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
|
|
||||||
Carryable = carryable;
|
Carryable = carryable;
|
||||||
State = CarryallState.Carrying;
|
State = CarryallState.Carrying;
|
||||||
|
self.World.ScreenMap.AddOrUpdate(self);
|
||||||
|
|
||||||
CarryableOffset = OffsetForCarryable(self, carryable);
|
CarryableOffset = OffsetForCarryable(self, carryable);
|
||||||
return true;
|
return true;
|
||||||
@@ -134,6 +145,7 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
public virtual void DetachCarryable(Actor self)
|
public virtual void DetachCarryable(Actor self)
|
||||||
{
|
{
|
||||||
UnreserveCarryable(self);
|
UnreserveCarryable(self);
|
||||||
|
self.World.ScreenMap.AddOrUpdate(self);
|
||||||
|
|
||||||
carryablePreview = null;
|
carryablePreview = null;
|
||||||
CarryableOffset = WVec.Zero;
|
CarryableOffset = WVec.Zero;
|
||||||
|
|||||||
@@ -127,6 +127,7 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
void ITickRender.TickRender(WorldRenderer wr, Actor self)
|
void ITickRender.TickRender(WorldRenderer wr, Actor self)
|
||||||
{
|
{
|
||||||
IRenderable[] renderables = null;
|
IRenderable[] renderables = null;
|
||||||
|
Rectangle[] bounds = null;
|
||||||
for (var playerIndex = 0; playerIndex < frozenStates.Count; playerIndex++)
|
for (var playerIndex = 0; playerIndex < frozenStates.Count; playerIndex++)
|
||||||
{
|
{
|
||||||
var frozen = frozenStates[playerIndex].FrozenActor;
|
var frozen = frozenStates[playerIndex].FrozenActor;
|
||||||
@@ -137,11 +138,15 @@ namespace OpenRA.Mods.Common.Traits
|
|||||||
{
|
{
|
||||||
isRendering = true;
|
isRendering = true;
|
||||||
renderables = self.Render(wr).ToArray();
|
renderables = self.Render(wr).ToArray();
|
||||||
|
bounds = self.ScreenBounds(wr).ToArray();
|
||||||
|
|
||||||
isRendering = false;
|
isRendering = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
frozen.NeedRenderables = false;
|
frozen.NeedRenderables = false;
|
||||||
frozen.Renderables = renderables;
|
frozen.Renderables = renderables;
|
||||||
|
frozen.ScreenBounds = bounds;
|
||||||
|
self.World.ScreenMap.AddOrUpdate(self.World.Players[playerIndex], frozen);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -101,6 +101,10 @@ namespace OpenRA.Mods.Common.Traits.Render
|
|||||||
public readonly bool IsPlayerPalette;
|
public readonly bool IsPlayerPalette;
|
||||||
public PaletteReference PaletteReference { get; private set; }
|
public PaletteReference PaletteReference { get; private set; }
|
||||||
|
|
||||||
|
bool cachedVisible;
|
||||||
|
WVec cachedOffset;
|
||||||
|
ISpriteSequence cachedSequence;
|
||||||
|
|
||||||
public AnimationWrapper(AnimationWithOffset animation, string palette, bool isPlayerPalette)
|
public AnimationWrapper(AnimationWithOffset animation, string palette, bool isPlayerPalette)
|
||||||
{
|
{
|
||||||
Animation = animation;
|
Animation = animation;
|
||||||
@@ -127,6 +131,24 @@ namespace OpenRA.Mods.Common.Traits.Render
|
|||||||
return Animation.DisableFunc == null || !Animation.DisableFunc();
|
return Animation.DisableFunc == null || !Animation.DisableFunc();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public bool Tick()
|
||||||
|
{
|
||||||
|
// Tick the animation
|
||||||
|
Animation.Animation.Tick();
|
||||||
|
|
||||||
|
// Return to the caller whether the renderable position or size has changed
|
||||||
|
var visible = IsVisible;
|
||||||
|
var offset = Animation.OffsetFunc != null ? Animation.OffsetFunc() : WVec.Zero;
|
||||||
|
var sequence = Animation.Animation.CurrentSequence;
|
||||||
|
|
||||||
|
var updated = visible != cachedVisible || offset != cachedOffset || sequence != cachedSequence;
|
||||||
|
cachedVisible = visible;
|
||||||
|
cachedOffset = offset;
|
||||||
|
cachedSequence = sequence;
|
||||||
|
|
||||||
|
return updated;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
readonly string faction;
|
readonly string faction;
|
||||||
@@ -196,8 +218,12 @@ namespace OpenRA.Mods.Common.Traits.Render
|
|||||||
|
|
||||||
protected virtual void Tick(Actor self)
|
protected virtual void Tick(Actor self)
|
||||||
{
|
{
|
||||||
|
var updated = false;
|
||||||
foreach (var a in anims)
|
foreach (var a in anims)
|
||||||
a.Animation.Animation.Tick();
|
updated |= a.Tick();
|
||||||
|
|
||||||
|
if (updated)
|
||||||
|
self.World.ScreenMap.AddOrUpdate(self);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Add(AnimationWithOffset anim, string palette = null, bool isPlayerPalette = false)
|
public void Add(AnimationWithOffset anim, string palette = null, bool isPlayerPalette = false)
|
||||||
|
|||||||
@@ -70,9 +70,36 @@ namespace OpenRA.Mods.Common.Traits.Render
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public class RenderVoxels : IRender, INotifyOwnerChanged
|
public class RenderVoxels : IRender, ITick, INotifyOwnerChanged
|
||||||
{
|
{
|
||||||
|
class AnimationWrapper
|
||||||
|
{
|
||||||
|
readonly ModelAnimation model;
|
||||||
|
bool cachedVisible;
|
||||||
|
WVec cachedOffset;
|
||||||
|
|
||||||
|
public AnimationWrapper(ModelAnimation model)
|
||||||
|
{
|
||||||
|
this.model = model;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool Tick()
|
||||||
|
{
|
||||||
|
// Return to the caller whether the renderable position or size has changed
|
||||||
|
var visible = model.IsVisible;
|
||||||
|
var offset = model.OffsetFunc != null ? model.OffsetFunc() : WVec.Zero;
|
||||||
|
|
||||||
|
var updated = visible != cachedVisible || offset != cachedOffset;
|
||||||
|
cachedVisible = visible;
|
||||||
|
cachedOffset = offset;
|
||||||
|
|
||||||
|
return updated;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
readonly List<ModelAnimation> components = new List<ModelAnimation>();
|
readonly List<ModelAnimation> components = new List<ModelAnimation>();
|
||||||
|
readonly Dictionary<ModelAnimation, AnimationWrapper> wrappers = new Dictionary<ModelAnimation, AnimationWrapper>();
|
||||||
|
|
||||||
readonly Actor self;
|
readonly Actor self;
|
||||||
readonly RenderVoxelsInfo info;
|
readonly RenderVoxelsInfo info;
|
||||||
readonly BodyOrientation body;
|
readonly BodyOrientation body;
|
||||||
@@ -91,6 +118,16 @@ namespace OpenRA.Mods.Common.Traits.Render
|
|||||||
bool initializePalettes = true;
|
bool initializePalettes = true;
|
||||||
public void OnOwnerChanged(Actor self, Player oldOwner, Player newOwner) { initializePalettes = true; }
|
public void OnOwnerChanged(Actor self, Player oldOwner, Player newOwner) { initializePalettes = true; }
|
||||||
|
|
||||||
|
void ITick.Tick(Actor self)
|
||||||
|
{
|
||||||
|
var updated = false;
|
||||||
|
foreach (var w in wrappers.Values)
|
||||||
|
updated |= w.Tick();
|
||||||
|
|
||||||
|
if (updated)
|
||||||
|
self.World.ScreenMap.AddOrUpdate(self);
|
||||||
|
}
|
||||||
|
|
||||||
protected PaletteReference colorPalette, normalsPalette, shadowPalette;
|
protected PaletteReference colorPalette, normalsPalette, shadowPalette;
|
||||||
IEnumerable<IRenderable> IRender.Render(Actor self, WorldRenderer wr)
|
IEnumerable<IRenderable> IRender.Render(Actor self, WorldRenderer wr)
|
||||||
{
|
{
|
||||||
@@ -118,7 +155,16 @@ namespace OpenRA.Mods.Common.Traits.Render
|
|||||||
}
|
}
|
||||||
|
|
||||||
public string Image { get { return info.Image ?? self.Info.Name; } }
|
public string Image { get { return info.Image ?? self.Info.Name; } }
|
||||||
public void Add(ModelAnimation v) { components.Add(v); }
|
public void Add(ModelAnimation m)
|
||||||
public void Remove(ModelAnimation v) { components.Remove(v); }
|
{
|
||||||
|
components.Add(m);
|
||||||
|
wrappers.Add(m, new AnimationWrapper(m));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Remove(ModelAnimation m)
|
||||||
|
{
|
||||||
|
components.Remove(m);
|
||||||
|
wrappers.Remove(m);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user