Merge pull request #2905 from pchote/baserange

Add building range and structure placement rate limiting to C&C
This commit is contained in:
Matthias Mailänder
2013-03-30 03:15:22 -07:00
8 changed files with 159 additions and 19 deletions

View File

@@ -0,0 +1,88 @@
#region Copyright & License Information
/*
* Copyright 2007-2011 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.Drawing;
using OpenRA.Graphics;
using OpenRA.Traits;
namespace OpenRA.Mods.RA.Buildings
{
public class BaseProviderInfo : ITraitInfo
{
public readonly float Range = 10;
public readonly int Cooldown = 0;
public readonly int InitialDelay = 0;
public object Create(ActorInitializer init) { return new BaseProvider(init.self, this); }
}
public class BaseProvider : ITick, IPreRenderSelection, ISelectionBar
{
public readonly BaseProviderInfo Info;
DeveloperMode devMode;
Actor self;
int total;
int progress;
public BaseProvider(Actor self, BaseProviderInfo info)
{
Info = info;
this.self = self;
devMode = self.Owner.PlayerActor.Trait<DeveloperMode>();
progress = total = info.InitialDelay;
}
public void Tick(Actor self)
{
if (progress > 0)
progress--;
}
public void BeginCooldown()
{
progress = total = Info.Cooldown;
}
public bool Ready()
{
return devMode.FastBuild || progress == 0;
}
// Range circle
public void RenderBeforeWorld(WorldRenderer wr, Actor self)
{
// Visible to player and allies
if (self.World.RenderedPlayer != null && self.Owner.Stances[self.World.RenderedPlayer] != Stance.Ally)
return;
wr.DrawRangeCircleWithContrast(
Color.FromArgb(128, Ready() ? Color.White : Color.Red),
self.CenterLocation.ToFloat2(), Info.Range,
Color.FromArgb(96, Color.Black), 1);
}
// Selection bar
public float GetValue()
{
// Visible to player and allies
if (self.World.RenderedPlayer != null && self.Owner.Stances[self.World.RenderedPlayer] != Stance.Ally)
return 0f;
// Ready or delay disabled
if (progress == 0 || total == 0 || devMode.FastBuild)
return 0f;
return (float)progress / total;
}
public Color GetColor() { return Color.Purple; }
}
}

View File

@@ -31,41 +31,64 @@ namespace OpenRA.Mods.RA.Buildings
[Desc("x means space it blocks, _ is a part that is passable by actors.")]
public readonly string Footprint = "x";
public readonly int2 Dimensions = new int2(1, 1);
public readonly bool RequiresBaseProvider = false;
public readonly string[] BuildSounds = {"placbldg.aud", "build5.aud"};
public readonly string[] SellSounds = {"cashturn.aud"};
public object Create(ActorInitializer init) { return new Building(init, this); }
public PPos CenterLocation(CPos topLeft)
{
return (PPos)((2 * topLeft.ToInt2() + Dimensions) * Game.CellSize / 2);
}
bool HasBaseProvider(World world, Player p, CPos topLeft)
{
var center = CenterLocation(topLeft);
foreach (var bp in world.ActorsWithTrait<BaseProvider>())
{
if (bp.Actor.Owner.Stances[p] != Stance.Ally || !bp.Trait.Ready())
continue;
if (Combat.IsInRange(center, bp.Trait.Info.Range, bp.Actor.CenterLocation))
return true;
}
return false;
}
public bool IsCloseEnoughToBase(World world, Player p, string buildingName, CPos topLeft)
{
if (p.PlayerActor.Trait<DeveloperMode>().BuildAnywhere)
return true;
if (RequiresBaseProvider && !HasBaseProvider(world, p, topLeft))
return false;
var buildingMaxBounds = (CVec)Dimensions;
if( Rules.Info[ buildingName ].Traits.Contains<BibInfo>() )
if (Rules.Info[buildingName].Traits.Contains<BibInfo>())
buildingMaxBounds += new CVec(0, 1);
var scanStart = world.ClampToWorld( topLeft - new CVec( Adjacent, Adjacent ) );
var scanStart = world.ClampToWorld(topLeft - new CVec(Adjacent, Adjacent));
var scanEnd = world.ClampToWorld(topLeft + buildingMaxBounds + new CVec(Adjacent, Adjacent));
var nearnessCandidates = new List<CPos>();
var bi = world.WorldActor.Trait<BuildingInfluence>();
for( int y = scanStart.Y ; y < scanEnd.Y ; y++ )
for( int x = scanStart.X ; x < scanEnd.X ; x++ )
for (var y = scanStart.Y; y < scanEnd.Y; y++)
for (var x = scanStart.X; x < scanEnd.X; x++)
{
var at = bi.GetBuildingAt( new CPos( x, y ) );
if( at != null && at.Owner.Stances[ p ] == Stance.Ally && at.HasTrait<GivesBuildableArea>() )
nearnessCandidates.Add( new CPos( x, y ) );
var at = bi.GetBuildingAt(new CPos(x, y));
if (at != null && at.Owner.Stances[p] == Stance.Ally && at.HasTrait<GivesBuildableArea>())
nearnessCandidates.Add(new CPos(x, y));
}
var buildingTiles = FootprintUtils.Tiles( buildingName, this, topLeft ).ToList();
var buildingTiles = FootprintUtils.Tiles(buildingName, this, topLeft).ToList();
return nearnessCandidates
.Any( a => buildingTiles
.Any( b => Math.Abs( a.X - b.X ) <= Adjacent
&& Math.Abs( a.Y - b.Y ) <= Adjacent ) );
.Any(a => buildingTiles
.Any(b => Math.Abs(a.X - b.X) <= Adjacent
&& Math.Abs(a.Y - b.Y) <= Adjacent));
}
}
@@ -103,7 +126,7 @@ namespace OpenRA.Mods.RA.Buildings
occupiedCells = FootprintUtils.UnpathableTiles( self.Info.Name, Info, TopLeft )
.Select(c => Pair.New(c, SubCell.FullCell)).ToArray();
pxPosition = (PPos) (( 2 * topLeft.ToInt2() + Info.Dimensions ) * Game.CellSize / 2);
pxPosition = Info.CenterLocation(topLeft);
}
public int GetPowerUsage()

View File

@@ -422,6 +422,7 @@
<Compile Include="Infiltrates.cs" />
<Compile Include="Armament.cs" />
<Compile Include="DebugMuzzlePositions.cs" />
<Compile Include="Buildings\BaseProvider.cs" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\OpenRA.FileFormats\OpenRA.FileFormats.csproj">

View File

@@ -80,6 +80,22 @@ namespace OpenRA.Mods.RA
queue.FinishProduction();
if (buildingInfo.RequiresBaseProvider)
{
var center = buildingInfo.CenterLocation(order.TargetLocation);
foreach (var bp in w.ActorsWithTrait<BaseProvider>())
{
if (bp.Actor.Owner.Stances[self.Owner] != Stance.Ally || !bp.Trait.Ready())
continue;
if (Combat.IsInRange(center, bp.Trait.Info.Range, bp.Actor.CenterLocation))
{
bp.Trait.BeginCooldown();
break;
}
}
}
if (GetNumBuildables(self.Owner) > prevItems)
w.Add(new DelayedAction(10,
() => Sound.PlayNotification(order.Player, "Speech", "NewOptions", order.Player.Country.Race)));

View File

@@ -19,7 +19,7 @@ using OpenRA.Mods.RA.Activities;
namespace OpenRA.Mods.RA.Render
{
public class RenderBuildingInfo : RenderSimpleInfo
public class RenderBuildingInfo : RenderSimpleInfo, Requires<BuildingInfo>, IPlaceBuildingDecoration
{
public readonly bool HasMakeAnimation = true;
public readonly float2 Origin = float2.Zero;
@@ -30,6 +30,15 @@ namespace OpenRA.Mods.RA.Render
return base.RenderPreview(building, pr)
.Select(a => a.WithPos(a.Pos + building.Traits.Get<RenderBuildingInfo>().Origin));
}
public void Render(WorldRenderer wr, World w, ActorInfo ai, PPos centerLocation)
{
if (!ai.Traits.Get<BuildingInfo>().RequiresBaseProvider)
return;
foreach (var a in w.ActorsWithTrait<BaseProvider>())
a.Trait.RenderBeforeWorld(wr, a.Actor);
}
}
public class RenderBuilding : RenderSimple, INotifyDamageStateChanged, IRenderModifier

View File

@@ -232,6 +232,7 @@
RepairPercent: 40
RepairStep: 14
Building:
RequiresBaseProvider: true
Dimensions: 1,1
Footprint: x
BuildSounds: constru2.aud, hvydoor1.aud

View File

@@ -45,6 +45,8 @@ FACT:
ReadyAudio: ConstructionComplete
BaseBuilding:
ProductionBar:
BaseProvider:
Cooldown: 200
NUKE:
Inherits: ^Building

View File

@@ -260,24 +260,24 @@ CRATE:
TerrainTypes: Clear, Rough, Road, Tiberium, BlueTiberium, Beach
GiveCashCrateAction:
Amount: 1000
SelectionShares: 20000
SelectionShares: 20
UseCashTick: yes
RevealMapCrateAction:
SelectionShares: 1000
SelectionShares: 1
Effect: reveal-map
ExplodeCrateAction@fire:
Weapon: Napalm.Crate
SelectionShares: 5000
SelectionShares: 5
CloakCrateAction:
SelectionShares: 5000
SelectionShares: 5
InitialDelay: 15
CloakDelay: 80
CloakSound: trans1.aud
UncloakSound: trans1.aud
Effect: cloak
GiveMcvCrateAction:
SelectionShares: 1
NoBaseSelectionShares: 9000001
SelectionShares: 2
NoBaseSelectionShares: 9001
Unit: mcv
RenderSimple:
BelowUnits: