Initial commit: OpenRA game engine
Fork from OpenRA/OpenRA with one-click launch script (start-ra.cmd) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
143
OpenRA.Mods.Common/Traits/SpawnActorsOnSell.cs
Normal file
143
OpenRA.Mods.Common/Traits/SpawnActorsOnSell.cs
Normal file
@@ -0,0 +1,143 @@
|
||||
#region Copyright & License Information
|
||||
/*
|
||||
* Copyright (c) The OpenRA Developers and Contributors
|
||||
* 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, either version 3 of
|
||||
* the License, or (at your option) any later version. For more
|
||||
* information, see COPYING.
|
||||
*/
|
||||
#endregion
|
||||
|
||||
using System.Collections.Frozen;
|
||||
using System.Collections.Immutable;
|
||||
using System.Linq;
|
||||
using OpenRA.Traits;
|
||||
|
||||
namespace OpenRA.Mods.Common.Traits
|
||||
{
|
||||
[Desc("Spawn new actors when sold.")]
|
||||
public class SpawnActorsOnSellInfo : ConditionalTraitInfo
|
||||
{
|
||||
public readonly int ValuePercent = 40;
|
||||
public readonly int MinHpPercent = 30;
|
||||
|
||||
[ActorReference]
|
||||
[FieldLoader.Require]
|
||||
[Desc("Actor types to spawn on sell, amount and type based on ValuePercent. Be sure to use lowercase.")]
|
||||
public readonly ImmutableArray<string> ActorTypes = default;
|
||||
|
||||
[ActorReference]
|
||||
[Desc("Actors to spawn on sell. Be sure to use lowercase.")]
|
||||
public readonly ImmutableArray<string> GuaranteedActorTypes = [];
|
||||
|
||||
[Desc("Spawns actors only if the selling player's faction is in this list. " +
|
||||
"Leave empty to allow all factions by default.")]
|
||||
public readonly FrozenSet<string> Factions = FrozenSet<string>.Empty;
|
||||
|
||||
public override object Create(ActorInitializer init) { return new SpawnActorsOnSell(init.Self, this); }
|
||||
}
|
||||
|
||||
public class SpawnActorsOnSell : ConditionalTrait<SpawnActorsOnSellInfo>, INotifySold
|
||||
{
|
||||
readonly bool correctFaction;
|
||||
|
||||
public SpawnActorsOnSell(Actor self, SpawnActorsOnSellInfo info)
|
||||
: base(info)
|
||||
{
|
||||
var factionsList = info.Factions;
|
||||
correctFaction = factionsList.Count == 0 || factionsList.Contains(self.Owner.Faction.InternalName);
|
||||
}
|
||||
|
||||
void INotifySold.Selling(Actor self) { }
|
||||
|
||||
void Emit(Actor self)
|
||||
{
|
||||
if (IsTraitDisabled || !correctFaction)
|
||||
return;
|
||||
|
||||
var buildingInfo = self.Info.TraitInfoOrDefault<BuildingInfo>();
|
||||
if (buildingInfo == null)
|
||||
return;
|
||||
|
||||
var csv = self.Info.TraitInfoOrDefault<CustomSellValueInfo>();
|
||||
var valued = self.Info.TraitInfoOrDefault<ValuedInfo>();
|
||||
var cost = csv?.Value ?? valued?.Cost ?? 0;
|
||||
|
||||
var health = self.TraitOrDefault<IHealth>();
|
||||
var dudesValue = Info.ValuePercent * cost / 100;
|
||||
if (health != null)
|
||||
{
|
||||
// Cast to long to avoid overflow when multiplying by the health
|
||||
if (100L * health.HP >= Info.MinHpPercent * (long)health.MaxHP)
|
||||
dudesValue = (int)((long)health.HP * dudesValue / health.MaxHP);
|
||||
else
|
||||
dudesValue = 0;
|
||||
}
|
||||
|
||||
var eligibleLocations = buildingInfo.Tiles(self.Location).ToList();
|
||||
|
||||
if (eligibleLocations.Count == 0)
|
||||
return;
|
||||
|
||||
if (Info.GuaranteedActorTypes.Length > 0)
|
||||
{
|
||||
var guaranteedActorTypes = Info.GuaranteedActorTypes.Select(a =>
|
||||
{
|
||||
var av = self.World.Map.Rules.Actors[a].TraitInfoOrDefault<ValuedInfo>();
|
||||
return new
|
||||
{
|
||||
Name = a,
|
||||
Cost = av?.Cost ?? 0
|
||||
};
|
||||
}).ToList();
|
||||
|
||||
while (eligibleLocations.Count > 0 && guaranteedActorTypes.Count > 0)
|
||||
{
|
||||
var at = guaranteedActorTypes.Random(self.World.SharedRandom);
|
||||
var loc = eligibleLocations.Random(self.World.SharedRandom);
|
||||
|
||||
eligibleLocations.Remove(loc);
|
||||
guaranteedActorTypes.Remove(at);
|
||||
dudesValue -= at.Cost;
|
||||
|
||||
self.World.AddFrameEndTask(w => w.CreateActor(at.Name,
|
||||
[
|
||||
new LocationInit(loc),
|
||||
new OwnerInit(self.Owner),
|
||||
]));
|
||||
}
|
||||
|
||||
if (eligibleLocations.Count == 0)
|
||||
return;
|
||||
}
|
||||
|
||||
var actorTypes = Info.ActorTypes.Select(a =>
|
||||
{
|
||||
var av = self.World.Map.Rules.Actors[a].TraitInfoOrDefault<ValuedInfo>();
|
||||
return new
|
||||
{
|
||||
Name = a,
|
||||
Cost = av?.Cost ?? 0
|
||||
};
|
||||
}).ToList();
|
||||
|
||||
while (eligibleLocations.Count > 0 && actorTypes.Any(a => a.Cost <= dudesValue))
|
||||
{
|
||||
var at = actorTypes.Where(a => a.Cost <= dudesValue).Random(self.World.SharedRandom);
|
||||
var loc = eligibleLocations.Random(self.World.SharedRandom);
|
||||
|
||||
eligibleLocations.Remove(loc);
|
||||
dudesValue -= at.Cost;
|
||||
|
||||
self.World.AddFrameEndTask(w => w.CreateActor(at.Name,
|
||||
[
|
||||
new LocationInit(loc),
|
||||
new OwnerInit(self.Owner),
|
||||
]));
|
||||
}
|
||||
}
|
||||
|
||||
void INotifySold.Sold(Actor self) { Emit(self); }
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user