mirror of
https://github.com/dcs-liberation/dcs_liberation.git
synced 2025-11-10 14:22:26 +00:00
The previous flight plan only makes sense if the convoy will make it a significant distance from its starting point. At road speeds over the typical mission duration this is not true, so we can actually plan this as if it was a strike mission near the origin point and that's close enough. There's some cleanup work to do here that I've added todos for. Fixes https://github.com/Khopa/dcs_liberation/issues/996
95 lines
2.9 KiB
Python
95 lines
2.9 KiB
Python
from __future__ import annotations
|
|
|
|
import itertools
|
|
from typing import Dict, TYPE_CHECKING, Type
|
|
|
|
from dcs import Mission
|
|
from dcs.mapping import Point
|
|
from dcs.point import PointAction
|
|
from dcs.unit import Vehicle
|
|
from dcs.unitgroup import VehicleGroup
|
|
from dcs.unittype import VehicleType
|
|
|
|
from game.transfers import RoadTransferOrder
|
|
from game.unitmap import UnitMap
|
|
from game.utils import kph
|
|
|
|
if TYPE_CHECKING:
|
|
from game import Game
|
|
|
|
|
|
class ConvoyGenerator:
|
|
def __init__(self, mission: Mission, game: Game, unit_map: UnitMap) -> None:
|
|
self.mission = mission
|
|
self.game = game
|
|
self.unit_map = unit_map
|
|
self.count = itertools.count()
|
|
|
|
def generate(self) -> None:
|
|
# Reset the count to make generation deterministic.
|
|
for transfer in self.game.transfers:
|
|
self.generate_convoy_for(transfer)
|
|
|
|
def generate_convoy_for(self, transfer: RoadTransferOrder) -> VehicleGroup:
|
|
next_hop = transfer.path()[0]
|
|
origin = transfer.position.convoy_spawns[next_hop]
|
|
destination = next_hop.convoy_spawns[transfer.position]
|
|
|
|
group = self._create_mixed_unit_group(
|
|
transfer.name,
|
|
origin,
|
|
transfer.units,
|
|
transfer.player,
|
|
)
|
|
group.add_waypoint(
|
|
destination, speed=kph(40).kph, move_formation=PointAction.OnRoad
|
|
)
|
|
self.make_drivable(group)
|
|
self.unit_map.add_convoy_units(group, transfer)
|
|
return group
|
|
|
|
def _create_mixed_unit_group(
|
|
self,
|
|
name: str,
|
|
position: Point,
|
|
units: Dict[Type[VehicleType], int],
|
|
for_player: bool,
|
|
) -> VehicleGroup:
|
|
country = self.mission.country(
|
|
self.game.player_country if for_player else self.game.enemy_country
|
|
)
|
|
|
|
unit_types = list(units.items())
|
|
main_unit_type, main_unit_count = unit_types[0]
|
|
|
|
group = self.mission.vehicle_group(
|
|
country,
|
|
name,
|
|
main_unit_type,
|
|
position=position,
|
|
group_size=main_unit_count,
|
|
move_formation=PointAction.OnRoad,
|
|
)
|
|
|
|
unit_name_counter = itertools.count(main_unit_count + 1)
|
|
# pydcs spreads units out by 20 in the Y axis by default. Pick up where it left
|
|
# off.
|
|
y = itertools.count(position.y + main_unit_count * 20, 20)
|
|
for unit_type, count in unit_types[1:]:
|
|
for i in range(count):
|
|
v = self.mission.vehicle(
|
|
f"{name} Unit #{next(unit_name_counter)}", unit_type
|
|
)
|
|
v.position.x = position.x
|
|
v.position.y = next(y)
|
|
v.heading = 0
|
|
group.add_unit(v)
|
|
|
|
return group
|
|
|
|
@staticmethod
|
|
def make_drivable(group: VehicleGroup) -> None:
|
|
for v in group.units:
|
|
if isinstance(v, Vehicle):
|
|
v.player_can_drive = True
|