clean up CrateDrop
This commit is contained in:
@@ -18,7 +18,7 @@ using OpenRA.Traits;
|
|||||||
|
|
||||||
namespace OpenRA.Mods.RA
|
namespace OpenRA.Mods.RA
|
||||||
{
|
{
|
||||||
public class CrateDropInfo : TraitInfo<CrateDrop>
|
public class CrateDropInfo : ITraitInfo
|
||||||
{
|
{
|
||||||
public readonly int Minimum = 1; // Minumum number of crates
|
public readonly int Minimum = 1; // Minumum number of crates
|
||||||
public readonly int Maximum = 255; // Maximum number of crates
|
public readonly int Maximum = 255; // Maximum number of crates
|
||||||
@@ -26,67 +26,80 @@ namespace OpenRA.Mods.RA
|
|||||||
public readonly string[] ValidWater = {"Water"};
|
public readonly string[] ValidWater = {"Water"};
|
||||||
public readonly int SpawnInterval = 180; // Average time (seconds) between crate spawn
|
public readonly int SpawnInterval = 180; // Average time (seconds) between crate spawn
|
||||||
public readonly float WaterChance = .2f; // Chance of generating a water crate instead of a land crate
|
public readonly float WaterChance = .2f; // Chance of generating a water crate instead of a land crate
|
||||||
|
|
||||||
|
public object Create (ActorInitializer init) { return new CrateDrop(this); }
|
||||||
}
|
}
|
||||||
|
|
||||||
public class CrateDrop : ITick
|
public class CrateDrop : ITick
|
||||||
{
|
{
|
||||||
List<Actor> crates = new List<Actor>();
|
List<Actor> crates = new List<Actor>();
|
||||||
int ticks = 0;
|
int ticks = 0;
|
||||||
|
CrateDropInfo Info;
|
||||||
|
|
||||||
|
public CrateDrop(CrateDropInfo info) { Info = info; }
|
||||||
|
|
||||||
public void Tick(Actor self)
|
public void Tick(Actor self)
|
||||||
{
|
{
|
||||||
if (--ticks <= 0)
|
if (--ticks <= 0)
|
||||||
{
|
{
|
||||||
var info = self.Info.Traits.Get<CrateDropInfo>();
|
ticks = Info.SpawnInterval * 25; // todo: randomize
|
||||||
ticks = info.SpawnInterval * 25; // todo: randomize
|
|
||||||
|
|
||||||
crates.RemoveAll(x => !x.IsInWorld); // BUG: this removes crates that are cargo of a BADR!
|
crates.RemoveAll(x => !x.IsInWorld); // BUG: this removes crates that are cargo of a BADR!
|
||||||
|
|
||||||
var toSpawn = Math.Max(0, info.Minimum - crates.Count)
|
var toSpawn = Math.Max(0, Info.Minimum - crates.Count)
|
||||||
+ (crates.Count < info.Maximum ? 1 : 0);
|
+ (crates.Count < Info.Maximum ? 1 : 0);
|
||||||
|
|
||||||
for (var n = 0; n < toSpawn; n++)
|
for (var n = 0; n < toSpawn; n++)
|
||||||
SpawnCrate(self, info);
|
SpawnCrate(self);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void SpawnCrate(Actor self, CrateDropInfo info)
|
int2? ChooseDropCell(Actor self, bool inWater, int maxTries)
|
||||||
{
|
{
|
||||||
var threshold = 100;
|
for( var n = 0; n < maxTries; n++ )
|
||||||
var inWater = self.World.SharedRandom.NextDouble() < info.WaterChance;
|
|
||||||
|
|
||||||
for (var n = 0; n < threshold; n++)
|
|
||||||
{
|
{
|
||||||
var p = self.World.ChooseRandomCell(self.World.SharedRandom);
|
var p = self.World.ChooseRandomCell(self.World.SharedRandom);
|
||||||
|
|
||||||
// Is this valid terrain?
|
// Is this valid terrain?
|
||||||
var terrainType = self.World.GetTerrainType(p);
|
var terrainType = self.World.GetTerrainType(p);
|
||||||
if (!(inWater ? info.ValidWater : info.ValidGround).Contains(terrainType)) continue;
|
if (!(inWater ? Info.ValidWater : Info.ValidGround).Contains(terrainType)) continue;
|
||||||
|
|
||||||
// Don't drop on any actors
|
// Don't drop on any actors
|
||||||
if (self.World.WorldActor.Trait<BuildingInfluence>().GetBuildingAt(p) != null) continue;
|
if (self.World.WorldActor.Trait<BuildingInfluence>().GetBuildingAt(p) != null) continue;
|
||||||
if (self.World.WorldActor.Trait<UnitInfluence>().GetUnitsAt(p).Any()) continue;
|
if (self.World.WorldActor.Trait<UnitInfluence>().GetUnitsAt(p).Any()) continue;
|
||||||
|
|
||||||
self.World.AddFrameEndTask(w =>
|
return p;
|
||||||
{
|
|
||||||
var crate = w.CreateActor(false, "crate", new TypeDictionary { new OwnerInit(w.WorldActor.Owner) });
|
|
||||||
crates.Add(crate);
|
|
||||||
|
|
||||||
var startPos = w.ChooseRandomEdgeCell();
|
|
||||||
var plane = w.CreateActor("badr", new TypeDictionary
|
|
||||||
{
|
|
||||||
new LocationInit( startPos ),
|
|
||||||
new OwnerInit( w.WorldActor.Owner),
|
|
||||||
new FacingInit( Util.GetFacing(p - startPos, 0) ),
|
|
||||||
new AltitudeInit( Rules.Info["badr"].Traits.Get<AircraftInfo>().CruiseAltitude ),
|
|
||||||
});
|
|
||||||
plane.CancelActivity();
|
|
||||||
plane.QueueActivity(new FlyAttack(Target.FromCell(p)));
|
|
||||||
plane.Trait<ParaDrop>().SetLZ(p);
|
|
||||||
plane.Trait<Cargo>().Load(plane, crate);
|
|
||||||
});
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SpawnCrate(Actor self)
|
||||||
|
{
|
||||||
|
var inWater = self.World.SharedRandom.NextDouble() < Info.WaterChance;
|
||||||
|
var pp = ChooseDropCell(self, inWater, 100);
|
||||||
|
if (pp == null) return;
|
||||||
|
|
||||||
|
var p = pp.Value; //
|
||||||
|
self.World.AddFrameEndTask(w =>
|
||||||
|
{
|
||||||
|
var crate = w.CreateActor(false, "crate", new TypeDictionary { new OwnerInit(w.WorldActor.Owner) });
|
||||||
|
crates.Add(crate);
|
||||||
|
|
||||||
|
var startPos = w.ChooseRandomEdgeCell();
|
||||||
|
var plane = w.CreateActor("badr", new TypeDictionary
|
||||||
|
{
|
||||||
|
new LocationInit( startPos ),
|
||||||
|
new OwnerInit( w.WorldActor.Owner),
|
||||||
|
new FacingInit( Util.GetFacing(p - startPos, 0) ),
|
||||||
|
new AltitudeInit( Rules.Info["badr"].Traits.Get<AircraftInfo>().CruiseAltitude ),
|
||||||
|
});
|
||||||
|
|
||||||
|
plane.CancelActivity();
|
||||||
|
plane.QueueActivity(new FlyAttack(Target.FromCell(p)));
|
||||||
|
plane.Trait<ParaDrop>().SetLZ(p);
|
||||||
|
plane.Trait<Cargo>().Load(plane, crate);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user